diff options
author | Erik Kundt <bitshift@posteo.org> | 2018-03-17 00:06:38 +0800 |
---|---|---|
committer | Erik Kundt <bitshift@posteo.org> | 2018-05-29 18:53:02 +0800 |
commit | 7156a01acc822ab66c189435421564afc8b1c922 (patch) | |
tree | b838519c5c24c16dd3bcc1564ec91cfa4f85b1f9 /libsolidity/codegen | |
parent | e9dcfb0b624e5443942451fc865c154a2c5a73d7 (diff) | |
download | dexon-solidity-7156a01acc822ab66c189435421564afc8b1c922.tar.gz dexon-solidity-7156a01acc822ab66c189435421564afc8b1c922.tar.zst dexon-solidity-7156a01acc822ab66c189435421564afc8b1c922.zip |
Implements pop() for byte arrays.
Diffstat (limited to 'libsolidity/codegen')
-rw-r--r-- | libsolidity/codegen/ArrayUtils.cpp | 75 |
1 files changed, 65 insertions, 10 deletions
diff --git a/libsolidity/codegen/ArrayUtils.cpp b/libsolidity/codegen/ArrayUtils.cpp index b434fddd..096b2d4e 100644 --- a/libsolidity/codegen/ArrayUtils.cpp +++ b/libsolidity/codegen/ArrayUtils.cpp @@ -830,19 +830,74 @@ void ArrayUtils::popStorageArrayElement(ArrayType const& _type) const if (!_type.isByteArray() && _type.baseType()->storageBytes() < 32) solAssert(_type.baseType()->isValueType(), "Invalid storage size for non-value type."); - // stack: ArrayReference - retrieveLength(_type); - // stack: ArrayReference oldLength - m_context << Instruction::DUP1; - // stack: ArrayReference oldLength oldLength - m_context << Instruction::ISZERO; - m_context.appendConditionalInvalid(); - if (_type.isByteArray()) { + m_context.appendInlineAssembly(R"({ + let slot_value := sload(ref) + switch and(slot_value, 1) + case 0 { + // short byte array + let length := and(div(slot_value, 2), 0x3f) + if iszero(length) { invalid() } + + // Zero-out the suffix of the byte array by masking it. + // Do not zero-out the least significant byte, but mask the + // higher bits of the length. + // (((1<<(8 * (32 - length))) - 1) << 8) + 128 + let mask := add(mul(0x100, sub(exp(0x100, sub(32, length)), 1)), 0x80) + slot_value := and(not(mask), slot_value) + + // Reduce the length by 1 + slot_value := sub(slot_value, 2) + sstore(ref, slot_value) + } + case 1 { + // long byte array + let length := div(slot_value, 2) + mstore(0, ref) + + switch length + case 32 + { + let slot := keccak256(0, 0x20) + let data := sload(slot) + sstore(slot, 0) + data := and(data, not(0xff)) + sstore(ref, or(data, 62)) + } + default + { + let slot := div(sub(length, 1), 32) + let offset := and(sub(length, 1), 0x1f) + slot := add(keccak256(0, 0x20), slot) + let data := sload(slot) + + // Zero-out the suffix of the byte array by masking it. + // ((1<<(8 * (32 - offset))) - 1) + let mask := sub(exp(0x100, sub(32, offset)), 1) + data := and(not(mask), data) + sstore(slot, data) + + // Reduce the length by 1 + slot_value := sub(slot_value, 2) + sstore(ref, slot_value) + } + } + })", {"ref"}); + m_context << Instruction::POP; } else - { + { + + // stack: ArrayReference + retrieveLength(_type); + // stack: ArrayReference oldLength + m_context << Instruction::DUP1; + // stack: ArrayReference oldLength oldLength + m_context << Instruction::ISZERO; + m_context.appendConditionalInvalid(); + + // Stack: ArrayReference oldLength m_context << u256(1) << Instruction::SWAP1 << Instruction::SUB; // Stack ArrayReference newLength @@ -852,7 +907,7 @@ void ArrayUtils::popStorageArrayElement(ArrayType const& _type) const // Stack: ArrayReference newLength storage_slot byte_offset StorageItem(m_context, _type).setToZero(SourceLocation(), true); // Stack: ArrayReference newLength - m_context << Instruction::SSTORE; + m_context << Instruction::SWAP1 << Instruction::SSTORE; } } |