mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	* 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>
		
			
				
	
	
		
			293 lines
		
	
	
	
		
			9.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			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)
 |