Merge pull request #8930 from ethereum/switch-single-default-warn

Warn on YUL switch statement with only default statement
This commit is contained in:
chriseth 2020-06-09 14:58:17 +02:00 committed by GitHub
commit bec9b24c5b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 73 additions and 41 deletions

View File

@ -4,6 +4,7 @@ Language Features:
Compiler Features: Compiler Features:
* Yul: Raise warning for switch statements that only have a default and no other cases.
Bugfixes: Bugfixes:

View File

@ -359,6 +359,13 @@ void AsmAnalyzer::operator()(Switch const& _switch)
{ {
yulAssert(_switch.expression, ""); yulAssert(_switch.expression, "");
if (_switch.cases.size() == 1 && !_switch.cases[0].value)
m_errorReporter.warning(
9592_error,
_switch.location,
"\"switch\" statement with only a default case."
);
YulString valueType = expectExpression(*_switch.expression); YulString valueType = expectExpression(*_switch.expression);
set<u256> cases; set<u256> cases;

View File

@ -33,14 +33,14 @@
{ {
"id": 4, "id": 4,
"nodeType": "Block", "nodeType": "Block",
"src": "42:48:1", "src": "42:58:1",
"statements": "statements":
[ [
{ {
"AST": "AST":
{ {
"nodeType": "YulBlock", "nodeType": "YulBlock",
"src": "61:23:1", "src": "61:33:1",
"statements": "statements":
[ [
{ {
@ -50,11 +50,29 @@
"body": "body":
{ {
"nodeType": "YulBlock", "nodeType": "YulBlock",
"src": "80:2:1", "src": "79:2:1",
"statements": [] "statements": []
}, },
"nodeType": "YulCase", "nodeType": "YulCase",
"src": "72:10:1", "src": "72:9:1",
"value":
{
"kind": "number",
"nodeType": "YulLiteral",
"src": "77:1:1",
"type": "",
"value": "0"
}
},
{
"body":
{
"nodeType": "YulBlock",
"src": "90:2:1",
"statements": []
},
"nodeType": "YulCase",
"src": "82:10:1",
"value": "default" "value": "default"
} }
], ],
@ -67,7 +85,7 @@
"value": "0" "value": "0"
}, },
"nodeType": "YulSwitch", "nodeType": "YulSwitch",
"src": "63:19:1" "src": "63:29:1"
} }
] ]
}, },
@ -75,7 +93,7 @@
"externalReferences": [], "externalReferences": [],
"id": 3, "id": 3,
"nodeType": "InlineAssembly", "nodeType": "InlineAssembly",
"src": "52:32:1" "src": "52:42:1"
} }
] ]
}, },
@ -103,15 +121,15 @@
"src": "42:0:1" "src": "42:0:1"
}, },
"scope": 6, "scope": 6,
"src": "17:73:1", "src": "17:83:1",
"stateMutability": "view", "stateMutability": "view",
"virtual": false, "virtual": false,
"visibility": "public" "visibility": "public"
} }
], ],
"scope": 7, "scope": 7,
"src": "0:92:1" "src": "0:102:1"
} }
], ],
"src": "0:93:1" "src": "0:103:1"
} }

View File

@ -1,6 +1,6 @@
contract C { contract C {
function g() view public { function g() view public {
assembly { switch 0 default {} } assembly { switch 0 case 0 {} default {} }
} }
} }

View File

@ -95,30 +95,30 @@
[ [
null null
], ],
"operations": "{\n switch 0\n default { }\n}" "operations": "{\n switch 0\n case 0 { }\n default { }\n}"
}, },
"children": [], "children": [],
"id": 3, "id": 3,
"name": "InlineAssembly", "name": "InlineAssembly",
"src": "52:32:1" "src": "52:42:1"
} }
], ],
"id": 4, "id": 4,
"name": "Block", "name": "Block",
"src": "42:48:1" "src": "42:58:1"
} }
], ],
"id": 5, "id": 5,
"name": "FunctionDefinition", "name": "FunctionDefinition",
"src": "17:73:1" "src": "17:83:1"
} }
], ],
"id": 6, "id": 6,
"name": "ContractDefinition", "name": "ContractDefinition",
"src": "0:92:1" "src": "0:102:1"
} }
], ],
"id": 7, "id": 7,
"name": "SourceUnit", "name": "SourceUnit",
"src": "0:93:1" "src": "0:103:1"
} }

View File

@ -314,9 +314,17 @@ BOOST_AUTO_TEST_CASE(switch_duplicate_case)
BOOST_AUTO_TEST_CASE(switch_invalid_expression) BOOST_AUTO_TEST_CASE(switch_invalid_expression)
{ {
CHECK_PARSE_ERROR("{ switch {} default {} }", ParserError, "Literal or identifier expected."); CHECK_PARSE_ERROR("{ switch {} case 1 {} default {} }", ParserError, "Literal or identifier expected.");
CHECK_PARSE_ERROR("{ switch mload default {} }", ParserError, "Expected '(' but got reserved keyword 'default'"); CHECK_PARSE_ERROR(
CHECK_PARSE_ERROR("{ switch mstore(1, 1) default {} }", TypeError, "Expected expression to evaluate to one value, but got 0 values instead."); "{ switch mload case 1 {} default {} }",
ParserError,
"Expected '(' but got reserved keyword 'case'"
);
CHECK_PARSE_ERROR(
"{ switch mstore(1, 1) case 1 {} default {} }",
TypeError,
"Expected expression to evaluate to one value, but got 0 values instead."
);
} }
BOOST_AUTO_TEST_CASE(switch_default_before_case) BOOST_AUTO_TEST_CASE(switch_default_before_case)

View File

@ -1,15 +1,7 @@
contract C { contract C {
struct S { bool f; } struct S { bool f; }
S s; S s;
function f(uint256 a) internal pure { function f(bool flag) internal pure {
S storage c;
assembly {
switch a
default { c_slot := s_slot }
}
c;
}
function g(bool flag) internal pure {
S storage c; S storage c;
assembly { assembly {
switch flag switch flag
@ -18,7 +10,7 @@ contract C {
} }
c; c;
} }
function h(uint256 a) internal pure { function g(uint256 a) internal pure {
S storage c; S storage c;
assembly { assembly {
switch a switch a

View File

@ -1,20 +1,14 @@
contract C { contract C {
struct S { bool f; } struct S { bool f; }
S s; S s;
function f(uint256 a) internal pure returns (S storage c) { function f(bool flag) internal pure returns (S storage c) {
assembly {
switch a
default { c_slot := s_slot }
}
}
function g(bool flag) internal pure returns (S storage c) {
assembly { assembly {
switch flag switch flag
case 0 { c_slot := s_slot } case 0 { c_slot := s_slot }
default { c_slot := s_slot } default { c_slot := s_slot }
} }
} }
function h(uint256 a) internal pure returns (S storage c) { function g(uint256 a) internal pure returns (S storage c) {
assembly { assembly {
switch a switch a
case 0 { revert(0, 0) } case 0 { revert(0, 0) }

View File

@ -0,0 +1,12 @@
contract C {
struct S { bool f; }
S s;
function f(uint256 a) internal pure returns (S storage c) {
assembly {
switch a
default { c_slot := s_slot }
}
}
}
// ----
// Warning: (142-195): "switch" statement with only a default case.

View File

@ -87,12 +87,12 @@ pair<shared_ptr<Block>, shared_ptr<yul::AsmAnalysisInfo>> yul::test::parse(
shared_ptr<Object> parserResult = yul::ObjectParser(errorReporter, _dialect).parse(scanner, false); shared_ptr<Object> parserResult = yul::ObjectParser(errorReporter, _dialect).parse(scanner, false);
if (!parserResult) if (!parserResult)
return {}; return {};
if (!parserResult->code || !errorReporter.errors().empty()) if (!parserResult->code || errorReporter.hasErrors())
return {}; return {};
shared_ptr<AsmAnalysisInfo> analysisInfo = make_shared<AsmAnalysisInfo>(); shared_ptr<AsmAnalysisInfo> analysisInfo = make_shared<AsmAnalysisInfo>();
AsmAnalyzer analyzer(*analysisInfo, errorReporter, _dialect, {}, parserResult->dataNames()); AsmAnalyzer analyzer(*analysisInfo, errorReporter, _dialect, {}, parserResult->dataNames());
// TODO this should be done recursively. // TODO this should be done recursively.
if (!analyzer.analyze(*parserResult->code) || !errorReporter.errors().empty()) if (!analyzer.analyze(*parserResult->code) || errorReporter.hasErrors())
return {}; return {};
return {std::move(parserResult->code), std::move(analysisInfo)}; return {std::move(parserResult->code), std::move(analysisInfo)};
} }

View File

@ -332,7 +332,7 @@ BOOST_FIXTURE_TEST_CASE(if_statement_custom_weights, CustomWeightFixture)
BOOST_AUTO_TEST_CASE(switch_statement_tiny) BOOST_AUTO_TEST_CASE(switch_statement_tiny)
{ {
BOOST_CHECK_EQUAL(codeSize( BOOST_CHECK_EQUAL(codeSize(
"{ switch calldatasize() default {} }" "{ switch calldatasize() case 0 {} }"
), 4); ), 4);
} }

View File

@ -362,7 +362,7 @@ bool YulOptimizerTest::parse(ostream& _stream, string const& _linePrefix, bool c
ErrorList errors; ErrorList errors;
soltestAssert(m_dialect, ""); soltestAssert(m_dialect, "");
std::tie(m_ast, m_analysisInfo) = yul::test::parse(m_source, *m_dialect, errors); std::tie(m_ast, m_analysisInfo) = yul::test::parse(m_source, *m_dialect, errors);
if (!m_ast || !m_analysisInfo || !errors.empty()) if (!m_ast || !m_analysisInfo || !Error::containsOnlyWarnings(errors))
{ {
AnsiColorized(_stream, _formatted, {formatting::BOLD, formatting::RED}) << _linePrefix << "Error parsing source." << endl; AnsiColorized(_stream, _formatted, {formatting::BOLD, formatting::RED}) << _linePrefix << "Error parsing source." << endl;
printErrors(_stream, errors); printErrors(_stream, errors);