aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlex Beregszaszi <alex@rtfs.hu>2018-02-27 19:02:56 +0800
committerAlex Beregszaszi <alex@rtfs.hu>2018-03-26 18:35:17 +0800
commitef3595b0001614c19174c61d29d221844ed83a0e (patch)
treefe46d4e1ba7ae882c026145086482f9de75608f9
parentba209fe485ba40ea3926800bc90932bec40cd16f (diff)
downloaddexon-solidity-ef3595b0001614c19174c61d29d221844ed83a0e.tar.gz
dexon-solidity-ef3595b0001614c19174c61d29d221844ed83a0e.tar.zst
dexon-solidity-ef3595b0001614c19174c61d29d221844ed83a0e.zip
Allow overriding external functions in interfaces with public in an implementing contract
-rw-r--r--Changelog.md1
-rw-r--r--libsolidity/analysis/TypeChecker.cpp15
-rw-r--r--test/libsolidity/SolidityNameAndTypeResolution.cpp14
3 files changed, 30 insertions, 0 deletions
diff --git a/Changelog.md b/Changelog.md
index c5577b86..7a4149f3 100644
--- a/Changelog.md
+++ b/Changelog.md
@@ -2,6 +2,7 @@
Features:
* General: Support accessing dynamic return data in post-byzantium EVMs.
+ * Interfaces: Allow overriding external functions in interfaces with public in an implementing contract.
Bugfixes:
* Code Generator: Allow ``block.blockhash`` without being called.
diff --git a/libsolidity/analysis/TypeChecker.cpp b/libsolidity/analysis/TypeChecker.cpp
index 999a2a97..487959c5 100644
--- a/libsolidity/analysis/TypeChecker.cpp
+++ b/libsolidity/analysis/TypeChecker.cpp
@@ -366,6 +366,16 @@ void TypeChecker::checkContractIllegalOverrides(ContractDefinition const& _contr
}
}
+namespace {
+
+bool functionIsInInterface(FunctionDefinition const& _function)
+{
+ return dynamic_cast<ContractDefinition const*>(_function.scope()) &&
+ dynamic_cast<ContractDefinition const*>(_function.scope())->contractKind() == ContractDefinition::ContractKind::Interface;
+}
+
+}
+
void TypeChecker::checkFunctionOverride(FunctionDefinition const& function, FunctionDefinition const& super)
{
FunctionType functionType(function);
@@ -378,7 +388,12 @@ void TypeChecker::checkFunctionOverride(FunctionDefinition const& function, Func
function.annotation().superFunction = &super;
if (function.visibility() != super.visibility())
+ {
+ // visibility is enforced to be external in interfaces, but a contract can override that with public
+ if (functionIsInInterface(super) && !functionIsInInterface(function) && function.visibility() == FunctionDefinition::Visibility::Public)
+ return;
overrideError(function, super, "Overriding function visibility differs.");
+ }
else if (function.stateMutability() != super.stateMutability())
overrideError(
diff --git a/test/libsolidity/SolidityNameAndTypeResolution.cpp b/test/libsolidity/SolidityNameAndTypeResolution.cpp
index 50ee2b2e..e30ea8a4 100644
--- a/test/libsolidity/SolidityNameAndTypeResolution.cpp
+++ b/test/libsolidity/SolidityNameAndTypeResolution.cpp
@@ -6784,6 +6784,20 @@ BOOST_AUTO_TEST_CASE(using_interface_complex)
CHECK_SUCCESS(text);
}
+BOOST_AUTO_TEST_CASE(interface_implement_public_contract)
+{
+ char const* text = R"(
+ interface I {
+ function f() external;
+ }
+ contract C is I {
+ function f() public {
+ }
+ }
+ )";
+ CHECK_SUCCESS_NO_WARNINGS(text);
+}
+
BOOST_AUTO_TEST_CASE(warn_about_throw)
{
char const* text = R"(