mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Enlarge storage references to two stack slots.
This commit is contained in:
parent
c34e1da6db
commit
fff3f98f58
@ -34,8 +34,10 @@ using namespace solidity;
|
|||||||
|
|
||||||
void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType const& _sourceType) const
|
void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType const& _sourceType) const
|
||||||
{
|
{
|
||||||
// stack layout: [source_ref] target_ref (top)
|
// this copies source to target and also clears target if it was larger
|
||||||
// need to leave target_ref on the stack at the end
|
// need to leave "target_ref target_byte_off" on the stack at the end
|
||||||
|
|
||||||
|
// stack layout: [source_ref] [source_byte_off] [source length] target_ref target_byte_off (top)
|
||||||
solAssert(_targetType.getLocation() == ArrayType::Location::Storage, "");
|
solAssert(_targetType.getLocation() == ArrayType::Location::Storage, "");
|
||||||
solAssert(
|
solAssert(
|
||||||
_sourceType.getLocation() == ArrayType::Location::CallData ||
|
_sourceType.getLocation() == ArrayType::Location::CallData ||
|
||||||
@ -47,14 +49,20 @@ void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType cons
|
|||||||
Type const* targetBaseType = _targetType.isByteArray() ? &uint256 : &(*_targetType.getBaseType());
|
Type const* targetBaseType = _targetType.isByteArray() ? &uint256 : &(*_targetType.getBaseType());
|
||||||
Type const* sourceBaseType = _sourceType.isByteArray() ? &uint256 : &(*_sourceType.getBaseType());
|
Type const* sourceBaseType = _sourceType.isByteArray() ? &uint256 : &(*_sourceType.getBaseType());
|
||||||
|
|
||||||
// this copies source to target and also clears target if it was larger
|
|
||||||
|
|
||||||
// TODO unroll loop for small sizes
|
// TODO unroll loop for small sizes
|
||||||
|
|
||||||
// stack: source_ref [source_length] target_ref
|
bool sourceIsStorage = _sourceType.getLocation() == ArrayType::Location::Storage;
|
||||||
|
|
||||||
|
// stack: source_ref [source_byte_off] [source_length] target_ref target_byte_off
|
||||||
// store target_ref
|
// store target_ref
|
||||||
|
m_context << eth::Instruction::POP; //@todo
|
||||||
for (unsigned i = _sourceType.getSizeOnStack(); i > 0; --i)
|
for (unsigned i = _sourceType.getSizeOnStack(); i > 0; --i)
|
||||||
m_context << eth::swapInstruction(i);
|
m_context << eth::swapInstruction(i);
|
||||||
|
// stack: target_ref source_ref [source_byte_off] [source_length]
|
||||||
|
if (sourceIsStorage)
|
||||||
|
m_context << eth::Instruction::POP; //@todo
|
||||||
|
// stack: target_ref source_ref [source_length]
|
||||||
|
// retrieve source length
|
||||||
if (_sourceType.getLocation() != ArrayType::Location::CallData || !_sourceType.isDynamicallySized())
|
if (_sourceType.getLocation() != ArrayType::Location::CallData || !_sourceType.isDynamicallySized())
|
||||||
retrieveLength(_sourceType); // otherwise, length is already there
|
retrieveLength(_sourceType); // otherwise, length is already there
|
||||||
// stack: target_ref source_ref source_length
|
// stack: target_ref source_ref source_length
|
||||||
@ -73,6 +81,7 @@ void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType cons
|
|||||||
m_context
|
m_context
|
||||||
<< eth::Instruction::POP << eth::Instruction::POP
|
<< eth::Instruction::POP << eth::Instruction::POP
|
||||||
<< eth::Instruction::POP << eth::Instruction::POP;
|
<< eth::Instruction::POP << eth::Instruction::POP;
|
||||||
|
m_context << u256(0); //@todo
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// compute hashes (data positions)
|
// compute hashes (data positions)
|
||||||
@ -109,32 +118,37 @@ void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType cons
|
|||||||
// copy
|
// copy
|
||||||
if (sourceBaseType->getCategory() == Type::Category::Array)
|
if (sourceBaseType->getCategory() == Type::Category::Array)
|
||||||
{
|
{
|
||||||
m_context << eth::Instruction::DUP3 << eth::Instruction::DUP3;
|
//@todo
|
||||||
|
m_context << eth::Instruction::DUP3;
|
||||||
|
if (sourceIsStorage)
|
||||||
|
m_context << u256(0);
|
||||||
|
m_context << eth::dupInstruction(sourceIsStorage ? 4 : 3) << u256(0);
|
||||||
copyArrayToStorage(
|
copyArrayToStorage(
|
||||||
dynamic_cast<ArrayType const&>(*targetBaseType),
|
dynamic_cast<ArrayType const&>(*targetBaseType),
|
||||||
dynamic_cast<ArrayType const&>(*sourceBaseType)
|
dynamic_cast<ArrayType const&>(*sourceBaseType)
|
||||||
);
|
);
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_context << eth::Instruction::DUP3;
|
m_context << eth::Instruction::DUP3;
|
||||||
if (_sourceType.getLocation() == ArrayType::Location::Storage)
|
if (_sourceType.getLocation() == ArrayType::Location::Storage)
|
||||||
|
{
|
||||||
|
m_context << u256(0);
|
||||||
StorageItem(m_context, *sourceBaseType).retrieveValue(SourceLocation(), true);
|
StorageItem(m_context, *sourceBaseType).retrieveValue(SourceLocation(), true);
|
||||||
|
}
|
||||||
else if (sourceBaseType->isValueType())
|
else if (sourceBaseType->isValueType())
|
||||||
CompilerUtils(m_context).loadFromMemoryDynamic(*sourceBaseType, true, true, false);
|
CompilerUtils(m_context).loadFromMemoryDynamic(*sourceBaseType, true, true, false);
|
||||||
else
|
else
|
||||||
solAssert(false, "Copying of unknown type requested: " + sourceBaseType->toString());
|
solAssert(false, "Copying of unknown type requested: " + sourceBaseType->toString());
|
||||||
solAssert(2 + sourceBaseType->getSizeOnStack() <= 16, "Stack too deep.");
|
solAssert(2 + sourceBaseType->getSizeOnStack() <= 16, "Stack too deep.");
|
||||||
m_context << eth::dupInstruction(2 + sourceBaseType->getSizeOnStack());
|
m_context << eth::dupInstruction(2 + sourceBaseType->getSizeOnStack()) << u256(0);
|
||||||
StorageItem(m_context, *targetBaseType).storeValue(*sourceBaseType, SourceLocation(), true);
|
StorageItem(m_context, *targetBaseType).storeValue(*sourceBaseType, SourceLocation(), true);
|
||||||
}
|
}
|
||||||
// increment source
|
// increment source
|
||||||
m_context
|
m_context
|
||||||
<< eth::Instruction::SWAP2
|
<< eth::Instruction::SWAP2
|
||||||
<< (_sourceType.getLocation() == ArrayType::Location::Storage ?
|
<< (sourceIsStorage ? sourceBaseType->getStorageSize() : sourceBaseType->getCalldataEncodedSize())
|
||||||
sourceBaseType->getStorageSize() :
|
|
||||||
sourceBaseType->getCalldataEncodedSize())
|
|
||||||
<< eth::Instruction::ADD
|
<< eth::Instruction::ADD
|
||||||
<< eth::Instruction::SWAP2;
|
<< eth::Instruction::SWAP2;
|
||||||
// increment target
|
// increment target
|
||||||
@ -147,40 +161,48 @@ void ArrayUtils::copyArrayToStorage(ArrayType const& _targetType, ArrayType cons
|
|||||||
m_context << copyLoopEnd;
|
m_context << copyLoopEnd;
|
||||||
|
|
||||||
// zero-out leftovers in target
|
// zero-out leftovers in target
|
||||||
// stack: target_ref target_data_end source_data_pos target_data_pos source_data_end
|
// stack: target_ref target_data_end source_data_pos target_data_pos_updated source_data_end
|
||||||
m_context << eth::Instruction::POP << eth::Instruction::SWAP1 << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::SWAP1 << eth::Instruction::POP;
|
||||||
// stack: target_ref target_data_end target_data_pos_updated
|
// stack: target_ref target_data_end target_data_pos_updated
|
||||||
clearStorageLoop(*targetBaseType);
|
clearStorageLoop(*targetBaseType);
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP;
|
||||||
|
m_context << u256(0); //@todo
|
||||||
}
|
}
|
||||||
|
|
||||||
void ArrayUtils::clearArray(ArrayType const& _type) const
|
void ArrayUtils::clearArray(ArrayType const& _type) const
|
||||||
{
|
{
|
||||||
|
unsigned stackHeightStart = m_context.getStackHeight();
|
||||||
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
||||||
if (_type.isDynamicallySized())
|
if (_type.isDynamicallySized())
|
||||||
|
{
|
||||||
|
m_context << eth::Instruction::POP; // remove byte offset
|
||||||
clearDynamicArray(_type);
|
clearDynamicArray(_type);
|
||||||
|
}
|
||||||
else if (_type.getLength() == 0 || _type.getBaseType()->getCategory() == Type::Category::Mapping)
|
else if (_type.getLength() == 0 || _type.getBaseType()->getCategory() == Type::Category::Mapping)
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
else if (_type.getLength() < 5) // unroll loop for small arrays @todo choose a good value
|
else if (_type.getLength() < 5) // unroll loop for small arrays @todo choose a good value
|
||||||
{
|
{
|
||||||
solAssert(!_type.isByteArray(), "");
|
solAssert(!_type.isByteArray(), "");
|
||||||
for (unsigned i = 1; i < _type.getLength(); ++i)
|
for (unsigned i = 1; i < _type.getLength(); ++i)
|
||||||
{
|
{
|
||||||
StorageItem(m_context, *_type.getBaseType()).setToZero(SourceLocation(), false);
|
StorageItem(m_context, *_type.getBaseType()).setToZero(SourceLocation(), false);
|
||||||
|
m_context << eth::Instruction::SWAP1;
|
||||||
m_context << u256(_type.getBaseType()->getStorageSize()) << eth::Instruction::ADD;
|
m_context << u256(_type.getBaseType()->getStorageSize()) << eth::Instruction::ADD;
|
||||||
|
m_context << eth::Instruction::SWAP1;
|
||||||
}
|
}
|
||||||
StorageItem(m_context, *_type.getBaseType()).setToZero(SourceLocation(), true);
|
StorageItem(m_context, *_type.getBaseType()).setToZero(SourceLocation(), true);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(!_type.isByteArray(), "");
|
solAssert(!_type.isByteArray(), "");
|
||||||
m_context
|
m_context << eth::Instruction::SWAP1;
|
||||||
<< eth::Instruction::DUP1 << u256(_type.getLength())
|
m_context << eth::Instruction::DUP1 << _type.getLength();
|
||||||
<< u256(_type.getBaseType()->getStorageSize())
|
convertLengthToSize(_type);
|
||||||
<< eth::Instruction::MUL << eth::Instruction::ADD << eth::Instruction::SWAP1;
|
m_context << eth::Instruction::ADD << eth::Instruction::SWAP1;
|
||||||
clearStorageLoop(*_type.getBaseType());
|
clearStorageLoop(*_type.getBaseType());
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
|
solAssert(m_context.getStackHeight() == stackHeightStart - 2, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ArrayUtils::clearDynamicArray(ArrayType const& _type) const
|
void ArrayUtils::clearDynamicArray(ArrayType const& _type) const
|
||||||
@ -188,6 +210,7 @@ void ArrayUtils::clearDynamicArray(ArrayType const& _type) const
|
|||||||
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
||||||
solAssert(_type.isDynamicallySized(), "");
|
solAssert(_type.isDynamicallySized(), "");
|
||||||
|
|
||||||
|
unsigned stackHeightStart = m_context.getStackHeight();
|
||||||
// fetch length
|
// fetch length
|
||||||
m_context << eth::Instruction::DUP1 << eth::Instruction::SLOAD;
|
m_context << eth::Instruction::DUP1 << eth::Instruction::SLOAD;
|
||||||
// set length to zero
|
// set length to zero
|
||||||
@ -197,7 +220,7 @@ void ArrayUtils::clearDynamicArray(ArrayType const& _type) const
|
|||||||
// compute data positions
|
// compute data positions
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context << eth::Instruction::SWAP1;
|
||||||
CompilerUtils(m_context).computeHashStatic();
|
CompilerUtils(m_context).computeHashStatic();
|
||||||
// stack: len data_pos (len is in slots for byte array and in items for other arrays)
|
// stack: len data_pos
|
||||||
m_context << eth::Instruction::SWAP1 << eth::Instruction::DUP2 << eth::Instruction::ADD
|
m_context << eth::Instruction::SWAP1 << eth::Instruction::DUP2 << eth::Instruction::ADD
|
||||||
<< eth::Instruction::SWAP1;
|
<< eth::Instruction::SWAP1;
|
||||||
// stack: data_pos_end data_pos
|
// stack: data_pos_end data_pos
|
||||||
@ -207,6 +230,7 @@ void ArrayUtils::clearDynamicArray(ArrayType const& _type) const
|
|||||||
clearStorageLoop(*_type.getBaseType());
|
clearStorageLoop(*_type.getBaseType());
|
||||||
// cleanup
|
// cleanup
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP;
|
||||||
|
solAssert(m_context.getStackHeight() == stackHeightStart - 1, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ArrayUtils::resizeDynamicArray(const ArrayType& _type) const
|
void ArrayUtils::resizeDynamicArray(const ArrayType& _type) const
|
||||||
@ -214,6 +238,7 @@ void ArrayUtils::resizeDynamicArray(const ArrayType& _type) const
|
|||||||
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
solAssert(_type.getLocation() == ArrayType::Location::Storage, "");
|
||||||
solAssert(_type.isDynamicallySized(), "");
|
solAssert(_type.isDynamicallySized(), "");
|
||||||
|
|
||||||
|
unsigned stackHeightStart = m_context.getStackHeight();
|
||||||
eth::AssemblyItem resizeEnd = m_context.newTag();
|
eth::AssemblyItem resizeEnd = m_context.newTag();
|
||||||
|
|
||||||
// stack: ref new_length
|
// stack: ref new_length
|
||||||
@ -249,10 +274,12 @@ void ArrayUtils::resizeDynamicArray(const ArrayType& _type) const
|
|||||||
m_context << resizeEnd;
|
m_context << resizeEnd;
|
||||||
// cleanup
|
// cleanup
|
||||||
m_context << eth::Instruction::POP << eth::Instruction::POP << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
|
solAssert(m_context.getStackHeight() == stackHeightStart - 2, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ArrayUtils::clearStorageLoop(Type const& _type) const
|
void ArrayUtils::clearStorageLoop(Type const& _type) const
|
||||||
{
|
{
|
||||||
|
unsigned stackHeightStart = m_context.getStackHeight();
|
||||||
if (_type.getCategory() == Type::Category::Mapping)
|
if (_type.getCategory() == Type::Category::Mapping)
|
||||||
{
|
{
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP;
|
||||||
@ -267,13 +294,16 @@ void ArrayUtils::clearStorageLoop(Type const& _type) const
|
|||||||
eth::AssemblyItem zeroLoopEnd = m_context.newTag();
|
eth::AssemblyItem zeroLoopEnd = m_context.newTag();
|
||||||
m_context.appendConditionalJumpTo(zeroLoopEnd);
|
m_context.appendConditionalJumpTo(zeroLoopEnd);
|
||||||
// delete
|
// delete
|
||||||
|
m_context << u256(0); //@todo
|
||||||
StorageItem(m_context, _type).setToZero(SourceLocation(), false);
|
StorageItem(m_context, _type).setToZero(SourceLocation(), false);
|
||||||
|
m_context << eth::Instruction::POP;
|
||||||
// increment
|
// increment
|
||||||
m_context << u256(1) << eth::Instruction::ADD;
|
m_context << u256(1) << eth::Instruction::ADD;
|
||||||
m_context.appendJumpTo(loopStart);
|
m_context.appendJumpTo(loopStart);
|
||||||
// cleanup
|
// cleanup
|
||||||
m_context << zeroLoopEnd;
|
m_context << zeroLoopEnd;
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP;
|
||||||
|
solAssert(m_context.getStackHeight() == stackHeightStart - 1, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ArrayUtils::convertLengthToSize(ArrayType const& _arrayType, bool _pad) const
|
void ArrayUtils::convertLengthToSize(ArrayType const& _arrayType, bool _pad) const
|
||||||
|
12
ArrayUtils.h
12
ArrayUtils.h
@ -41,19 +41,19 @@ public:
|
|||||||
|
|
||||||
/// Copies an array to an array in storage. The arrays can be of different types only if
|
/// Copies an array to an array in storage. The arrays can be of different types only if
|
||||||
/// their storage representation is the same.
|
/// their storage representation is the same.
|
||||||
/// Stack pre: [source_reference] target_reference
|
/// Stack pre: source_reference [source_byte_offset/source_length] target_reference target_byte_offset
|
||||||
/// Stack post: target_reference
|
/// Stack post: target_reference target_byte_offset
|
||||||
void copyArrayToStorage(ArrayType const& _targetType, ArrayType const& _sourceType) const;
|
void copyArrayToStorage(ArrayType const& _targetType, ArrayType const& _sourceType) const;
|
||||||
/// Clears the given dynamic or static array.
|
/// Clears the given dynamic or static array.
|
||||||
/// Stack pre: reference
|
/// Stack pre: storage_ref storage_byte_offset
|
||||||
/// Stack post:
|
/// Stack post:
|
||||||
void clearArray(ArrayType const& _type) const;
|
void clearArray(ArrayType const& _type) const;
|
||||||
/// Clears the length and data elements of the array referenced on the stack.
|
/// Clears the length and data elements of the array referenced on the stack.
|
||||||
/// Stack pre: reference
|
/// Stack pre: reference (excludes byte offset)
|
||||||
/// Stack post:
|
/// Stack post:
|
||||||
void clearDynamicArray(ArrayType const& _type) const;
|
void clearDynamicArray(ArrayType const& _type) const;
|
||||||
/// Changes the size of a dynamic array and clears the tail if it is shortened.
|
/// Changes the size of a dynamic array and clears the tail if it is shortened.
|
||||||
/// Stack pre: reference new_length
|
/// Stack pre: reference (excludes byte offset) new_length
|
||||||
/// Stack post:
|
/// Stack post:
|
||||||
void resizeDynamicArray(ArrayType const& _type) const;
|
void resizeDynamicArray(ArrayType const& _type) const;
|
||||||
/// Appends a loop that clears a sequence of storage slots of the given type (excluding end).
|
/// Appends a loop that clears a sequence of storage slots of the given type (excluding end).
|
||||||
@ -67,7 +67,7 @@ public:
|
|||||||
void convertLengthToSize(ArrayType const& _arrayType, bool _pad = false) const;
|
void convertLengthToSize(ArrayType const& _arrayType, bool _pad = false) const;
|
||||||
/// Retrieves the length (number of elements) of the array ref on the stack. This also
|
/// Retrieves the length (number of elements) of the array ref on the stack. This also
|
||||||
/// works for statically-sized arrays.
|
/// works for statically-sized arrays.
|
||||||
/// Stack pre: reference
|
/// Stack pre: reference (excludes byte offset for dynamic storage arrays)
|
||||||
/// Stack post: reference length
|
/// Stack post: reference length
|
||||||
void retrieveLength(ArrayType const& _arrayType) const;
|
void retrieveLength(ArrayType const& _arrayType) const;
|
||||||
|
|
||||||
|
@ -93,6 +93,7 @@ void CompilerUtils::storeInMemoryDynamic(Type const& _type, bool _padToWordBound
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(type.getLocation() == ArrayType::Location::Storage, "Memory arrays not yet implemented.");
|
solAssert(type.getLocation() == ArrayType::Location::Storage, "Memory arrays not yet implemented.");
|
||||||
|
m_context << eth::Instruction::POP; //@todo
|
||||||
m_context << eth::Instruction::DUP1 << eth::Instruction::SLOAD;
|
m_context << eth::Instruction::DUP1 << eth::Instruction::SLOAD;
|
||||||
// stack here: memory_offset storage_offset length_bytes
|
// stack here: memory_offset storage_offset length_bytes
|
||||||
// jump to end if length is zero
|
// jump to end if length is zero
|
||||||
|
@ -81,6 +81,7 @@ void ExpressionCompiler::appendStateVariableAccessor(VariableDeclaration const&
|
|||||||
returnType = dynamic_cast<MappingType const&>(*returnType).getValueType();
|
returnType = dynamic_cast<MappingType const&>(*returnType).getValueType();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
m_context << u256(0); // @todo
|
||||||
unsigned retSizeOnStack = 0;
|
unsigned retSizeOnStack = 0;
|
||||||
solAssert(accessorType.getReturnParameterTypes().size() >= 1, "");
|
solAssert(accessorType.getReturnParameterTypes().size() >= 1, "");
|
||||||
if (StructType const* structType = dynamic_cast<StructType const*>(returnType.get()))
|
if (StructType const* structType = dynamic_cast<StructType const*>(returnType.get()))
|
||||||
@ -90,15 +91,18 @@ void ExpressionCompiler::appendStateVariableAccessor(VariableDeclaration const&
|
|||||||
// struct
|
// struct
|
||||||
for (size_t i = 0; i < names.size(); ++i)
|
for (size_t i = 0; i < names.size(); ++i)
|
||||||
{
|
{
|
||||||
m_context << eth::Instruction::DUP1
|
if (types[i]->getCategory() == Type::Category::Mapping)
|
||||||
<< structType->getStorageOffsetOfMember(names[i])
|
continue;
|
||||||
<< eth::Instruction::ADD;
|
m_context
|
||||||
|
<< eth::Instruction::DUP2 << structType->getStorageOffsetOfMember(names[i])
|
||||||
|
<< eth::Instruction::ADD;
|
||||||
|
m_context << u256(0); //@todo
|
||||||
StorageItem(m_context, *types[i]).retrieveValue(SourceLocation(), true);
|
StorageItem(m_context, *types[i]).retrieveValue(SourceLocation(), true);
|
||||||
solAssert(types[i]->getSizeOnStack() == 1, "Returning struct elements with stack size != 1 not yet implemented.");
|
solAssert(types[i]->getSizeOnStack() == 1, "Returning struct elements with stack size != 1 not yet implemented.");
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context << eth::Instruction::SWAP2 << eth::Instruction::SWAP1;
|
||||||
retSizeOnStack += types[i]->getSizeOnStack();
|
retSizeOnStack += types[i]->getSizeOnStack();
|
||||||
}
|
}
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -280,23 +284,24 @@ bool ExpressionCompiler::visit(UnaryOperation const& _unaryOperation)
|
|||||||
case Token::Dec: // -- (pre- or postfix)
|
case Token::Dec: // -- (pre- or postfix)
|
||||||
solAssert(!!m_currentLValue, "LValue not retrieved.");
|
solAssert(!!m_currentLValue, "LValue not retrieved.");
|
||||||
m_currentLValue->retrieveValue(_unaryOperation.getLocation());
|
m_currentLValue->retrieveValue(_unaryOperation.getLocation());
|
||||||
solAssert(m_currentLValue->sizeOnStack() <= 1, "Not implemented.");
|
|
||||||
if (!_unaryOperation.isPrefixOperation())
|
if (!_unaryOperation.isPrefixOperation())
|
||||||
{
|
{
|
||||||
if (m_currentLValue->sizeOnStack() == 1)
|
// store value for later
|
||||||
m_context << eth::Instruction::SWAP1 << eth::Instruction::DUP2;
|
solAssert(_unaryOperation.getType()->getSizeOnStack() == 1, "Stack size != 1 not implemented.");
|
||||||
else
|
m_context << eth::Instruction::DUP1;
|
||||||
m_context << eth::Instruction::DUP1;
|
if (m_currentLValue->sizeOnStack() > 0)
|
||||||
|
for (unsigned i = 1 + m_currentLValue->sizeOnStack(); i > 0; --i)
|
||||||
|
m_context << eth::swapInstruction(i);
|
||||||
}
|
}
|
||||||
m_context << u256(1);
|
m_context << u256(1);
|
||||||
if (_unaryOperation.getOperator() == Token::Inc)
|
if (_unaryOperation.getOperator() == Token::Inc)
|
||||||
m_context << eth::Instruction::ADD;
|
m_context << eth::Instruction::ADD;
|
||||||
else
|
else
|
||||||
m_context << eth::Instruction::SWAP1 << eth::Instruction::SUB; // @todo avoid the swap
|
m_context << eth::Instruction::SWAP1 << eth::Instruction::SUB;
|
||||||
// Stack for prefix: [ref] (*ref)+-1
|
// Stack for prefix: [ref...] (*ref)+-1
|
||||||
// Stack for postfix: *ref [ref] (*ref)+-1
|
// Stack for postfix: *ref [ref...] (*ref)+-1
|
||||||
if (m_currentLValue->sizeOnStack() == 1)
|
for (unsigned i = m_currentLValue->sizeOnStack(); i > 0; --i)
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context << eth::swapInstruction(i);
|
||||||
m_currentLValue->storeValue(
|
m_currentLValue->storeValue(
|
||||||
*_unaryOperation.getType(), _unaryOperation.getLocation(),
|
*_unaryOperation.getType(), _unaryOperation.getLocation(),
|
||||||
!_unaryOperation.isPrefixOperation());
|
!_unaryOperation.isPrefixOperation());
|
||||||
@ -661,7 +666,10 @@ void ExpressionCompiler::endVisit(MemberAccess const& _memberAccess)
|
|||||||
case Type::Category::Struct:
|
case Type::Category::Struct:
|
||||||
{
|
{
|
||||||
StructType const& type = dynamic_cast<StructType const&>(*_memberAccess.getExpression().getType());
|
StructType const& type = dynamic_cast<StructType const&>(*_memberAccess.getExpression().getType());
|
||||||
|
m_context << eth::Instruction::POP; //@todo
|
||||||
m_context << type.getStorageOffsetOfMember(member) << eth::Instruction::ADD;
|
m_context << type.getStorageOffsetOfMember(member) << eth::Instruction::ADD;
|
||||||
|
//@todo
|
||||||
|
m_context << u256(0);
|
||||||
setLValueToStorageItem(_memberAccess);
|
setLValueToStorageItem(_memberAccess);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -729,20 +737,22 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
Type const& baseType = *_indexAccess.getBaseExpression().getType();
|
Type const& baseType = *_indexAccess.getBaseExpression().getType();
|
||||||
if (baseType.getCategory() == Type::Category::Mapping)
|
if (baseType.getCategory() == Type::Category::Mapping)
|
||||||
{
|
{
|
||||||
|
// storage byte offset is ignored for mappings, it should be zero.
|
||||||
|
m_context << eth::Instruction::POP;
|
||||||
|
// stack: storage_base_ref
|
||||||
Type const& keyType = *dynamic_cast<MappingType const&>(baseType).getKeyType();
|
Type const& keyType = *dynamic_cast<MappingType const&>(baseType).getKeyType();
|
||||||
m_context << u256(0);
|
m_context << u256(0); // memory position
|
||||||
solAssert(_indexAccess.getIndexExpression(), "Index expression expected.");
|
solAssert(_indexAccess.getIndexExpression(), "Index expression expected.");
|
||||||
appendExpressionCopyToMemory(keyType, *_indexAccess.getIndexExpression());
|
appendExpressionCopyToMemory(keyType, *_indexAccess.getIndexExpression());
|
||||||
solAssert(baseType.getSizeOnStack() == 1,
|
|
||||||
"Unexpected: Not exactly one stack slot taken by subscriptable expression.");
|
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context << eth::Instruction::SWAP1;
|
||||||
appendTypeMoveToMemory(IntegerType(256));
|
appendTypeMoveToMemory(IntegerType(256));
|
||||||
m_context << u256(0) << eth::Instruction::SHA3;
|
m_context << u256(0) << eth::Instruction::SHA3;
|
||||||
|
m_context << u256(0);
|
||||||
setLValueToStorageItem( _indexAccess);
|
setLValueToStorageItem( _indexAccess);
|
||||||
}
|
}
|
||||||
else if (baseType.getCategory() == Type::Category::Array)
|
else if (baseType.getCategory() == Type::Category::Array)
|
||||||
{
|
{
|
||||||
// stack layout: <base_ref> [<length>] <index>
|
// stack layout: <base_ref> [storage_byte_offset] [<length>] <index>
|
||||||
ArrayType const& arrayType = dynamic_cast<ArrayType const&>(baseType);
|
ArrayType const& arrayType = dynamic_cast<ArrayType const&>(baseType);
|
||||||
solAssert(_indexAccess.getIndexExpression(), "Index expression expected.");
|
solAssert(_indexAccess.getIndexExpression(), "Index expression expected.");
|
||||||
ArrayType::Location location = arrayType.getLocation();
|
ArrayType::Location location = arrayType.getLocation();
|
||||||
@ -758,9 +768,11 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
else if (location == ArrayType::Location::CallData)
|
else if (location == ArrayType::Location::CallData)
|
||||||
// length is stored on the stack
|
// length is stored on the stack
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context << eth::Instruction::SWAP1;
|
||||||
|
else if (location == ArrayType::Location::Storage)
|
||||||
|
m_context << eth::Instruction::DUP3 << load;
|
||||||
else
|
else
|
||||||
m_context << eth::Instruction::DUP2 << load;
|
m_context << eth::Instruction::DUP2 << load;
|
||||||
// stack: <base_ref> <index> <length>
|
// stack: <base_ref> [storage_byte_offset] <index> <length>
|
||||||
// check out-of-bounds access
|
// check out-of-bounds access
|
||||||
m_context << eth::Instruction::DUP2 << eth::Instruction::LT;
|
m_context << eth::Instruction::DUP2 << eth::Instruction::LT;
|
||||||
eth::AssemblyItem legalAccess = m_context.appendConditionalJump();
|
eth::AssemblyItem legalAccess = m_context.appendConditionalJump();
|
||||||
@ -768,7 +780,7 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
m_context << eth::Instruction::STOP;
|
m_context << eth::Instruction::STOP;
|
||||||
|
|
||||||
m_context << legalAccess;
|
m_context << legalAccess;
|
||||||
// stack: <base_ref> <index>
|
// stack: <base_ref> [storage_byte_offset] <index>
|
||||||
if (arrayType.isByteArray())
|
if (arrayType.isByteArray())
|
||||||
// byte array is packed differently, especially in storage
|
// byte array is packed differently, especially in storage
|
||||||
switch (location)
|
switch (location)
|
||||||
@ -776,14 +788,15 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
case ArrayType::Location::Storage:
|
case ArrayType::Location::Storage:
|
||||||
// byte array index storage lvalue on stack (goal):
|
// byte array index storage lvalue on stack (goal):
|
||||||
// <ref> <byte_number> = <base_ref + index / 32> <index % 32>
|
// <ref> <byte_number> = <base_ref + index / 32> <index % 32>
|
||||||
m_context << u256(32) << eth::Instruction::SWAP2;
|
m_context << u256(32) << eth::Instruction::SWAP3;
|
||||||
CompilerUtils(m_context).computeHashStatic();
|
CompilerUtils(m_context).computeHashStatic();
|
||||||
// stack: 32 index data_ref
|
// stack: 32 storage_byte_offset index data_ref
|
||||||
m_context
|
m_context
|
||||||
<< eth::Instruction::DUP3 << eth::Instruction::DUP3
|
<< eth::Instruction::DUP4 << eth::Instruction::DUP3
|
||||||
<< eth::Instruction::DIV << eth::Instruction::ADD
|
<< eth::Instruction::DIV << eth::Instruction::ADD
|
||||||
// stack: 32 index (data_ref + index / 32)
|
// stack: 32 storage_byte_offset index (data_ref + index / 32)
|
||||||
<< eth::Instruction::SWAP2 << eth::Instruction::SWAP1 << eth::Instruction::MOD;
|
<< eth::Instruction::SWAP3 << eth::Instruction::SWAP2
|
||||||
|
<< eth::Instruction::POP << eth::Instruction::MOD;
|
||||||
setLValue<StorageByteArrayElement>(_indexAccess);
|
setLValue<StorageByteArrayElement>(_indexAccess);
|
||||||
break;
|
break;
|
||||||
case ArrayType::Location::CallData:
|
case ArrayType::Location::CallData:
|
||||||
@ -797,6 +810,10 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
// stack: <base_ref> [storage_byte_offset] <index>
|
||||||
|
if (location == ArrayType::Location::Storage)
|
||||||
|
//@todo use byte offset, remove it for now
|
||||||
|
m_context << eth::Instruction::SWAP1 << eth::Instruction::POP;
|
||||||
u256 elementSize =
|
u256 elementSize =
|
||||||
location == ArrayType::Location::Storage ?
|
location == ArrayType::Location::Storage ?
|
||||||
arrayType.getBaseType()->getStorageSize() :
|
arrayType.getBaseType()->getStorageSize() :
|
||||||
@ -822,6 +839,7 @@ bool ExpressionCompiler::visit(IndexAccess const& _indexAccess)
|
|||||||
CompilerUtils(m_context).loadFromMemoryDynamic(*arrayType.getBaseType(), true, true, false);
|
CompilerUtils(m_context).loadFromMemoryDynamic(*arrayType.getBaseType(), true, true, false);
|
||||||
break;
|
break;
|
||||||
case ArrayType::Location::Storage:
|
case ArrayType::Location::Storage:
|
||||||
|
m_context << u256(0); // @todo
|
||||||
setLValueToStorageItem(_indexAccess);
|
setLValueToStorageItem(_indexAccess);
|
||||||
break;
|
break;
|
||||||
case ArrayType::Location::Memory:
|
case ArrayType::Location::Memory:
|
||||||
@ -1141,7 +1159,7 @@ void ExpressionCompiler::setLValueFromDeclaration(Declaration const& _declaratio
|
|||||||
if (m_context.isLocalVariable(&_declaration))
|
if (m_context.isLocalVariable(&_declaration))
|
||||||
setLValue<StackVariable>(_expression, _declaration);
|
setLValue<StackVariable>(_expression, _declaration);
|
||||||
else if (m_context.isStateVariable(&_declaration))
|
else if (m_context.isStateVariable(&_declaration))
|
||||||
setLValue<StorageItem>(_expression, _declaration);
|
setLValue<StorageItem>(_expression, _declaration);
|
||||||
else
|
else
|
||||||
BOOST_THROW_EXCEPTION(InternalCompilerError()
|
BOOST_THROW_EXCEPTION(InternalCompilerError()
|
||||||
<< errinfo_sourceLocation(_expression.getLocation())
|
<< errinfo_sourceLocation(_expression.getLocation())
|
||||||
|
@ -140,7 +140,6 @@ void ExpressionCompiler::setLValue(Expression const& _expression, _Arguments con
|
|||||||
m_currentLValue = move(lvalue);
|
m_currentLValue = move(lvalue);
|
||||||
else
|
else
|
||||||
lvalue->retrieveValue(_expression.getLocation(), true);
|
lvalue->retrieveValue(_expression.getLocation(), true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
128
LValue.cpp
128
LValue.cpp
@ -77,7 +77,7 @@ void StackVariable::setToZero(SourceLocation const& _location, bool) const
|
|||||||
StorageItem::StorageItem(CompilerContext& _compilerContext, Declaration const& _declaration):
|
StorageItem::StorageItem(CompilerContext& _compilerContext, Declaration const& _declaration):
|
||||||
StorageItem(_compilerContext, *_declaration.getType())
|
StorageItem(_compilerContext, *_declaration.getType())
|
||||||
{
|
{
|
||||||
m_context << m_context.getStorageLocationOfVariable(_declaration);
|
m_context << m_context.getStorageLocationOfVariable(_declaration) << u256(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
StorageItem::StorageItem(CompilerContext& _compilerContext, Type const& _type):
|
StorageItem::StorageItem(CompilerContext& _compilerContext, Type const& _type):
|
||||||
@ -86,62 +86,42 @@ StorageItem::StorageItem(CompilerContext& _compilerContext, Type const& _type):
|
|||||||
if (m_dataType.isValueType())
|
if (m_dataType.isValueType())
|
||||||
{
|
{
|
||||||
solAssert(m_dataType.getStorageSize() == m_dataType.getSizeOnStack(), "");
|
solAssert(m_dataType.getStorageSize() == m_dataType.getSizeOnStack(), "");
|
||||||
solAssert(m_dataType.getStorageSize() <= numeric_limits<unsigned>::max(),
|
//@todo the meaning of getStorageSize() probably changes
|
||||||
"The storage size of " + m_dataType.toString() + " should fit in an unsigned");
|
solAssert(m_dataType.getStorageSize() == 1, "Invalid storage size.");
|
||||||
m_size = unsigned(m_dataType.getStorageSize());
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
m_size = 0; // unused
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void StorageItem::retrieveValue(SourceLocation const&, bool _remove) const
|
void StorageItem::retrieveValue(SourceLocation const&, bool _remove) const
|
||||||
{
|
{
|
||||||
|
// stack: storage_key storage_offset
|
||||||
if (!m_dataType.isValueType())
|
if (!m_dataType.isValueType())
|
||||||
return; // no distinction between value and reference for non-value types
|
return; // no distinction between value and reference for non-value types
|
||||||
if (!_remove)
|
if (!_remove)
|
||||||
m_context << eth::Instruction::DUP1;
|
CompilerUtils(m_context).copyToStackTop(sizeOnStack(), sizeOnStack());
|
||||||
if (m_size == 1)
|
m_context
|
||||||
m_context << eth::Instruction::SLOAD;
|
<< eth::Instruction::SWAP1 << eth::Instruction::SLOAD << eth::Instruction::SWAP1
|
||||||
else
|
<< u256(2) << eth::Instruction::EXP << eth::Instruction::SWAP1 << eth::Instruction::DIV;
|
||||||
for (unsigned i = 0; i < m_size; ++i)
|
//@todo higher order bits might be dirty. Is this bad?
|
||||||
{
|
//@todo this does not work for types that are left-aligned on the stack.
|
||||||
m_context << eth::Instruction::DUP1 << eth::Instruction::SLOAD << eth::Instruction::SWAP1;
|
// make those types right-aligned?
|
||||||
if (i + 1 < m_size)
|
|
||||||
m_context << u256(1) << eth::Instruction::ADD;
|
|
||||||
else
|
|
||||||
m_context << eth::Instruction::POP;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void StorageItem::storeValue(Type const& _sourceType, SourceLocation const& _location, bool _move) const
|
void StorageItem::storeValue(Type const& _sourceType, SourceLocation const& _location, bool _move) const
|
||||||
{
|
{
|
||||||
// stack layout: value value ... value target_ref
|
// stack: value storage_key storage_offset
|
||||||
if (m_dataType.isValueType())
|
if (m_dataType.isValueType())
|
||||||
{
|
{
|
||||||
if (!_move) // copy values
|
//@todo OR the value into the storage like it is done for ByteArrayElement
|
||||||
{
|
m_context
|
||||||
if (m_size + 1 > 16)
|
<< u256(2) << eth::Instruction::EXP << eth::Instruction::DUP3 << eth::Instruction::MUL
|
||||||
BOOST_THROW_EXCEPTION(CompilerError()
|
<< eth::Instruction::SWAP1 << eth::Instruction::SSTORE;
|
||||||
<< errinfo_sourceLocation(_location) << errinfo_comment("Stack too deep."));
|
if (_move)
|
||||||
for (unsigned i = 0; i < m_size; ++i)
|
m_context << eth::Instruction::POP;
|
||||||
m_context << eth::dupInstruction(m_size + 1) << eth::Instruction::SWAP1;
|
|
||||||
}
|
|
||||||
if (m_size > 1) // store high index value first
|
|
||||||
m_context << u256(m_size - 1) << eth::Instruction::ADD;
|
|
||||||
for (unsigned i = 0; i < m_size; ++i)
|
|
||||||
{
|
|
||||||
if (i + 1 >= m_size)
|
|
||||||
m_context << eth::Instruction::SSTORE;
|
|
||||||
else
|
|
||||||
// stack here: value value ... value value (target_ref+offset)
|
|
||||||
m_context << eth::Instruction::SWAP1 << eth::Instruction::DUP2
|
|
||||||
<< eth::Instruction::SSTORE
|
|
||||||
<< u256(1) << eth::Instruction::SWAP1 << eth::Instruction::SUB;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(_sourceType.getCategory() == m_dataType.getCategory(),
|
solAssert(
|
||||||
|
_sourceType.getCategory() == m_dataType.getCategory(),
|
||||||
"Wrong type conversation for assignment.");
|
"Wrong type conversation for assignment.");
|
||||||
if (m_dataType.getCategory() == Type::Category::Array)
|
if (m_dataType.getCategory() == Type::Category::Array)
|
||||||
{
|
{
|
||||||
@ -149,40 +129,48 @@ void StorageItem::storeValue(Type const& _sourceType, SourceLocation const& _loc
|
|||||||
dynamic_cast<ArrayType const&>(m_dataType),
|
dynamic_cast<ArrayType const&>(m_dataType),
|
||||||
dynamic_cast<ArrayType const&>(_sourceType));
|
dynamic_cast<ArrayType const&>(_sourceType));
|
||||||
if (_move)
|
if (_move)
|
||||||
m_context << eth::Instruction::POP;
|
CompilerUtils(m_context).popStackElement(_sourceType);
|
||||||
}
|
}
|
||||||
else if (m_dataType.getCategory() == Type::Category::Struct)
|
else if (m_dataType.getCategory() == Type::Category::Struct)
|
||||||
{
|
{
|
||||||
// stack layout: source_ref target_ref
|
// stack layout: source_ref source_offset target_ref target_offset
|
||||||
auto const& structType = dynamic_cast<StructType const&>(m_dataType);
|
auto const& structType = dynamic_cast<StructType const&>(m_dataType);
|
||||||
solAssert(structType == _sourceType, "Struct assignment with conversion.");
|
solAssert(structType == _sourceType, "Struct assignment with conversion.");
|
||||||
for (auto const& member: structType.getMembers())
|
for (auto const& member: structType.getMembers())
|
||||||
{
|
{
|
||||||
|
//@todo actually use offsets
|
||||||
// assign each member that is not a mapping
|
// assign each member that is not a mapping
|
||||||
TypePointer const& memberType = member.second;
|
TypePointer const& memberType = member.second;
|
||||||
if (memberType->getCategory() == Type::Category::Mapping)
|
if (memberType->getCategory() == Type::Category::Mapping)
|
||||||
continue;
|
continue;
|
||||||
m_context << structType.getStorageOffsetOfMember(member.first)
|
m_context << structType.getStorageOffsetOfMember(member.first)
|
||||||
<< eth::Instruction::DUP3 << eth::Instruction::DUP2 << eth::Instruction::ADD;
|
<< eth::Instruction::DUP5 << eth::Instruction::DUP2 << eth::Instruction::ADD;
|
||||||
// stack: source_ref target_ref member_offset source_member_ref
|
m_context << u256(0); // zero offset
|
||||||
|
// stack: source_ref source_off target_ref target_off member_offset source_member_ref source_member_off
|
||||||
StorageItem(m_context, *memberType).retrieveValue(_location, true);
|
StorageItem(m_context, *memberType).retrieveValue(_location, true);
|
||||||
// stack: source_ref target_ref member_offset source_value...
|
// stack: source_ref source_off target_ref target_off member_offset source_value...
|
||||||
solAssert(2 + memberType->getSizeOnStack() <= 16, "Stack too deep.");
|
solAssert(2 + memberType->getSizeOnStack() <= 16, "Stack too deep.");
|
||||||
m_context << eth::dupInstruction(2 + memberType->getSizeOnStack())
|
m_context << eth::dupInstruction(3 + memberType->getSizeOnStack())
|
||||||
<< eth::dupInstruction(2 + memberType->getSizeOnStack()) << eth::Instruction::ADD;
|
<< eth::dupInstruction(2 + memberType->getSizeOnStack()) << eth::Instruction::ADD;
|
||||||
// stack: source_ref target_ref member_offset source_value... target_member_ref
|
// stack: source_ref source_off target_ref target_off member_offset source_value... target_member_ref
|
||||||
|
m_context << u256(0); // zero offset
|
||||||
StorageItem(m_context, *memberType).storeValue(*memberType, _location, true);
|
StorageItem(m_context, *memberType).storeValue(*memberType, _location, true);
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
if (_move)
|
if (_move)
|
||||||
m_context << eth::Instruction::POP;
|
m_context
|
||||||
|
<< eth::Instruction::POP << eth::Instruction::POP
|
||||||
|
<< eth::Instruction::POP << eth::Instruction::POP;
|
||||||
else
|
else
|
||||||
m_context << eth::Instruction::SWAP1;
|
m_context
|
||||||
m_context << eth::Instruction::POP;
|
<< eth::Instruction::SWAP2 << eth::Instruction::POP
|
||||||
|
<< eth::Instruction::SWAP2 << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
BOOST_THROW_EXCEPTION(InternalCompilerError()
|
BOOST_THROW_EXCEPTION(
|
||||||
<< errinfo_sourceLocation(_location) << errinfo_comment("Invalid non-value type for assignment."));
|
InternalCompilerError()
|
||||||
|
<< errinfo_sourceLocation(_location)
|
||||||
|
<< errinfo_comment("Invalid non-value type for assignment."));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,12 +179,12 @@ void StorageItem::setToZero(SourceLocation const&, bool _removeReference) const
|
|||||||
if (m_dataType.getCategory() == Type::Category::Array)
|
if (m_dataType.getCategory() == Type::Category::Array)
|
||||||
{
|
{
|
||||||
if (!_removeReference)
|
if (!_removeReference)
|
||||||
m_context << eth::Instruction::DUP1;
|
CompilerUtils(m_context).copyToStackTop(sizeOnStack(), sizeOnStack());
|
||||||
ArrayUtils(m_context).clearArray(dynamic_cast<ArrayType const&>(m_dataType));
|
ArrayUtils(m_context).clearArray(dynamic_cast<ArrayType const&>(m_dataType));
|
||||||
}
|
}
|
||||||
else if (m_dataType.getCategory() == Type::Category::Struct)
|
else if (m_dataType.getCategory() == Type::Category::Struct)
|
||||||
{
|
{
|
||||||
// stack layout: ref
|
// stack layout: storage_key storage_offset
|
||||||
auto const& structType = dynamic_cast<StructType const&>(m_dataType);
|
auto const& structType = dynamic_cast<StructType const&>(m_dataType);
|
||||||
for (auto const& member: structType.getMembers())
|
for (auto const& member: structType.getMembers())
|
||||||
{
|
{
|
||||||
@ -204,33 +192,23 @@ void StorageItem::setToZero(SourceLocation const&, bool _removeReference) const
|
|||||||
TypePointer const& memberType = member.second;
|
TypePointer const& memberType = member.second;
|
||||||
if (memberType->getCategory() == Type::Category::Mapping)
|
if (memberType->getCategory() == Type::Category::Mapping)
|
||||||
continue;
|
continue;
|
||||||
m_context << structType.getStorageOffsetOfMember(member.first)
|
// @todo actually use offset
|
||||||
<< eth::Instruction::DUP2 << eth::Instruction::ADD;
|
m_context
|
||||||
|
<< structType.getStorageOffsetOfMember(member.first)
|
||||||
|
<< eth::Instruction::DUP3 << eth::Instruction::ADD;
|
||||||
|
m_context << u256(0);
|
||||||
StorageItem(m_context, *memberType).setToZero();
|
StorageItem(m_context, *memberType).setToZero();
|
||||||
}
|
}
|
||||||
if (_removeReference)
|
if (_removeReference)
|
||||||
m_context << eth::Instruction::POP;
|
m_context << eth::Instruction::POP << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(m_dataType.isValueType(), "Clearing of unsupported type requested: " + m_dataType.toString());
|
solAssert(m_dataType.isValueType(), "Clearing of unsupported type requested: " + m_dataType.toString());
|
||||||
if (m_size == 0 && _removeReference)
|
// @todo actually use offset
|
||||||
m_context << eth::Instruction::POP;
|
if (!_removeReference)
|
||||||
else if (m_size == 1)
|
CompilerUtils(m_context).copyToStackTop(sizeOnStack(), sizeOnStack());
|
||||||
m_context
|
m_context << eth::Instruction::POP << u256(0) << eth::Instruction::SWAP1 << eth::Instruction::SSTORE;
|
||||||
<< u256(0) << (_removeReference ? eth::Instruction::SWAP1 : eth::Instruction::DUP2)
|
|
||||||
<< eth::Instruction::SSTORE;
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!_removeReference)
|
|
||||||
m_context << eth::Instruction::DUP1;
|
|
||||||
for (unsigned i = 0; i < m_size; ++i)
|
|
||||||
if (i + 1 >= m_size)
|
|
||||||
m_context << u256(0) << eth::Instruction::SWAP1 << eth::Instruction::SSTORE;
|
|
||||||
else
|
|
||||||
m_context << u256(0) << eth::Instruction::DUP2 << eth::Instruction::SSTORE
|
|
||||||
<< u256(1) << eth::Instruction::ADD;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -300,6 +278,8 @@ StorageArrayLength::StorageArrayLength(CompilerContext& _compilerContext, const
|
|||||||
m_arrayType(_arrayType)
|
m_arrayType(_arrayType)
|
||||||
{
|
{
|
||||||
solAssert(m_arrayType.isDynamicallySized(), "");
|
solAssert(m_arrayType.isDynamicallySized(), "");
|
||||||
|
// storage byte offset must be zero
|
||||||
|
m_context << eth::Instruction::POP;
|
||||||
}
|
}
|
||||||
|
|
||||||
void StorageArrayLength::retrieveValue(SourceLocation const&, bool _remove) const
|
void StorageArrayLength::retrieveValue(SourceLocation const&, bool _remove) const
|
||||||
|
10
LValue.h
10
LValue.h
@ -98,7 +98,9 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reference to some item in storage. The (starting) position of the item is stored on the stack.
|
* Reference to some item in storage. On the stack this is <storage key> <offset_inside_value>,
|
||||||
|
* where 0 <= offset_inside_value < 32 and an offset of i means that the value is multiplied
|
||||||
|
* by 2**i before storing it.
|
||||||
*/
|
*/
|
||||||
class StorageItem: public LValue
|
class StorageItem: public LValue
|
||||||
{
|
{
|
||||||
@ -107,6 +109,7 @@ public:
|
|||||||
StorageItem(CompilerContext& _compilerContext, Declaration const& _declaration);
|
StorageItem(CompilerContext& _compilerContext, Declaration const& _declaration);
|
||||||
/// Constructs the LValue and assumes that the storage reference is already on the stack.
|
/// Constructs the LValue and assumes that the storage reference is already on the stack.
|
||||||
StorageItem(CompilerContext& _compilerContext, Type const& _type);
|
StorageItem(CompilerContext& _compilerContext, Type const& _type);
|
||||||
|
virtual unsigned sizeOnStack() const { return 2; }
|
||||||
virtual void retrieveValue(SourceLocation const& _location, bool _remove = false) const override;
|
virtual void retrieveValue(SourceLocation const& _location, bool _remove = false) const override;
|
||||||
virtual void storeValue(
|
virtual void storeValue(
|
||||||
Type const& _sourceType,
|
Type const& _sourceType,
|
||||||
@ -117,11 +120,6 @@ public:
|
|||||||
SourceLocation const& _location = SourceLocation(),
|
SourceLocation const& _location = SourceLocation(),
|
||||||
bool _removeReference = true
|
bool _removeReference = true
|
||||||
) const override;
|
) const override;
|
||||||
|
|
||||||
private:
|
|
||||||
/// Number of stack elements occupied by the value (not the reference).
|
|
||||||
/// Only used for value types.
|
|
||||||
unsigned m_size;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -644,6 +644,9 @@ unsigned ArrayType::getSizeOnStack() const
|
|||||||
if (m_location == Location::CallData)
|
if (m_location == Location::CallData)
|
||||||
// offset [length] (stack top)
|
// offset [length] (stack top)
|
||||||
return 1 + (isDynamicallySized() ? 1 : 0);
|
return 1 + (isDynamicallySized() ? 1 : 0);
|
||||||
|
else if (m_location == Location::Storage)
|
||||||
|
// storage_key storage_offset
|
||||||
|
return 2;
|
||||||
else
|
else
|
||||||
// offset
|
// offset
|
||||||
return 1;
|
return 1;
|
||||||
|
7
Types.h
7
Types.h
@ -284,6 +284,9 @@ public:
|
|||||||
/**
|
/**
|
||||||
* The type of an array. The flavours are byte array (bytes), statically- (<type>[<length>])
|
* The type of an array. The flavours are byte array (bytes), statically- (<type>[<length>])
|
||||||
* and dynamically-sized array (<type>[]).
|
* and dynamically-sized array (<type>[]).
|
||||||
|
* In storage, all arrays are packed tightly (as long as more than one elementary type fits in
|
||||||
|
* one slot). Dynamically sized arrays (including byte arrays) start with their size as a uint and
|
||||||
|
* thus start on their own slot.
|
||||||
*/
|
*/
|
||||||
class ArrayType: public Type
|
class ArrayType: public Type
|
||||||
{
|
{
|
||||||
@ -384,7 +387,7 @@ public:
|
|||||||
virtual bool operator==(Type const& _other) const override;
|
virtual bool operator==(Type const& _other) const override;
|
||||||
virtual u256 getStorageSize() const override;
|
virtual u256 getStorageSize() const override;
|
||||||
virtual bool canLiveOutsideStorage() const override;
|
virtual bool canLiveOutsideStorage() const override;
|
||||||
virtual unsigned getSizeOnStack() const override { return 1; /*@todo*/ }
|
virtual unsigned getSizeOnStack() const override { return 2; }
|
||||||
virtual std::string toString() const override;
|
virtual std::string toString() const override;
|
||||||
|
|
||||||
virtual MemberList const& getMembers() const override;
|
virtual MemberList const& getMembers() const override;
|
||||||
@ -527,6 +530,7 @@ private:
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The type of a mapping, there is one distinct type per key/value type pair.
|
* The type of a mapping, there is one distinct type per key/value type pair.
|
||||||
|
* Mappings always occupy their own storage slot, but do not actually use it.
|
||||||
*/
|
*/
|
||||||
class MappingType: public Type
|
class MappingType: public Type
|
||||||
{
|
{
|
||||||
@ -537,6 +541,7 @@ public:
|
|||||||
|
|
||||||
virtual bool operator==(Type const& _other) const override;
|
virtual bool operator==(Type const& _other) const override;
|
||||||
virtual std::string toString() const override;
|
virtual std::string toString() const override;
|
||||||
|
virtual unsigned getSizeOnStack() const override { return 2; }
|
||||||
virtual bool canLiveOutsideStorage() const override { return false; }
|
virtual bool canLiveOutsideStorage() const override { return false; }
|
||||||
|
|
||||||
TypePointer const& getKeyType() const { return m_keyType; }
|
TypePointer const& getKeyType() const { return m_keyType; }
|
||||||
|
Loading…
Reference in New Issue
Block a user