mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #9432 from ethereum/develop
Merge develop into breaking.
This commit is contained in:
commit
f945163909
@ -7,18 +7,21 @@
|
|||||||
# - ems: Emscripten
|
# - ems: Emscripten
|
||||||
version: 2.1
|
version: 2.1
|
||||||
parameters:
|
parameters:
|
||||||
ubuntu-1804-docker-image-rev:
|
ubuntu-1804-docker-image:
|
||||||
type: string
|
type: string
|
||||||
default: "4"
|
default: "solbuildpackpusher/solidity-buildpack-deps@sha256:4484ac3da8fdc337cc77a7a7be1af71cd0f28f9c890d934f1d6ae7532beb66b1"
|
||||||
ubuntu-2004-docker-image-rev:
|
ubuntu-2004-docker-image:
|
||||||
type: string
|
type: string
|
||||||
default: "2"
|
default: "solbuildpackpusher/solidity-buildpack-deps@sha256:48b5bb6b91ac7dddfe9345c88876ebed126c652258800f114caed69db73b29bf"
|
||||||
ubuntu-2004-clang-docker-image-rev:
|
ubuntu-2004-clang-docker-image:
|
||||||
type: string
|
type: string
|
||||||
default: "2"
|
default: "solbuildpackpusher/solidity-buildpack-deps@sha256:d8775de58167db5a11690fdb6ef639317fe1e579ec5d46e9732d2d903b55d135"
|
||||||
ubuntu-1604-clang-ossfuzz-docker-image-rev:
|
ubuntu-1604-clang-ossfuzz-docker-image:
|
||||||
type: string
|
type: string
|
||||||
default: "2"
|
default: "solbuildpackpusher/solidity-buildpack-deps@sha256:db52f3257396814215744a19904e42c07e040ab36b68be72a27ba71ad2f1083c"
|
||||||
|
emscripten-docker-image:
|
||||||
|
type: string
|
||||||
|
default: "solbuildpackpusher/solidity-buildpack-deps@sha256:d557d015918c3cf68b0d22839bab41013f0757b651a7fef21595f89721dbebcc"
|
||||||
|
|
||||||
defaults:
|
defaults:
|
||||||
|
|
||||||
@ -33,26 +36,11 @@ defaults:
|
|||||||
|
|
||||||
- run_build: &run_build
|
- run_build: &run_build
|
||||||
name: Build
|
name: Build
|
||||||
command: |
|
command: scripts/ci/build.sh
|
||||||
set -ex
|
|
||||||
if [ "$CIRCLE_BRANCH" = release -o -n "$CIRCLE_TAG" -o -n "$FORCE_RELEASE" ]; then echo -n > prerelease.txt; else date -u +"nightly.%Y.%-m.%-d" > prerelease.txt; fi
|
|
||||||
echo -n "$CIRCLE_SHA1" > commit_hash.txt
|
|
||||||
mkdir -p build
|
|
||||||
cd build
|
|
||||||
[ -n "$COVERAGE" -a "$CIRCLE_BRANCH" != release -a -z "$CIRCLE_TAG" ] && CMAKE_OPTIONS="$CMAKE_OPTIONS -DCOVERAGE=ON"
|
|
||||||
cmake .. -DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE:-Release} $CMAKE_OPTIONS -G "Unix Makefiles"
|
|
||||||
make -j4
|
|
||||||
|
|
||||||
- run_build_ossfuzz: &run_build_ossfuzz
|
- run_build_ossfuzz: &run_build_ossfuzz
|
||||||
name: Build_ossfuzz
|
name: Build_ossfuzz
|
||||||
command: |
|
command: scripts/ci/build_ossfuzz.sh
|
||||||
mkdir -p build
|
|
||||||
cd build
|
|
||||||
protoc --proto_path=../test/tools/ossfuzz yulProto.proto --cpp_out=../test/tools/ossfuzz
|
|
||||||
protoc --proto_path=../test/tools/ossfuzz abiV2Proto.proto --cpp_out=../test/tools/ossfuzz
|
|
||||||
protoc --proto_path=../test/tools/ossfuzz solProto.proto --cpp_out=../test/tools/ossfuzz
|
|
||||||
cmake .. -DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE:-Release} $CMAKE_OPTIONS
|
|
||||||
make ossfuzz ossfuzz_proto ossfuzz_abiv2 -j4
|
|
||||||
|
|
||||||
- run_proofs: &run_proofs
|
- run_proofs: &run_proofs
|
||||||
name: Correctness proofs for optimization rules
|
name: Correctness proofs for optimization rules
|
||||||
@ -131,7 +119,7 @@ defaults:
|
|||||||
|
|
||||||
- test_ubuntu1604_clang: &test_ubuntu1604_clang
|
- test_ubuntu1604_clang: &test_ubuntu1604_clang
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu1604-clang-ossfuzz-<< pipeline.parameters.ubuntu-1604-clang-ossfuzz-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-1604-clang-ossfuzz-docker-image >>
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- attach_workspace:
|
- attach_workspace:
|
||||||
@ -142,7 +130,7 @@ defaults:
|
|||||||
|
|
||||||
- test_ubuntu2004_clang: &test_ubuntu2004_clang
|
- test_ubuntu2004_clang: &test_ubuntu2004_clang
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-clang-<< pipeline.parameters.ubuntu-2004-clang-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-clang-docker-image >>
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- attach_workspace:
|
- attach_workspace:
|
||||||
@ -153,7 +141,8 @@ defaults:
|
|||||||
|
|
||||||
- test_ubuntu2004: &test_ubuntu2004
|
- test_ubuntu2004: &test_ubuntu2004
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
|
parallelism: 6
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- attach_workspace:
|
- attach_workspace:
|
||||||
@ -388,7 +377,7 @@ jobs:
|
|||||||
|
|
||||||
chk_docs_pragma_min_version:
|
chk_docs_pragma_min_version:
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
TERM: xterm
|
TERM: xterm
|
||||||
steps:
|
steps:
|
||||||
@ -397,7 +386,7 @@ jobs:
|
|||||||
|
|
||||||
b_ubu_clang: &build_ubuntu2004_clang
|
b_ubu_clang: &build_ubuntu2004_clang
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-clang-<< pipeline.parameters.ubuntu-2004-clang-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-clang-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
CC: clang
|
CC: clang
|
||||||
CXX: clang++
|
CXX: clang++
|
||||||
@ -410,7 +399,7 @@ jobs:
|
|||||||
|
|
||||||
b_ubu_asan_clang: &build_ubuntu2004_clang
|
b_ubu_asan_clang: &build_ubuntu2004_clang
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-clang-<< pipeline.parameters.ubuntu-2004-clang-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-clang-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
CC: clang
|
CC: clang
|
||||||
CXX: clang++
|
CXX: clang++
|
||||||
@ -422,8 +411,11 @@ jobs:
|
|||||||
- persist_to_workspace: *artifacts_executables
|
- persist_to_workspace: *artifacts_executables
|
||||||
|
|
||||||
b_ubu: &build_ubuntu2004
|
b_ubu: &build_ubuntu2004
|
||||||
|
resource_class: xlarge
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
|
environment:
|
||||||
|
MAKEFLAGS: -j 10
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- run: *run_build
|
- run: *run_build
|
||||||
@ -438,7 +430,7 @@ jobs:
|
|||||||
|
|
||||||
b_ubu18: &build_ubuntu1804
|
b_ubu18: &build_ubuntu1804
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu1804-<< pipeline.parameters.ubuntu-1804-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-1804-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
CMAKE_OPTIONS: -DCMAKE_CXX_FLAGS=-O2
|
CMAKE_OPTIONS: -DCMAKE_CXX_FLAGS=-O2
|
||||||
CMAKE_BUILD_TYPE: RelWithDebugInfo
|
CMAKE_BUILD_TYPE: RelWithDebugInfo
|
||||||
@ -492,7 +484,7 @@ jobs:
|
|||||||
|
|
||||||
b_ubu_ossfuzz: &build_ubuntu1604_clang
|
b_ubu_ossfuzz: &build_ubuntu1604_clang
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu1604-clang-ossfuzz-<< pipeline.parameters.ubuntu-1604-clang-ossfuzz-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-1604-clang-ossfuzz-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
CC: clang
|
CC: clang
|
||||||
CXX: clang++
|
CXX: clang++
|
||||||
@ -598,9 +590,11 @@ jobs:
|
|||||||
- store_artifacts: *artifacts_test_results
|
- store_artifacts: *artifacts_test_results
|
||||||
|
|
||||||
b_ems:
|
b_ems:
|
||||||
|
resource_class: xlarge
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:emsdk-1.39.15-2
|
- image: << pipeline.parameters.emscripten-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
|
MAKEFLAGS: -j 10
|
||||||
TERM: xterm
|
TERM: xterm
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
@ -634,7 +628,7 @@ jobs:
|
|||||||
|
|
||||||
b_docs:
|
b_docs:
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- run: *setup_prerelease_commit_hash
|
- run: *setup_prerelease_commit_hash
|
||||||
@ -650,7 +644,7 @@ jobs:
|
|||||||
|
|
||||||
t_ubu_soltest_enforce_yul: &t_ubu_soltest_enforce_yul
|
t_ubu_soltest_enforce_yul: &t_ubu_soltest_enforce_yul
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
EVM: constantinople
|
EVM: constantinople
|
||||||
SOLTEST_FLAGS: --enforce-via-yul
|
SOLTEST_FLAGS: --enforce-via-yul
|
||||||
@ -676,7 +670,7 @@ jobs:
|
|||||||
|
|
||||||
t_ubu_cli: &t_ubu_cli
|
t_ubu_cli: &t_ubu_cli
|
||||||
docker:
|
docker:
|
||||||
- image: ethereum/solidity-buildpack-deps:ubuntu2004-<< pipeline.parameters.ubuntu-2004-docker-image-rev >>
|
- image: << pipeline.parameters.ubuntu-2004-docker-image >>
|
||||||
environment:
|
environment:
|
||||||
TERM: xterm
|
TERM: xterm
|
||||||
steps:
|
steps:
|
||||||
|
@ -28,10 +28,49 @@ set -e
|
|||||||
|
|
||||||
REPODIR="$(realpath $(dirname $0)/..)"
|
REPODIR="$(realpath $(dirname $0)/..)"
|
||||||
|
|
||||||
EVM=istanbul OPTIMIZE=1 ABI_ENCODER_V2=1 ${REPODIR}/.circleci/soltest.sh
|
EVM_VALUES=(homestead byzantium constantinople petersburg istanbul)
|
||||||
|
OPTIMIZE_VALUES=(0 1)
|
||||||
|
STEPS=$(( 1 + ${#EVM_VALUES[@]} * ${#OPTIMIZE_VALUES[@]} ))
|
||||||
|
|
||||||
for OPTIMIZE in 0 1; do
|
if (( $CIRCLE_NODE_TOTAL )) && (( $CIRCLE_NODE_TOTAL > 1 ))
|
||||||
for EVM in homestead byzantium constantinople petersburg istanbul; do
|
then
|
||||||
EVM=$EVM OPTIMIZE=$OPTIMIZE BOOST_TEST_ARGS="-t !@nooptions" ${REPODIR}/.circleci/soltest.sh
|
# Run step 1 as the only step on the first executor
|
||||||
|
# and evenly distribute the other steps among
|
||||||
|
# the other executors.
|
||||||
|
# The first step takes much longer than the other steps.
|
||||||
|
if (( $CIRCLE_NODE_INDEX == 0 ))
|
||||||
|
then
|
||||||
|
RUN_STEPS="1"
|
||||||
|
else
|
||||||
|
export CIRCLE_NODE_INDEX=$(($CIRCLE_NODE_INDEX - 1))
|
||||||
|
export CIRCLE_NODE_TOTAL=$(($CIRCLE_NODE_TOTAL - 1))
|
||||||
|
RUN_STEPS=$(seq 2 "$STEPS" | circleci tests split)
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
RUN_STEPS=$(seq "$STEPS")
|
||||||
|
fi
|
||||||
|
|
||||||
|
# turn newlines into spaces
|
||||||
|
RUN_STEPS=$(echo $RUN_STEPS)
|
||||||
|
|
||||||
|
echo "Running steps $RUN_STEPS..."
|
||||||
|
|
||||||
|
STEP=1
|
||||||
|
|
||||||
|
[[ " $RUN_STEPS " =~ " $STEP " ]] && EVM=istanbul OPTIMIZE=1 ABI_ENCODER_V2=1 "${REPODIR}/.circleci/soltest.sh"
|
||||||
|
STEP=$(($STEP + 1))
|
||||||
|
|
||||||
|
for OPTIMIZE in ${OPTIMIZE_VALUES[@]}
|
||||||
|
do
|
||||||
|
for EVM in ${EVM_VALUES[@]}
|
||||||
|
do
|
||||||
|
[[ " $RUN_STEPS " =~ " $STEP " ]] && EVM="$EVM" OPTIMIZE="$OPTIMIZE" BOOST_TEST_ARGS="-t !@nooptions" "${REPODIR}/.circleci/soltest.sh"
|
||||||
|
STEP=$(($STEP + 1))
|
||||||
done
|
done
|
||||||
done
|
done
|
||||||
|
|
||||||
|
if (($STEP != $STEPS + 1))
|
||||||
|
then
|
||||||
|
echo "Step counter not properly adjusted!" >2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
43
.github/workflows/buildpack-deps.yml
vendored
Normal file
43
.github/workflows/buildpack-deps.yml
vendored
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
name: buildpack-deps
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [ develop ]
|
||||||
|
paths:
|
||||||
|
- 'scripts/docker/buildpack-deps/Dockerfile.emscripten'
|
||||||
|
- 'scripts/docker/buildpack-deps/Dockerfile.ubuntu1604.clang.ossfuzz'
|
||||||
|
- 'scripts/docker/buildpack-deps/Dockerfile.ubuntu1804'
|
||||||
|
- 'scripts/docker/buildpack-deps/Dockerfile.ubuntu2004.clang'
|
||||||
|
- 'scripts/docker/buildpack-deps/Dockerfile.ubuntu2004'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
buildpack-deps:
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
DOCKERHUB_TOKEN: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
DOCKER_REPOSITORY: solbuildpackpusher/solidity-buildpack-deps
|
||||||
|
IMAGE_NAME: buildpack-deps
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
image_variant: [emscripten, ubuntu1604.clang.ossfuzz, ubuntu1804, ubuntu2004.clang, ubuntu2004]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Upgrade ${{ env.IMAGE_NAME }}-${{ matrix.image_variant }}
|
||||||
|
run: |
|
||||||
|
echo ${DOCKERHUB_TOKEN} | docker login -u solbuildpackpusher --password-stdin
|
||||||
|
scripts/ci/docker_upgrade.sh ${{ env.IMAGE_NAME }} ${{ matrix.image_variant }} ${{ env.DOCKER_REPOSITORY }}
|
||||||
|
docker logout
|
||||||
|
|
||||||
|
- name: comment PR
|
||||||
|
if: "env.DOCKER_IMAGE"
|
||||||
|
uses: aarlt/comment-on-pr@v1.2.0
|
||||||
|
with:
|
||||||
|
msg: "`${{ env.DOCKER_IMAGE }} ${{ env.DOCKER_REPO_DIGEST }}`."
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -32,7 +32,7 @@ prerelease.txt
|
|||||||
|
|
||||||
# Build directory
|
# Build directory
|
||||||
build/
|
build/
|
||||||
build*/
|
/build*/
|
||||||
emscripten_build/
|
emscripten_build/
|
||||||
docs/_build
|
docs/_build
|
||||||
__pycache__
|
__pycache__
|
||||||
|
@ -47,6 +47,7 @@ env:
|
|||||||
- SOLC_TESTS=On
|
- SOLC_TESTS=On
|
||||||
- SOLC_STOREBYTECODE=Off
|
- SOLC_STOREBYTECODE=Off
|
||||||
- SOLC_DOCKER=Off
|
- SOLC_DOCKER=Off
|
||||||
|
- MAKEFLAGS="-j 4"
|
||||||
|
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
@ -112,7 +113,7 @@ matrix:
|
|||||||
before_install:
|
before_install:
|
||||||
- nvm install 10
|
- nvm install 10
|
||||||
- nvm use 10
|
- nvm use 10
|
||||||
- docker pull ethereum/solidity-buildpack-deps:emsdk-1.39.15-2
|
- docker pull solbuildpackpusher/solidity-buildpack-deps@sha256:d557d015918c3cf68b0d22839bab41013f0757b651a7fef21595f89721dbebcc
|
||||||
env:
|
env:
|
||||||
- SOLC_EMSCRIPTEN=On
|
- SOLC_EMSCRIPTEN=On
|
||||||
- SOLC_INSTALL_DEPS_TRAVIS=Off
|
- SOLC_INSTALL_DEPS_TRAVIS=Off
|
||||||
|
@ -31,16 +31,24 @@ Bugfixes:
|
|||||||
|
|
||||||
### 0.6.12 (unreleased)
|
### 0.6.12 (unreleased)
|
||||||
|
|
||||||
|
Language Features:
|
||||||
|
* Wasm backend: Add ``i32.ctz``, ``i64.ctz``, ``i32.popcnt``, and ``i64.popcnt``.
|
||||||
|
|
||||||
Compiler Features:
|
Compiler Features:
|
||||||
* Code Generator: Evaluate ``keccak256`` of string literals at compile-time.
|
* Code Generator: Evaluate ``keccak256`` of string literals at compile-time.
|
||||||
|
* Peephole Optimizer: Remove unnecessary masking of tags.
|
||||||
|
* Yul EVM Code Transform: Free stack slots directly after visiting the right-hand-side of variable declarations instead of at the end of the statement only.
|
||||||
|
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
|
* SMTChecker: Fix internal error when using bitwise operators on fixed bytes type.
|
||||||
* Type Checker: Fix overload resolution in combination with ``{value: ...}``.
|
* Type Checker: Fix overload resolution in combination with ``{value: ...}``.
|
||||||
* Type Checker: Fix internal compiler error related to oversized types.
|
* Type Checker: Fix internal compiler error related to oversized types.
|
||||||
|
* Code Generator: Avoid double cleanup when copying to memory.
|
||||||
|
|
||||||
Compiler Features:
|
Compiler Features:
|
||||||
* Build System: Update internal dependency of jsoncpp to 1.9.3.
|
* Build System: Update internal dependency of jsoncpp to 1.9.3.
|
||||||
* Optimizer: Add rule to remove shifts inside the byte opcode.
|
* Optimizer: Add rule to remove shifts inside the byte opcode.
|
||||||
|
* Peephole Optimizer: Add rule to remove swap after dup.
|
||||||
|
|
||||||
|
|
||||||
### 0.6.11 (2020-07-07)
|
### 0.6.11 (2020-07-07)
|
||||||
|
@ -174,16 +174,16 @@ enum class Instruction: uint8_t
|
|||||||
LOG3, ///< Makes a log entry; 3 topics.
|
LOG3, ///< Makes a log entry; 3 topics.
|
||||||
LOG4, ///< Makes a log entry; 4 topics.
|
LOG4, ///< Makes a log entry; 4 topics.
|
||||||
|
|
||||||
JUMPTO = 0xb0, ///< alter the program counter to a jumpdest -- not part of Instructions.cpp
|
EIP615_JUMPTO = 0xb0, ///< alter the program counter to a jumpdest -- not part of Instructions.cpp
|
||||||
JUMPIF, ///< conditionally alter the program counter -- not part of Instructions.cpp
|
EIP615_JUMPIF, ///< conditionally alter the program counter -- not part of Instructions.cpp
|
||||||
JUMPV, ///< alter the program counter to a jumpdest -- not part of Instructions.cpp
|
EIP615_JUMPV, ///< alter the program counter to a jumpdest -- not part of Instructions.cpp
|
||||||
JUMPSUB, ///< alter the program counter to a beginsub -- not part of Instructions.cpp
|
EIP615_JUMPSUB, ///< alter the program counter to a beginsub -- not part of Instructions.cpp
|
||||||
JUMPSUBV, ///< alter the program counter to a beginsub -- not part of Instructions.cpp
|
EIP615_JUMPSUBV, ///< alter the program counter to a beginsub -- not part of Instructions.cpp
|
||||||
BEGINSUB, ///< set a potential jumpsub destination -- not part of Instructions.cpp
|
EIP615_BEGINSUB, ///< set a potential jumpsub destination -- not part of Instructions.cpp
|
||||||
BEGINDATA, ///< begin the data section -- not part of Instructions.cpp
|
EIP615_BEGINDATA, ///< begin the data section -- not part of Instructions.cpp
|
||||||
RETURNSUB, ///< return to subroutine jumped from -- not part of Instructions.cpp
|
EIP615_RETURNSUB, ///< return to subroutine jumped from -- not part of Instructions.cpp
|
||||||
PUTLOCAL, ///< pop top of stack to local variable -- not part of Instructions.cpp
|
EIP615_PUTLOCAL, ///< pop top of stack to local variable -- not part of Instructions.cpp
|
||||||
GETLOCAL, ///< push local variable to top of stack -- not part of Instructions.cpp
|
EIP615_GETLOCAL, ///< push local variable to top of stack -- not part of Instructions.cpp
|
||||||
|
|
||||||
CREATE = 0xf0, ///< create a new account with associated code
|
CREATE = 0xf0, ///< create a new account with associated code
|
||||||
CALL, ///< message-call into an account
|
CALL, ///< message-call into an account
|
||||||
|
@ -205,6 +205,30 @@ struct SwapComparison: SimplePeepholeOptimizerMethod<SwapComparison, 2>
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// Remove swapN after dupN
|
||||||
|
struct DupSwap: SimplePeepholeOptimizerMethod<DupSwap, 2>
|
||||||
|
{
|
||||||
|
static size_t applySimple(
|
||||||
|
AssemblyItem const& _dupN,
|
||||||
|
AssemblyItem const& _swapN,
|
||||||
|
std::back_insert_iterator<AssemblyItems> _out
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (
|
||||||
|
SemanticInformation::isDupInstruction(_dupN) &&
|
||||||
|
SemanticInformation::isSwapInstruction(_swapN) &&
|
||||||
|
getDupNumber(_dupN.instruction()) == getSwapNumber(_swapN.instruction())
|
||||||
|
)
|
||||||
|
{
|
||||||
|
*_out = _dupN;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
struct IsZeroIsZeroJumpI: SimplePeepholeOptimizerMethod<IsZeroIsZeroJumpI, 4>
|
struct IsZeroIsZeroJumpI: SimplePeepholeOptimizerMethod<IsZeroIsZeroJumpI, 4>
|
||||||
{
|
{
|
||||||
static size_t applySimple(
|
static size_t applySimple(
|
||||||
@ -266,9 +290,10 @@ struct TagConjunctions: SimplePeepholeOptimizerMethod<TagConjunctions, 3>
|
|||||||
std::back_insert_iterator<AssemblyItems> _out
|
std::back_insert_iterator<AssemblyItems> _out
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
if (_and != Instruction::AND)
|
||||||
|
return false;
|
||||||
if (
|
if (
|
||||||
_pushTag.type() == PushTag &&
|
_pushTag.type() == PushTag &&
|
||||||
_and == Instruction::AND &&
|
|
||||||
_pushConstant.type() == Push &&
|
_pushConstant.type() == Push &&
|
||||||
(_pushConstant.data() & u256(0xFFFFFFFF)) == u256(0xFFFFFFFF)
|
(_pushConstant.data() & u256(0xFFFFFFFF)) == u256(0xFFFFFFFF)
|
||||||
)
|
)
|
||||||
@ -276,6 +301,16 @@ struct TagConjunctions: SimplePeepholeOptimizerMethod<TagConjunctions, 3>
|
|||||||
*_out = _pushTag;
|
*_out = _pushTag;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
else if (
|
||||||
|
// tag and constant are swapped
|
||||||
|
_pushConstant.type() == PushTag &&
|
||||||
|
_pushTag.type() == Push &&
|
||||||
|
(_pushTag.data() & u256(0xFFFFFFFF)) == u256(0xFFFFFFFF)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
*_out = _pushConstant;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -357,7 +392,7 @@ bool PeepholeOptimiser::optimise()
|
|||||||
applyMethods(
|
applyMethods(
|
||||||
state,
|
state,
|
||||||
PushPop(), OpPop(), DoublePush(), DoubleSwap(), CommutativeSwap(), SwapComparison(),
|
PushPop(), OpPop(), DoublePush(), DoubleSwap(), CommutativeSwap(), SwapComparison(),
|
||||||
IsZeroIsZeroJumpI(), JumpToNext(), UnreachableCode(),
|
DupSwap(), IsZeroIsZeroJumpI(), JumpToNext(), UnreachableCode(),
|
||||||
TagConjunctions(), TruthyAnd(), Identity()
|
TagConjunctions(), TruthyAnd(), Identity()
|
||||||
);
|
);
|
||||||
if (m_optimisedItems.size() < m_items.size() || (
|
if (m_optimisedItems.size() < m_items.size() || (
|
||||||
|
@ -59,10 +59,10 @@ class Expression
|
|||||||
public:
|
public:
|
||||||
explicit Expression(bool _v): Expression(_v ? "true" : "false", Kind::Bool) {}
|
explicit Expression(bool _v): Expression(_v ? "true" : "false", Kind::Bool) {}
|
||||||
explicit Expression(std::shared_ptr<SortSort> _sort, std::string _name = ""): Expression(std::move(_name), {}, _sort) {}
|
explicit Expression(std::shared_ptr<SortSort> _sort, std::string _name = ""): Expression(std::move(_name), {}, _sort) {}
|
||||||
Expression(size_t _number): Expression(std::to_string(_number), Kind::Int) {}
|
Expression(size_t _number): Expression(std::to_string(_number), {}, SortProvider::sintSort) {}
|
||||||
Expression(u256 const& _number): Expression(_number.str(), Kind::Int) {}
|
Expression(u256 const& _number): Expression(_number.str(), {}, SortProvider::sintSort) {}
|
||||||
Expression(s256 const& _number): Expression(_number.str(), Kind::Int) {}
|
Expression(s256 const& _number): Expression(_number.str(), {}, SortProvider::sintSort) {}
|
||||||
Expression(bigint const& _number): Expression(_number.str(), Kind::Int) {}
|
Expression(bigint const& _number): Expression(_number.str(), {}, SortProvider::sintSort) {}
|
||||||
|
|
||||||
Expression(Expression const&) = default;
|
Expression(Expression const&) = default;
|
||||||
Expression(Expression&&) = default;
|
Expression(Expression&&) = default;
|
||||||
@ -262,23 +262,28 @@ public:
|
|||||||
}
|
}
|
||||||
friend Expression operator+(Expression _a, Expression _b)
|
friend Expression operator+(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
return Expression("+", std::move(_a), std::move(_b), Kind::Int);
|
auto intSort = _a.sort;
|
||||||
|
return Expression("+", {std::move(_a), std::move(_b)}, intSort);
|
||||||
}
|
}
|
||||||
friend Expression operator-(Expression _a, Expression _b)
|
friend Expression operator-(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
return Expression("-", std::move(_a), std::move(_b), Kind::Int);
|
auto intSort = _a.sort;
|
||||||
|
return Expression("-", {std::move(_a), std::move(_b)}, intSort);
|
||||||
}
|
}
|
||||||
friend Expression operator*(Expression _a, Expression _b)
|
friend Expression operator*(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
return Expression("*", std::move(_a), std::move(_b), Kind::Int);
|
auto intSort = _a.sort;
|
||||||
|
return Expression("*", {std::move(_a), std::move(_b)}, intSort);
|
||||||
}
|
}
|
||||||
friend Expression operator/(Expression _a, Expression _b)
|
friend Expression operator/(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
return Expression("/", std::move(_a), std::move(_b), Kind::Int);
|
auto intSort = _a.sort;
|
||||||
|
return Expression("/", {std::move(_a), std::move(_b)}, intSort);
|
||||||
}
|
}
|
||||||
friend Expression operator%(Expression _a, Expression _b)
|
friend Expression operator%(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
return Expression("mod", std::move(_a), std::move(_b), Kind::Int);
|
auto intSort = _a.sort;
|
||||||
|
return Expression("mod", {std::move(_a), std::move(_b)}, intSort);
|
||||||
}
|
}
|
||||||
friend Expression operator&(Expression _a, Expression _b)
|
friend Expression operator&(Expression _a, Expression _b)
|
||||||
{
|
{
|
||||||
|
@ -60,6 +60,7 @@ bool ContractLevelChecker::check(ContractDefinition const& _contract)
|
|||||||
checkLibraryRequirements(_contract);
|
checkLibraryRequirements(_contract);
|
||||||
checkBaseABICompatibility(_contract);
|
checkBaseABICompatibility(_contract);
|
||||||
checkPayableFallbackWithoutReceive(_contract);
|
checkPayableFallbackWithoutReceive(_contract);
|
||||||
|
checkStorageSize(_contract);
|
||||||
|
|
||||||
return Error::containsOnlyWarnings(m_errorReporter.errors());
|
return Error::containsOnlyWarnings(m_errorReporter.errors());
|
||||||
}
|
}
|
||||||
@ -459,3 +460,20 @@ void ContractLevelChecker::checkPayableFallbackWithoutReceive(ContractDefinition
|
|||||||
SecondarySourceLocation{}.append("The payable fallback function is defined here.", fallback->location())
|
SecondarySourceLocation{}.append("The payable fallback function is defined here.", fallback->location())
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ContractLevelChecker::checkStorageSize(ContractDefinition const& _contract)
|
||||||
|
{
|
||||||
|
bigint size = 0;
|
||||||
|
vector<VariableDeclaration const*> variables;
|
||||||
|
for (ContractDefinition const* contract: boost::adaptors::reverse(_contract.annotation().linearizedBaseContracts))
|
||||||
|
for (VariableDeclaration const* variable: contract->stateVariables())
|
||||||
|
if (!(variable->isConstant() || variable->immutable()))
|
||||||
|
{
|
||||||
|
size += variable->annotation().type->storageSizeUpperBound();
|
||||||
|
if (size >= bigint(1) << 256)
|
||||||
|
{
|
||||||
|
m_errorReporter.typeError(7676_error, _contract.location(), "Contract too large for storage.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -83,6 +83,8 @@ private:
|
|||||||
|
|
||||||
/// Warns if the contract has a payable fallback, but no receive ether function.
|
/// Warns if the contract has a payable fallback, but no receive ether function.
|
||||||
void checkPayableFallbackWithoutReceive(ContractDefinition const& _contract);
|
void checkPayableFallbackWithoutReceive(ContractDefinition const& _contract);
|
||||||
|
/// Error if the contract requires too much storage
|
||||||
|
void checkStorageSize(ContractDefinition const& _contract);
|
||||||
|
|
||||||
OverrideChecker m_overrideChecker;
|
OverrideChecker m_overrideChecker;
|
||||||
langutil::ErrorReporter& m_errorReporter;
|
langutil::ErrorReporter& m_errorReporter;
|
||||||
|
@ -92,19 +92,7 @@ bool TypeChecker::visit(ContractDefinition const& _contract)
|
|||||||
for (auto const& n: _contract.subNodes())
|
for (auto const& n: _contract.subNodes())
|
||||||
n->accept(*this);
|
n->accept(*this);
|
||||||
|
|
||||||
bigint size = 0;
|
m_currentContract = nullptr;
|
||||||
vector<VariableDeclaration const*> variables;
|
|
||||||
for (ContractDefinition const* contract: boost::adaptors::reverse(m_currentContract->annotation().linearizedBaseContracts))
|
|
||||||
for (VariableDeclaration const* variable: contract->stateVariables())
|
|
||||||
if (!(variable->isConstant() || variable->immutable()))
|
|
||||||
{
|
|
||||||
size += storageSizeUpperBound(*(variable->annotation().type));
|
|
||||||
if (size >= bigint(1) << 256)
|
|
||||||
{
|
|
||||||
m_errorReporter.typeError(7676_error, m_currentContract->location(), "Contract too large for storage.");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -353,7 +341,7 @@ bool TypeChecker::visit(FunctionDefinition const& _function)
|
|||||||
if (_function.libraryFunction())
|
if (_function.libraryFunction())
|
||||||
m_errorReporter.typeError(7708_error, _function.location(), "Library functions cannot be payable.");
|
m_errorReporter.typeError(7708_error, _function.location(), "Library functions cannot be payable.");
|
||||||
if (_function.isOrdinary() && !_function.isPartOfExternalInterface())
|
if (_function.isOrdinary() && !_function.isPartOfExternalInterface())
|
||||||
m_errorReporter.typeError(5587_error, _function.location(), "Internal functions cannot be payable.");
|
m_errorReporter.typeError(5587_error, _function.location(), "\"internal\" and \"private\" functions cannot be payable.");
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<VariableDeclaration const*> internalParametersInConstructor;
|
vector<VariableDeclaration const*> internalParametersInConstructor;
|
||||||
@ -598,12 +586,16 @@ bool TypeChecker::visit(VariableDeclaration const& _variable)
|
|||||||
" in small quantities per transaction.";
|
" in small quantities per transaction.";
|
||||||
};
|
};
|
||||||
|
|
||||||
if (storageSizeUpperBound(*varType) >= bigint(1) << 64)
|
if (varType->storageSizeUpperBound() >= bigint(1) << 64)
|
||||||
{
|
{
|
||||||
if (_variable.isStateVariable())
|
if (_variable.isStateVariable())
|
||||||
m_errorReporter.warning(3408_error, _variable.location(), collisionMessage(_variable.name(), true));
|
m_errorReporter.warning(3408_error, _variable.location(), collisionMessage(_variable.name(), true));
|
||||||
else
|
else
|
||||||
m_errorReporter.warning(2332_error, _variable.typeName()->location(), collisionMessage(varType->canonicalName(), false));
|
m_errorReporter.warning(
|
||||||
|
2332_error,
|
||||||
|
_variable.typeName() ? _variable.typeName()->location() : _variable.location(),
|
||||||
|
collisionMessage(varType->canonicalName(), false)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
vector<Type const*> oversizedSubtypes = frontend::oversizedSubtypes(*varType);
|
vector<Type const*> oversizedSubtypes = frontend::oversizedSubtypes(*varType);
|
||||||
for (Type const* subtype: oversizedSubtypes)
|
for (Type const* subtype: oversizedSubtypes)
|
||||||
|
@ -63,37 +63,6 @@ struct TypeComp
|
|||||||
};
|
};
|
||||||
using TypeSet = std::set<Type const*, TypeComp>;
|
using TypeSet = std::set<Type const*, TypeComp>;
|
||||||
|
|
||||||
bigint storageSizeUpperBoundInner(
|
|
||||||
Type const& _type,
|
|
||||||
set<StructDefinition const*>& _structsSeen
|
|
||||||
)
|
|
||||||
{
|
|
||||||
switch (_type.category())
|
|
||||||
{
|
|
||||||
case Type::Category::Array:
|
|
||||||
{
|
|
||||||
auto const& t = dynamic_cast<ArrayType const&>(_type);
|
|
||||||
if (!t.isDynamicallySized())
|
|
||||||
return storageSizeUpperBoundInner(*t.baseType(), _structsSeen) * t.length();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case Type::Category::Struct:
|
|
||||||
{
|
|
||||||
auto const& t = dynamic_cast<StructType const&>(_type);
|
|
||||||
solAssert(!_structsSeen.count(&t.structDefinition()), "Recursive struct.");
|
|
||||||
bigint size = 1;
|
|
||||||
_structsSeen.insert(&t.structDefinition());
|
|
||||||
for (auto const& m: t.members(nullptr))
|
|
||||||
size += storageSizeUpperBoundInner(*m.type, _structsSeen);
|
|
||||||
_structsSeen.erase(&t.structDefinition());
|
|
||||||
return size;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return bigint(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
void oversizedSubtypesInner(
|
void oversizedSubtypesInner(
|
||||||
Type const& _type,
|
Type const& _type,
|
||||||
bool _includeType,
|
bool _includeType,
|
||||||
@ -106,7 +75,7 @@ void oversizedSubtypesInner(
|
|||||||
case Type::Category::Array:
|
case Type::Category::Array:
|
||||||
{
|
{
|
||||||
auto const& t = dynamic_cast<ArrayType const&>(_type);
|
auto const& t = dynamic_cast<ArrayType const&>(_type);
|
||||||
if (_includeType && storageSizeUpperBound(t) >= bigint(1) << 64)
|
if (_includeType && t.storageSizeUpperBound() >= bigint(1) << 64)
|
||||||
_oversizedSubtypes.insert(&t);
|
_oversizedSubtypes.insert(&t);
|
||||||
oversizedSubtypesInner(*t.baseType(), t.isDynamicallySized(), _structsSeen, _oversizedSubtypes);
|
oversizedSubtypesInner(*t.baseType(), t.isDynamicallySized(), _structsSeen, _oversizedSubtypes);
|
||||||
break;
|
break;
|
||||||
@ -116,7 +85,7 @@ void oversizedSubtypesInner(
|
|||||||
auto const& t = dynamic_cast<StructType const&>(_type);
|
auto const& t = dynamic_cast<StructType const&>(_type);
|
||||||
if (_structsSeen.count(&t.structDefinition()))
|
if (_structsSeen.count(&t.structDefinition()))
|
||||||
return;
|
return;
|
||||||
if (_includeType && storageSizeUpperBound(t) >= bigint(1) << 64)
|
if (_includeType && t.storageSizeUpperBound() >= bigint(1) << 64)
|
||||||
_oversizedSubtypes.insert(&t);
|
_oversizedSubtypes.insert(&t);
|
||||||
_structsSeen.insert(&t.structDefinition());
|
_structsSeen.insert(&t.structDefinition());
|
||||||
for (auto const& m: t.members(nullptr))
|
for (auto const& m: t.members(nullptr))
|
||||||
@ -231,12 +200,6 @@ util::Result<TypePointers> transformParametersToExternal(TypePointers const& _pa
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bigint solidity::frontend::storageSizeUpperBound(frontend::Type const& _type)
|
|
||||||
{
|
|
||||||
set<StructDefinition const*> structsSeen;
|
|
||||||
return storageSizeUpperBoundInner(_type, structsSeen);
|
|
||||||
}
|
|
||||||
|
|
||||||
vector<frontend::Type const*> solidity::frontend::oversizedSubtypes(frontend::Type const& _type)
|
vector<frontend::Type const*> solidity::frontend::oversizedSubtypes(frontend::Type const& _type)
|
||||||
{
|
{
|
||||||
set<StructDefinition const*> structsSeen;
|
set<StructDefinition const*> structsSeen;
|
||||||
@ -1867,7 +1830,7 @@ BoolResult ArrayType::validForLocation(DataLocation _loc) const
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DataLocation::Storage:
|
case DataLocation::Storage:
|
||||||
if (storageSizeUpperBound(*this) >= bigint(1) << 256)
|
if (storageSizeUpperBound() >= bigint(1) << 256)
|
||||||
return BoolResult::err("Type too large for storage.");
|
return BoolResult::err("Type too large for storage.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1908,6 +1871,14 @@ bool ArrayType::isDynamicallyEncoded() const
|
|||||||
return isDynamicallySized() || baseType()->isDynamicallyEncoded();
|
return isDynamicallySized() || baseType()->isDynamicallyEncoded();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bigint ArrayType::storageSizeUpperBound() const
|
||||||
|
{
|
||||||
|
if (isDynamicallySized())
|
||||||
|
return 1;
|
||||||
|
else
|
||||||
|
return length() * baseType()->storageSizeUpperBound();
|
||||||
|
}
|
||||||
|
|
||||||
u256 ArrayType::storageSize() const
|
u256 ArrayType::storageSize() const
|
||||||
{
|
{
|
||||||
if (isDynamicallySized())
|
if (isDynamicallySized())
|
||||||
@ -2381,6 +2352,14 @@ u256 StructType::memoryDataSize() const
|
|||||||
return size;
|
return size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bigint StructType::storageSizeUpperBound() const
|
||||||
|
{
|
||||||
|
bigint size = 1;
|
||||||
|
for (auto const& member: members(nullptr))
|
||||||
|
size += member.type->storageSizeUpperBound();
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
u256 StructType::storageSize() const
|
u256 StructType::storageSize() const
|
||||||
{
|
{
|
||||||
return max<u256>(1, members(nullptr).storageSize());
|
return max<u256>(1, members(nullptr).storageSize());
|
||||||
@ -2558,7 +2537,7 @@ BoolResult StructType::validForLocation(DataLocation _loc) const
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
_loc == DataLocation::Storage &&
|
_loc == DataLocation::Storage &&
|
||||||
storageSizeUpperBound(*this) >= bigint(1) << 256
|
storageSizeUpperBound() >= bigint(1) << 256
|
||||||
)
|
)
|
||||||
return BoolResult::err("Type too large for storage.");
|
return BoolResult::err("Type too large for storage.");
|
||||||
|
|
||||||
|
@ -59,7 +59,6 @@ using BoolResult = util::Result<bool>;
|
|||||||
namespace solidity::frontend
|
namespace solidity::frontend
|
||||||
{
|
{
|
||||||
|
|
||||||
bigint storageSizeUpperBound(frontend::Type const& _type);
|
|
||||||
std::vector<frontend::Type const*> oversizedSubtypes(frontend::Type const& _type);
|
std::vector<frontend::Type const*> oversizedSubtypes(frontend::Type const& _type);
|
||||||
|
|
||||||
inline rational makeRational(bigint const& _numerator, bigint const& _denominator)
|
inline rational makeRational(bigint const& _numerator, bigint const& _denominator)
|
||||||
@ -251,6 +250,10 @@ public:
|
|||||||
/// @returns the number of storage slots required to hold this value in storage.
|
/// @returns the number of storage slots required to hold this value in storage.
|
||||||
/// For dynamically "allocated" types, it returns the size of the statically allocated head,
|
/// For dynamically "allocated" types, it returns the size of the statically allocated head,
|
||||||
virtual u256 storageSize() const { return 1; }
|
virtual u256 storageSize() const { return 1; }
|
||||||
|
/// @returns an upper bound on the total storage size required by this type, descending
|
||||||
|
/// into structs and statically-sized arrays. This is mainly to ensure that the storage
|
||||||
|
/// slot allocation algorithm does not overflow, it is not a protection against collisions.
|
||||||
|
virtual bigint storageSizeUpperBound() const { return 1; }
|
||||||
/// Multiple small types can be packed into a single storage slot. If such a packing is possible
|
/// Multiple small types can be packed into a single storage slot. If such a packing is possible
|
||||||
/// this function @returns the size in bytes smaller than 32. Data is moved to the next slot if
|
/// this function @returns the size in bytes smaller than 32. Data is moved to the next slot if
|
||||||
/// it does not fit.
|
/// it does not fit.
|
||||||
@ -790,6 +793,7 @@ public:
|
|||||||
unsigned calldataEncodedTailSize() const override;
|
unsigned calldataEncodedTailSize() const override;
|
||||||
bool isDynamicallySized() const override { return m_hasDynamicLength; }
|
bool isDynamicallySized() const override { return m_hasDynamicLength; }
|
||||||
bool isDynamicallyEncoded() const override;
|
bool isDynamicallyEncoded() const override;
|
||||||
|
bigint storageSizeUpperBound() const override;
|
||||||
u256 storageSize() const override;
|
u256 storageSize() const override;
|
||||||
bool containsNestedMapping() const override { return m_baseType->containsNestedMapping(); }
|
bool containsNestedMapping() const override { return m_baseType->containsNestedMapping(); }
|
||||||
bool nameable() const override { return true; }
|
bool nameable() const override { return true; }
|
||||||
@ -954,6 +958,7 @@ public:
|
|||||||
unsigned calldataEncodedTailSize() const override;
|
unsigned calldataEncodedTailSize() const override;
|
||||||
bool isDynamicallyEncoded() const override;
|
bool isDynamicallyEncoded() const override;
|
||||||
u256 memoryDataSize() const override;
|
u256 memoryDataSize() const override;
|
||||||
|
bigint storageSizeUpperBound() const override;
|
||||||
u256 storageSize() const override;
|
u256 storageSize() const override;
|
||||||
bool containsNestedMapping() const override;
|
bool containsNestedMapping() const override;
|
||||||
bool nameable() const override { return true; }
|
bool nameable() const override { return true; }
|
||||||
|
@ -180,7 +180,7 @@ void CompilerUtils::storeInMemory(unsigned _offset)
|
|||||||
m_context << u256(_offset) << Instruction::MSTORE;
|
m_context << u256(_offset) << Instruction::MSTORE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CompilerUtils::storeInMemoryDynamic(Type const& _type, bool _padToWordBoundaries)
|
void CompilerUtils::storeInMemoryDynamic(Type const& _type, bool _padToWordBoundaries, bool _cleanup)
|
||||||
{
|
{
|
||||||
// process special types (Reference, StringLiteral, Function)
|
// process special types (Reference, StringLiteral, Function)
|
||||||
if (auto ref = dynamic_cast<ReferenceType const*>(&_type))
|
if (auto ref = dynamic_cast<ReferenceType const*>(&_type))
|
||||||
@ -189,7 +189,7 @@ void CompilerUtils::storeInMemoryDynamic(Type const& _type, bool _padToWordBound
|
|||||||
ref->location() == DataLocation::Memory,
|
ref->location() == DataLocation::Memory,
|
||||||
"Only in-memory reference type can be stored."
|
"Only in-memory reference type can be stored."
|
||||||
);
|
);
|
||||||
storeInMemoryDynamic(*TypeProvider::uint256(), _padToWordBoundaries);
|
storeInMemoryDynamic(*TypeProvider::uint256(), _padToWordBoundaries, _cleanup);
|
||||||
}
|
}
|
||||||
else if (auto str = dynamic_cast<StringLiteralType const*>(&_type))
|
else if (auto str = dynamic_cast<StringLiteralType const*>(&_type))
|
||||||
{
|
{
|
||||||
@ -212,7 +212,7 @@ void CompilerUtils::storeInMemoryDynamic(Type const& _type, bool _padToWordBound
|
|||||||
}
|
}
|
||||||
else if (_type.isValueType())
|
else if (_type.isValueType())
|
||||||
{
|
{
|
||||||
unsigned numBytes = prepareMemoryStore(_type, _padToWordBoundaries);
|
unsigned numBytes = prepareMemoryStore(_type, _padToWordBoundaries, _cleanup);
|
||||||
m_context << Instruction::DUP2 << Instruction::MSTORE;
|
m_context << Instruction::DUP2 << Instruction::MSTORE;
|
||||||
m_context << u256(numBytes) << Instruction::ADD;
|
m_context << u256(numBytes) << Instruction::ADD;
|
||||||
}
|
}
|
||||||
@ -463,6 +463,7 @@ void CompilerUtils::encodeToMemory(
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
bool needCleanup = true;
|
||||||
copyToStackTop(argSize - stackPos + dynPointers + 2, _givenTypes[i]->sizeOnStack());
|
copyToStackTop(argSize - stackPos + dynPointers + 2, _givenTypes[i]->sizeOnStack());
|
||||||
solAssert(!!targetType, "Externalable type expected.");
|
solAssert(!!targetType, "Externalable type expected.");
|
||||||
TypePointer type = targetType;
|
TypePointer type = targetType;
|
||||||
@ -481,7 +482,11 @@ void CompilerUtils::encodeToMemory(
|
|||||||
)
|
)
|
||||||
type = _givenTypes[i]; // delay conversion
|
type = _givenTypes[i]; // delay conversion
|
||||||
else
|
else
|
||||||
|
{
|
||||||
convertType(*_givenTypes[i], *targetType, true);
|
convertType(*_givenTypes[i], *targetType, true);
|
||||||
|
needCleanup = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (auto arrayType = dynamic_cast<ArrayType const*>(type))
|
if (auto arrayType = dynamic_cast<ArrayType const*>(type))
|
||||||
ArrayUtils(m_context).copyArrayToMemory(*arrayType, _padToWordBoundaries);
|
ArrayUtils(m_context).copyArrayToMemory(*arrayType, _padToWordBoundaries);
|
||||||
else if (auto arraySliceType = dynamic_cast<ArraySliceType const*>(type))
|
else if (auto arraySliceType = dynamic_cast<ArraySliceType const*>(type))
|
||||||
@ -495,7 +500,7 @@ void CompilerUtils::encodeToMemory(
|
|||||||
ArrayUtils(m_context).copyArrayToMemory(arraySliceType->arrayType(), _padToWordBoundaries);
|
ArrayUtils(m_context).copyArrayToMemory(arraySliceType->arrayType(), _padToWordBoundaries);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
storeInMemoryDynamic(*type, _padToWordBoundaries);
|
storeInMemoryDynamic(*type, _padToWordBoundaries, needCleanup);
|
||||||
}
|
}
|
||||||
stackPos += _givenTypes[i]->sizeOnStack();
|
stackPos += _givenTypes[i]->sizeOnStack();
|
||||||
}
|
}
|
||||||
@ -1498,7 +1503,7 @@ void CompilerUtils::rightShiftNumberOnStack(unsigned _bits)
|
|||||||
m_context << (u256(1) << _bits) << Instruction::SWAP1 << Instruction::DIV;
|
m_context << (u256(1) << _bits) << Instruction::SWAP1 << Instruction::DIV;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned CompilerUtils::prepareMemoryStore(Type const& _type, bool _padToWords)
|
unsigned CompilerUtils::prepareMemoryStore(Type const& _type, bool _padToWords, bool _cleanup)
|
||||||
{
|
{
|
||||||
solAssert(
|
solAssert(
|
||||||
_type.sizeOnStack() == 1,
|
_type.sizeOnStack() == 1,
|
||||||
@ -1521,7 +1526,9 @@ unsigned CompilerUtils::prepareMemoryStore(Type const& _type, bool _padToWords)
|
|||||||
|
|
||||||
bool leftAligned = _type.category() == Type::Category::FixedBytes;
|
bool leftAligned = _type.category() == Type::Category::FixedBytes;
|
||||||
|
|
||||||
|
if (_cleanup)
|
||||||
convertType(_type, _type, true);
|
convertType(_type, _type, true);
|
||||||
|
|
||||||
if (numBytes != 32 && !leftAligned && !_padToWords)
|
if (numBytes != 32 && !leftAligned && !_padToWords)
|
||||||
// shift the value accordingly before storing
|
// shift the value accordingly before storing
|
||||||
leftShiftNumberOnStack((32 - numBytes) * 8);
|
leftShiftNumberOnStack((32 - numBytes) * 8);
|
||||||
|
@ -107,10 +107,11 @@ public:
|
|||||||
/// and also updates that. For reference types, only copies the data pointer. Fails for
|
/// and also updates that. For reference types, only copies the data pointer. Fails for
|
||||||
/// non-memory-references.
|
/// non-memory-references.
|
||||||
/// @param _padToWords if true, adds zeros to pad to multiple of 32 bytes. Array elements
|
/// @param _padToWords if true, adds zeros to pad to multiple of 32 bytes. Array elements
|
||||||
|
/// @param _cleanup if true, adds code to cleanup the value before storing it.
|
||||||
/// are always padded (except for byte arrays), regardless of this parameter.
|
/// are always padded (except for byte arrays), regardless of this parameter.
|
||||||
/// Stack pre: memory_offset value...
|
/// Stack pre: memory_offset value...
|
||||||
/// Stack post: (memory_offset+length)
|
/// Stack post: (memory_offset+length)
|
||||||
void storeInMemoryDynamic(Type const& _type, bool _padToWords = true);
|
void storeInMemoryDynamic(Type const& _type, bool _padToWords = true, bool _cleanup = true);
|
||||||
|
|
||||||
/// Creates code that unpacks the arguments according to their types specified by a vector of TypePointers.
|
/// Creates code that unpacks the arguments according to their types specified by a vector of TypePointers.
|
||||||
/// From memory if @a _fromMemory is true, otherwise from call data.
|
/// From memory if @a _fromMemory is true, otherwise from call data.
|
||||||
@ -309,7 +310,8 @@ private:
|
|||||||
void cleanHigherOrderBits(IntegerType const& _typeOnStack);
|
void cleanHigherOrderBits(IntegerType const& _typeOnStack);
|
||||||
|
|
||||||
/// Prepares the given type for storing in memory by shifting it if necessary.
|
/// Prepares the given type for storing in memory by shifting it if necessary.
|
||||||
unsigned prepareMemoryStore(Type const& _type, bool _padToWords);
|
/// @param _cleanup if true, also cleanup the value when preparing to store it in memory
|
||||||
|
unsigned prepareMemoryStore(Type const& _type, bool _padToWords, bool _cleanup = true);
|
||||||
/// Loads type from memory assuming memory offset is on stack top.
|
/// Loads type from memory assuming memory offset is on stack top.
|
||||||
unsigned loadFromMemoryHelper(Type const& _type, bool _fromCalldata, bool _padToWords);
|
unsigned loadFromMemoryHelper(Type const& _type, bool _fromCalldata, bool _padToWords);
|
||||||
|
|
||||||
|
@ -1376,9 +1376,11 @@ void SMTEncoder::bitwiseOperation(BinaryOperation const& _op)
|
|||||||
bvSize = fixedType->numBits();
|
bvSize = fixedType->numBits();
|
||||||
isSigned = fixedType->isSigned();
|
isSigned = fixedType->isSigned();
|
||||||
}
|
}
|
||||||
|
else if (auto const* fixedBytesType = dynamic_cast<FixedBytesType const*>(commonType))
|
||||||
|
bvSize = fixedBytesType->numBytes() * 8;
|
||||||
|
|
||||||
auto bvLeft = smtutil::Expression::int2bv(expr(_op.leftExpression()), bvSize);
|
auto bvLeft = smtutil::Expression::int2bv(expr(_op.leftExpression(), commonType), bvSize);
|
||||||
auto bvRight = smtutil::Expression::int2bv(expr(_op.rightExpression()), bvSize);
|
auto bvRight = smtutil::Expression::int2bv(expr(_op.rightExpression(), commonType), bvSize);
|
||||||
|
|
||||||
optional<smtutil::Expression> result;
|
optional<smtutil::Expression> result;
|
||||||
if (_op.getOperator() == Token::BitAnd)
|
if (_op.getOperator() == Token::BitAnd)
|
||||||
|
@ -441,7 +441,11 @@ optional<smtutil::Expression> symbolicTypeConversion(TypePointer _from, TypePoin
|
|||||||
// case they'd need to be encoded as numbers.
|
// case they'd need to be encoded as numbers.
|
||||||
if (auto strType = dynamic_cast<StringLiteralType const*>(_from))
|
if (auto strType = dynamic_cast<StringLiteralType const*>(_from))
|
||||||
if (_to->category() == frontend::Type::Category::FixedBytes)
|
if (_to->category() == frontend::Type::Category::FixedBytes)
|
||||||
|
{
|
||||||
|
if (strType->value().empty())
|
||||||
|
return smtutil::Expression(size_t(0));
|
||||||
return smtutil::Expression(u256(toHex(util::asBytes(strType->value()), util::HexPrefix::Add)));
|
return smtutil::Expression(u256(toHex(util::asBytes(strType->value()), util::HexPrefix::Add)));
|
||||||
|
}
|
||||||
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
@ -102,7 +102,7 @@ void EVMAssembly::appendJumpTo(LabelID _labelId, int _stackDiffAfter, JumpType _
|
|||||||
{
|
{
|
||||||
if (m_evm15)
|
if (m_evm15)
|
||||||
{
|
{
|
||||||
m_bytecode.push_back(uint8_t(evmasm::Instruction::JUMPTO));
|
m_bytecode.push_back(uint8_t(evmasm::Instruction::EIP615_JUMPTO));
|
||||||
appendLabelReferenceInternal(_labelId);
|
appendLabelReferenceInternal(_labelId);
|
||||||
m_stackHeight += _stackDiffAfter;
|
m_stackHeight += _stackDiffAfter;
|
||||||
}
|
}
|
||||||
@ -117,7 +117,7 @@ void EVMAssembly::appendJumpToIf(LabelID _labelId, JumpType)
|
|||||||
{
|
{
|
||||||
if (m_evm15)
|
if (m_evm15)
|
||||||
{
|
{
|
||||||
m_bytecode.push_back(uint8_t(evmasm::Instruction::JUMPIF));
|
m_bytecode.push_back(uint8_t(evmasm::Instruction::EIP615_JUMPIF));
|
||||||
appendLabelReferenceInternal(_labelId);
|
appendLabelReferenceInternal(_labelId);
|
||||||
m_stackHeight--;
|
m_stackHeight--;
|
||||||
}
|
}
|
||||||
@ -133,7 +133,7 @@ void EVMAssembly::appendBeginsub(LabelID _labelId, int _arguments)
|
|||||||
yulAssert(m_evm15, "BEGINSUB used for EVM 1.0");
|
yulAssert(m_evm15, "BEGINSUB used for EVM 1.0");
|
||||||
yulAssert(_arguments >= 0, "");
|
yulAssert(_arguments >= 0, "");
|
||||||
setLabelToCurrentPosition(_labelId);
|
setLabelToCurrentPosition(_labelId);
|
||||||
m_bytecode.push_back(uint8_t(evmasm::Instruction::BEGINSUB));
|
m_bytecode.push_back(uint8_t(evmasm::Instruction::EIP615_BEGINSUB));
|
||||||
m_stackHeight += _arguments;
|
m_stackHeight += _arguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -141,7 +141,7 @@ void EVMAssembly::appendJumpsub(LabelID _labelId, int _arguments, int _returns)
|
|||||||
{
|
{
|
||||||
yulAssert(m_evm15, "JUMPSUB used for EVM 1.0");
|
yulAssert(m_evm15, "JUMPSUB used for EVM 1.0");
|
||||||
yulAssert(_arguments >= 0 && _returns >= 0, "");
|
yulAssert(_arguments >= 0 && _returns >= 0, "");
|
||||||
m_bytecode.push_back(uint8_t(evmasm::Instruction::JUMPSUB));
|
m_bytecode.push_back(uint8_t(evmasm::Instruction::EIP615_JUMPSUB));
|
||||||
appendLabelReferenceInternal(_labelId);
|
appendLabelReferenceInternal(_labelId);
|
||||||
m_stackHeight += _returns - _arguments;
|
m_stackHeight += _returns - _arguments;
|
||||||
}
|
}
|
||||||
@ -150,7 +150,7 @@ void EVMAssembly::appendReturnsub(int _returns, int _stackDiffAfter)
|
|||||||
{
|
{
|
||||||
yulAssert(m_evm15, "RETURNSUB used for EVM 1.0");
|
yulAssert(m_evm15, "RETURNSUB used for EVM 1.0");
|
||||||
yulAssert(_returns >= 0, "");
|
yulAssert(_returns >= 0, "");
|
||||||
m_bytecode.push_back(uint8_t(evmasm::Instruction::RETURNSUB));
|
m_bytecode.push_back(uint8_t(evmasm::Instruction::EIP615_RETURNSUB));
|
||||||
m_stackHeight += _stackDiffAfter - _returns;
|
m_stackHeight += _stackDiffAfter - _returns;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -141,7 +141,7 @@ bool CodeTransform::unreferenced(Scope::Variable const& _var) const
|
|||||||
return !m_context->variableReferences.count(&_var) || m_context->variableReferences[&_var] == 0;
|
return !m_context->variableReferences.count(&_var) || m_context->variableReferences[&_var] == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::freeUnusedVariables()
|
void CodeTransform::freeUnusedVariables(bool _popUnusedSlotsAtStackTop)
|
||||||
{
|
{
|
||||||
if (!m_allowStackOpt)
|
if (!m_allowStackOpt)
|
||||||
return;
|
return;
|
||||||
@ -154,6 +154,7 @@ void CodeTransform::freeUnusedVariables()
|
|||||||
deleteVariable(var);
|
deleteVariable(var);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (_popUnusedSlotsAtStackTop)
|
||||||
while (m_unusedStackSlots.count(m_assembly.stackHeight() - 1))
|
while (m_unusedStackSlots.count(m_assembly.stackHeight() - 1))
|
||||||
{
|
{
|
||||||
yulAssert(m_unusedStackSlots.erase(m_assembly.stackHeight() - 1), "");
|
yulAssert(m_unusedStackSlots.erase(m_assembly.stackHeight() - 1), "");
|
||||||
@ -181,6 +182,7 @@ void CodeTransform::operator()(VariableDeclaration const& _varDecl)
|
|||||||
{
|
{
|
||||||
std::visit(*this, *_varDecl.value);
|
std::visit(*this, *_varDecl.value);
|
||||||
expectDeposit(static_cast<int>(numVariables), static_cast<int>(heightAtStart));
|
expectDeposit(static_cast<int>(numVariables), static_cast<int>(heightAtStart));
|
||||||
|
freeUnusedVariables(false);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -161,8 +161,9 @@ protected:
|
|||||||
bool unreferenced(Scope::Variable const& _var) const;
|
bool unreferenced(Scope::Variable const& _var) const;
|
||||||
/// Marks slots of variables that are not used anymore
|
/// Marks slots of variables that are not used anymore
|
||||||
/// and were defined in the current scope for reuse.
|
/// and were defined in the current scope for reuse.
|
||||||
/// Also POPs unused topmost stack slots.
|
/// Also POPs unused topmost stack slots,
|
||||||
void freeUnusedVariables();
|
/// unless @a _popUnusedSlotsAtStackTop is set to false.
|
||||||
|
void freeUnusedVariables(bool _popUnusedSlotsAtStackTop = true);
|
||||||
/// Marks the stack slot of @a _var to be reused.
|
/// Marks the stack slot of @a _var to be reused.
|
||||||
void deleteVariable(Scope::Variable const& _var);
|
void deleteVariable(Scope::Variable const& _var);
|
||||||
|
|
||||||
|
@ -38,13 +38,18 @@ WasmDialect::WasmDialect()
|
|||||||
"add",
|
"add",
|
||||||
"sub",
|
"sub",
|
||||||
"mul",
|
"mul",
|
||||||
|
// TODO: div_s
|
||||||
"div_u",
|
"div_u",
|
||||||
|
// TODO: rem_s
|
||||||
"rem_u",
|
"rem_u",
|
||||||
"and",
|
"and",
|
||||||
"or",
|
"or",
|
||||||
"xor",
|
"xor",
|
||||||
"shl",
|
"shl",
|
||||||
|
// TODO: shr_s
|
||||||
"shr_u",
|
"shr_u",
|
||||||
|
// TODO: rotl
|
||||||
|
// TODO: rotr
|
||||||
})
|
})
|
||||||
addFunction(t.str() + "." + name, {t, t}, {t});
|
addFunction(t.str() + "." + name, {t, t}, {t});
|
||||||
|
|
||||||
@ -52,9 +57,13 @@ WasmDialect::WasmDialect()
|
|||||||
for (auto const& name: {
|
for (auto const& name: {
|
||||||
"eq",
|
"eq",
|
||||||
"ne",
|
"ne",
|
||||||
|
// TODO: lt_s
|
||||||
"lt_u",
|
"lt_u",
|
||||||
|
// TODO: gt_s
|
||||||
"gt_u",
|
"gt_u",
|
||||||
|
// TODO: le_s
|
||||||
"le_u",
|
"le_u",
|
||||||
|
// TODO: ge_s
|
||||||
"ge_u"
|
"ge_u"
|
||||||
})
|
})
|
||||||
addFunction(t.str() + "." + name, {t, t}, {i32});
|
addFunction(t.str() + "." + name, {t, t}, {i32});
|
||||||
@ -62,8 +71,13 @@ WasmDialect::WasmDialect()
|
|||||||
addFunction("i32.eqz", {i32}, {i32});
|
addFunction("i32.eqz", {i32}, {i32});
|
||||||
addFunction("i64.eqz", {i64}, {i32});
|
addFunction("i64.eqz", {i64}, {i32});
|
||||||
|
|
||||||
addFunction("i32.clz", {i32}, {i32});
|
for (auto t: types)
|
||||||
addFunction("i64.clz", {i64}, {i64});
|
for (auto const& name: {
|
||||||
|
"clz",
|
||||||
|
"ctz",
|
||||||
|
"popcnt",
|
||||||
|
})
|
||||||
|
addFunction(t.str() + "." + name, {t}, {t});
|
||||||
|
|
||||||
addFunction("i32.wrap_i64", {i64}, {i32});
|
addFunction("i32.wrap_i64", {i64}, {i32});
|
||||||
|
|
||||||
@ -73,6 +87,7 @@ WasmDialect::WasmDialect()
|
|||||||
m_functions["i32.store"_yulstring].sideEffects.invalidatesStorage = false;
|
m_functions["i32.store"_yulstring].sideEffects.invalidatesStorage = false;
|
||||||
addFunction("i64.store", {i32, i64}, {}, false);
|
addFunction("i64.store", {i32, i64}, {}, false);
|
||||||
m_functions["i64.store"_yulstring].sideEffects.invalidatesStorage = false;
|
m_functions["i64.store"_yulstring].sideEffects.invalidatesStorage = false;
|
||||||
|
// TODO: add i32.store16, i64.store8, i64.store16, i64.store32
|
||||||
|
|
||||||
addFunction("i32.store8", {i32, i32}, {}, false);
|
addFunction("i32.store8", {i32, i32}, {}, false);
|
||||||
m_functions["i32.store8"_yulstring].sideEffects.invalidatesStorage = false;
|
m_functions["i32.store8"_yulstring].sideEffects.invalidatesStorage = false;
|
||||||
@ -89,6 +104,7 @@ WasmDialect::WasmDialect()
|
|||||||
m_functions["i64.load"_yulstring].sideEffects.invalidatesMemory = false;
|
m_functions["i64.load"_yulstring].sideEffects.invalidatesMemory = false;
|
||||||
m_functions["i64.load"_yulstring].sideEffects.sideEffectFree = true;
|
m_functions["i64.load"_yulstring].sideEffects.sideEffectFree = true;
|
||||||
m_functions["i64.load"_yulstring].sideEffects.sideEffectFreeIfNoMSize = true;
|
m_functions["i64.load"_yulstring].sideEffects.sideEffectFreeIfNoMSize = true;
|
||||||
|
// TODO: add i32.load8, i32.load16, i64.load8, i64.load16, i64.load32
|
||||||
|
|
||||||
// Drop is actually overloaded for all types, but Yul does not support that.
|
// Drop is actually overloaded for all types, but Yul does not support that.
|
||||||
// Because of that, we introduce "i32.drop" and "i64.drop".
|
// Because of that, we introduce "i32.drop" and "i64.drop".
|
||||||
|
@ -30,6 +30,47 @@ using namespace solidity;
|
|||||||
using namespace solidity::yul;
|
using namespace solidity::yul;
|
||||||
using namespace solidity::util;
|
using namespace solidity::util;
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
// TODO: This algorithm is non-optimal.
|
||||||
|
struct CallGraphCycleFinder
|
||||||
|
{
|
||||||
|
CallGraph const& callGraph;
|
||||||
|
set<YulString> containedInCycle{};
|
||||||
|
set<YulString> visited{};
|
||||||
|
vector<YulString> currentPath{};
|
||||||
|
|
||||||
|
void visit(YulString _function)
|
||||||
|
{
|
||||||
|
if (visited.count(_function))
|
||||||
|
return;
|
||||||
|
if (
|
||||||
|
auto it = find(currentPath.begin(), currentPath.end(), _function);
|
||||||
|
it != currentPath.end()
|
||||||
|
)
|
||||||
|
containedInCycle.insert(it, currentPath.end());
|
||||||
|
else
|
||||||
|
{
|
||||||
|
currentPath.emplace_back(_function);
|
||||||
|
if (callGraph.functionCalls.count(_function))
|
||||||
|
for (auto const& child: callGraph.functionCalls.at(_function))
|
||||||
|
visit(child);
|
||||||
|
currentPath.pop_back();
|
||||||
|
visited.insert(_function);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
set<YulString> CallGraph::recursiveFunctions() const
|
||||||
|
{
|
||||||
|
CallGraphCycleFinder cycleFinder{*this};
|
||||||
|
// Visiting the root only is not enough, since there may be disconnected recursive functions.
|
||||||
|
for (auto const& call: functionCalls)
|
||||||
|
cycleFinder.visit(call.first);
|
||||||
|
return cycleFinder.containedInCycle;
|
||||||
|
}
|
||||||
|
|
||||||
CallGraph CallGraphGenerator::callGraph(Block const& _ast)
|
CallGraph CallGraphGenerator::callGraph(Block const& _ast)
|
||||||
{
|
{
|
||||||
CallGraphGenerator gen;
|
CallGraphGenerator gen;
|
||||||
|
@ -35,6 +35,10 @@ struct CallGraph
|
|||||||
{
|
{
|
||||||
std::map<YulString, std::set<YulString>> functionCalls;
|
std::map<YulString, std::set<YulString>> functionCalls;
|
||||||
std::set<YulString> functionsWithLoops;
|
std::set<YulString> functionsWithLoops;
|
||||||
|
/// @returns the set of functions contained in cycles in the call graph, i.e.
|
||||||
|
/// functions that are part of a (mutual) recursion.
|
||||||
|
/// Note that this does not include functions that merely call recursive functions.
|
||||||
|
std::set<YulString> recursiveFunctions() const;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -102,33 +102,13 @@ map<YulString, SideEffects> SideEffectsPropagator::sideEffects(
|
|||||||
// is actually a bit different from "not movable".
|
// is actually a bit different from "not movable".
|
||||||
|
|
||||||
map<YulString, SideEffects> ret;
|
map<YulString, SideEffects> ret;
|
||||||
for (auto const& function: _directCallGraph.functionsWithLoops)
|
for (auto const& function: _directCallGraph.functionsWithLoops + _directCallGraph.recursiveFunctions())
|
||||||
{
|
{
|
||||||
ret[function].movable = false;
|
ret[function].movable = false;
|
||||||
ret[function].sideEffectFree = false;
|
ret[function].sideEffectFree = false;
|
||||||
ret[function].sideEffectFreeIfNoMSize = false;
|
ret[function].sideEffectFreeIfNoMSize = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Detect recursive functions.
|
|
||||||
for (auto const& call: _directCallGraph.functionCalls)
|
|
||||||
{
|
|
||||||
// TODO we could shortcut the search as soon as we find a
|
|
||||||
// function that has as bad side-effects as we can
|
|
||||||
// ever achieve via recursion.
|
|
||||||
auto search = [&](YulString const& _functionName, util::CycleDetector<YulString>& _cycleDetector, size_t) {
|
|
||||||
for (auto const& callee: _directCallGraph.functionCalls.at(_functionName))
|
|
||||||
if (!_dialect.builtin(callee))
|
|
||||||
if (_cycleDetector.run(callee))
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
if (util::CycleDetector<YulString>(search).run(call.first))
|
|
||||||
{
|
|
||||||
ret[call.first].movable = false;
|
|
||||||
ret[call.first].sideEffectFree = false;
|
|
||||||
ret[call.first].sideEffectFreeIfNoMSize = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto const& call: _directCallGraph.functionCalls)
|
for (auto const& call: _directCallGraph.functionCalls)
|
||||||
{
|
{
|
||||||
YulString funName = call.first;
|
YulString funName = call.first;
|
||||||
|
@ -34,5 +34,6 @@ else
|
|||||||
BUILD_DIR="$1"
|
BUILD_DIR="$1"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
docker run -v $(pwd):/root/project -w /root/project ethereum/solidity-buildpack-deps:emsdk-1.39.15-2 \
|
docker run -v $(pwd):/root/project -w /root/project \
|
||||||
|
solbuildpackpusher/solidity-buildpack-deps@sha256:d557d015918c3cf68b0d22839bab41013f0757b651a7fef21595f89721dbebcc \
|
||||||
./scripts/travis-emscripten/build_emscripten.sh $BUILD_DIR
|
./scripts/travis-emscripten/build_emscripten.sh $BUILD_DIR
|
||||||
|
23
scripts/ci/build.sh
Executable file
23
scripts/ci/build.sh
Executable file
@ -0,0 +1,23 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -ex
|
||||||
|
|
||||||
|
ROOTDIR="$(dirname "$0")/../.."
|
||||||
|
cd "${ROOTDIR}"
|
||||||
|
|
||||||
|
# shellcheck disable=SC2166
|
||||||
|
if [ "$CIRCLE_BRANCH" = release -o -n "$CIRCLE_TAG" -o -n "$FORCE_RELEASE" ]; then echo -n >prerelease.txt; else date -u +"nightly.%Y.%-m.%-d" >prerelease.txt; fi
|
||||||
|
if [ -n "$CIRCLE_SHA1" ]
|
||||||
|
then
|
||||||
|
echo -n "$CIRCLE_SHA1" >commit_hash.txt
|
||||||
|
fi
|
||||||
|
|
||||||
|
mkdir -p build
|
||||||
|
cd build
|
||||||
|
|
||||||
|
# shellcheck disable=SC2166
|
||||||
|
[ -n "$COVERAGE" -a "$CIRCLE_BRANCH" != release -a -z "$CIRCLE_TAG" ] && CMAKE_OPTIONS="$CMAKE_OPTIONS -DCOVERAGE=ON"
|
||||||
|
|
||||||
|
# shellcheck disable=SC2086
|
||||||
|
cmake .. -DCMAKE_BUILD_TYPE="${CMAKE_BUILD_TYPE:-Release}" $CMAKE_OPTIONS -G "Unix Makefiles"
|
||||||
|
|
||||||
|
make
|
15
scripts/ci/build_ossfuzz.sh
Executable file
15
scripts/ci/build_ossfuzz.sh
Executable file
@ -0,0 +1,15 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
ROOTDIR="$(dirname "$0")/../.."
|
||||||
|
BUILDDIR="${ROOTDIR}/build"
|
||||||
|
|
||||||
|
mkdir -p "${BUILDDIR}"
|
||||||
|
cd "${BUILDDIR}"
|
||||||
|
|
||||||
|
protoc --proto_path=../test/tools/ossfuzz yulProto.proto --cpp_out=../test/tools/ossfuzz
|
||||||
|
protoc --proto_path=../test/tools/ossfuzz abiV2Proto.proto --cpp_out=../test/tools/ossfuzz
|
||||||
|
protoc --proto_path=../test/tools/ossfuzz solProto.proto --cpp_out=../test/tools/ossfuzz
|
||||||
|
cmake .. -DCMAKE_BUILD_TYPE="${CMAKE_BUILD_TYPE:-Release}" -DCMAKE_TOOLCHAIN_FILE=../cmake/toolchains/libfuzzer.cmake
|
||||||
|
|
||||||
|
make ossfuzz ossfuzz_proto ossfuzz_abiv2 -j 4
|
1
scripts/ci/buildpack-deps_test_emscripten.sh
Symbolic link
1
scripts/ci/buildpack-deps_test_emscripten.sh
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
../../scripts/travis-emscripten/build_emscripten.sh
|
1
scripts/ci/buildpack-deps_test_ubuntu1604.clang.ossfuzz.sh
Symbolic link
1
scripts/ci/buildpack-deps_test_ubuntu1604.clang.ossfuzz.sh
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
build_ossfuzz.sh
|
1
scripts/ci/buildpack-deps_test_ubuntu1804.sh
Symbolic link
1
scripts/ci/buildpack-deps_test_ubuntu1804.sh
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
build.sh
|
1
scripts/ci/buildpack-deps_test_ubuntu2004.clang.sh
Symbolic link
1
scripts/ci/buildpack-deps_test_ubuntu2004.clang.sh
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
build.sh
|
1
scripts/ci/buildpack-deps_test_ubuntu2004.sh
Symbolic link
1
scripts/ci/buildpack-deps_test_ubuntu2004.sh
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
build.sh
|
65
scripts/ci/docker_upgrade.sh
Executable file
65
scripts/ci/docker_upgrade.sh
Executable file
@ -0,0 +1,65 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
function error() {
|
||||||
|
echo >&2 "ERROR: ${1} Aborting." && false
|
||||||
|
}
|
||||||
|
|
||||||
|
function warning() {
|
||||||
|
echo >&2 "WARNING: ${1}"
|
||||||
|
}
|
||||||
|
|
||||||
|
[[ $# == 3 ]] || error "Expected exactly 3 parameters: '${0} <IMAGE_NAME> <IMAGE_VARIANT> <DOCKER_REPOSITORY>'."
|
||||||
|
|
||||||
|
IMAGE_NAME="${1}"
|
||||||
|
IMAGE_VARIANT="${2}"
|
||||||
|
DOCKER_REPOSITORY="${3}"
|
||||||
|
DOCKERFILE="scripts/docker/${IMAGE_NAME}/Dockerfile.${IMAGE_VARIANT}"
|
||||||
|
|
||||||
|
echo "-- check_dockerfile_was_changed"
|
||||||
|
|
||||||
|
# exit, if the dockerfile was not changed.
|
||||||
|
if git diff --quiet origin/develop HEAD -- "${DOCKERFILE}"; then
|
||||||
|
echo "${DOCKERFILE} was not changed. Nothing to do."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "-- check_version"
|
||||||
|
|
||||||
|
PREV_VERSION=$(git diff origin/develop HEAD -- "${DOCKERFILE}" | grep -e '^\s*-LABEL\s\+version=".*"\s*$' | awk -F'"' '{ print $2 }')
|
||||||
|
NEXT_VERSION=$(git diff origin/develop HEAD -- "${DOCKERFILE}" | grep -e '^\s*+LABEL\s\+version=".*"\s*$' | awk -F'"' '{ print $2 }')
|
||||||
|
|
||||||
|
[[ $NEXT_VERSION != "" ]] || error "No version label defined in Dockerfile. You may need to add 'LABEL version' in '${DOCKERFILE}'."
|
||||||
|
|
||||||
|
[[ $PREV_VERSION != "" ]] || {
|
||||||
|
warning "no previous version found. Will set \$PREV_VERSION = 0."
|
||||||
|
PREV_VERSION=0
|
||||||
|
}
|
||||||
|
|
||||||
|
if [[ $((PREV_VERSION + 1)) != $((NEXT_VERSION)) ]]; then
|
||||||
|
error "Version label in Dockerfile was not incremented. You may need to change 'LABEL version' in '${DOCKERFILE}'."
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "-- build_docker"
|
||||||
|
|
||||||
|
# This is a workaround: we run `docker build` twice to prevent the `layer does not exist` problem.
|
||||||
|
# See https://github.com/moby/moby/issues/37965.
|
||||||
|
docker build "scripts/docker/${IMAGE_NAME}" --file "scripts/docker/${IMAGE_NAME}/Dockerfile.${IMAGE_VARIANT}" --tag "${IMAGE_NAME}" ||
|
||||||
|
docker build "scripts/docker/${IMAGE_NAME}" --file "scripts/docker/${IMAGE_NAME}/Dockerfile.${IMAGE_VARIANT}" --tag "${IMAGE_NAME}"
|
||||||
|
|
||||||
|
echo "-- test_docker @ '${PWD}'"
|
||||||
|
|
||||||
|
docker run --rm --volume "${PWD}:/root/project" "${IMAGE_NAME}" "/root/project/scripts/ci/${IMAGE_NAME}_test_${IMAGE_VARIANT}.sh"
|
||||||
|
|
||||||
|
echo "-- push_docker"
|
||||||
|
|
||||||
|
VERSION=$(docker inspect --format='{{.Config.Labels.version}}' "${IMAGE_NAME}")
|
||||||
|
DOCKER_IMAGE_ID="${DOCKER_REPOSITORY}:${IMAGE_VARIANT}"
|
||||||
|
|
||||||
|
docker tag "${IMAGE_NAME}" "${DOCKER_IMAGE_ID}-${VERSION}"
|
||||||
|
docker push "${DOCKER_IMAGE_ID}-${VERSION}"
|
||||||
|
|
||||||
|
REPO_DIGEST=$(docker inspect --format='{{.RepoDigests}}' "${DOCKER_IMAGE_ID}-${VERSION}")
|
||||||
|
|
||||||
|
echo "::set-env name=DOCKER_IMAGE::${DOCKER_IMAGE_ID}-${VERSION}"
|
||||||
|
echo "::set-env name=DOCKER_REPO_DIGEST::${REPO_DIGEST}"
|
@ -29,6 +29,7 @@
|
|||||||
# make version=1.39.15 build
|
# make version=1.39.15 build
|
||||||
#
|
#
|
||||||
FROM emscripten/emsdk:1.39.15 AS base
|
FROM emscripten/emsdk:1.39.15 AS base
|
||||||
|
LABEL version="1"
|
||||||
|
|
||||||
ADD emscripten.jam /usr/src
|
ADD emscripten.jam /usr/src
|
||||||
RUN set -ex; \
|
RUN set -ex; \
|
||||||
@ -63,3 +64,4 @@ RUN set -ex; \
|
|||||||
cxxflags="-s DISABLE_EXCEPTION_CATCHING=0 -Wno-unused-local-typedef -Wno-variadic-macros -Wno-c99-extensions -Wno-all" \
|
cxxflags="-s DISABLE_EXCEPTION_CATCHING=0 -Wno-unused-local-typedef -Wno-variadic-macros -Wno-c99-extensions -Wno-all" \
|
||||||
--prefix=/emsdk/emscripten/sdk/system install; \
|
--prefix=/emsdk/emscripten/sdk/system install; \
|
||||||
rm -r /usr/src/boost_1_73_0
|
rm -r /usr/src/boost_1_73_0
|
||||||
|
|
@ -22,6 +22,7 @@
|
|||||||
# (c) 2016-2019 solidity contributors.
|
# (c) 2016-2019 solidity contributors.
|
||||||
#------------------------------------------------------------------------------
|
#------------------------------------------------------------------------------
|
||||||
FROM gcr.io/oss-fuzz-base/base-clang as base
|
FROM gcr.io/oss-fuzz-base/base-clang as base
|
||||||
|
LABEL version="1"
|
||||||
|
|
||||||
ARG DEBIAN_FRONTEND=noninteractive
|
ARG DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
@ -99,3 +100,4 @@ FROM base
|
|||||||
COPY --from=libraries /usr/lib /usr/lib
|
COPY --from=libraries /usr/lib /usr/lib
|
||||||
COPY --from=libraries /usr/bin /usr/bin
|
COPY --from=libraries /usr/bin /usr/bin
|
||||||
COPY --from=libraries /usr/include /usr/include
|
COPY --from=libraries /usr/include /usr/include
|
||||||
|
|
@ -22,6 +22,7 @@
|
|||||||
# (c) 2016-2019 solidity contributors.
|
# (c) 2016-2019 solidity contributors.
|
||||||
#------------------------------------------------------------------------------
|
#------------------------------------------------------------------------------
|
||||||
FROM buildpack-deps:bionic AS base
|
FROM buildpack-deps:bionic AS base
|
||||||
|
LABEL version="1"
|
||||||
|
|
||||||
ARG DEBIAN_FRONTEND=noninteractive
|
ARG DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
@ -91,3 +92,4 @@ FROM base
|
|||||||
COPY --from=libraries /usr/lib /usr/lib
|
COPY --from=libraries /usr/lib /usr/lib
|
||||||
COPY --from=libraries /usr/bin /usr/bin
|
COPY --from=libraries /usr/bin /usr/bin
|
||||||
COPY --from=libraries /usr/include /usr/include
|
COPY --from=libraries /usr/include /usr/include
|
||||||
|
|
@ -22,6 +22,7 @@
|
|||||||
# (c) 2016-2019 solidity contributors.
|
# (c) 2016-2019 solidity contributors.
|
||||||
#------------------------------------------------------------------------------
|
#------------------------------------------------------------------------------
|
||||||
FROM buildpack-deps:focal AS base
|
FROM buildpack-deps:focal AS base
|
||||||
|
LABEL version="1"
|
||||||
|
|
||||||
ARG DEBIAN_FRONTEND=noninteractive
|
ARG DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
@ -60,3 +61,4 @@ FROM base
|
|||||||
COPY --from=libraries /usr/lib /usr/lib
|
COPY --from=libraries /usr/lib /usr/lib
|
||||||
COPY --from=libraries /usr/bin /usr/bin
|
COPY --from=libraries /usr/bin /usr/bin
|
||||||
COPY --from=libraries /usr/include /usr/include
|
COPY --from=libraries /usr/include /usr/include
|
||||||
|
|
@ -22,6 +22,7 @@
|
|||||||
# (c) 2016-2019 solidity contributors.
|
# (c) 2016-2019 solidity contributors.
|
||||||
#------------------------------------------------------------------------------
|
#------------------------------------------------------------------------------
|
||||||
FROM buildpack-deps:focal AS base
|
FROM buildpack-deps:focal AS base
|
||||||
|
LABEL version="1"
|
||||||
|
|
||||||
ARG DEBIAN_FRONTEND=noninteractive
|
ARG DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
@ -62,3 +63,4 @@ FROM base
|
|||||||
COPY --from=libraries /usr/lib /usr/lib
|
COPY --from=libraries /usr/lib /usr/lib
|
||||||
COPY --from=libraries /usr/bin /usr/bin
|
COPY --from=libraries /usr/bin /usr/bin
|
||||||
COPY --from=libraries /usr/include /usr/include
|
COPY --from=libraries /usr/include /usr/include
|
||||||
|
|
35
scripts/docker/buildpack-deps/README.md
Normal file
35
scripts/docker/buildpack-deps/README.md
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# buildpack-deps docker images
|
||||||
|
|
||||||
|
The `buildpack-deps` docker images are used to compile and test solidity within our CI.
|
||||||
|
|
||||||
|
## GitHub Workflow
|
||||||
|
|
||||||
|
The creation of the images are triggered by a single workflow, defined in `.github/workflows/buildpack-deps.yml`.
|
||||||
|
For each resulting `buildpack-deps` docker image a strategy is defined in the workflow file - the image variant.
|
||||||
|
The workflow gets triggered, if any Dockerfile defined in `scripts/docker/buildpack-deps/Dockerfile.*` were changed
|
||||||
|
within the PR.
|
||||||
|
|
||||||
|
### Versioning
|
||||||
|
|
||||||
|
The version of the docker images can be defined within the Dockerfile with `LABEL version`. A new docker image
|
||||||
|
will only be created and pushed, if the new version is incremented by `1` compared with the version of the Dockerfile
|
||||||
|
located in `develop`.
|
||||||
|
|
||||||
|
### Build, Test & Push
|
||||||
|
|
||||||
|
Note that the whole workflow - including all defined strategies (image variants) - will be triggered,
|
||||||
|
even if only a single Dockerfile was change. The full workflow will only gets executed, if the corresponding
|
||||||
|
Dockerfile was changed. The execution of workflows of unchanged Dockerfiles will not continue and just return success.
|
||||||
|
See `scripts/ci/docker_upgrade.sh`.
|
||||||
|
|
||||||
|
If the version check was successful, the docker image will be built using the Dockerfile located in
|
||||||
|
`scripts/docker/buildpack-deps/Dockerfile.*`.
|
||||||
|
|
||||||
|
The resulting docker image will be tested by executing
|
||||||
|
the corresponding `scripts/ci/buildpack-deps_test_*` scripts. These scripts are normally symlinked to `scripts/ci/build.sh`,
|
||||||
|
except for the `buildpack-deps-ubuntu1604.clang.ossfuzz` docker image, that is symlinked to `scripts/ci/build_ossfuzz.sh`.
|
||||||
|
These scripts `scripts/ci/build.sh` and `scripts/ci/build_ossfuzz.sh` are also used by CircleCI, see `.circleci/config.yml`.
|
||||||
|
|
||||||
|
If the tests passed successfully, the docker image will get tagged by the version defined within the corresponding `Dockerfile`.
|
||||||
|
Finally, a comment will be added to the PR that contains the full repository, version and repository digest
|
||||||
|
of the freshly created docker image.
|
@ -53,7 +53,7 @@ cmake \
|
|||||||
-DBoost_USE_STATIC_RUNTIME=1 \
|
-DBoost_USE_STATIC_RUNTIME=1 \
|
||||||
-DTESTS=0 \
|
-DTESTS=0 \
|
||||||
..
|
..
|
||||||
make -j 4 soljson
|
make soljson
|
||||||
# Patch soljson.js for backwards compatibility.
|
# Patch soljson.js for backwards compatibility.
|
||||||
# TODO: remove this with 0.7.
|
# TODO: remove this with 0.7.
|
||||||
# "viiiii" encodes the signature of the callback function.
|
# "viiiii" encodes the signature of the callback function.
|
||||||
|
1
test/cmdlineTests/dup_opt_peephole/args
Normal file
1
test/cmdlineTests/dup_opt_peephole/args
Normal file
@ -0,0 +1 @@
|
|||||||
|
--asm
|
5
test/cmdlineTests/dup_opt_peephole/err
Normal file
5
test/cmdlineTests/dup_opt_peephole/err
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
Warning: SPDX license identifier not provided in source file. Before publishing, consider adding a comment containing "SPDX-License-Identifier: <SPDX-License>" to each source file. Use "SPDX-License-Identifier: UNLICENSED" for non-open-source code. Please see https://spdx.org for more information.
|
||||||
|
--> dup_opt_peephole/input.sol
|
||||||
|
|
||||||
|
Warning: Source file does not specify required compiler version!
|
||||||
|
--> dup_opt_peephole/input.sol
|
9
test/cmdlineTests/dup_opt_peephole/input.sol
Normal file
9
test/cmdlineTests/dup_opt_peephole/input.sol
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
contract C {
|
||||||
|
fallback() external {
|
||||||
|
assembly {
|
||||||
|
let x := calldataload(0)
|
||||||
|
x := x
|
||||||
|
sstore(0, x)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
54
test/cmdlineTests/dup_opt_peephole/output
Normal file
54
test/cmdlineTests/dup_opt_peephole/output
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
|
||||||
|
======= dup_opt_peephole/input.sol:C =======
|
||||||
|
EVM assembly:
|
||||||
|
/* "dup_opt_peephole/input.sol":0:111 contract C {... */
|
||||||
|
mstore(0x40, 0x80)
|
||||||
|
callvalue
|
||||||
|
dup1
|
||||||
|
iszero
|
||||||
|
tag_1
|
||||||
|
jumpi
|
||||||
|
0x00
|
||||||
|
dup1
|
||||||
|
revert
|
||||||
|
tag_1:
|
||||||
|
pop
|
||||||
|
dataSize(sub_0)
|
||||||
|
dup1
|
||||||
|
dataOffset(sub_0)
|
||||||
|
0x00
|
||||||
|
codecopy
|
||||||
|
0x00
|
||||||
|
return
|
||||||
|
stop
|
||||||
|
|
||||||
|
sub_0: assembly {
|
||||||
|
/* "dup_opt_peephole/input.sol":0:111 contract C {... */
|
||||||
|
mstore(0x40, 0x80)
|
||||||
|
callvalue
|
||||||
|
dup1
|
||||||
|
iszero
|
||||||
|
tag_3
|
||||||
|
jumpi
|
||||||
|
0x00
|
||||||
|
dup1
|
||||||
|
revert
|
||||||
|
tag_3:
|
||||||
|
pop
|
||||||
|
/* "dup_opt_peephole/input.sol":74:75 0 */
|
||||||
|
0x00
|
||||||
|
/* "dup_opt_peephole/input.sol":61:76 calldataload(0) */
|
||||||
|
calldataload
|
||||||
|
/* "dup_opt_peephole/input.sol":100:101 x */
|
||||||
|
dup1
|
||||||
|
/* "dup_opt_peephole/input.sol":97:98 0 */
|
||||||
|
0x00
|
||||||
|
/* "dup_opt_peephole/input.sol":90:102 sstore(0, x) */
|
||||||
|
sstore
|
||||||
|
/* "dup_opt_peephole/input.sol":47:106 {... */
|
||||||
|
pop
|
||||||
|
/* "dup_opt_peephole/input.sol":0:111 contract C {... */
|
||||||
|
stop
|
||||||
|
|
||||||
|
auxdata: AUXDATA REMOVED
|
||||||
|
}
|
@ -708,6 +708,19 @@ BOOST_AUTO_TEST_CASE(shift_optimizer_bug)
|
|||||||
compareVersions("g(uint256)", u256(-1));
|
compareVersions("g(uint256)", u256(-1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(avoid_double_cleanup)
|
||||||
|
{
|
||||||
|
char const* sourceCode = R"(
|
||||||
|
contract C {
|
||||||
|
receive() external payable {
|
||||||
|
abi.encodePacked(uint200(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
compileBothVersions(sourceCode, 0, "C", 50);
|
||||||
|
// Check that there is no double AND instruction in the resulting code
|
||||||
|
BOOST_CHECK_EQUAL(numInstructions(m_nonOptimizedBytecode, Instruction::AND), 1);
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_SUITE_END()
|
BOOST_AUTO_TEST_SUITE_END()
|
||||||
|
|
||||||
|
22
test/libsolidity/semanticTests/strings/unicode_escapes.sol
Normal file
22
test/libsolidity/semanticTests/strings/unicode_escapes.sol
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
contract C {
|
||||||
|
function oneByteUTF8() public pure returns (string memory) {
|
||||||
|
return "aaa\u0024aaa"; // usdollar
|
||||||
|
}
|
||||||
|
|
||||||
|
function twoBytesUTF8() public pure returns (string memory) {
|
||||||
|
return "aaa\u00A2aaa"; // cent
|
||||||
|
}
|
||||||
|
|
||||||
|
function threeBytesUTF8() public pure returns (string memory) {
|
||||||
|
return "aaa\u20ACaaa"; // euro
|
||||||
|
}
|
||||||
|
|
||||||
|
function combined() public pure returns (string memory) {
|
||||||
|
return "\u0024\u00A2\u20AC";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// oneByteUTF8() -> 0x20, 7, "aaa$aaa"
|
||||||
|
// twoBytesUTF8() -> 0x20, 8, "aaa\xc2\xa2aaa"
|
||||||
|
// threeBytesUTF8() -> 0x20, 9, "aaa\xe2\x82\xacaaa"
|
||||||
|
// combined() -> 0x20, 6, "$\xc2\xa2\xe2\x82\xac"
|
@ -0,0 +1,7 @@
|
|||||||
|
contract C {
|
||||||
|
function f() public pure returns (string memory) {
|
||||||
|
return "😃, 😭, and 😈";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// f() -> 0x20, 0x14, "\xf0\x9f\x98\x83, \xf0\x9f\x98\xad, and \xf0\x9f\x98\x88"
|
@ -0,0 +1,8 @@
|
|||||||
|
pragma experimental SMTChecker;
|
||||||
|
contract C {
|
||||||
|
function f() public pure returns (byte) {
|
||||||
|
return (byte("") & (""));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Warning 5084: (101-109): Type conversion is not yet fully supported and might yield false positives.
|
@ -0,0 +1,9 @@
|
|||||||
|
pragma experimental SMTChecker;
|
||||||
|
contract Simp {
|
||||||
|
function f3() public pure returns (byte) {
|
||||||
|
bytes memory y = "def";
|
||||||
|
return y[0] ^ "e";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Warning 1093: (142-152): Assertion checker does not yet implement this bitwise operator.
|
@ -0,0 +1,15 @@
|
|||||||
|
contract b {
|
||||||
|
struct c {
|
||||||
|
uint [2 ** 253] a;
|
||||||
|
}
|
||||||
|
|
||||||
|
c d;
|
||||||
|
function e() public {
|
||||||
|
var d = d;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Warning 2519: (105-110): This declaration shadows an existing declaration.
|
||||||
|
// Warning 3408: (66-69): Variable "d" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
||||||
|
// Warning 2332: (105-110): Type "b.c" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
||||||
|
// SyntaxError 1719: (105-114): Use of the "var" keyword is disallowed. Use explicit declaration `struct b.c storage pointer d = ...´ instead.
|
@ -4,5 +4,5 @@ contract C {
|
|||||||
uint[2**255][2] a;
|
uint[2**255][2] a;
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError 1534: (77-94): Type too large for storage.
|
|
||||||
// TypeError 7676: (60-97): Contract too large for storage.
|
// TypeError 7676: (60-97): Contract too large for storage.
|
||||||
|
// TypeError 1534: (77-94): Type too large for storage.
|
||||||
|
@ -5,6 +5,6 @@ contract C {
|
|||||||
uint[2**255] b;
|
uint[2**255] b;
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
|
// TypeError 7676: (60-114): Contract too large for storage.
|
||||||
// Warning 3408: (77-91): Variable "a" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
// Warning 3408: (77-91): Variable "a" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
||||||
// Warning 3408: (97-111): Variable "b" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
// Warning 3408: (97-111): Variable "b" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
||||||
// TypeError 7676: (60-114): Contract too large for storage.
|
|
||||||
|
@ -0,0 +1,12 @@
|
|||||||
|
// SPDX-License-Identifier: GPL-3.0
|
||||||
|
pragma solidity >= 0.0;
|
||||||
|
contract C {
|
||||||
|
uint[2**255] a;
|
||||||
|
}
|
||||||
|
contract D is C {
|
||||||
|
uint[2**255] b;
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 7676: (95-134): Contract too large for storage.
|
||||||
|
// Warning 3408: (77-91): Variable "a" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
||||||
|
// Warning 3408: (117-131): Variable "b" covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
|
@ -8,5 +8,5 @@ contract C {
|
|||||||
S s;
|
S s;
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError 1534: (146-149): Type too large for storage.
|
|
||||||
// TypeError 7676: (60-152): Contract too large for storage.
|
// TypeError 7676: (60-152): Contract too large for storage.
|
||||||
|
// TypeError 1534: (146-149): Type too large for storage.
|
||||||
|
@ -2,4 +2,4 @@ contract test {
|
|||||||
function f() payable internal {}
|
function f() payable internal {}
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError 5587: (20-52): Internal functions cannot be payable.
|
// TypeError 5587: (20-52): "internal" and "private" functions cannot be payable.
|
||||||
|
@ -2,4 +2,4 @@ contract test {
|
|||||||
function f() payable private {}
|
function f() payable private {}
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError 5587: (20-51): Internal functions cannot be payable.
|
// TypeError 5587: (20-51): "internal" and "private" functions cannot be payable.
|
||||||
|
@ -0,0 +1,5 @@
|
|||||||
|
contract C {
|
||||||
|
string s = hex"a000";
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 7407: (28-37): Type literal_string (contains invalid UTF-8 sequence at position 0) is not implicitly convertible to expected type string storage ref.
|
@ -0,0 +1,6 @@
|
|||||||
|
contract C {
|
||||||
|
bytes b1 = "\xa0\x00";
|
||||||
|
bytes32 b2 = "\xa0\x00";
|
||||||
|
bytes b3 = hex"a000";
|
||||||
|
}
|
||||||
|
// ----
|
6
test/libsolidity/syntaxTests/string/string_unicode.sol
Normal file
6
test/libsolidity/syntaxTests/string/string_unicode.sol
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
contract test {
|
||||||
|
function f() public pure returns (string memory) {
|
||||||
|
return "😃, 😭, and 😈";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
@ -1,5 +1,4 @@
|
|||||||
contract test {
|
contract test {
|
||||||
|
|
||||||
function oneByteUTF8() public pure returns (bytes32) {
|
function oneByteUTF8() public pure returns (bytes32) {
|
||||||
bytes32 usdollar = "aaa\u0024aaa";
|
bytes32 usdollar = "aaa\u0024aaa";
|
||||||
return usdollar;
|
return usdollar;
|
||||||
@ -15,17 +14,8 @@ contract test {
|
|||||||
return eur;
|
return eur;
|
||||||
}
|
}
|
||||||
|
|
||||||
function together() public pure returns (bytes32) {
|
function combined() public pure returns (bytes32) {
|
||||||
bytes32 res = "\u0024\u00A2\u20AC";
|
bytes32 res = "\u0024\u00A2\u20AC";
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
// this function returns an invalid unicode character
|
|
||||||
function invalidLiteral() public pure returns(bytes32) {
|
|
||||||
bytes32 invalid = "\u00xx";
|
|
||||||
return invalid;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
// ----
|
|
||||||
// ParserError 8936: (678-681): Invalid escape sequence.
|
|
@ -0,0 +1,7 @@
|
|||||||
|
contract test {
|
||||||
|
function f() public pure returns (string memory) {
|
||||||
|
return "\xc1";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError 6359: (86-92): Return argument type literal_string (contains invalid UTF-8 sequence at position 0) is not implicitly convertible to expected type (type of first return variable) string memory.
|
@ -0,0 +1,10 @@
|
|||||||
|
contract test {
|
||||||
|
// this function returns an invalid unicode character
|
||||||
|
function invalidLiteral() public pure returns (bytes32) {
|
||||||
|
bytes32 invalid = "\u00xx";
|
||||||
|
return invalid;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// ParserError 8936: (162-165): Invalid escape sequence.
|
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
#include <liblangutil/Common.h>
|
#include <liblangutil/Common.h>
|
||||||
|
|
||||||
|
#include <libsolutil/CommonData.h>
|
||||||
#include <libsolutil/StringUtils.h>
|
#include <libsolutil/StringUtils.h>
|
||||||
|
|
||||||
#include <boost/algorithm/string.hpp>
|
#include <boost/algorithm/string.hpp>
|
||||||
@ -198,8 +199,7 @@ string BytesUtils::formatString(bytes const& _bytes, size_t _cutOff)
|
|||||||
if (isprint(v))
|
if (isprint(v))
|
||||||
os << v;
|
os << v;
|
||||||
else
|
else
|
||||||
os << "\\x" << setw(2) << setfill('0') << hex << v;
|
os << "\\x" << toHex(v);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
os << "\"";
|
os << "\"";
|
||||||
|
@ -99,11 +99,9 @@ string EwasmTranslationTest::interpret()
|
|||||||
InterpreterState state;
|
InterpreterState state;
|
||||||
state.maxTraceSize = 10000;
|
state.maxTraceSize = 10000;
|
||||||
state.maxSteps = 100000;
|
state.maxSteps = 100000;
|
||||||
WasmDialect dialect;
|
|
||||||
Interpreter interpreter(state, dialect);
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
interpreter(*m_object->code);
|
Interpreter::run(state, WasmDialect{}, *m_object->code);
|
||||||
}
|
}
|
||||||
catch (InterpreterTerminatedGeneric const&)
|
catch (InterpreterTerminatedGeneric const&)
|
||||||
{
|
{
|
||||||
|
@ -345,6 +345,69 @@ BOOST_AUTO_TEST_CASE(reuse_slots_function_with_gaps)
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(reuse_on_decl_assign_to_last_used)
|
||||||
|
{
|
||||||
|
string in = R"({
|
||||||
|
let x := 5
|
||||||
|
let y := x // y should reuse the stack slot of x
|
||||||
|
sstore(y, y)
|
||||||
|
})";
|
||||||
|
BOOST_CHECK_EQUAL(assemble(in),
|
||||||
|
"PUSH1 0x5 "
|
||||||
|
"DUP1 SWAP1 POP "
|
||||||
|
"DUP1 DUP2 SSTORE "
|
||||||
|
"POP "
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(reuse_on_decl_assign_to_last_used_expr)
|
||||||
|
{
|
||||||
|
string in = R"({
|
||||||
|
let x := 5
|
||||||
|
let y := add(x, 2) // y should reuse the stack slot of x
|
||||||
|
sstore(y, y)
|
||||||
|
})";
|
||||||
|
BOOST_CHECK_EQUAL(assemble(in),
|
||||||
|
"PUSH1 0x5 "
|
||||||
|
"PUSH1 0x2 DUP2 ADD "
|
||||||
|
"SWAP1 POP "
|
||||||
|
"DUP1 DUP2 SSTORE "
|
||||||
|
"POP "
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(reuse_on_decl_assign_to_not_last_used)
|
||||||
|
{
|
||||||
|
string in = R"({
|
||||||
|
let x := 5
|
||||||
|
let y := x // y should not reuse the stack slot of x, since x is still used below
|
||||||
|
sstore(y, x)
|
||||||
|
})";
|
||||||
|
BOOST_CHECK_EQUAL(assemble(in),
|
||||||
|
"PUSH1 0x5 "
|
||||||
|
"DUP1 "
|
||||||
|
"DUP2 DUP2 SSTORE "
|
||||||
|
"POP POP "
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(reuse_on_decl_assign_not_same_scope)
|
||||||
|
{
|
||||||
|
string in = R"({
|
||||||
|
let x := 5
|
||||||
|
{
|
||||||
|
let y := x // y should not reuse the stack slot of x, since x is not in the same scope
|
||||||
|
sstore(y, y)
|
||||||
|
}
|
||||||
|
})";
|
||||||
|
BOOST_CHECK_EQUAL(assemble(in),
|
||||||
|
"PUSH1 0x5 "
|
||||||
|
"DUP1 "
|
||||||
|
"DUP1 DUP2 SSTORE "
|
||||||
|
"POP POP "
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
BOOST_AUTO_TEST_SUITE_END()
|
BOOST_AUTO_TEST_SUITE_END()
|
||||||
|
|
||||||
|
@ -88,10 +88,9 @@ string YulInterpreterTest::interpret()
|
|||||||
InterpreterState state;
|
InterpreterState state;
|
||||||
state.maxTraceSize = 10000;
|
state.maxTraceSize = 10000;
|
||||||
state.maxSteps = 10000;
|
state.maxSteps = 10000;
|
||||||
Interpreter interpreter(state, EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}));
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
interpreter(*m_ast);
|
Interpreter::run(state, EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}), *m_ast);
|
||||||
}
|
}
|
||||||
catch (InterpreterTerminatedGeneric const&)
|
catch (InterpreterTerminatedGeneric const&)
|
||||||
{
|
{
|
||||||
|
@ -44,12 +44,11 @@ yulFuzzerUtil::TerminationReason yulFuzzerUtil::interpret(
|
|||||||
0xc7, 0x60, 0x5f, 0x7c, 0xcd, 0xfb, 0x92, 0xcd,
|
0xc7, 0x60, 0x5f, 0x7c, 0xcd, 0xfb, 0x92, 0xcd,
|
||||||
0x8e, 0xf3, 0x9b, 0xe4, 0x4f, 0x6c, 0x14, 0xde
|
0x8e, 0xf3, 0x9b, 0xe4, 0x4f, 0x6c, 0x14, 0xde
|
||||||
};
|
};
|
||||||
Interpreter interpreter(state, _dialect);
|
|
||||||
|
|
||||||
TerminationReason reason = TerminationReason::None;
|
TerminationReason reason = TerminationReason::None;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
interpreter(*_ast);
|
Interpreter::run(state, _dialect, *_ast);
|
||||||
}
|
}
|
||||||
catch (StepLimitReached const&)
|
catch (StepLimitReached const&)
|
||||||
{
|
{
|
||||||
|
@ -409,17 +409,17 @@ u256 EVMInstructionInterpreter::eval(
|
|||||||
case Instruction::SWAP14:
|
case Instruction::SWAP14:
|
||||||
case Instruction::SWAP15:
|
case Instruction::SWAP15:
|
||||||
case Instruction::SWAP16:
|
case Instruction::SWAP16:
|
||||||
// --------------- EVM 2.0 ---------------
|
// --------------- EIP-615 ---------------
|
||||||
case Instruction::JUMPTO:
|
case Instruction::EIP615_JUMPTO:
|
||||||
case Instruction::JUMPIF:
|
case Instruction::EIP615_JUMPIF:
|
||||||
case Instruction::JUMPV:
|
case Instruction::EIP615_JUMPV:
|
||||||
case Instruction::JUMPSUB:
|
case Instruction::EIP615_JUMPSUB:
|
||||||
case Instruction::JUMPSUBV:
|
case Instruction::EIP615_JUMPSUBV:
|
||||||
case Instruction::BEGINSUB:
|
case Instruction::EIP615_BEGINSUB:
|
||||||
case Instruction::BEGINDATA:
|
case Instruction::EIP615_BEGINDATA:
|
||||||
case Instruction::RETURNSUB:
|
case Instruction::EIP615_RETURNSUB:
|
||||||
case Instruction::PUTLOCAL:
|
case Instruction::EIP615_PUTLOCAL:
|
||||||
case Instruction::GETLOCAL:
|
case Instruction::EIP615_GETLOCAL:
|
||||||
{
|
{
|
||||||
yulAssert(false, "");
|
yulAssert(false, "");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -67,6 +67,50 @@ uint64_t clz64(uint64_t _v)
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Count trailing zeros for uint32. Following WebAssembly rules, it returns 32 for @a _v being zero.
|
||||||
|
/// NOTE: the ctz builtin of the compiler may or may not do this
|
||||||
|
uint32_t ctz32(uint32_t _v)
|
||||||
|
{
|
||||||
|
if (_v == 0)
|
||||||
|
return 32;
|
||||||
|
|
||||||
|
uint32_t r = 0;
|
||||||
|
while (!(_v & 1))
|
||||||
|
{
|
||||||
|
r++;
|
||||||
|
_v >>= 1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Count trailing zeros for uint64. Following WebAssembly rules, it returns 64 for @a _v being zero.
|
||||||
|
/// NOTE: the ctz builtin of the compiler may or may not do this
|
||||||
|
uint64_t ctz64(uint64_t _v)
|
||||||
|
{
|
||||||
|
if (_v == 0)
|
||||||
|
return 64;
|
||||||
|
|
||||||
|
uint64_t r = 0;
|
||||||
|
while (!(_v & 1))
|
||||||
|
{
|
||||||
|
r++;
|
||||||
|
_v >>= 1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Count number of bits set for uint64
|
||||||
|
uint64_t popcnt(uint64_t _v)
|
||||||
|
{
|
||||||
|
uint64_t r = 0;
|
||||||
|
while (_v)
|
||||||
|
{
|
||||||
|
r += (_v & 1);
|
||||||
|
_v >>= 1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
using u512 = boost::multiprecision::number<boost::multiprecision::cpp_int_backend<512, 256, boost::multiprecision::unsigned_magnitude, boost::multiprecision::unchecked, void>>;
|
using u512 = boost::multiprecision::number<boost::multiprecision::cpp_int_backend<512, 256, boost::multiprecision::unsigned_magnitude, boost::multiprecision::unchecked, void>>;
|
||||||
@ -77,11 +121,12 @@ u256 EwasmBuiltinInterpreter::evalBuiltin(YulString _fun, vector<u256> const& _a
|
|||||||
for (u256 const& a: _arguments)
|
for (u256 const& a: _arguments)
|
||||||
arg.emplace_back(uint64_t(a & uint64_t(-1)));
|
arg.emplace_back(uint64_t(a & uint64_t(-1)));
|
||||||
|
|
||||||
if (_fun == "datasize"_yulstring)
|
string fun = _fun.str();
|
||||||
|
if (fun == "datasize")
|
||||||
return u256(keccak256(h256(_arguments.at(0)))) & 0xfff;
|
return u256(keccak256(h256(_arguments.at(0)))) & 0xfff;
|
||||||
else if (_fun == "dataoffset"_yulstring)
|
else if (fun == "dataoffset")
|
||||||
return u256(keccak256(h256(_arguments.at(0) + 2))) & 0xfff;
|
return u256(keccak256(h256(_arguments.at(0) + 2))) & 0xfff;
|
||||||
else if (_fun == "datacopy"_yulstring)
|
else if (fun == "datacopy")
|
||||||
{
|
{
|
||||||
// This is identical to codecopy.
|
// This is identical to codecopy.
|
||||||
if (accessMemory(_arguments.at(0), _arguments.at(2)))
|
if (accessMemory(_arguments.at(0), _arguments.at(2)))
|
||||||
@ -94,42 +139,42 @@ u256 EwasmBuiltinInterpreter::evalBuiltin(YulString _fun, vector<u256> const& _a
|
|||||||
);
|
);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else if (_fun == "i32.drop"_yulstring || _fun == "i64.drop"_yulstring || _fun == "nop"_yulstring)
|
else if (fun == "i32.drop" || fun == "i64.drop" || fun == "nop")
|
||||||
return {};
|
return {};
|
||||||
else if (_fun == "i32.wrap_i64"_yulstring)
|
else if (fun == "i32.wrap_i64")
|
||||||
return arg.at(0) & uint32_t(-1);
|
return arg.at(0) & uint32_t(-1);
|
||||||
else if (_fun == "i64.extend_i32_u"_yulstring)
|
else if (fun == "i64.extend_i32_u")
|
||||||
// Return the same as above because everything is u256 anyway.
|
// Return the same as above because everything is u256 anyway.
|
||||||
return arg.at(0) & uint32_t(-1);
|
return arg.at(0) & uint32_t(-1);
|
||||||
else if (_fun == "unreachable"_yulstring)
|
else if (fun == "unreachable")
|
||||||
{
|
{
|
||||||
logTrace(evmasm::Instruction::INVALID, {});
|
logTrace(evmasm::Instruction::INVALID, {});
|
||||||
throw ExplicitlyTerminated();
|
throw ExplicitlyTerminated();
|
||||||
}
|
}
|
||||||
else if (_fun == "i64.store"_yulstring)
|
else if (fun == "i64.store")
|
||||||
{
|
{
|
||||||
accessMemory(arg[0], 8);
|
accessMemory(arg[0], 8);
|
||||||
writeMemoryWord(arg[0], arg[1]);
|
writeMemoryWord(arg[0], arg[1]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else if (_fun == "i64.store8"_yulstring || _fun == "i32.store8"_yulstring)
|
else if (fun == "i64.store8" || fun == "i32.store8")
|
||||||
{
|
{
|
||||||
accessMemory(arg[0], 1);
|
accessMemory(arg[0], 1);
|
||||||
writeMemoryByte(arg[0], static_cast<uint8_t>(arg[1] & 0xff));
|
writeMemoryByte(arg[0], static_cast<uint8_t>(arg[1] & 0xff));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else if (_fun == "i64.load"_yulstring)
|
else if (fun == "i64.load")
|
||||||
{
|
{
|
||||||
accessMemory(arg[0], 8);
|
accessMemory(arg[0], 8);
|
||||||
return readMemoryWord(arg[0]);
|
return readMemoryWord(arg[0]);
|
||||||
}
|
}
|
||||||
else if (_fun == "i32.store"_yulstring)
|
else if (fun == "i32.store")
|
||||||
{
|
{
|
||||||
accessMemory(arg[0], 4);
|
accessMemory(arg[0], 4);
|
||||||
writeMemoryHalfWord(arg[0], arg[1]);
|
writeMemoryHalfWord(arg[0], arg[1]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else if (_fun == "i32.load"_yulstring)
|
else if (fun == "i32.load")
|
||||||
{
|
{
|
||||||
accessMemory(arg[0], 4);
|
accessMemory(arg[0], 4);
|
||||||
return readMemoryHalfWord(arg[0]);
|
return readMemoryHalfWord(arg[0]);
|
||||||
@ -139,8 +184,12 @@ u256 EwasmBuiltinInterpreter::evalBuiltin(YulString _fun, vector<u256> const& _a
|
|||||||
return clz64(arg[0] & uint32_t(-1)) - 32;
|
return clz64(arg[0] & uint32_t(-1)) - 32;
|
||||||
else if (_fun == "i64.clz"_yulstring)
|
else if (_fun == "i64.clz"_yulstring)
|
||||||
return clz64(arg[0]);
|
return clz64(arg[0]);
|
||||||
|
else if (_fun == "i32.ctz"_yulstring)
|
||||||
|
return ctz32(uint32_t(arg[0] & uint32_t(-1)));
|
||||||
|
else if (_fun == "i64.ctz"_yulstring)
|
||||||
|
return ctz64(arg[0]);
|
||||||
|
|
||||||
string prefix = _fun.str();
|
string prefix = fun;
|
||||||
string suffix;
|
string suffix;
|
||||||
auto dot = prefix.find(".");
|
auto dot = prefix.find(".");
|
||||||
if (dot != string::npos)
|
if (dot != string::npos)
|
||||||
@ -207,6 +256,8 @@ u256 EwasmBuiltinInterpreter::evalWasmBuiltin(string const& _fun, vector<Word> c
|
|||||||
return arg[0] != arg[1] ? 1 : 0;
|
return arg[0] != arg[1] ? 1 : 0;
|
||||||
else if (_fun == "eqz")
|
else if (_fun == "eqz")
|
||||||
return arg[0] == 0 ? 1 : 0;
|
return arg[0] == 0 ? 1 : 0;
|
||||||
|
else if (_fun == "popcnt")
|
||||||
|
return popcnt(arg[0]);
|
||||||
else if (_fun == "lt_u")
|
else if (_fun == "lt_u")
|
||||||
return arg[0] < arg[1] ? 1 : 0;
|
return arg[0] < arg[1] ? 1 : 0;
|
||||||
else if (_fun == "gt_u")
|
else if (_fun == "gt_u")
|
||||||
|
@ -64,6 +64,12 @@ void InterpreterState::dumpTraceAndState(ostream& _out) const
|
|||||||
_out << " " << slot.first.hex() << ": " << slot.second.hex() << endl;
|
_out << " " << slot.first.hex() << ": " << slot.second.hex() << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Interpreter::run(InterpreterState& _state, Dialect const& _dialect, Block const& _ast)
|
||||||
|
{
|
||||||
|
Scope scope;
|
||||||
|
Interpreter{_state, _dialect, scope}(_ast);
|
||||||
|
}
|
||||||
|
|
||||||
void Interpreter::operator()(ExpressionStatement const& _expressionStatement)
|
void Interpreter::operator()(ExpressionStatement const& _expressionStatement)
|
||||||
{
|
{
|
||||||
evaluateMulti(_expressionStatement.expression);
|
evaluateMulti(_expressionStatement.expression);
|
||||||
@ -94,8 +100,7 @@ void Interpreter::operator()(VariableDeclaration const& _declaration)
|
|||||||
YulString varName = _declaration.variables.at(i).name;
|
YulString varName = _declaration.variables.at(i).name;
|
||||||
solAssert(!m_variables.count(varName), "");
|
solAssert(!m_variables.count(varName), "");
|
||||||
m_variables[varName] = values.at(i);
|
m_variables[varName] = values.at(i);
|
||||||
solAssert(!m_scopes.back().count(varName), "");
|
m_scope->names.emplace(varName, nullptr);
|
||||||
m_scopes.back().emplace(varName, nullptr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -128,8 +133,8 @@ void Interpreter::operator()(ForLoop const& _forLoop)
|
|||||||
{
|
{
|
||||||
solAssert(_forLoop.condition, "");
|
solAssert(_forLoop.condition, "");
|
||||||
|
|
||||||
openScope();
|
enterScope(_forLoop.pre);
|
||||||
ScopeGuard g([this]{ closeScope(); });
|
ScopeGuard g([this]{ leaveScope(); });
|
||||||
|
|
||||||
for (auto const& statement: _forLoop.pre.statements)
|
for (auto const& statement: _forLoop.pre.statements)
|
||||||
{
|
{
|
||||||
@ -176,14 +181,13 @@ void Interpreter::operator()(Block const& _block)
|
|||||||
m_state.trace.emplace_back("Interpreter execution step limit reached.");
|
m_state.trace.emplace_back("Interpreter execution step limit reached.");
|
||||||
throw StepLimitReached();
|
throw StepLimitReached();
|
||||||
}
|
}
|
||||||
openScope();
|
enterScope(_block);
|
||||||
// Register functions.
|
// Register functions.
|
||||||
for (auto const& statement: _block.statements)
|
for (auto const& statement: _block.statements)
|
||||||
if (holds_alternative<FunctionDefinition>(statement))
|
if (holds_alternative<FunctionDefinition>(statement))
|
||||||
{
|
{
|
||||||
FunctionDefinition const& funDef = std::get<FunctionDefinition>(statement);
|
FunctionDefinition const& funDef = std::get<FunctionDefinition>(statement);
|
||||||
solAssert(!m_scopes.back().count(funDef.name), "");
|
m_scope->names.emplace(funDef.name, &funDef);
|
||||||
m_scopes.back().emplace(funDef.name, &funDef);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto const& statement: _block.statements)
|
for (auto const& statement: _block.statements)
|
||||||
@ -193,29 +197,41 @@ void Interpreter::operator()(Block const& _block)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
closeScope();
|
leaveScope();
|
||||||
}
|
}
|
||||||
|
|
||||||
u256 Interpreter::evaluate(Expression const& _expression)
|
u256 Interpreter::evaluate(Expression const& _expression)
|
||||||
{
|
{
|
||||||
ExpressionEvaluator ev(m_state, m_dialect, m_variables, m_scopes);
|
ExpressionEvaluator ev(m_state, m_dialect, *m_scope, m_variables);
|
||||||
ev.visit(_expression);
|
ev.visit(_expression);
|
||||||
return ev.value();
|
return ev.value();
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<u256> Interpreter::evaluateMulti(Expression const& _expression)
|
vector<u256> Interpreter::evaluateMulti(Expression const& _expression)
|
||||||
{
|
{
|
||||||
ExpressionEvaluator ev(m_state, m_dialect, m_variables, m_scopes);
|
ExpressionEvaluator ev(m_state, m_dialect, *m_scope, m_variables);
|
||||||
ev.visit(_expression);
|
ev.visit(_expression);
|
||||||
return ev.values();
|
return ev.values();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Interpreter::closeScope()
|
void Interpreter::enterScope(Block const& _block)
|
||||||
{
|
{
|
||||||
for (auto const& [var, funDeclaration]: m_scopes.back())
|
if (!m_scope->subScopes.count(&_block))
|
||||||
|
m_scope->subScopes[&_block] = make_unique<Scope>(Scope{
|
||||||
|
{},
|
||||||
|
{},
|
||||||
|
m_scope
|
||||||
|
});
|
||||||
|
m_scope = m_scope->subScopes[&_block].get();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interpreter::leaveScope()
|
||||||
|
{
|
||||||
|
for (auto const& [var, funDeclaration]: m_scope->names)
|
||||||
if (!funDeclaration)
|
if (!funDeclaration)
|
||||||
solAssert(m_variables.erase(var) == 1, "");
|
m_variables.erase(var);
|
||||||
m_scopes.pop_back();
|
m_scope = m_scope->parent;
|
||||||
|
yulAssert(m_scope, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExpressionEvaluator::operator()(Literal const& _literal)
|
void ExpressionEvaluator::operator()(Literal const& _literal)
|
||||||
@ -253,10 +269,15 @@ void ExpressionEvaluator::operator()(FunctionCall const& _funCall)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto [functionScopes, fun] = findFunctionAndScope(_funCall.functionName.name);
|
Scope* scope = &m_scope;
|
||||||
|
for (; scope; scope = scope->parent)
|
||||||
|
if (scope->names.count(_funCall.functionName.name))
|
||||||
|
break;
|
||||||
|
yulAssert(scope, "");
|
||||||
|
|
||||||
solAssert(fun, "Function not found.");
|
FunctionDefinition const* fun = scope->names.at(_funCall.functionName.name);
|
||||||
solAssert(m_values.size() == fun->parameters.size(), "");
|
yulAssert(fun, "Function not found.");
|
||||||
|
yulAssert(m_values.size() == fun->parameters.size(), "");
|
||||||
map<YulString, u256> variables;
|
map<YulString, u256> variables;
|
||||||
for (size_t i = 0; i < fun->parameters.size(); ++i)
|
for (size_t i = 0; i < fun->parameters.size(); ++i)
|
||||||
variables[fun->parameters.at(i).name] = m_values.at(i);
|
variables[fun->parameters.at(i).name] = m_values.at(i);
|
||||||
@ -264,7 +285,7 @@ void ExpressionEvaluator::operator()(FunctionCall const& _funCall)
|
|||||||
variables[fun->returnVariables.at(i).name] = 0;
|
variables[fun->returnVariables.at(i).name] = 0;
|
||||||
|
|
||||||
m_state.controlFlowState = ControlFlowState::Default;
|
m_state.controlFlowState = ControlFlowState::Default;
|
||||||
Interpreter interpreter(m_state, m_dialect, variables, functionScopes);
|
Interpreter interpreter(m_state, m_dialect, *scope, std::move(variables));
|
||||||
interpreter(fun->body);
|
interpreter(fun->body);
|
||||||
m_state.controlFlowState = ControlFlowState::Default;
|
m_state.controlFlowState = ControlFlowState::Default;
|
||||||
|
|
||||||
@ -297,27 +318,3 @@ void ExpressionEvaluator::evaluateArgs(vector<Expression> const& _expr)
|
|||||||
m_values = std::move(values);
|
m_values = std::move(values);
|
||||||
std::reverse(m_values.begin(), m_values.end());
|
std::reverse(m_values.begin(), m_values.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
pair<
|
|
||||||
vector<map<YulString, FunctionDefinition const*>>,
|
|
||||||
FunctionDefinition const*
|
|
||||||
> ExpressionEvaluator::findFunctionAndScope(YulString _functionName) const
|
|
||||||
{
|
|
||||||
FunctionDefinition const* fun = nullptr;
|
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>> newScopes;
|
|
||||||
for (auto const& scope: m_scopes)
|
|
||||||
{
|
|
||||||
// Copy over all functions.
|
|
||||||
newScopes.emplace_back();
|
|
||||||
for (auto const& [name, funDef]: scope)
|
|
||||||
if (funDef)
|
|
||||||
newScopes.back().emplace(name, funDef);
|
|
||||||
// Stop at the called function.
|
|
||||||
if (scope.count(_functionName))
|
|
||||||
{
|
|
||||||
fun = scope.at(_functionName);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {move(newScopes), fun};
|
|
||||||
}
|
|
||||||
|
@ -96,23 +96,37 @@ struct InterpreterState
|
|||||||
void dumpTraceAndState(std::ostream& _out) const;
|
void dumpTraceAndState(std::ostream& _out) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scope structure built and maintained during execution.
|
||||||
|
*/
|
||||||
|
struct Scope
|
||||||
|
{
|
||||||
|
/// Used for variables and functions. Value is nullptr for variables.
|
||||||
|
std::map<YulString, FunctionDefinition const*> names;
|
||||||
|
std::map<Block const*, std::unique_ptr<Scope>> subScopes;
|
||||||
|
Scope* parent = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Yul interpreter.
|
* Yul interpreter.
|
||||||
*/
|
*/
|
||||||
class Interpreter: public ASTWalker
|
class Interpreter: public ASTWalker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
static void run(InterpreterState& _state, Dialect const& _dialect, Block const& _ast);
|
||||||
|
|
||||||
Interpreter(
|
Interpreter(
|
||||||
InterpreterState& _state,
|
InterpreterState& _state,
|
||||||
Dialect const& _dialect,
|
Dialect const& _dialect,
|
||||||
std::map<YulString, u256> _variables = {},
|
Scope& _scope,
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>> _scopes = {}
|
std::map<YulString, u256> _variables = {}
|
||||||
):
|
):
|
||||||
m_dialect(_dialect),
|
m_dialect(_dialect),
|
||||||
m_state(_state),
|
m_state(_state),
|
||||||
m_variables(std::move(_variables)),
|
m_variables(std::move(_variables)),
|
||||||
m_scopes(std::move(_scopes))
|
m_scope(&_scope)
|
||||||
{}
|
{
|
||||||
|
}
|
||||||
|
|
||||||
void operator()(ExpressionStatement const& _statement) override;
|
void operator()(ExpressionStatement const& _statement) override;
|
||||||
void operator()(Assignment const& _assignment) override;
|
void operator()(Assignment const& _assignment) override;
|
||||||
@ -136,18 +150,14 @@ private:
|
|||||||
/// Evaluates the expression and returns its value.
|
/// Evaluates the expression and returns its value.
|
||||||
std::vector<u256> evaluateMulti(Expression const& _expression);
|
std::vector<u256> evaluateMulti(Expression const& _expression);
|
||||||
|
|
||||||
void openScope() { m_scopes.emplace_back(); }
|
void enterScope(Block const& _block);
|
||||||
/// Unregisters variables and functions.
|
void leaveScope();
|
||||||
void closeScope();
|
|
||||||
|
|
||||||
Dialect const& m_dialect;
|
Dialect const& m_dialect;
|
||||||
InterpreterState& m_state;
|
InterpreterState& m_state;
|
||||||
/// Values of variables.
|
/// Values of variables.
|
||||||
std::map<YulString, u256> m_variables;
|
std::map<YulString, u256> m_variables;
|
||||||
/// Scopes of variables and functions. Used for lookup, clearing at end of blocks
|
Scope* m_scope;
|
||||||
/// and passing over the visible functions across function calls.
|
|
||||||
/// The pointer is nullptr if and only if the key is a variable.
|
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>> m_scopes;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -159,13 +169,13 @@ public:
|
|||||||
ExpressionEvaluator(
|
ExpressionEvaluator(
|
||||||
InterpreterState& _state,
|
InterpreterState& _state,
|
||||||
Dialect const& _dialect,
|
Dialect const& _dialect,
|
||||||
std::map<YulString, u256> const& _variables,
|
Scope& _scope,
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>> const& _scopes
|
std::map<YulString, u256> const& _variables
|
||||||
):
|
):
|
||||||
m_state(_state),
|
m_state(_state),
|
||||||
m_dialect(_dialect),
|
m_dialect(_dialect),
|
||||||
m_variables(_variables),
|
m_variables(_variables),
|
||||||
m_scopes(_scopes)
|
m_scope(_scope)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
void operator()(Literal const&) override;
|
void operator()(Literal const&) override;
|
||||||
@ -184,19 +194,11 @@ private:
|
|||||||
/// stores it in m_value.
|
/// stores it in m_value.
|
||||||
void evaluateArgs(std::vector<Expression> const& _expr);
|
void evaluateArgs(std::vector<Expression> const& _expr);
|
||||||
|
|
||||||
/// Finds the function called @a _functionName in the current scope stack and returns
|
|
||||||
/// the function's scope stack (with variables removed) and definition.
|
|
||||||
std::pair<
|
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>>,
|
|
||||||
FunctionDefinition const*
|
|
||||||
> findFunctionAndScope(YulString _functionName) const;
|
|
||||||
|
|
||||||
InterpreterState& m_state;
|
InterpreterState& m_state;
|
||||||
Dialect const& m_dialect;
|
Dialect const& m_dialect;
|
||||||
/// Values of variables.
|
/// Values of variables.
|
||||||
std::map<YulString, u256> const& m_variables;
|
std::map<YulString, u256> const& m_variables;
|
||||||
/// Stack of scopes in the current context.
|
Scope& m_scope;
|
||||||
std::vector<std::map<YulString, FunctionDefinition const*>> const& m_scopes;
|
|
||||||
/// Current value of the expression
|
/// Current value of the expression
|
||||||
std::vector<u256> m_values;
|
std::vector<u256> m_values;
|
||||||
};
|
};
|
||||||
|
@ -88,11 +88,10 @@ void interpret(string const& _source)
|
|||||||
|
|
||||||
InterpreterState state;
|
InterpreterState state;
|
||||||
state.maxTraceSize = 10000;
|
state.maxTraceSize = 10000;
|
||||||
Dialect const& dialect(EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}));
|
|
||||||
Interpreter interpreter(state, dialect);
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
interpreter(*ast);
|
Dialect const& dialect(EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}));
|
||||||
|
Interpreter::run(state, dialect, *ast);
|
||||||
}
|
}
|
||||||
catch (InterpreterTerminatedGeneric const&)
|
catch (InterpreterTerminatedGeneric const&)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user