mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
bytes parameters for events and sha3.
This commit is contained in:
parent
a44bcb6909
commit
a6d08950c6
11
AST.cpp
11
AST.cpp
@ -450,14 +450,11 @@ void FunctionDefinition::checkTypeRequirements()
|
|||||||
{
|
{
|
||||||
if (!var->getType()->canLiveOutsideStorage())
|
if (!var->getType()->canLiveOutsideStorage())
|
||||||
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
||||||
|
// todo delete when will be implemented arrays as parameter type in internal functions
|
||||||
|
if (getVisibility() == Visibility::Public && var->getType()->getCategory() == Type::Category::Array)
|
||||||
|
BOOST_THROW_EXCEPTION(var->createTypeError("Arrays only implemented for external functions."));
|
||||||
if (getVisibility() >= Visibility::Public && !(var->getType()->externalType()))
|
if (getVisibility() >= Visibility::Public && !(var->getType()->externalType()))
|
||||||
{
|
BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed for public and external functions."));
|
||||||
// todo delete when will be implemented arrays as parameter type in internal functions
|
|
||||||
if (getVisibility() == Visibility::Public && var->getType()->getCategory() == Type::Category::Array)
|
|
||||||
BOOST_THROW_EXCEPTION(var->createTypeError("Arrays only implemented for external functions."));
|
|
||||||
else
|
|
||||||
BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed for public and external functions."));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
|
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
|
||||||
modifier->checkTypeRequirements(isConstructor() ?
|
modifier->checkTypeRequirements(isConstructor() ?
|
||||||
|
@ -155,6 +155,13 @@ void CompilerUtils::copyToStackTop(unsigned _stackDepth, unsigned _itemSize)
|
|||||||
m_context << eth::dupInstruction(_stackDepth);
|
m_context << eth::dupInstruction(_stackDepth);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CompilerUtils::moveToStackTop(unsigned _stackDepth)
|
||||||
|
{
|
||||||
|
solAssert(_stackDepth <= 15, "Stack too deep.");
|
||||||
|
for (unsigned i = 0; i < _stackDepth; ++i)
|
||||||
|
m_context << eth::swapInstruction(1 + i);
|
||||||
|
}
|
||||||
|
|
||||||
void CompilerUtils::popStackElement(Type const& _type)
|
void CompilerUtils::popStackElement(Type const& _type)
|
||||||
{
|
{
|
||||||
popStackSlots(_type.getSizeOnStack());
|
popStackSlots(_type.getSizeOnStack());
|
||||||
|
@ -77,6 +77,8 @@ public:
|
|||||||
/// Copies an item that occupies @a _itemSize stack slots from a stack depth of @a _stackDepth
|
/// Copies an item that occupies @a _itemSize stack slots from a stack depth of @a _stackDepth
|
||||||
/// to the top of the stack.
|
/// to the top of the stack.
|
||||||
void copyToStackTop(unsigned _stackDepth, unsigned _itemSize);
|
void copyToStackTop(unsigned _stackDepth, unsigned _itemSize);
|
||||||
|
/// Moves a single stack element (with _stackDepth items on top of it) to the top of the stack.
|
||||||
|
void moveToStackTop(unsigned _stackDepth);
|
||||||
/// Removes the current value from the top of the stack.
|
/// Removes the current value from the top of the stack.
|
||||||
void popStackElement(Type const& _type);
|
void popStackElement(Type const& _type);
|
||||||
/// Removes element from the top of the stack _amount times.
|
/// Removes element from the top of the stack _amount times.
|
||||||
|
@ -532,7 +532,7 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
|
|||||||
case Location::SHA3:
|
case Location::SHA3:
|
||||||
{
|
{
|
||||||
m_context << u256(0);
|
m_context << u256(0);
|
||||||
appendArgumentsCopyToMemory(arguments, TypePointers(), function.padArguments());
|
appendArgumentsCopyToMemory(arguments, TypePointers(), function.padArguments(), false, true);
|
||||||
m_context << u256(0) << eth::Instruction::SHA3;
|
m_context << u256(0) << eth::Instruction::SHA3;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -575,9 +575,15 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
|
|||||||
solAssert(numIndexed <= 4, "Too many indexed arguments.");
|
solAssert(numIndexed <= 4, "Too many indexed arguments.");
|
||||||
// Copy all non-indexed arguments to memory (data)
|
// Copy all non-indexed arguments to memory (data)
|
||||||
m_context << u256(0);
|
m_context << u256(0);
|
||||||
|
vector<ASTPointer<Expression const>> nonIndexedArgs;
|
||||||
|
TypePointers nonIndexedTypes;
|
||||||
for (unsigned arg = 0; arg < arguments.size(); ++arg)
|
for (unsigned arg = 0; arg < arguments.size(); ++arg)
|
||||||
if (!event.getParameters()[arg]->isIndexed())
|
if (!event.getParameters()[arg]->isIndexed())
|
||||||
appendExpressionCopyToMemory(*function.getParameterTypes()[arg], *arguments[arg]);
|
{
|
||||||
|
nonIndexedArgs.push_back(arguments[arg]);
|
||||||
|
nonIndexedTypes.push_back(function.getParameterTypes()[arg]);
|
||||||
|
}
|
||||||
|
appendArgumentsCopyToMemory(nonIndexedArgs, nonIndexedTypes);
|
||||||
m_context << u256(0) << eth::logInstruction(numIndexed);
|
m_context << u256(0) << eth::logInstruction(numIndexed);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1046,8 +1052,14 @@ void ExpressionCompiler::appendExternalFunctionCall(FunctionType const& _functio
|
|||||||
|
|
||||||
// For bare call, activate "4 byte pad exception": If the first argument has exactly 4 bytes,
|
// For bare call, activate "4 byte pad exception": If the first argument has exactly 4 bytes,
|
||||||
// do not pad it to 32 bytes.
|
// do not pad it to 32 bytes.
|
||||||
appendArgumentsCopyToMemory(_arguments, _functionType.getParameterTypes(),
|
// If the function takes arbitrary parameters, copy dynamic length data in place.
|
||||||
_functionType.padArguments(), bare);
|
appendArgumentsCopyToMemory(
|
||||||
|
_arguments,
|
||||||
|
_functionType.getParameterTypes(),
|
||||||
|
_functionType.padArguments(),
|
||||||
|
bare,
|
||||||
|
_functionType.takesArbitraryParameters()
|
||||||
|
);
|
||||||
|
|
||||||
// CALL arguments: outSize, outOff, inSize, (already present up to here)
|
// CALL arguments: outSize, outOff, inSize, (already present up to here)
|
||||||
// inOff, value, addr, gas (stack top)
|
// inOff, value, addr, gas (stack top)
|
||||||
@ -1089,20 +1101,72 @@ void ExpressionCompiler::appendArgumentsCopyToMemory(
|
|||||||
vector<ASTPointer<Expression const>> const& _arguments,
|
vector<ASTPointer<Expression const>> const& _arguments,
|
||||||
TypePointers const& _types,
|
TypePointers const& _types,
|
||||||
bool _padToWordBoundaries,
|
bool _padToWordBoundaries,
|
||||||
bool _padExceptionIfFourBytes
|
bool _padExceptionIfFourBytes,
|
||||||
|
bool _copyDynamicDataInPlace
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
solAssert(_types.empty() || _types.size() == _arguments.size(), "");
|
solAssert(_types.empty() || _types.size() == _arguments.size(), "");
|
||||||
|
TypePointers types = _types;
|
||||||
|
if (_types.empty())
|
||||||
|
for (ASTPointer<Expression const> const& argument: _arguments)
|
||||||
|
types.push_back(argument->getType()->getRealType());
|
||||||
|
|
||||||
|
vector<size_t> dynamicArguments;
|
||||||
|
unsigned stackSizeOfDynamicTypes = 0;
|
||||||
for (size_t i = 0; i < _arguments.size(); ++i)
|
for (size_t i = 0; i < _arguments.size(); ++i)
|
||||||
{
|
{
|
||||||
_arguments[i]->accept(*this);
|
_arguments[i]->accept(*this);
|
||||||
TypePointer const& expectedType = _types.empty() ? _arguments[i]->getType()->getRealType() : _types[i];
|
TypePointer argType = types[i]->externalType();
|
||||||
appendTypeConversion(*_arguments[i]->getType(), *expectedType, true);
|
solAssert(!!argType, "Externalable type expected.");
|
||||||
|
if (argType->isValueType())
|
||||||
|
appendTypeConversion(*_arguments[i]->getType(), *argType, true);
|
||||||
|
else
|
||||||
|
argType = _arguments[i]->getType()->getRealType()->externalType();
|
||||||
|
solAssert(!!argType, "Externalable type expected.");
|
||||||
bool pad = _padToWordBoundaries;
|
bool pad = _padToWordBoundaries;
|
||||||
// Do not pad if the first argument has exactly four bytes
|
// Do not pad if the first argument has exactly four bytes
|
||||||
if (i == 0 && pad && _padExceptionIfFourBytes && expectedType->getCalldataEncodedSize(false) == 4)
|
if (i == 0 && pad && _padExceptionIfFourBytes && argType->getCalldataEncodedSize(false) == 4)
|
||||||
pad = false;
|
pad = false;
|
||||||
appendTypeMoveToMemory(*expectedType, pad);
|
if (!_copyDynamicDataInPlace && argType->isDynamicallySized())
|
||||||
|
{
|
||||||
|
solAssert(argType->getCategory() == Type::Category::Array, "Unknown dynamic type.");
|
||||||
|
auto const& arrayType = dynamic_cast<ArrayType const&>(*_arguments[i]->getType());
|
||||||
|
// move memory reference to top of stack
|
||||||
|
CompilerUtils(m_context).moveToStackTop(arrayType.getSizeOnStack());
|
||||||
|
if (arrayType.getLocation() == ArrayType::Location::CallData)
|
||||||
|
m_context << eth::Instruction::DUP2; // length is on stack
|
||||||
|
else if (arrayType.getLocation() == ArrayType::Location::Storage)
|
||||||
|
m_context << eth::Instruction::DUP3 << eth::Instruction::SLOAD;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
solAssert(arrayType.getLocation() == ArrayType::Location::Memory, "");
|
||||||
|
m_context << eth::Instruction::DUP2 << eth::Instruction::MLOAD;
|
||||||
|
}
|
||||||
|
appendTypeMoveToMemory(IntegerType(256), true);
|
||||||
|
stackSizeOfDynamicTypes += arrayType.getSizeOnStack();
|
||||||
|
dynamicArguments.push_back(i);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
appendTypeMoveToMemory(*argType, pad);
|
||||||
|
}
|
||||||
|
|
||||||
|
// copy dynamic values to memory
|
||||||
|
unsigned dynStackPointer = stackSizeOfDynamicTypes;
|
||||||
|
// stack layout: <dyn arg 1> ... <dyn arg m> <memory pointer>
|
||||||
|
for (size_t i: dynamicArguments)
|
||||||
|
{
|
||||||
|
auto const& arrayType = dynamic_cast<ArrayType const&>(*_arguments[i]->getType());
|
||||||
|
CompilerUtils(m_context).copyToStackTop(1 + dynStackPointer, arrayType.getSizeOnStack());
|
||||||
|
dynStackPointer -= arrayType.getSizeOnStack();
|
||||||
|
appendTypeMoveToMemory(arrayType, true);
|
||||||
|
}
|
||||||
|
solAssert(dynStackPointer == 0, "");
|
||||||
|
|
||||||
|
// remove dynamic values (and retain memory pointer)
|
||||||
|
if (stackSizeOfDynamicTypes > 0)
|
||||||
|
{
|
||||||
|
m_context << eth::swapInstruction(stackSizeOfDynamicTypes);
|
||||||
|
CompilerUtils(m_context).popStackSlots(stackSizeOfDynamicTypes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1114,8 +1178,13 @@ void ExpressionCompiler::appendTypeMoveToMemory(Type const& _type, bool _padToWo
|
|||||||
void ExpressionCompiler::appendExpressionCopyToMemory(Type const& _expectedType, Expression const& _expression)
|
void ExpressionCompiler::appendExpressionCopyToMemory(Type const& _expectedType, Expression const& _expression)
|
||||||
{
|
{
|
||||||
_expression.accept(*this);
|
_expression.accept(*this);
|
||||||
appendTypeConversion(*_expression.getType(), _expectedType, true);
|
if (_expectedType.isValueType())
|
||||||
appendTypeMoveToMemory(_expectedType);
|
{
|
||||||
|
appendTypeConversion(*_expression.getType(), _expectedType, true);
|
||||||
|
appendTypeMoveToMemory(_expectedType);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
appendTypeMoveToMemory(*_expression.getType()->getRealType());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExpressionCompiler::setLValueFromDeclaration(Declaration const& _declaration, Expression const& _expression)
|
void ExpressionCompiler::setLValueFromDeclaration(Declaration const& _declaration, Expression const& _expression)
|
||||||
|
@ -100,12 +100,18 @@ private:
|
|||||||
/// Appends code to call a function of the given type with the given arguments.
|
/// Appends code to call a function of the given type with the given arguments.
|
||||||
void appendExternalFunctionCall(FunctionType const& _functionType, std::vector<ASTPointer<Expression const>> const& _arguments,
|
void appendExternalFunctionCall(FunctionType const& _functionType, std::vector<ASTPointer<Expression const>> const& _arguments,
|
||||||
bool bare = false);
|
bool bare = false);
|
||||||
/// Appends code that evaluates the given arguments and moves the result to memory. The memory offset is
|
/// Appends code that evaluates the given arguments and moves the result to memory encoded as
|
||||||
/// expected to be on the stack and is updated by this call.
|
/// specified by the ABI. The memory offset is expected to be on the stack and is updated by
|
||||||
void appendArgumentsCopyToMemory(std::vector<ASTPointer<Expression const>> const& _arguments,
|
/// this call. If @a _padToWordBoundaries is set to false, all values are concatenated without
|
||||||
TypePointers const& _types = {},
|
/// padding. If @a _copyDynamicDataInPlace is set, dynamic types is stored (without length)
|
||||||
bool _padToWordBoundaries = true,
|
/// together with fixed-length data.
|
||||||
bool _padExceptionIfFourBytes = false);
|
void appendArgumentsCopyToMemory(
|
||||||
|
std::vector<ASTPointer<Expression const>> const& _arguments,
|
||||||
|
TypePointers const& _types = {},
|
||||||
|
bool _padToWordBoundaries = true,
|
||||||
|
bool _padExceptionIfFourBytes = false,
|
||||||
|
bool _copyDynamicDataInPlace = false
|
||||||
|
);
|
||||||
/// Appends code that moves a stack element of the given type to memory. The memory offset is
|
/// Appends code that moves a stack element of the given type to memory. The memory offset is
|
||||||
/// expected below the stack element and is updated by this call.
|
/// expected below the stack element and is updated by this call.
|
||||||
void appendTypeMoveToMemory(Type const& _type, bool _padToWordBoundaries = true);
|
void appendTypeMoveToMemory(Type const& _type, bool _padToWordBoundaries = true);
|
||||||
|
@ -745,8 +745,6 @@ string ArrayType::toString() const
|
|||||||
|
|
||||||
TypePointer ArrayType::externalType() const
|
TypePointer ArrayType::externalType() const
|
||||||
{
|
{
|
||||||
if (m_location != Location::CallData)
|
|
||||||
return TypePointer();
|
|
||||||
if (m_isByteArray)
|
if (m_isByteArray)
|
||||||
return shared_from_this();
|
return shared_from_this();
|
||||||
if (!m_baseType->externalType())
|
if (!m_baseType->externalType())
|
||||||
@ -1218,7 +1216,9 @@ string FunctionType::externalSignature(std::string const& _name) const
|
|||||||
}
|
}
|
||||||
string ret = funcName + "(";
|
string ret = funcName + "(";
|
||||||
|
|
||||||
TypePointers externalParameterTypes = externalFunctionType()->getParameterTypes();
|
FunctionTypePointer external = externalFunctionType();
|
||||||
|
solAssert(!!external, "External function type requested.");
|
||||||
|
TypePointers externalParameterTypes = external->getParameterTypes();
|
||||||
for (auto it = externalParameterTypes.cbegin(); it != externalParameterTypes.cend(); ++it)
|
for (auto it = externalParameterTypes.cbegin(); it != externalParameterTypes.cend(); ++it)
|
||||||
{
|
{
|
||||||
solAssert(!!(*it), "Parameter should have external type");
|
solAssert(!!(*it), "Parameter should have external type");
|
||||||
|
6
Types.h
6
Types.h
@ -430,7 +430,7 @@ public:
|
|||||||
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
||||||
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
||||||
virtual bool operator==(Type const& _other) const override;
|
virtual bool operator==(Type const& _other) const override;
|
||||||
virtual unsigned getCalldataEncodedSize(bool _padded = true) const override
|
virtual unsigned getCalldataEncodedSize(bool _padded ) const override
|
||||||
{
|
{
|
||||||
return externalType()->getCalldataEncodedSize(_padded);
|
return externalType()->getCalldataEncodedSize(_padded);
|
||||||
}
|
}
|
||||||
@ -506,7 +506,7 @@ public:
|
|||||||
explicit EnumType(EnumDefinition const& _enum): m_enum(_enum) {}
|
explicit EnumType(EnumDefinition const& _enum): m_enum(_enum) {}
|
||||||
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
||||||
virtual bool operator==(Type const& _other) const override;
|
virtual bool operator==(Type const& _other) const override;
|
||||||
virtual unsigned getCalldataEncodedSize(bool _padded = true) const override
|
virtual unsigned getCalldataEncodedSize(bool _padded) const override
|
||||||
{
|
{
|
||||||
return externalType()->getCalldataEncodedSize(_padded);
|
return externalType()->getCalldataEncodedSize(_padded);
|
||||||
}
|
}
|
||||||
@ -558,7 +558,7 @@ public:
|
|||||||
/// appropriate external types of input/return parameters of current function.
|
/// appropriate external types of input/return parameters of current function.
|
||||||
/// Returns an empty shared pointer if one of the input/return parameters does not have an
|
/// Returns an empty shared pointer if one of the input/return parameters does not have an
|
||||||
/// external type.
|
/// external type.
|
||||||
virtual FunctionTypePointer externalFunctionType() const;
|
FunctionTypePointer externalFunctionType() const;
|
||||||
virtual TypePointer externalType() const override { return externalFunctionType(); }
|
virtual TypePointer externalType() const override { return externalFunctionType(); }
|
||||||
|
|
||||||
explicit FunctionType(FunctionDefinition const& _function, bool _isInternal = true);
|
explicit FunctionType(FunctionDefinition const& _function, bool _isInternal = true);
|
||||||
|
Loading…
Reference in New Issue
Block a user