Merge pull request #1263 from ethereum/1116

Do not push code for `L` in `L.Foo`, an enum Foo defined in a library L
This commit is contained in:
chriseth 2016-10-24 15:53:11 +02:00 committed by GitHub
commit d651a506fa
3 changed files with 49 additions and 1 deletions

View File

@ -14,6 +14,7 @@ Bugfixes:
* Type checker: Proper type checking for bound functions. * Type checker: Proper type checking for bound functions.
* Type checker: fix crash related to invalid fixed point constants * Type checker: fix crash related to invalid fixed point constants
* Code generator: expect zero stack increase after ``super`` as an expression. * Code generator: expect zero stack increase after ``super`` as an expression.
* Code Generator: fixed an internal compiler error for ``L.Foo`` for ``enum Foo`` defined in library ``L``.
* Inline assembly: support the ``address`` opcode. * Inline assembly: support the ``address`` opcode.
* Inline assembly: fix parsing of assignment after a label. * Inline assembly: fix parsing of assignment after a label.
* Inline assembly: external variables of unsupported type (such as ``this``, ``super``, etc.) * Inline assembly: external variables of unsupported type (such as ``this``, ``super``, etc.)

View File

@ -861,11 +861,12 @@ bool ExpressionCompiler::visit(MemberAccess const& _memberAccess)
} }
// Special processing for TypeType because we do not want to visit the library itself // Special processing for TypeType because we do not want to visit the library itself
// for internal functions. // for internal functions, or enum/struct definitions.
if (TypeType const* type = dynamic_cast<TypeType const*>(_memberAccess.expression().annotation().type.get())) if (TypeType const* type = dynamic_cast<TypeType const*>(_memberAccess.expression().annotation().type.get()))
{ {
if (dynamic_cast<ContractType const*>(type->actualType().get())) if (dynamic_cast<ContractType const*>(type->actualType().get()))
{ {
solAssert(_memberAccess.annotation().type, "_memberAccess has no type");
if (auto funType = dynamic_cast<FunctionType const*>(_memberAccess.annotation().type.get())) if (auto funType = dynamic_cast<FunctionType const*>(_memberAccess.annotation().type.get()))
{ {
if (funType->location() != FunctionType::Location::Internal) if (funType->location() != FunctionType::Location::Internal)
@ -883,6 +884,10 @@ bool ExpressionCompiler::visit(MemberAccess const& _memberAccess)
m_context << m_context.functionEntryLabel(*function).pushTag(); m_context << m_context.functionEntryLabel(*function).pushTag();
} }
} }
else if (dynamic_cast<TypeType const*>(_memberAccess.annotation().type.get()))
{
// no-op
}
else else
_memberAccess.expression().accept(*this); _memberAccess.expression().accept(*this);
} }

View File

@ -5905,6 +5905,48 @@ BOOST_AUTO_TEST_CASE(using_library_structs)
BOOST_CHECK(callContractFunction("f()") == encodeArgs(u256(7), u256(8))); BOOST_CHECK(callContractFunction("f()") == encodeArgs(u256(7), u256(8)));
} }
BOOST_AUTO_TEST_CASE(library_struct_as_an_expression)
{
char const* sourceCode = R"(
library Arst {
struct Foo {
int Things;
int Stuff;
}
}
contract Tsra {
function f() returns(uint) {
Arst.Foo;
return 1;
}
}
)";
compileAndRun(sourceCode, 0, "Tsra");
BOOST_CHECK(callContractFunction("f()") == encodeArgs(u256(1)));
}
BOOST_AUTO_TEST_CASE(library_enum_as_an_expression)
{
char const* sourceCode = R"(
library Arst {
enum Foo {
Things,
Stuff
}
}
contract Tsra {
function f() returns(uint) {
Arst.Foo;
return 1;
}
}
)";
compileAndRun(sourceCode, 0, "Tsra");
BOOST_CHECK(callContractFunction("f()") == encodeArgs(u256(1)));
}
BOOST_AUTO_TEST_CASE(short_strings) BOOST_AUTO_TEST_CASE(short_strings)
{ {
// This test verifies that the byte array encoding that combines length and data works // This test verifies that the byte array encoding that combines length and data works