Public state variables are implementing external functions.

This commit is contained in:
chriseth 2018-12-01 00:08:42 +01:00
parent cc00d8172b
commit 0668a9ecfb
7 changed files with 66 additions and 21 deletions

View File

@ -27,6 +27,7 @@ Bugfixes:
* Type Checker: Fixed internal error when trying to create abstract contract in some cases. * Type Checker: Fixed internal error when trying to create abstract contract in some cases.
* Type Checker: Fixed internal error related to double declaration of events. * Type Checker: Fixed internal error related to double declaration of events.
* Type Checker: Disallow inline arrays of mapping type. * Type Checker: Disallow inline arrays of mapping type.
* Type Checker: Consider abstract function to be implemented by public state variable.
Build System: Build System:
* Emscripten: Upgrade to Emscripten SDK 1.37.21 and boost 1.67. * Emscripten: Upgrade to Emscripten SDK 1.37.21 and boost 1.67.

View File

@ -219,28 +219,39 @@ void ContractLevelChecker::checkAbstractFunctions(ContractDefinition const& _con
using FunTypeAndFlag = std::pair<FunctionTypePointer, bool>; using FunTypeAndFlag = std::pair<FunctionTypePointer, bool>;
map<string, vector<FunTypeAndFlag>> functions; map<string, vector<FunTypeAndFlag>> functions;
// Search from base to derived auto registerFunction = [&](Declaration const& _declaration, FunctionTypePointer const& _type, bool _implemented)
for (ContractDefinition const* contract: boost::adaptors::reverse(_contract.annotation().linearizedBaseContracts))
for (FunctionDefinition const* function: contract->definedFunctions())
{ {
// Take constructors out of overload hierarchy auto& overloads = functions[_declaration.name()];
if (function->isConstructor())
continue;
auto& overloads = functions[function->name()];
FunctionTypePointer funType = make_shared<FunctionType>(*function)->asCallableFunction(false);
auto it = find_if(overloads.begin(), overloads.end(), [&](FunTypeAndFlag const& _funAndFlag) auto it = find_if(overloads.begin(), overloads.end(), [&](FunTypeAndFlag const& _funAndFlag)
{ {
return funType->hasEqualParameterTypes(*_funAndFlag.first); return _type->hasEqualParameterTypes(*_funAndFlag.first);
}); });
if (it == overloads.end()) if (it == overloads.end())
overloads.push_back(make_pair(funType, function->isImplemented())); overloads.push_back(make_pair(_type, _implemented));
else if (it->second) else if (it->second)
{ {
if (!function->isImplemented()) if (!_implemented)
m_errorReporter.typeError(function->location(), "Redeclaring an already implemented function as abstract"); m_errorReporter.typeError(_declaration.location(), "Redeclaring an already implemented function as abstract");
} }
else if (function->isImplemented()) else if (_implemented)
it->second = true; it->second = true;
};
// Search from base to derived, collect all functions and update
// the 'implemented' flag.
for (ContractDefinition const* contract: boost::adaptors::reverse(_contract.annotation().linearizedBaseContracts))
{
for (VariableDeclaration const* v: contract->stateVariables())
if (v->isPartOfExternalInterface())
registerFunction(*v, make_shared<FunctionType>(*v), true);
for (FunctionDefinition const* function: contract->definedFunctions())
if (!function->isConstructor())
registerFunction(
*function,
make_shared<FunctionType>(*function)->asCallableFunction(false),
function->isImplemented()
);
} }
// Set to not fully implemented if at least one flag is false. // Set to not fully implemented if at least one flag is false.

View File

@ -0,0 +1,7 @@
interface X { function test() external returns (uint256); }
contract Y is X {
uint256 public test = 42;
}
contract T {
constructor() public { new Y(); }
}

View File

@ -0,0 +1,9 @@
contract X { function test() internal returns (uint256); }
contract Y is X {
uint256 public test = 42;
}
contract T {
constructor() public { new Y(); }
}
// ----
// DeclarationError: (81-105): Identifier already declared.

View File

@ -0,0 +1,7 @@
contract X { function test() private returns (uint256); }
contract Y is X {
uint256 public test = 42;
}
contract T {
constructor() public { new Y(); }
}

View File

@ -0,0 +1,9 @@
contract X { function test() public returns (uint256); }
contract Y is X {
uint256 public test = 42;
}
contract T {
constructor() public { new Y(); }
}
// ----
// DeclarationError: (79-103): Identifier already declared.

View File

@ -6,3 +6,4 @@ contract C is A {
} }
// ---- // ----
// DeclarationError: (50-85): Identifier already declared. // DeclarationError: (50-85): Identifier already declared.
// TypeError: (50-85): Redeclaring an already implemented function as abstract