mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #624 from chriseth/inaccessibleDynType
Inaccessible dynamic types
This commit is contained in:
commit
0a0fc04641
@ -2037,29 +2037,20 @@ FunctionTypePointer FunctionType::asMemberFunction(bool _inLibrary, bool _bound)
|
|||||||
location = Location::DelegateCall;
|
location = Location::DelegateCall;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePointers returnParameterTypes;
|
TypePointers returnParameterTypes = m_returnParameterTypes;
|
||||||
vector<string> returnParameterNames;
|
if (location != Location::Internal)
|
||||||
if (location == Location::Internal)
|
|
||||||
{
|
{
|
||||||
returnParameterNames = m_returnParameterNames;
|
// Alter dynamic types to be non-accessible.
|
||||||
returnParameterTypes = m_returnParameterTypes;
|
for (auto& param: returnParameterTypes)
|
||||||
}
|
if (param->isDynamicallySized())
|
||||||
else
|
param = make_shared<InaccessibleDynamicType>();
|
||||||
{
|
|
||||||
// Removes dynamic types.
|
|
||||||
for (size_t i = 0; i < m_returnParameterTypes.size(); ++i)
|
|
||||||
if (!m_returnParameterTypes[i]->isDynamicallySized())
|
|
||||||
{
|
|
||||||
returnParameterTypes.push_back(m_returnParameterTypes[i]);
|
|
||||||
returnParameterNames.push_back(m_returnParameterNames[i]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return make_shared<FunctionType>(
|
return make_shared<FunctionType>(
|
||||||
parameterTypes,
|
parameterTypes,
|
||||||
returnParameterTypes,
|
returnParameterTypes,
|
||||||
m_parameterNames,
|
m_parameterNames,
|
||||||
returnParameterNames,
|
m_returnParameterNames,
|
||||||
location,
|
location,
|
||||||
m_arbitraryParameters,
|
m_arbitraryParameters,
|
||||||
m_declaration,
|
m_declaration,
|
||||||
|
@ -137,7 +137,8 @@ public:
|
|||||||
{
|
{
|
||||||
Integer, RationalNumber, StringLiteral, Bool, FixedPoint, Array,
|
Integer, RationalNumber, StringLiteral, Bool, FixedPoint, Array,
|
||||||
FixedBytes, Contract, Struct, Function, Enum, Tuple,
|
FixedBytes, Contract, Struct, Function, Enum, Tuple,
|
||||||
Mapping, TypeType, Modifier, Magic, Module
|
Mapping, TypeType, Modifier, Magic, Module,
|
||||||
|
InaccessibleDynamic
|
||||||
};
|
};
|
||||||
|
|
||||||
/// @{
|
/// @{
|
||||||
@ -1081,5 +1082,25 @@ private:
|
|||||||
Kind m_kind;
|
Kind m_kind;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special type that is used for dynamic types in returns from external function calls
|
||||||
|
* (The EVM currently cannot access dynamically-sized return values).
|
||||||
|
*/
|
||||||
|
class InaccessibleDynamicType: public Type
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual Category category() const override { return Category::InaccessibleDynamic; }
|
||||||
|
|
||||||
|
virtual bool isImplicitlyConvertibleTo(Type const&) const override { return false; }
|
||||||
|
virtual bool isExplicitlyConvertibleTo(Type const&) const override { return false; }
|
||||||
|
virtual unsigned calldataEncodedSize(bool _padded) const override { (void)_padded; return 32; }
|
||||||
|
virtual bool canBeStored() const override { return false; }
|
||||||
|
virtual bool canLiveOutsideStorage() const override { return false; }
|
||||||
|
virtual bool isValueType() const override { return true; }
|
||||||
|
virtual unsigned sizeOnStack() const override { return 1; }
|
||||||
|
virtual std::string toString(bool) const override { return "inaccessible dynamic type"; }
|
||||||
|
virtual TypePointer decodingType() const override { return std::make_shared<IntegerType>(256); }
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6793,6 +6793,51 @@ BOOST_AUTO_TEST_CASE(cleanup_bytes_types)
|
|||||||
BOOST_CHECK(callContractFunction("f(bytes2,uint16)", string("abc"), u256(0x040102)) == encodeArgs(0));
|
BOOST_CHECK(callContractFunction("f(bytes2,uint16)", string("abc"), u256(0x040102)) == encodeArgs(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(skip_dynamic_types)
|
||||||
|
{
|
||||||
|
// The EVM cannot provide access to dynamically-sized return values, so we have to skip them.
|
||||||
|
char const* sourceCode = R"(
|
||||||
|
contract C {
|
||||||
|
function f() returns (uint, uint[], uint) {
|
||||||
|
return (7, new uint[](2), 8);
|
||||||
|
}
|
||||||
|
function g() returns (uint, uint) {
|
||||||
|
// Previous implementation "moved" b to the second place and did not skip.
|
||||||
|
var (a, _, b) = this.f();
|
||||||
|
return (a, b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
compileAndRun(sourceCode, 0, "C");
|
||||||
|
BOOST_CHECK(callContractFunction("g()") == encodeArgs(u256(7), u256(8)));
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(skip_dynamic_types_for_structs)
|
||||||
|
{
|
||||||
|
// For accessors, the dynamic types are already removed in the external signature itself.
|
||||||
|
char const* sourceCode = R"(
|
||||||
|
contract C {
|
||||||
|
struct S {
|
||||||
|
uint x;
|
||||||
|
string a; // this is present in the accessor
|
||||||
|
uint[] b; // this is not present
|
||||||
|
uint y;
|
||||||
|
}
|
||||||
|
S public s;
|
||||||
|
function g() returns (uint, uint) {
|
||||||
|
s.x = 2;
|
||||||
|
s.a = "abc";
|
||||||
|
s.b = [7, 8, 9];
|
||||||
|
s.y = 6;
|
||||||
|
var (x, a, y) = this.s();
|
||||||
|
return (x, y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
compileAndRun(sourceCode, 0, "C");
|
||||||
|
BOOST_CHECK(callContractFunction("g()") == encodeArgs(u256(2), u256(6)));
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_SUITE_END()
|
BOOST_AUTO_TEST_SUITE_END()
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -2154,6 +2154,8 @@ BOOST_AUTO_TEST_CASE(dynamic_return_types_not_possible)
|
|||||||
function f(uint) returns (string);
|
function f(uint) returns (string);
|
||||||
function g() {
|
function g() {
|
||||||
var (x,) = this.f(2);
|
var (x,) = this.f(2);
|
||||||
|
// we can assign to x but it is not usable.
|
||||||
|
bytes(x).length;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
|
Loading…
Reference in New Issue
Block a user