2016-01-14 01:58:09 +00:00
/*
2016-11-18 23:13:20 +00:00
This file is part of solidity .
2016-01-14 01:58:09 +00:00
2016-11-18 23:13:20 +00:00
solidity is free software : you can redistribute it and / or modify
2016-01-14 01:58:09 +00:00
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
2016-11-18 23:13:20 +00:00
solidity is distributed in the hope that it will be useful ,
2016-01-14 01:58:09 +00:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2016-11-18 23:13:20 +00:00
along with solidity . If not , see < http : //www.gnu.org/licenses/>.
2016-01-14 01:58:09 +00:00
*/
2020-07-17 14:54:12 +00:00
// SPDX-License-Identifier: GPL-3.0
2016-01-14 01:58:09 +00:00
2016-08-19 17:57:21 +00:00
# include <libsolidity/analysis/SyntaxChecker.h>
2018-12-17 11:30:08 +00:00
2016-01-14 01:58:09 +00:00
# include <libsolidity/ast/AST.h>
2017-08-08 13:09:57 +00:00
# include <libsolidity/ast/ExperimentalFeatures.h>
2016-08-19 17:57:21 +00:00
# include <libsolidity/interface/Version.h>
2016-01-14 01:58:09 +00:00
2019-05-27 12:01:53 +00:00
# include <libyul/optimiser/Semantics.h>
2020-10-29 14:00:27 +00:00
# include <libyul/AST.h>
2019-05-27 12:01:53 +00:00
2018-12-17 11:30:08 +00:00
# include <liblangutil/ErrorReporter.h>
2019-04-17 11:02:30 +00:00
# include <liblangutil/SemVerHandler.h>
2018-12-17 11:30:08 +00:00
2020-07-02 16:39:04 +00:00
# include <libsolutil/UTF8.h>
2017-10-25 08:12:07 +00:00
# include <string>
2019-12-11 16:31:36 +00:00
using namespace solidity ;
using namespace solidity : : langutil ;
using namespace solidity : : frontend ;
2020-07-02 16:39:04 +00:00
using namespace solidity : : util ;
2016-01-14 01:58:09 +00:00
2017-01-27 17:27:59 +00:00
bool SyntaxChecker : : checkSyntax ( ASTNode const & _astRoot )
2016-01-14 01:58:09 +00:00
{
2017-01-27 17:27:59 +00:00
_astRoot . accept ( * this ) ;
2021-06-30 12:48:45 +00:00
return ! Error : : containsErrors ( m_errorReporter . errors ( ) ) ;
2016-01-14 01:58:09 +00:00
}
2017-07-08 23:10:22 +00:00
bool SyntaxChecker : : visit ( SourceUnit const & _sourceUnit )
2016-08-19 17:57:21 +00:00
{
m_versionPragmaFound = false ;
2017-07-08 23:10:22 +00:00
m_sourceUnit = & _sourceUnit ;
2016-08-19 17:57:21 +00:00
return true ;
}
void SyntaxChecker : : endVisit ( SourceUnit const & _sourceUnit )
{
if ( ! m_versionPragmaFound )
{
2023-08-14 08:37:11 +00:00
std : : string errorString ( " Source file does not specify required compiler version! " ) ;
SemVerVersion recommendedVersion { std : : string ( VersionString ) } ;
2016-10-05 09:58:25 +00:00
if ( ! recommendedVersion . isPrerelease ( ) )
errorString + =
2018-10-08 20:58:34 +00:00
" Consider adding \" pragma solidity ^ " +
2023-08-14 08:37:11 +00:00
std : : to_string ( recommendedVersion . major ( ) ) +
std : : string ( " . " ) +
std : : to_string ( recommendedVersion . minor ( ) ) +
std : : string ( " . " ) +
std : : to_string ( recommendedVersion . patch ( ) ) +
std : : string ( " ; \" " ) ;
2016-10-05 09:58:25 +00:00
2020-02-07 01:36:51 +00:00
// when reporting the warning, print the source name only
2021-06-29 12:38:59 +00:00
m_errorReporter . warning ( 3420 _error , { - 1 , - 1 , _sourceUnit . location ( ) . sourceName } , errorString ) ;
2016-08-19 17:57:21 +00:00
}
2020-10-22 17:19:14 +00:00
if ( ! m_sourceUnit - > annotation ( ) . useABICoderV2 . set ( ) )
2020-11-18 19:22:51 +00:00
m_sourceUnit - > annotation ( ) . useABICoderV2 = true ;
2017-07-08 23:10:22 +00:00
m_sourceUnit = nullptr ;
2016-08-19 17:57:21 +00:00
}
bool SyntaxChecker : : visit ( PragmaDirective const & _pragma )
{
solAssert ( ! _pragma . tokens ( ) . empty ( ) , " " ) ;
solAssert ( _pragma . tokens ( ) . size ( ) = = _pragma . literals ( ) . size ( ) , " " ) ;
2017-07-08 23:10:22 +00:00
if ( _pragma . tokens ( ) [ 0 ] ! = Token : : Identifier )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 5226 _error , _pragma . location ( ) , " Invalid pragma \" " + _pragma . literals ( ) [ 0 ] + " \" " ) ;
2017-07-08 23:10:22 +00:00
else if ( _pragma . literals ( ) [ 0 ] = = " experimental " )
{
solAssert ( m_sourceUnit , " " ) ;
2023-08-14 08:37:11 +00:00
std : : vector < std : : string > literals ( _pragma . literals ( ) . begin ( ) + 1 , _pragma . literals ( ) . end ( ) ) ;
2018-10-09 03:29:37 +00:00
if ( literals . empty ( ) )
2017-07-08 23:10:22 +00:00
m_errorReporter . syntaxError (
2020-05-05 22:38:28 +00:00
9679 _error ,
2017-07-08 23:10:22 +00:00
_pragma . location ( ) ,
2017-08-04 22:28:28 +00:00
" Experimental feature name is missing. "
) ;
else if ( literals . size ( ) > 1 )
m_errorReporter . syntaxError (
2020-05-05 22:38:28 +00:00
6022 _error ,
2017-08-04 22:28:28 +00:00
_pragma . location ( ) ,
" Stray arguments. "
2017-07-08 23:10:22 +00:00
) ;
else
2017-08-02 19:05:35 +00:00
{
2023-08-14 08:37:11 +00:00
std : : string const literal = literals [ 0 ] ;
2017-08-04 22:28:28 +00:00
if ( literal . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 3250 _error , _pragma . location ( ) , " Empty experimental feature name is invalid. " ) ;
2017-08-08 13:09:57 +00:00
else if ( ! ExperimentalFeatureNames . count ( literal ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 8491 _error , _pragma . location ( ) , " Unsupported experimental feature name. " ) ;
2017-08-08 13:09:57 +00:00
else if ( m_sourceUnit - > annotation ( ) . experimentalFeatures . count ( ExperimentalFeatureNames . at ( literal ) ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 1231 _error , _pragma . location ( ) , " Duplicate experimental feature name. " ) ;
2017-08-04 22:28:28 +00:00
else
2017-07-08 23:10:22 +00:00
{
2018-02-21 15:33:59 +00:00
auto feature = ExperimentalFeatureNames . at ( literal ) ;
m_sourceUnit - > annotation ( ) . experimentalFeatures . insert ( feature ) ;
2019-11-20 11:29:08 +00:00
if ( ! ExperimentalFeatureWithoutWarning . count ( feature ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . warning ( 2264 _error , _pragma . location ( ) , " Experimental features are turned on. Do not use experimental features on live deployments. " ) ;
2020-10-22 17:19:14 +00:00
if ( feature = = ExperimentalFeature : : ABIEncoderV2 )
{
if ( m_sourceUnit - > annotation ( ) . useABICoderV2 . set ( ) )
{
if ( ! * m_sourceUnit - > annotation ( ) . useABICoderV2 )
m_errorReporter . syntaxError (
8273 _error ,
_pragma . location ( ) ,
2020-10-29 18:40:17 +00:00
" ABI coder v1 has already been selected through \" pragma abicoder v1 \" . "
2020-10-22 17:19:14 +00:00
) ;
}
else
m_sourceUnit - > annotation ( ) . useABICoderV2 = true ;
}
2017-07-08 23:10:22 +00:00
}
2017-08-02 19:05:35 +00:00
}
2017-07-08 23:10:22 +00:00
}
2020-10-22 17:19:14 +00:00
else if ( _pragma . literals ( ) [ 0 ] = = " abicoder " )
{
solAssert ( m_sourceUnit , " " ) ;
if (
_pragma . literals ( ) . size ( ) ! = 2 | |
2023-08-14 08:37:11 +00:00
! std : : set < std : : string > { " v1 " , " v2 " } . count ( _pragma . literals ( ) [ 1 ] )
2020-10-22 17:19:14 +00:00
)
m_errorReporter . syntaxError (
2745 _error ,
_pragma . location ( ) ,
" Expected either \" pragma abicoder v1 \" or \" pragma abicoder v2 \" . "
) ;
else if ( m_sourceUnit - > annotation ( ) . useABICoderV2 . set ( ) )
m_errorReporter . syntaxError (
3845 _error ,
_pragma . location ( ) ,
" ABI coder has already been selected for this source unit. "
) ;
else
m_sourceUnit - > annotation ( ) . useABICoderV2 = ( _pragma . literals ( ) [ 1 ] = = " v2 " ) ;
}
2017-07-08 23:10:22 +00:00
else if ( _pragma . literals ( ) [ 0 ] = = " solidity " )
2016-08-19 17:57:21 +00:00
{
2022-03-15 18:53:29 +00:00
try
{
2023-08-14 08:37:11 +00:00
std : : vector < Token > tokens ( _pragma . tokens ( ) . begin ( ) + 1 , _pragma . tokens ( ) . end ( ) ) ;
std : : vector < std : : string > literals ( _pragma . literals ( ) . begin ( ) + 1 , _pragma . literals ( ) . end ( ) ) ;
2022-03-15 18:53:29 +00:00
SemVerMatchExpressionParser parser ( tokens , literals ) ;
SemVerMatchExpression matchExpression = parser . parse ( ) ;
2023-08-14 08:37:11 +00:00
static SemVerVersion const currentVersion { std : : string ( VersionString ) } ;
2023-08-04 20:54:48 +00:00
solAssert ( matchExpression . matches ( currentVersion ) ) ;
2022-03-15 18:53:29 +00:00
m_versionPragmaFound = true ;
}
catch ( SemVerError const & )
{
// An unparsable version pragma is an unrecoverable fatal error in the parser.
solAssert ( false ) ;
}
2016-08-19 17:57:21 +00:00
}
2017-07-08 23:10:22 +00:00
else
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 4936 _error , _pragma . location ( ) , " Unknown pragma \" " + _pragma . literals ( ) [ 0 ] + " \" " ) ;
2020-10-22 17:19:14 +00:00
2016-08-19 17:57:21 +00:00
return true ;
}
2016-08-06 13:08:06 +00:00
bool SyntaxChecker : : visit ( ModifierDefinition const & )
{
m_placeholderFound = false ;
return true ;
}
void SyntaxChecker : : endVisit ( ModifierDefinition const & _modifier )
{
2020-04-15 10:42:15 +00:00
if ( _modifier . isImplemented ( ) & & ! m_placeholderFound )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 2883 _error , _modifier . body ( ) . location ( ) , " Modifier body does not contain '_'. " ) ;
2016-08-06 13:08:06 +00:00
m_placeholderFound = false ;
}
2018-09-04 10:14:04 +00:00
void SyntaxChecker : : checkSingleStatementVariableDeclaration ( ASTNode const & _statement )
2018-09-03 16:17:59 +00:00
{
2018-09-04 10:14:04 +00:00
auto varDecl = dynamic_cast < VariableDeclarationStatement const * > ( & _statement ) ;
2018-09-03 16:17:59 +00:00
if ( varDecl )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 9079 _error , _statement . location ( ) , " Variable declarations can only be used inside blocks. " ) ;
2018-09-03 16:17:59 +00:00
}
bool SyntaxChecker : : visit ( IfStatement const & _ifStatement )
{
2018-09-04 10:14:04 +00:00
checkSingleStatementVariableDeclaration ( _ifStatement . trueStatement ( ) ) ;
2018-09-03 16:17:59 +00:00
if ( Statement const * _statement = _ifStatement . falseStatement ( ) )
2018-09-04 10:14:04 +00:00
checkSingleStatementVariableDeclaration ( * _statement ) ;
2018-09-03 16:17:59 +00:00
return true ;
}
bool SyntaxChecker : : visit ( WhileStatement const & _whileStatement )
2016-01-14 01:58:09 +00:00
{
2016-01-19 02:16:13 +00:00
m_inLoopDepth + + ;
2018-09-04 10:14:04 +00:00
checkSingleStatementVariableDeclaration ( _whileStatement . body ( ) ) ;
2016-01-14 01:58:09 +00:00
return true ;
}
2017-08-15 12:22:50 +00:00
void SyntaxChecker : : endVisit ( WhileStatement const & )
2016-01-14 01:58:09 +00:00
{
2016-01-19 02:16:13 +00:00
m_inLoopDepth - - ;
2016-01-14 01:58:09 +00:00
}
2018-09-03 16:17:59 +00:00
bool SyntaxChecker : : visit ( ForStatement const & _forStatement )
2016-01-14 01:58:09 +00:00
{
2016-01-19 02:16:13 +00:00
m_inLoopDepth + + ;
2018-09-04 10:14:04 +00:00
checkSingleStatementVariableDeclaration ( _forStatement . body ( ) ) ;
2016-01-14 01:58:09 +00:00
return true ;
}
2016-01-19 02:18:01 +00:00
void SyntaxChecker : : endVisit ( ForStatement const & )
2016-01-14 01:58:09 +00:00
{
2016-01-19 02:16:13 +00:00
m_inLoopDepth - - ;
2016-01-14 01:58:09 +00:00
}
2020-07-22 08:28:04 +00:00
bool SyntaxChecker : : visit ( Block const & _block )
{
if ( _block . unchecked ( ) )
{
if ( m_uncheckedArithmetic )
m_errorReporter . syntaxError (
1941 _error ,
_block . location ( ) ,
" \" unchecked \" blocks cannot be nested. "
) ;
m_uncheckedArithmetic = true ;
}
return true ;
}
void SyntaxChecker : : endVisit ( Block const & _block )
{
if ( _block . unchecked ( ) )
m_uncheckedArithmetic = false ;
}
2016-01-14 01:58:09 +00:00
bool SyntaxChecker : : visit ( Continue const & _continueStatement )
{
2016-01-19 02:16:13 +00:00
if ( m_inLoopDepth < = 0 )
2016-01-14 01:58:09 +00:00
// we're not in a for/while loop, report syntax error
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 4123 _error , _continueStatement . location ( ) , " \" continue \" has to be in a \" for \" or \" while \" loop. " ) ;
2016-01-14 01:58:09 +00:00
return true ;
}
bool SyntaxChecker : : visit ( Break const & _breakStatement )
{
2016-01-19 02:16:13 +00:00
if ( m_inLoopDepth < = 0 )
2016-01-14 01:58:09 +00:00
// we're not in a for/while loop, report syntax error
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 6102 _error , _breakStatement . location ( ) , " \" break \" has to be in a \" for \" or \" while \" loop. " ) ;
2016-01-14 01:58:09 +00:00
return true ;
}
2017-07-05 17:45:12 +00:00
bool SyntaxChecker : : visit ( Throw const & _throwStatement )
{
2018-07-11 14:06:31 +00:00
m_errorReporter . syntaxError (
2020-05-05 22:38:28 +00:00
4538 _error ,
2018-07-11 14:06:31 +00:00
_throwStatement . location ( ) ,
" \" throw \" is deprecated in favour of \" revert() \" , \" require() \" and \" assert() \" . "
) ;
2017-07-05 17:45:12 +00:00
return true ;
}
2017-10-25 08:12:07 +00:00
bool SyntaxChecker : : visit ( Literal const & _literal )
{
2020-09-23 12:21:30 +00:00
size_t invalidSequence ;
if ( ( _literal . token ( ) = = Token : : UnicodeStringLiteral ) & & ! validateUTF8 ( _literal . value ( ) , invalidSequence ) )
2020-07-02 16:39:04 +00:00
m_errorReporter . syntaxError (
8452 _error ,
_literal . location ( ) ,
2020-09-23 12:21:30 +00:00
" Contains invalid UTF-8 sequence at position " + toString ( invalidSequence ) + " . "
2020-07-02 16:39:04 +00:00
) ;
2018-08-03 14:13:52 +00:00
if ( _literal . token ( ) ! = Token : : Number )
2017-10-25 08:12:07 +00:00
return true ;
2018-08-03 14:13:52 +00:00
ASTString const & value = _literal . value ( ) ;
solAssert ( ! value . empty ( ) , " " ) ;
2017-10-25 08:12:07 +00:00
2018-08-03 14:13:52 +00:00
// Generic checks no matter what base this number literal is of:
if ( value . back ( ) = = ' _ ' )
{
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 2090 _error , _literal . location ( ) , " Invalid use of underscores in number literal. No trailing underscores allowed. " ) ;
2017-10-25 08:12:07 +00:00
return true ;
2018-08-03 14:13:52 +00:00
}
2017-10-25 08:12:07 +00:00
2018-08-03 14:13:52 +00:00
if ( value . find ( " __ " ) ! = ASTString : : npos )
2017-10-25 08:12:07 +00:00
{
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 2990 _error , _literal . location ( ) , " Invalid use of underscores in number literal. Only one consecutive underscores between digits allowed. " ) ;
2018-08-03 14:13:52 +00:00
return true ;
2017-10-25 08:12:07 +00:00
}
2018-08-03 14:13:52 +00:00
if ( ! _literal . isHexNumber ( ) ) // decimal literal
2017-10-25 08:12:07 +00:00
{
2018-08-03 14:13:52 +00:00
if ( value . find ( " ._ " ) ! = ASTString : : npos )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 3891 _error , _literal . location ( ) , " Invalid use of underscores in number literal. No underscores in front of the fraction part allowed. " ) ;
2018-08-03 14:13:52 +00:00
if ( value . find ( " _. " ) ! = ASTString : : npos )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 1023 _error , _literal . location ( ) , " Invalid use of underscores in number literal. No underscores in front of the fraction part allowed. " ) ;
2018-08-03 14:13:52 +00:00
if ( value . find ( " _e " ) ! = ASTString : : npos )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 6415 _error , _literal . location ( ) , " Invalid use of underscores in number literal. No underscore at the end of the mantissa allowed. " ) ;
2018-08-03 14:13:52 +00:00
if ( value . find ( " e_ " ) ! = ASTString : : npos )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 6165 _error , _literal . location ( ) , " Invalid use of underscores in number literal. No underscore in front of exponent allowed. " ) ;
2017-10-25 08:12:07 +00:00
}
return true ;
}
2017-04-29 00:43:19 +00:00
bool SyntaxChecker : : visit ( UnaryOperation const & _operation )
{
2023-04-14 19:06:59 +00:00
solAssert ( _operation . getOperator ( ) ! = Token : : Add ) ;
2017-04-29 00:43:19 +00:00
return true ;
}
2019-05-27 12:01:53 +00:00
bool SyntaxChecker : : visit ( InlineAssembly const & _inlineAssembly )
{
2022-02-14 12:21:15 +00:00
if ( _inlineAssembly . flags ( ) )
for ( auto flag : * _inlineAssembly . flags ( ) )
{
if ( * flag = = " memory-safe " )
{
if ( _inlineAssembly . annotation ( ) . markedMemorySafe )
m_errorReporter . syntaxError (
7026 _error ,
_inlineAssembly . location ( ) ,
" Inline assembly marked memory-safe multiple times. "
) ;
_inlineAssembly . annotation ( ) . markedMemorySafe = true ;
}
else
m_errorReporter . warning (
4430 _error ,
_inlineAssembly . location ( ) ,
" Unknown inline assembly flag: \" " + * flag + " \" "
) ;
}
2019-05-27 12:01:53 +00:00
if ( ! m_useYulOptimizer )
return false ;
2019-08-13 11:34:33 +00:00
if ( yul : : MSizeFinder : : containsMSize ( _inlineAssembly . dialect ( ) , _inlineAssembly . operations ( ) ) )
2019-05-27 12:01:53 +00:00
m_errorReporter . syntaxError (
2020-05-05 22:38:28 +00:00
6553 _error ,
2019-05-27 12:01:53 +00:00
_inlineAssembly . location ( ) ,
" The msize instruction cannot be used when the Yul optimizer is activated because "
" it can change its semantics. Either disable the Yul optimizer or do not use the instruction. "
) ;
2020-05-12 15:09:25 +00:00
2019-05-27 12:01:53 +00:00
return false ;
}
2020-07-22 08:28:04 +00:00
bool SyntaxChecker : : visit ( PlaceholderStatement const & _placeholder )
2016-08-06 13:08:06 +00:00
{
2020-07-22 08:28:04 +00:00
if ( m_uncheckedArithmetic )
m_errorReporter . syntaxError (
2573 _error ,
_placeholder . location ( ) ,
" The placeholder statement \" _ \" cannot be used inside an \" unchecked \" block. "
) ;
2016-08-06 13:08:06 +00:00
m_placeholderFound = true ;
return true ;
}
2018-07-10 20:43:02 +00:00
bool SyntaxChecker : : visit ( ContractDefinition const & _contract )
{
2020-05-04 16:38:00 +00:00
m_currentContractKind = _contract . contractKind ( ) ;
2018-06-27 10:29:03 +00:00
ASTString const & contractName = _contract . name ( ) ;
for ( FunctionDefinition const * function : _contract . definedFunctions ( ) )
if ( function - > name ( ) = = contractName )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError (
2020-05-07 01:52:53 +00:00
5796 _error ,
function - > location ( ) ,
2018-06-27 10:29:03 +00:00
" Functions are not allowed to have the same name as the contract. "
" If you intend this to be a constructor, use \" constructor(...) { ... } \" to define it. "
) ;
2018-07-10 20:43:02 +00:00
return true ;
}
2020-05-04 16:38:00 +00:00
void SyntaxChecker : : endVisit ( ContractDefinition const & )
{
m_currentContractKind = std : : nullopt ;
2021-10-11 08:16:52 +00:00
}
bool SyntaxChecker : : visit ( UsingForDirective const & _usingFor )
{
2022-07-06 07:17:59 +00:00
if ( ! _usingFor . usesBraces ( ) )
solAssert (
_usingFor . functionsAndOperators ( ) . size ( ) = = 1 & &
2023-08-14 08:37:11 +00:00
! std : : get < 1 > ( _usingFor . functionsAndOperators ( ) . front ( ) )
2022-07-06 07:17:59 +00:00
) ;
2021-10-11 08:16:52 +00:00
if ( ! m_currentContractKind & & ! _usingFor . typeName ( ) )
m_errorReporter . syntaxError (
8118 _error ,
_usingFor . location ( ) ,
" The type has to be specified explicitly at file level (cannot use '*'). "
) ;
else if ( _usingFor . usesBraces ( ) & & ! _usingFor . typeName ( ) )
m_errorReporter . syntaxError (
3349 _error ,
_usingFor . location ( ) ,
" The type has to be specified explicitly when attaching specific functions. "
) ;
2021-11-16 16:01:09 +00:00
if ( _usingFor . global ( ) & & ! _usingFor . typeName ( ) )
m_errorReporter . syntaxError (
2854 _error ,
_usingFor . location ( ) ,
2022-08-30 09:53:02 +00:00
" Can only globally attach functions to specific types. "
2021-11-16 16:01:09 +00:00
) ;
if ( _usingFor . global ( ) & & m_currentContractKind )
m_errorReporter . syntaxError (
3367 _error ,
_usingFor . location ( ) ,
" \" global \" can only be used at file level. "
) ;
2021-10-11 08:16:52 +00:00
if ( m_currentContractKind = = ContractKind : : Interface )
m_errorReporter . syntaxError (
9088 _error ,
_usingFor . location ( ) ,
" The \" using for \" directive is not allowed inside interfaces. "
) ;
return true ;
2020-05-04 16:38:00 +00:00
}
2017-08-15 12:22:50 +00:00
bool SyntaxChecker : : visit ( FunctionDefinition const & _function )
{
2020-05-04 16:38:00 +00:00
solAssert ( _function . isFree ( ) = = ( m_currentContractKind = = std : : nullopt ) , " " ) ;
if ( ! _function . isFree ( ) & & ! _function . isConstructor ( ) & & _function . noVisibilitySpecified ( ) )
2018-07-10 20:43:02 +00:00
{
2023-08-14 08:37:11 +00:00
std : : string suggestedVisibility =
2020-05-04 16:38:00 +00:00
_function . isFallback ( ) | |
_function . isReceive ( ) | |
m_currentContractKind = = ContractKind : : Interface
? " external " : " public " ;
2018-07-10 20:43:02 +00:00
m_errorReporter . syntaxError (
2020-05-05 22:38:28 +00:00
4937 _error ,
2018-07-10 20:43:02 +00:00
_function . location ( ) ,
" No visibility specified. Did you intend to add \" " + suggestedVisibility + " \" ? "
) ;
}
2020-05-04 16:38:00 +00:00
else if ( _function . isFree ( ) )
{
if ( ! _function . noVisibilitySpecified ( ) )
m_errorReporter . syntaxError (
4126 _error ,
_function . location ( ) ,
" Free functions cannot have visibility. "
) ;
if ( ! _function . isImplemented ( ) )
m_errorReporter . typeError ( 4668 _error , _function . location ( ) , " Free functions must be implemented. " ) ;
}
2018-03-01 17:39:01 +00:00
2020-05-04 16:38:00 +00:00
if ( m_currentContractKind = = ContractKind : : Interface & & ! _function . modifiers ( ) . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 5842 _error , _function . location ( ) , " Functions in interfaces cannot have modifiers. " ) ;
2020-04-02 15:00:43 +00:00
else if ( ! _function . isImplemented ( ) & & ! _function . modifiers ( ) . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 2668 _error , _function . location ( ) , " Functions without implementation cannot have modifiers. " ) ;
2018-07-03 09:28:57 +00:00
2017-08-15 12:22:50 +00:00
return true ;
}
2017-06-24 17:09:19 +00:00
bool SyntaxChecker : : visit ( FunctionTypeName const & _node )
{
for ( auto const & decl : _node . parameterTypeList ( ) - > parameters ( ) )
if ( ! decl - > name ( ) . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . warning ( 6162 _error , decl - > location ( ) , " Naming function type parameters is deprecated. " ) ;
2017-06-24 17:09:19 +00:00
for ( auto const & decl : _node . returnParameterTypeList ( ) - > parameters ( ) )
if ( ! decl - > name ( ) . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 7304 _error , decl - > location ( ) , " Return parameters in function types may not be named. " ) ;
2017-06-24 17:09:19 +00:00
return true ;
}
2018-01-22 21:32:47 +00:00
2018-03-27 13:38:28 +00:00
bool SyntaxChecker : : visit ( StructDefinition const & _struct )
{
if ( _struct . members ( ) . empty ( ) )
2020-05-05 22:38:28 +00:00
m_errorReporter . syntaxError ( 5306 _error , _struct . location ( ) , " Defining empty structs is disallowed. " ) ;
2018-06-06 09:15:22 +00:00
2018-03-27 13:38:28 +00:00
return true ;
}