/* This file is part of cpp-ethereum. cpp-ethereum is free software: you can redistribute it and/or modify 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. cpp-ethereum is distributed in the hope that it will be useful, 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 along with cpp-ethereum. If not, see . */ /** * @author Christian * @date 2015 * Component that resolves type names to types and annotates the AST accordingly. */ #include #include #include #include #include using namespace std; using namespace dev; using namespace dev::solidity; ReferencesResolver::ReferencesResolver( ASTNode& _root, NameAndTypeResolver& _resolver, ContractDefinition const* _currentContract, ParameterList const* _returnParameters, bool _resolveInsideCode ): m_resolver(_resolver), m_currentContract(_currentContract), m_returnParameters(_returnParameters), m_resolveInsideCode(_resolveInsideCode) { _root.accept(*this); } bool ReferencesResolver::visit(Return const& _return) { _return.annotation().functionReturnParameters = m_returnParameters; return true; } bool ReferencesResolver::visit(UserDefinedTypeName const& _typeName) { auto declarations = m_resolver.nameFromCurrentScope(_typeName.name()); if (declarations.empty()) BOOST_THROW_EXCEPTION( DeclarationError() << errinfo_sourceLocation(_typeName.location()) << errinfo_comment("Undeclared identifier.") ); else if (declarations.size() > 1) BOOST_THROW_EXCEPTION( DeclarationError() << errinfo_sourceLocation(_typeName.location()) << errinfo_comment("Duplicate identifier.") ); Declaration const* declaration = *declarations.begin(); _typeName.annotation().referencedDeclaration = declaration; return true; } bool ReferencesResolver::visit(Identifier const& _identifier) { auto declarations = m_resolver.nameFromCurrentScope(_identifier.name()); if (declarations.empty()) BOOST_THROW_EXCEPTION( DeclarationError() << errinfo_sourceLocation(_identifier.location()) << errinfo_comment("Undeclared identifier.") ); else if (declarations.size() == 1) { _identifier.annotation().referencedDeclaration = declarations.front(); _identifier.annotation().contractScope = m_currentContract; } else _identifier.annotation().overloadedDeclarations = m_resolver.cleanedDeclarations(_identifier, declarations); return false; } void ReferencesResolver::endVisit(VariableDeclaration const& _variable) { if (_variable.annotation().type) return; TypePointer type; if (_variable.typeName()) { type = typeFor(*_variable.typeName()); using Location = VariableDeclaration::Location; Location loc = _variable.referenceLocation(); // References are forced to calldata for external function parameters (not return) // and memory for parameters (also return) of publicly visible functions. // They default to memory for function parameters and storage for local variables. // As an exception, "storage" is allowed for library functions. if (auto ref = dynamic_cast(type.get())) { if (_variable.isExternalCallableParameter()) { auto const& contract = dynamic_cast(*_variable.scope()->scope()); if (contract.isLibrary()) { if (loc == Location::Memory) BOOST_THROW_EXCEPTION(_variable.createTypeError( "Location has to be calldata or storage for external " "library functions (remove the \"memory\" keyword)." )); } else { // force location of external function parameters (not return) to calldata if (loc != Location::Default) BOOST_THROW_EXCEPTION(_variable.createTypeError( "Location has to be calldata for external functions " "(remove the \"memory\" or \"storage\" keyword)." )); } if (loc == Location::Default) type = ref->copyForLocation(DataLocation::CallData, true); } else if (_variable.isCallableParameter() && _variable.scope()->isPublic()) { auto const& contract = dynamic_cast(*_variable.scope()->scope()); // force locations of public or external function (return) parameters to memory if (loc == Location::Storage && !contract.isLibrary()) BOOST_THROW_EXCEPTION(_variable.createTypeError( "Location has to be memory for publicly visible functions " "(remove the \"storage\" keyword)." )); if (loc == Location::Default || !contract.isLibrary()) type = ref->copyForLocation(DataLocation::Memory, true); } else { if (_variable.isConstant()) { if (loc != Location::Default && loc != Location::Memory) BOOST_THROW_EXCEPTION(_variable.createTypeError( "Storage location has to be \"memory\" (or unspecified) for constants." )); loc = Location::Memory; } if (loc == Location::Default) loc = _variable.isCallableParameter() ? Location::Memory : Location::Storage; bool isPointer = !_variable.isStateVariable(); type = ref->copyForLocation( loc == Location::Memory ? DataLocation::Memory : DataLocation::Storage, isPointer ); } } else if (loc != Location::Default && !ref) BOOST_THROW_EXCEPTION(_variable.createTypeError( "Storage location can only be given for array or struct types." )); if (!type) BOOST_THROW_EXCEPTION(_variable.typeName()->createTypeError("Invalid type name.")); } else if (!_variable.canHaveAutoType()) BOOST_THROW_EXCEPTION(_variable.createTypeError("Explicit type needed.")); // otherwise we have a "var"-declaration whose type is resolved by the first assignment _variable.annotation().type = type; } TypePointer ReferencesResolver::typeFor(TypeName const& _typeName) { if (_typeName.annotation().type) return _typeName.annotation().type; TypePointer type; if (auto elemTypeName = dynamic_cast(&_typeName)) type = Type::fromElementaryTypeName(elemTypeName->typeName()); else if (auto typeName = dynamic_cast(&_typeName)) { Declaration const* declaration = typeName->annotation().referencedDeclaration; solAssert(!!declaration, ""); if (StructDefinition const* structDef = dynamic_cast(declaration)) type = make_shared(*structDef); else if (EnumDefinition const* enumDef = dynamic_cast(declaration)) type = make_shared(*enumDef); else if (ContractDefinition const* contract = dynamic_cast(declaration)) type = make_shared(*contract); else BOOST_THROW_EXCEPTION(typeName->createTypeError( "Name has to refer to a struct, enum or contract." )); } else if (auto mapping = dynamic_cast(&_typeName)) { TypePointer keyType = typeFor(mapping->keyType()); TypePointer valueType = typeFor(mapping->valueType()); // Convert key type to memory. keyType = ReferenceType::copyForLocationIfReference(DataLocation::Memory, keyType); // Convert value type to storage reference. valueType = ReferenceType::copyForLocationIfReference(DataLocation::Storage, valueType); type = make_shared(keyType, valueType); } else if (auto arrayType = dynamic_cast(&_typeName)) { TypePointer baseType = typeFor(arrayType->baseType()); if (baseType->storageBytes() == 0) BOOST_THROW_EXCEPTION(arrayType->baseType().createTypeError( "Illegal base type of storage size zero for array." )); if (Expression const* length = arrayType->length()) { if (!length->annotation().type) ConstantEvaluator e(*length); auto const* lengthType = dynamic_cast(length->annotation().type.get()); if (!lengthType) BOOST_THROW_EXCEPTION(length->createTypeError("Invalid array length.")); type = make_shared(DataLocation::Storage, baseType, lengthType->literalValue(nullptr)); } else type = make_shared(DataLocation::Storage, baseType); } return _typeName.annotation().type = move(type); }