mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #12836 from ethereum/fixUsingGlobal
Fix using global with libraries.
This commit is contained in:
commit
df29ea72f0
@ -11,6 +11,7 @@ Compiler Features:
|
|||||||
|
|
||||||
|
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
|
* Type Checker: Properly check restrictions of ``using ... global`` in conjunction with libraries.
|
||||||
* Assembly-Json: Fix assembly json export to store jump types of operations in `jumpType` field instead of `value`.
|
* Assembly-Json: Fix assembly json export to store jump types of operations in `jumpType` field instead of `value`.
|
||||||
* TypeChecker: Convert parameters of function type to how they would be called for ``abi.encodeCall``.
|
* TypeChecker: Convert parameters of function type to how they would be called for ``abi.encodeCall``.
|
||||||
* View Pure Checker: Mark ``returndatasize`` and ``returndatacopy`` as view to disallow them in inline assembly blocks in pure functions.
|
* View Pure Checker: Mark ``returndatasize`` and ``returndatacopy`` as view to disallow them in inline assembly blocks in pure functions.
|
||||||
|
@ -3638,11 +3638,36 @@ void TypeChecker::endVisit(Literal const& _literal)
|
|||||||
|
|
||||||
void TypeChecker::endVisit(UsingForDirective const& _usingFor)
|
void TypeChecker::endVisit(UsingForDirective const& _usingFor)
|
||||||
{
|
{
|
||||||
|
if (_usingFor.global())
|
||||||
|
{
|
||||||
|
if (m_currentContract || !_usingFor.typeName())
|
||||||
|
{
|
||||||
|
solAssert(m_errorReporter.hasErrors());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
solAssert(_usingFor.typeName()->annotation().type);
|
||||||
|
if (Declaration const* typeDefinition = _usingFor.typeName()->annotation().type->typeDefinition())
|
||||||
|
{
|
||||||
|
if (typeDefinition->scope() != m_currentSourceUnit)
|
||||||
|
m_errorReporter.typeError(
|
||||||
|
4117_error,
|
||||||
|
_usingFor.location(),
|
||||||
|
"Can only use \"global\" with types defined in the same source unit at file level."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_errorReporter.typeError(
|
||||||
|
8841_error,
|
||||||
|
_usingFor.location(),
|
||||||
|
"Can only use \"global\" with user-defined types."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (!_usingFor.usesBraces())
|
if (!_usingFor.usesBraces())
|
||||||
{
|
{
|
||||||
solAssert(_usingFor.functionsOrLibrary().size() == 1);
|
solAssert(_usingFor.functionsOrLibrary().size() == 1);
|
||||||
ContractDefinition const* library = dynamic_cast<ContractDefinition const*>(
|
ContractDefinition const* library = dynamic_cast<ContractDefinition const*>(
|
||||||
_usingFor.functionsOrLibrary().front()->annotation().referencedDeclaration
|
_usingFor.functionsOrLibrary().front()->annotation().referencedDeclaration
|
||||||
);
|
);
|
||||||
solAssert(library && library->isLibrary());
|
solAssert(library && library->isLibrary());
|
||||||
// No type checking for libraries
|
// No type checking for libraries
|
||||||
@ -3662,28 +3687,6 @@ void TypeChecker::endVisit(UsingForDirective const& _usingFor)
|
|||||||
);
|
);
|
||||||
solAssert(normalizedType);
|
solAssert(normalizedType);
|
||||||
|
|
||||||
if (_usingFor.global())
|
|
||||||
{
|
|
||||||
if (m_currentContract)
|
|
||||||
solAssert(m_errorReporter.hasErrors());
|
|
||||||
if (Declaration const* typeDefinition = _usingFor.typeName()->annotation().type->typeDefinition())
|
|
||||||
{
|
|
||||||
if (typeDefinition->scope() != m_currentSourceUnit)
|
|
||||||
m_errorReporter.typeError(
|
|
||||||
4117_error,
|
|
||||||
_usingFor.location(),
|
|
||||||
"Can only use \"global\" with types defined in the same source unit at file level."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
m_errorReporter.typeError(
|
|
||||||
8841_error,
|
|
||||||
_usingFor.location(),
|
|
||||||
"Can only use \"global\" with user-defined types."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
for (ASTPointer<IdentifierPath> const& path: _usingFor.functionsOrLibrary())
|
for (ASTPointer<IdentifierPath> const& path: _usingFor.functionsOrLibrary())
|
||||||
{
|
{
|
||||||
solAssert(path->annotation().referencedDeclaration);
|
solAssert(path->annotation().referencedDeclaration);
|
||||||
|
@ -0,0 +1,18 @@
|
|||||||
|
using L for I;
|
||||||
|
interface I { function f() external pure returns (uint); }
|
||||||
|
library L {
|
||||||
|
function execute(I i) internal pure returns (uint) {
|
||||||
|
return i.f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
contract C is I {
|
||||||
|
function x() public view returns (uint) {
|
||||||
|
I i = this;
|
||||||
|
return i.execute();
|
||||||
|
}
|
||||||
|
function f() public pure returns (uint) { return 7; }
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
|
// ----
|
||||||
|
// x() -> 7
|
@ -0,0 +1,38 @@
|
|||||||
|
==== Source: A ====
|
||||||
|
enum E {A, B}
|
||||||
|
struct S { uint x; }
|
||||||
|
type T is uint;
|
||||||
|
using L for E global;
|
||||||
|
using L for S global;
|
||||||
|
using L for T global;
|
||||||
|
library L {
|
||||||
|
function f(E e) internal pure returns (uint) {
|
||||||
|
return uint(e);
|
||||||
|
}
|
||||||
|
function f(S memory s) internal pure returns (uint) {
|
||||||
|
return s.x;
|
||||||
|
}
|
||||||
|
function f(T t) internal pure returns (uint) {
|
||||||
|
return T.unwrap(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
==== Source: B ====
|
||||||
|
contract C {
|
||||||
|
function f() public pure returns (uint a, uint b, uint c) {
|
||||||
|
E e = E.B;
|
||||||
|
a = e.f();
|
||||||
|
S memory s;
|
||||||
|
s.x = 7;
|
||||||
|
b = s.f();
|
||||||
|
T t = T.wrap(9);
|
||||||
|
c = t.f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
import {E, S, T} from "A";
|
||||||
|
|
||||||
|
// ====
|
||||||
|
// compileViaYul: also
|
||||||
|
// ----
|
||||||
|
// f() -> 1, 7, 9
|
@ -4,4 +4,3 @@ contract C {
|
|||||||
function f(uint) pure{}
|
function f(uint) pure{}
|
||||||
// ----
|
// ----
|
||||||
// SyntaxError 3367: (17-43): "global" can only be used at file level.
|
// SyntaxError 3367: (17-43): "global" can only be used at file level.
|
||||||
// TypeError 8841: (17-43): Can only use "global" with user-defined types.
|
|
||||||
|
@ -0,0 +1,9 @@
|
|||||||
|
using L for uint global;
|
||||||
|
using L for uint[] global;
|
||||||
|
using L for function() returns (uint) global;
|
||||||
|
library L {
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 8841: (0-24): Can only use "global" with user-defined types.
|
||||||
|
// TypeError 8841: (25-51): Can only use "global" with user-defined types.
|
||||||
|
// TypeError 8841: (52-97): Can only use "global" with user-defined types.
|
@ -0,0 +1,6 @@
|
|||||||
|
using L for L.S global;
|
||||||
|
library L {
|
||||||
|
struct S { uint x; }
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 4117: (0-23): Can only use "global" with types defined in the same source unit at file level.
|
@ -0,0 +1,6 @@
|
|||||||
|
interface I {}
|
||||||
|
using L for I global;
|
||||||
|
library L {
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 8841: (15-36): Can only use "global" with user-defined types.
|
@ -0,0 +1,8 @@
|
|||||||
|
==== Source: A ====
|
||||||
|
struct S { uint x; }
|
||||||
|
==== Source: B ====
|
||||||
|
library L {}
|
||||||
|
using L for S global;
|
||||||
|
import {S} from "A";
|
||||||
|
// ----
|
||||||
|
// TypeError 4117: (B:13-34): Can only use "global" with types defined in the same source unit at file level.
|
@ -0,0 +1,5 @@
|
|||||||
|
using L for * global;
|
||||||
|
library L {}
|
||||||
|
// ----
|
||||||
|
// SyntaxError 8118: (0-21): The type has to be specified explicitly at file level (cannot use '*').
|
||||||
|
// SyntaxError 2854: (0-21): Can only globally bind functions to specific types.
|
@ -0,0 +1,5 @@
|
|||||||
|
using {f} for * global;
|
||||||
|
function f(uint) pure{}
|
||||||
|
// ----
|
||||||
|
// SyntaxError 8118: (0-23): The type has to be specified explicitly at file level (cannot use '*').
|
||||||
|
// SyntaxError 2854: (0-23): Can only globally bind functions to specific types.
|
Loading…
Reference in New Issue
Block a user