Fix #11889 - private functions should not be overridden by other contracts (rebase)

This commit is contained in:
David Bar-On 2023-06-27 10:39:36 +03:00
parent 34d2383f28
commit 85d17c6f60
16 changed files with 151 additions and 2 deletions

View File

@ -933,7 +933,7 @@ OverrideChecker::OverrideProxyBySignatureMultiSet const& OverrideChecker::inheri
{ {
set<OverrideProxy, OverrideProxy::CompareBySignature> functionsInBase; set<OverrideProxy, OverrideProxy::CompareBySignature> functionsInBase;
for (FunctionDefinition const* fun: base->definedFunctions()) for (FunctionDefinition const* fun: base->definedFunctions())
if (!fun->isConstructor()) if (!fun->isConstructor() && fun->visibility() != Visibility::Private)
functionsInBase.emplace(OverrideProxy{fun}); functionsInBase.emplace(OverrideProxy{fun});
for (VariableDeclaration const* var: base->stateVariables()) for (VariableDeclaration const* var: base->stateVariables())
if (var->isPublic()) if (var->isPublic())

View File

@ -2791,6 +2791,7 @@ bool TypeChecker::visit(FunctionCall const& _functionCall)
case Type::Category::Function: case Type::Category::Function:
functionType = dynamic_cast<FunctionType const*>(expressionType); functionType = dynamic_cast<FunctionType const*>(expressionType);
funcCallAnno.kind = FunctionCallKind::FunctionCall; funcCallAnno.kind = FunctionCallKind::FunctionCall;
funcCallAnno.origCallingContract = dynamic_cast<ContractDefinition const*>(currentDefinitionScope());
if (auto memberAccess = dynamic_cast<MemberAccess const*>(&_functionCall.expression())) if (auto memberAccess = dynamic_cast<MemberAccess const*>(&_functionCall.expression()))
{ {

View File

@ -339,6 +339,9 @@ struct FunctionCallAnnotation: ExpressionAnnotation
util::SetOnce<FunctionCallKind> kind; util::SetOnce<FunctionCallKind> kind;
/// If true, this is the external call of a try statement. /// If true, this is the external call of a try statement.
bool tryCall = false; bool tryCall = false;
/// The original contract calling the function - actual calling contract may be different,
/// e.g. in case is inlining Internal library function call.
ContractDefinition const* origCallingContract = nullptr;
}; };
} }

View File

@ -696,6 +696,17 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
// Calling convention: Caller pushes return address and arguments // Calling convention: Caller pushes return address and arguments
// Callee removes them and pushes return values // Callee removes them and pushes return values
FunctionDefinition const* functionDef = ASTNode::resolveFunctionCall(_functionCall, &m_context.mostDerivedContract());
if (functionDef && !functionDef->isVisibleInDerivedContracts())
{
auto origCallingContract = _functionCall.annotation().origCallingContract;
solAssert (origCallingContract, "Original calling contract is not set");
solAssert(
functionDef->annotation().contract == origCallingContract,
"Attempt to call private function from other contract."
);
}
evmasm::AssemblyItem returnLabel = m_context.pushNewTag(); evmasm::AssemblyItem returnLabel = m_context.pushNewTag();
for (unsigned i = 0; i < arguments.size(); ++i) for (unsigned i = 0; i < arguments.size(); ++i)
acceptAndConvert(*arguments[i], *function.parameterTypes()[i]); acceptAndConvert(*arguments[i], *function.parameterTypes()[i]);

View File

@ -1012,6 +1012,16 @@ void IRGeneratorForStatements::endVisit(FunctionCall const& _functionCall)
{ {
solAssert(functionDef->isImplemented()); solAssert(functionDef->isImplemented());
if (!functionDef->isVisibleInDerivedContracts())
{
auto origCallingContract = _functionCall.annotation().origCallingContract;
solAssert (origCallingContract, "Original calling contract is not set");
solAssert(
functionDef->annotation().contract == origCallingContract,
"Attempt to call private function from other contract"
);
}
define(_functionCall) << define(_functionCall) <<
m_context.enqueueFunctionForCodeGeneration(*functionDef) << m_context.enqueueFunctionForCodeGeneration(*functionDef) <<
"(" << "(" <<

View File

@ -0,0 +1,16 @@
// Tests that private contract function does not shadow a free function,
// but that internal contract function does.
function foo(uint256 value) pure returns (uint256) { return 1; }
function bar(uint256 value) pure returns (uint256) { return 2; }
contract A {
function foo(uint256 value) private pure returns (uint256) { return 3; }
function bar(uint256 value) internal pure returns (uint256) { return 4; }
}
contract B is A {
using {foo} for uint256;
function test1(uint256 value) public pure returns (uint256) { return value.foo(); }
function test2(uint256 value) public pure returns (uint256) { return bar(value); }
}
// ----
// test1(uint256): 0 -> 1
// test2(uint256): 0 -> 4

View File

@ -0,0 +1,18 @@
// Tests that private contract functions are not overridden or visible by inheriting contracts,
// and that that functions functions do.
contract A {
function foo() private pure returns (uint256) { return 1; }
function bar() internal pure returns (uint256) { return foo(); }
}
contract B is A {
function foo() private pure returns (uint256) { return 2; }
function test1() public pure returns (uint256) { return bar(); }
function test2() public pure returns (uint256) { return foo(); }
}
contract C is B {
function foo() public pure returns (uint256) { return 3; }
}
// ----
// test1() -> 1
// test2() -> 2
// foo() -> 3

View File

@ -0,0 +1,13 @@
// Tests that private contract function is not inherited, but that internal function does.
contract A {
function f() private pure returns (uint) { return 7; }
function g() internal pure returns (uint) { return f(); }
}
contract B is A {
function f() private pure returns (uint) { return 42; }
function test() external pure returns (uint, uint) {
return (g(), f());
}
}
// ----
// test() -> 7, 0x2a

View File

@ -0,0 +1,18 @@
// Tests that private library functions are not overridden or visible contracts,
// and that that internal library functions do.
library L {
function foo(uint256 value) private pure returns (uint256) { return 1; }
function bar(uint256 value) internal pure returns (uint256) { return foo(value); }
}
contract A {
function foo(uint256 value) private pure returns (uint256) { return 2; }
}
contract B is A {
using L for uint256;
function foo(uint256 value) private pure returns (uint256) { return 3; }
function test1(uint256 value) public pure returns (uint256) { return value.bar(); }
function test2(uint256 value) public pure returns (uint256) { return foo(0); }
}
// ----
// library: L
// test1(uint256): 0 -> 1

View File

@ -0,0 +1,18 @@
// Tests that private functions are not overridden by inheriting contract's functions.
contract A {
function foo() private pure returns (uint256) { return 1; }
function foo(uint256 value) private pure returns (uint256) { return 2; }
function test1() public pure returns (uint256) { return foo(); }
function test2() public pure returns (uint256) { return foo(0); }
}
contract B is A {
function foo() private pure returns (uint256) { return 3; }
function foo(uint256 value) private pure returns (uint256) { return 4; }
function test3() public pure returns (uint256) { return foo(); }
function test4() public pure returns (uint256) { return foo(0); }
}
// ----
// test1() -> 1
// test2() -> 2
// test3() -> 3
// test4() -> 4

View File

@ -0,0 +1,13 @@
// Tests that private functions are not overridden by inheriting contracts, but that public functions does.
contract A {
function foo() private {}
function foo(uint128 value) private {}
function foo(uint256 value) public {}
}
contract B is A {
function foo(uint128 value) private {}
function foo(uint256 value) public {}
}
// ----
// TypeError 9456: (303-340): Overriding function is missing "override" specifier.
// TypeError 4334: (198-235): Trying to override non-virtual function. Did you forget to add "virtual"?

View File

@ -6,5 +6,5 @@ contract T {
constructor() { new Y(); } constructor() { new Y(); }
} }
// ---- // ----
// TypeError 5225: (97-130): Public state variables can only override functions with external visibility. // TypeError 7792: (112-120): Public state variable has override specified but does not override anything.
// TypeError 3942: (22-72): "virtual" and "private" cannot be used together. // TypeError 3942: (22-72): "virtual" and "private" cannot be used together.

View File

@ -0,0 +1,9 @@
// Tests that private functions are not overridden by inheriting contracts.
contract A {
function foo() private {}
}
contract B is A {
function foo() private override {}
}
// ----
// TypeError 7792: (160-168): Function has override specified but does not override anything.

View File

@ -5,4 +5,5 @@ abstract contract X is A {
function test() private override returns (uint256) {} function test() private override returns (uint256) {}
} }
// ---- // ----
// TypeError 7792: (128-136): Function has override specified but does not override anything.
// TypeError 3942: (23-73): "virtual" and "private" cannot be used together. // TypeError 3942: (23-73): "virtual" and "private" cannot be used together.

View File

@ -0,0 +1,11 @@
// Tests that private functions do not shadow, and are not shadowed by, inheriting contract's functions.
contract A {
function foo() private {}
}
contract B is A {
function foo() private {}
}
contract C is B {
function foo() public {}
}
// ----

View File

@ -0,0 +1,7 @@
// Tests that two private functions cannot be defined with the same name.
contract A {
function foo() private {}
function foo() private {}
}
// ----
// DeclarationError 1686: (88-113): Function with same name and parameter types defined twice.