1
0
Fork 0
mirror of https://github.com/ton-blockchain/ton synced 2025-02-13 03:32:22 +00:00
ton/crypto/func/auto-tests/stress_tester.py
EmelyanenkoK a1d8a5e4f3
Add complex assigns to FunC and fix UB (#574)
* Fixed complex funC setglob cases

* Forbid modifying local variables after using them in the same tensor

* Fix analyzing "while" in func

* Update funC version (#9)

* Update stress tester

* Fix using variable after move

Co-authored-by: krigga <krigga7@gmail.com>
Co-authored-by: SpyCheese <mikle98@yandex.ru>
2022-12-29 18:06:13 +03:00

293 lines
9.4 KiB
Python

import os
import os.path
import random
import subprocess
import sys
import tempfile
def getenv(name, default=None):
if name in os.environ:
return os.environ[name]
if default is not None:
return default
print("Environemnt variable", name, "is not set", file=sys.stderr)
exit(1)
VAR_CNT = 10
TMP_DIR = tempfile.mkdtemp()
FUNC_EXECUTABLE = getenv("FUNC_EXECUTABLE", "func")
FIFT_EXECUTABLE = getenv("FIFT_EXECUTABLE", "fift")
FIFT_LIBS = getenv("FIFT_LIBS")
MAGIC = 123456789
var_idx = 0
def gen_var_name():
global var_idx
var_idx += 1
return "i%d" % var_idx
class State:
def __init__(self, x):
self.x = x
self.vs = [0] * VAR_CNT
def copy(self):
s = State(self.x)
s.vs = self.vs.copy()
return s
def copy_from(self, s):
self.x = s.x
self.vs = s.vs.copy()
class Code:
pass
class CodeEmpty(Code):
def execute(self, state):
return None
def write(self, f, indent=0):
pass
class CodeReturn(Code):
def __init__(self, value):
self.value = value
def execute(self, state):
return [self.value] + state.vs
def write(self, f, indent=0):
print(" " * indent + "return (%d, %s);" % (self.value, ", ".join("v%d" % i for i in range(VAR_CNT))), file=f)
class CodeAdd(Code):
def __init__(self, i, value):
self.i = i
self.value = value
def execute(self, state):
state.vs[self.i] += self.value
return None
def write(self, f, indent=0):
print(" " * indent + "v%d += %d;" % (self.i, self.value), file=f)
class CodeBlock(Code):
def __init__(self, code):
self.code = code
def execute(self, state):
for c in self.code:
res = c.execute(state)
if res is not None:
return res
return None
def write(self, f, indent=0):
for c in self.code:
c.write(f, indent)
class CodeIfRange(Code):
def __init__(self, l, r, c1, c2):
self.l = l
self.r = r
self.c1 = c1
self.c2 = c2
def execute(self, state):
if self.l <= state.x < self.r:
return self.c1.execute(state)
else:
return self.c2.execute(state)
def write(self, f, indent=0):
print(" " * indent + "if (in(x, %d, %d)) {" % (self.l, self.r), file=f)
self.c1.write(f, indent + 1)
if isinstance(self.c2, CodeEmpty):
print(" " * indent + "}", file=f)
else:
print(" " * indent + "} else {", file=f)
self.c2.write(f, indent + 1)
print(" " * indent + "}", file=f)
class CodeRepeat(Code):
def __init__(self, n, c, loop_type):
if loop_type == 2:
n = max(n, 1)
self.n = n
self.c = c
self.loop_type = loop_type
def execute(self, state):
for _ in range(self.n):
res = self.c.execute(state)
if res is not None:
return res
return None
def write(self, f, indent=0):
if self.loop_type == 0:
print(" " * indent + "repeat (%d) {" % self.n, file=f)
self.c.write(f, indent + 1)
print(" " * indent + "}", file=f)
elif self.loop_type == 1:
var = gen_var_name()
print(" " * indent + "int %s = 0;" % var, file=f)
print(" " * indent + "while (%s < %d) {" % (var, self.n), file=f)
self.c.write(f, indent + 1)
print(" " * (indent + 1) + "%s += 1;" % var, file=f)
print(" " * indent + "}", file=f)
elif self.loop_type == 2:
var = gen_var_name()
print(" " * indent + "int %s = 0;" % var, file=f)
print(" " * indent + "do {", file=f)
self.c.write(f, indent + 1)
print(" " * (indent + 1) + "%s += 1;" % var, file=f)
print(" " * indent + "} until (%s >= %d);" % (var, self.n), file=f)
else:
var = gen_var_name()
print(" " * indent + "int %s = %d;" % (var, self.n - 1), file=f)
print(" " * indent + "while (%s >= 0) {" % var, file=f)
self.c.write(f, indent + 1)
print(" " * (indent + 1) + "%s -= 1;" % var, file=f)
print(" " * indent + "}", file=f)
class CodeThrow(Code):
def __init__(self):
pass
def execute(self, state):
return "EXCEPTION"
def write(self, f, indent=0):
print(" " * indent + "throw(42);", file=f)
class CodeTryCatch(Code):
def __init__(self, c1, c2):
self.c1 = c1
self.c2 = c2
def execute(self, state):
state0 = state.copy()
res = self.c1.execute(state)
if res == "EXCEPTION":
state.copy_from(state0)
return self.c2.execute(state)
else:
return res
def write(self, f, indent=0):
print(" " * indent + "try {", file=f)
self.c1.write(f, indent + 1)
print(" " * indent + "} catch (_, _) {", file=f)
self.c2.write(f, indent + 1)
print(" " * indent + "}", file=f)
def write_function(f, name, body, inline=False, inline_ref=False, method_id=None):
print("_ %s(int x)" % name, file=f, end="")
if inline:
print(" inline", file=f, end="")
if inline_ref:
print(" inline_ref", file=f, end="")
if method_id is not None:
print(" method_id(%d)" % method_id, file=f, end="")
print(" {", file=f)
for i in range(VAR_CNT):
print(" int v%d = 0;" % i, file=f)
body.write(f, 1)
print("}", file=f)
def gen_code(xl, xr, with_return, loop_depth=0, try_catch_depth=0, can_throw=False):
if try_catch_depth < 3 and random.randint(0, 5) == 0:
c1 = gen_code(xl, xr, with_return, loop_depth, try_catch_depth + 1, random.randint(0, 1) == 0)
c2 = gen_code(xl, xr, with_return, loop_depth, try_catch_depth + 1, can_throw)
return CodeTryCatch(c1, c2)
code = []
for _ in range(random.randint(0, 2)):
if random.randint(0, 3) == 0 and loop_depth < 3:
c = gen_code(xl, xr, False, loop_depth + 1, try_catch_depth, can_throw)
code.append(CodeRepeat(random.randint(0, 3), c, random.randint(0, 3)))
elif xr - xl > 1:
xmid = random.randrange(xl + 1, xr)
ret = random.choice((0, 0, 0, 0, 0, 1, 2))
c1 = gen_code(xl, xmid, ret == 1, loop_depth, try_catch_depth, can_throw)
if random.randrange(5) == 0:
c2 = CodeEmpty()
else:
c2 = gen_code(xmid, xr, ret == 2, loop_depth, try_catch_depth, can_throw)
code.append(CodeIfRange(xl, xmid, c1, c2))
if xr - xl == 1 and can_throw and random.randint(0, 5) == 0:
code.append(CodeThrow())
if with_return:
if xr - xl == 1:
code.append(CodeReturn(random.randrange(10**9)))
else:
xmid = random.randrange(xl + 1, xr)
c1 = gen_code(xl, xmid, True, loop_depth, try_catch_depth, can_throw)
c2 = gen_code(xmid, xr, True, loop_depth, try_catch_depth, can_throw)
code.append(CodeIfRange(xl, xmid, c1, c2))
for _ in range(random.randint(0, 3)):
pos = random.randint(0, len(code))
code.insert(pos, CodeAdd(random.randrange(VAR_CNT), random.randint(0, 10**6)))
if len(code) == 0:
return CodeEmpty()
return CodeBlock(code)
class ExecutionError(Exception):
pass
def compile_func(fc, fif):
res = subprocess.run([FUNC_EXECUTABLE, "-o", fif, "-SPA", fc], capture_output=True)
if res.returncode != 0:
raise ExecutionError(str(res.stderr, "utf-8"))
def runvm(compiled_fif, xl, xr):
runner = os.path.join(TMP_DIR, "runner.fif")
with open(runner, "w") as f:
print("\"%s\" include <s constant code" % compiled_fif, file=f)
for x in range(xl, xr):
print("%d 0 code 1 runvmx abort\"exitcode is not 0\" .s cr { drop } depth 1- times" % x, file=f)
res = subprocess.run([FIFT_EXECUTABLE, "-I", FIFT_LIBS, runner], capture_output=True)
if res.returncode != 0:
raise ExecutionError(str(res.stderr, "utf-8"))
output = []
for s in str(res.stdout, "utf-8").split("\n"):
if s.strip() != "":
output.append(list(map(int, s.split())))
return output
cnt_ok = 0
cnt_fail = 0
for test_id in range(0, 1000000):
random.seed(test_id)
inline = random.randint(0, 2)
xr = random.randint(1, 15)
var_idx = 0
code = gen_code(0, xr, True)
fc = os.path.join(TMP_DIR, "code.fc")
fif = os.path.join(TMP_DIR, "compiled.fif")
with open(fc, "w") as f:
print("int in(int x, int l, int r) impure { return (l <= x) & (x < r); }", file=f)
write_function(f, "foo", code, inline=(inline == 1), inline_ref=(inline == 2))
print("_ main(int x) {", file=f)
print(" (int ret, %s) = foo(x);" % ", ".join("int v%d" % i for i in range(VAR_CNT)), file=f)
print(" return (ret, %s, %d);" % (", ".join("v%d" % i for i in range(VAR_CNT)), MAGIC), file=f)
print("}", file=f)
compile_func(fc, fif)
ok = True
try:
output = runvm(fif, 0, xr)
for x in range(xr):
my_out = code.execute(State(x)) + [MAGIC]
fc_out = output[x]
if my_out != fc_out:
ok = False
break
except ExecutionError:
ok = False
if ok:
cnt_ok += 1
else:
cnt_fail += 1
print("Test %-6d %-6s ok:%-6d fail:%-6d" % (test_id, "OK" if ok else "FAIL", cnt_ok, cnt_fail), file=sys.stderr)