Merge pull request #10580 from ethereum/sized-types

Disallow leading zeroes in sized-types
This commit is contained in:
chriseth 2020-12-14 14:44:06 +01:00 committed by GitHub
commit e3b009d6a7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 71 additions and 19 deletions

View File

@ -26,6 +26,7 @@ Bugfixes:
* SMTChecker: Fix internal error when trying to generate counterexamples with old z3. * SMTChecker: Fix internal error when trying to generate counterexamples with old z3.
* SMTChecker: Fix segmentation fault that could occur on certain SMT-enabled sources when no SMT solver was available. * SMTChecker: Fix segmentation fault that could occur on certain SMT-enabled sources when no SMT solver was available.
* Type Checker: ``super`` is not available in libraries. * Type Checker: ``super`` is not available in libraries.
* Type Checker: Disallow leading zeroes in sized-types (e.g. ``bytes000032``), but allow them to be treated as identifiers.
* Yul Optimizer: Fix a bug in NameSimplifier where a new name created by NameSimplifier could also be created by NameDispenser. * Yul Optimizer: Fix a bug in NameSimplifier where a new name created by NameSimplifier could also be created by NameDispenser.
* Yul Optimizer: Removed NameSimplifier from optimization steps available to users. * Yul Optimizer: Removed NameSimplifier from optimization steps available to users.

View File

@ -33,7 +33,7 @@ Event: 'event';
External: 'external'; External: 'external';
Fallback: 'fallback'; Fallback: 'fallback';
False: 'false'; False: 'false';
Fixed: 'fixed' | ('fixed' [0-9]+ 'x' [0-9]+); Fixed: 'fixed' | ('fixed' [1-9][0-9]* 'x' [1-9][0-9]*);
From: 'from'; From: 'from';
/** /**
* Bytes types of fixed length. * Bytes types of fixed length.
@ -86,7 +86,7 @@ Struct: 'struct';
True: 'true'; True: 'true';
Try: 'try'; Try: 'try';
Type: 'type'; Type: 'type';
Ufixed: 'ufixed' | ('ufixed' [0-9]+ 'x' [0-9]+); Ufixed: 'ufixed' | ('ufixed' [1-9][0-9]+ 'x' [1-9][0-9]+);
/** /**
* Sized unsigned integer types. * Sized unsigned integer types.
* uint is an alias of uint256. * uint is an alias of uint256.

View File

@ -41,7 +41,6 @@
// along with solidity. If not, see <http://www.gnu.org/licenses/>. // along with solidity. If not, see <http://www.gnu.org/licenses/>.
#include <liblangutil/Token.h> #include <liblangutil/Token.h>
#include <boost/range/iterator_range.hpp>
#include <map> #include <map>
using namespace std; using namespace std;
@ -126,22 +125,6 @@ int precedence(Token tok)
} }
#undef T #undef T
namespace
{
int parseSize(string::const_iterator _begin, string::const_iterator _end)
{
try
{
int m = boost::lexical_cast<int>(boost::make_iterator_range(_begin, _end));
return m;
}
catch(boost::bad_lexical_cast const&)
{
return -1;
}
}
}
static Token keywordByName(string const& _name) static Token keywordByName(string const& _name)
{ {
// The following macros are used inside TOKEN_LIST and cause non-keyword tokens to be ignored // The following macros are used inside TOKEN_LIST and cause non-keyword tokens to be ignored
@ -162,6 +145,32 @@ bool isYulKeyword(string const& _literal)
tuple<Token, unsigned int, unsigned int> fromIdentifierOrKeyword(string const& _literal) tuple<Token, unsigned int, unsigned int> fromIdentifierOrKeyword(string const& _literal)
{ {
// Used for `bytesM`, `uintM`, `intM`, `fixedMxN`, `ufixedMxN`.
// M/N must be shortest representation. M can never be 0. N can be zero.
auto parseSize = [](string::const_iterator _begin, string::const_iterator _end) -> int
{
// No number.
if (distance(_begin, _end) == 0)
return -1;
// Disallow leading zero.
if (distance(_begin, _end) > 1 && *_begin == '0')
return -1;
int ret = 0;
for (auto it = _begin; it != _end; it++)
{
if (*it < '0' || *it > '9')
return -1;
// Overflow check. The largest acceptable value is 256 in the callers.
if (ret >= 256)
return -1;
ret *= 10;
ret += *it - '0';
}
return ret;
};
auto positionM = find_if(_literal.begin(), _literal.end(), ::isdigit); auto positionM = find_if(_literal.begin(), _literal.end(), ::isdigit);
if (positionM != _literal.end()) if (positionM != _literal.end())
{ {

View File

@ -0,0 +1,42 @@
contract C {
uint bytes01 = 0;
uint bytes000000001 = 0;
uint bytes000099000 = 0;
uint bytes0a = 0;
uint int0 = 0;
uint int01 = 0;
uint int000000001 = 0;
uint int000099000 = 0;
uint int0a = 0;
uint uint0 = 0;
uint uint01 = 0;
uint uint000000001 = 0;
uint uint000099000 = 0;
uint uint0a = 0;
uint fixed0x0 = 0;
uint fixed01x1 = 0;
uint fixed1x01 = 0;
uint fixed000000001x1 = 0;
uint fixed1x000000001 = 0;
uint fixed000099000x1 = 0;
uint fixed1x000099000 = 0;
uint fixed0ax1 = 0;
uint fixed1x0a = 0;
uint ufixed0x0 = 0;
uint ufixed01x1 = 0;
uint ufixed1x01 = 0;
uint ufixed000000001x1 = 0;
uint ufixed1x000000001 = 0;
uint ufixed000099000x1 = 0;
uint ufixed1x000099000 = 0;
uint ufixed0ax1 = 0;
uint ufixed1x0a = 0;
// overflow check
uint uint300 = 0;
}
// ----