diff --git a/libsolidity/codegen/YulUtilFunctions.cpp b/libsolidity/codegen/YulUtilFunctions.cpp index 4ac32a1fa..f976f8066 100644 --- a/libsolidity/codegen/YulUtilFunctions.cpp +++ b/libsolidity/codegen/YulUtilFunctions.cpp @@ -873,6 +873,12 @@ std::string YulUtilFunctions::resizeDynamicArrayFunction(ArrayType const& _type) let arrayDataStart := (array) let deleteStart := add(arrayDataStart, newSlotCount) let deleteEnd := add(arrayDataStart, oldSlotCount) + + // if we are dealing with packed array and offset is greater than zero + // we have to partially clear last slot that is still used, so decreasing start by one + let offset := mul(mod(newLen, ), ) + if gt(offset, 0) { (sub(deleteStart, 1), offset) } + (deleteStart, deleteEnd) } })") @@ -883,6 +889,10 @@ std::string YulUtilFunctions::resizeDynamicArrayFunction(ArrayType const& _type) ("dataPosition", arrayDataAreaFunction(_type)) ("clearStorageRange", clearStorageRangeFunction(*_type.baseType())) ("maxArrayLength", (u256(1) << 64).str()) + ("packed", _type.baseType()->storageBytes() <= 16) + ("itemsPerSlot", to_string(32 / _type.baseType()->storageBytes())) + ("storageBytes", to_string(_type.baseType()->storageBytes())) + ("partialClearStorageSlot", partialClearStorageSlotFunction()) .render(); }); } @@ -1060,11 +1070,29 @@ string YulUtilFunctions::storageArrayPushZeroFunction(ArrayType const& _type) }); } +string YulUtilFunctions::partialClearStorageSlotFunction() +{ + string functionName = "partial_clear_storage_slot"; + return m_functionCollector.createFunction(functionName, [&]() { + return Whiskers(R"( + function (slot, offset) { + let mask := (mul(8, sub(32, offset)), ) + sstore(slot, and(mask, sload(slot))) + } + )") + ("functionName", functionName) + ("ones", formatNumber((bigint(1) << 256) - 1)) + ("shr", shiftRightFunctionDynamic()) + .render(); + }); +} + string YulUtilFunctions::clearStorageRangeFunction(Type const& _type) { - string functionName = "clear_storage_range_" + _type.identifier(); + if (_type.storageBytes() < 32) + solAssert(_type.isValueType(), ""); - solAssert(_type.storageBytes() >= 32, "Expected smaller value for storage bytes"); + string functionName = "clear_storage_range_" + _type.identifier(); return m_functionCollector.createFunction(functionName, [&]() { return Whiskers(R"( @@ -1076,7 +1104,7 @@ string YulUtilFunctions::clearStorageRangeFunction(Type const& _type) } )") ("functionName", functionName) - ("setToZero", storageSetToZeroFunction(_type)) + ("setToZero", storageSetToZeroFunction(_type.storageBytes() < 32 ? *TypeProvider::uint256() : _type)) ("increment", _type.storageSize().str()) .render(); }); diff --git a/libsolidity/codegen/YulUtilFunctions.h b/libsolidity/codegen/YulUtilFunctions.h index 5437114f7..351c857d4 100644 --- a/libsolidity/codegen/YulUtilFunctions.h +++ b/libsolidity/codegen/YulUtilFunctions.h @@ -422,6 +422,11 @@ private: /// @returns a function that reads a reference type from storage to memory (performing a deep copy). std::string readFromStorageReferenceType(Type const& _type); + /// @returns the name of a function that will clear given storage slot + /// starting with given offset until the end of the slot + /// signature: (slot, offset) + std::string partialClearStorageSlotFunction(); + langutil::EVMVersion m_evmVersion; RevertStrings m_revertStrings; MultiUseYulFunctionCollector& m_functionCollector; diff --git a/test/libsolidity/semanticTests/array/delete_storage_array_packed.sol b/test/libsolidity/semanticTests/array/delete_storage_array_packed.sol new file mode 100644 index 000000000..281c9a0e5 --- /dev/null +++ b/test/libsolidity/semanticTests/array/delete_storage_array_packed.sol @@ -0,0 +1,18 @@ +contract C { + uint120[] data; + + function f() public returns (uint120, uint120, uint120) { + data.push(123); + data.push(234); + data.push(345); + delete data; + assembly { + sstore(data.slot, 3) + } + return (data[0], data[1], data[2]); + } +} +// ==== +// compileViaYul: also +// ---- +// f() -> 0, 0, 0