Consider all grandparents in override analysis.

This commit is contained in:
Daniel Kirchner 2022-02-02 17:18:38 +01:00
parent d29709e741
commit c47da51eab
9 changed files with 131 additions and 3 deletions

View File

@ -19,6 +19,7 @@ Bugfixes:
* Control Flow Graph: Perform proper virtual lookup for modifiers for uninitialized variable and unreachable code analysis. * Control Flow Graph: Perform proper virtual lookup for modifiers for uninitialized variable and unreachable code analysis.
* General: ``string.concat`` now properly takes strings as arguments and returns ``string memory``. It was accidentally introduced as a copy of ``bytes.concat`` before. * General: ``string.concat`` now properly takes strings as arguments and returns ``string memory``. It was accidentally introduced as a copy of ``bytes.concat`` before.
* Immutables: Fix wrong error when the constructor of a base contract uses ``return`` and the derived contract contains immutable variables. * Immutables: Fix wrong error when the constructor of a base contract uses ``return`` and the derived contract contains immutable variables.
* Inheritance: Consider functions in all ancestors during override analysis.
* IR Generator: Add missing cleanup during the conversion of fixed bytes types to smaller fixed bytes types. * IR Generator: Add missing cleanup during the conversion of fixed bytes types to smaller fixed bytes types.
* IR Generator: Add missing cleanup for indexed event arguments of value type. * IR Generator: Add missing cleanup for indexed event arguments of value type.
* IR Generator: Fix internal error when copying reference types in calldata and storage to struct or array members in memory. * IR Generator: Fix internal error when copying reference types in calldata and storage to struct or array members in memory.

View File

@ -897,10 +897,11 @@ OverrideChecker::OverrideProxyBySignatureMultiSet const& OverrideChecker::inheri
if (var->isPublic()) if (var->isPublic())
functionsInBase.emplace(OverrideProxy{var}); functionsInBase.emplace(OverrideProxy{var});
for (OverrideProxy const& func: inheritedFunctions(*base))
functionsInBase.insert(func);
result += functionsInBase; result += functionsInBase;
for (OverrideProxy const& func: inheritedFunctions(*base))
if (!functionsInBase.count(func))
result.insert(func);
} }
m_inheritedFunctions[&_contract] = result; m_inheritedFunctions[&_contract] = result;

View File

@ -0,0 +1,17 @@
interface IBase {
function foo() external view;
}
contract Base is IBase {
function foo() public virtual view {}
}
interface IExt is IBase {}
contract Ext is IExt, Base {}
contract T { function foo() public virtual view {} }
contract Impl is Ext, T {
function foo() public view override(IBase, Base, T) {}
}

View File

@ -0,0 +1,17 @@
abstract contract IBase {
function foo() external view virtual;
}
contract Base is IBase {
function foo() public virtual override view {}
}
abstract contract IExt is IBase {}
contract Ext is IExt, Base {}
contract T { function foo() public virtual view {} }
contract Impl is Ext, T {
function foo() public view override(IBase, Base, T) {}
}

View File

@ -0,0 +1,18 @@
interface IBase {
function foo() external view;
}
contract Base is IBase {
function foo() public virtual view {}
}
interface IExt is IBase {}
contract Ext is IExt, Base {}
contract Impl is Ext {
function foo() public view {}
}
// ----
// TypeError 9456: (211-240): Overriding function is missing "override" specifier.
// TypeError 4327: (211-240): Function needs to specify overridden contracts "Base" and "IBase".

View File

@ -0,0 +1,16 @@
interface IBase {
function foo() external view;
}
contract Base is IBase {
function foo() public virtual view {}
}
interface IExt is IBase {}
contract Ext is IExt, Base {}
contract Impl is Ext {
function foo() public view override (IBase, Base) {}
}
// ----

View File

@ -0,0 +1,22 @@
interface IBase {
function foo() external view;
}
contract Base1 is IBase { function foo() public virtual view {} }
contract Base2 is IBase { function foo() public virtual view {} }
interface IExt1a is IBase {}
interface IExt1b is IBase {}
interface IExt2a is IBase {}
interface IExt2b is IBase {}
contract Ext1 is IExt1a, IExt1b, Base1 {}
contract Ext2 is IExt2a, IExt2b, Base2 {}
contract Impl is Ext1, Ext2 {
function foo() public view {}
}
// ----
// TypeError 9456: (424-453): Overriding function is missing "override" specifier.
// TypeError 9456: (424-453): Overriding function is missing "override" specifier.
// TypeError 4327: (424-453): Function needs to specify overridden contracts "Base1", "Base2" and "IBase".

View File

@ -0,0 +1,18 @@
interface IBase {
function foo() external view;
}
contract Base1 is IBase { function foo() public virtual view {} }
contract Base2 is IBase { function foo() public virtual view {} }
interface IExt1a is IBase {}
interface IExt1b is IBase {}
interface IExt2a is IBase {}
interface IExt2b is IBase {}
contract Ext1 is IExt1a, IExt1b, Base1 {}
contract Ext2 is IExt2a, IExt2b, Base2 {}
contract Impl is Ext1, Ext2 {
function foo() public view override (IBase, Base1, Base2) {}
}

View File

@ -0,0 +1,18 @@
interface IBase {
function foo() external view;
}
contract Base1 is IBase { function foo() public virtual view {} }
contract Base2 is IBase { function foo() public virtual view {} }
interface IExt1a is IBase {}
abstract contract IExt1b is IBase {}
abstract contract IExt2a is IBase {}
interface IExt2b is IBase {}
contract Ext1 is IExt1a, IExt1b, Base1 {}
contract Ext2 is IExt2a, IExt2b, Base2 {}
contract Impl is Ext1, Ext2 {
function foo() public view override (IBase, Base1, Base2) {}
}