aboutsummaryrefslogtreecommitdiff
path: root/tests/runner.py
diff options
context:
space:
mode:
Diffstat (limited to 'tests/runner.py')
-rw-r--r--tests/runner.py473
1 files changed, 473 insertions, 0 deletions
diff --git a/tests/runner.py b/tests/runner.py
new file mode 100644
index 00000000..e675815d
--- /dev/null
+++ b/tests/runner.py
@@ -0,0 +1,473 @@
+'''
+Simple test runner
+
+See settings.cfg file for options&params. Edit as needed.
+'''
+
+from subprocess import Popen, PIPE, STDOUT
+import os, unittest, tempfile, shutil, time
+
+# Params
+
+def path_from_root(pathelems):
+ return os.path.join(os.path.sep, *(((os.path.abspath(os.path.dirname(__file__)).split(os.sep)))[:-1] + pathelems))
+
+exec(open(os.path.join(os.path.abspath(os.path.dirname(__file__)), 'settings.cfg'), 'r').read())
+
+def timeout_run(proc, timeout, note):
+ start = time.time()
+ while time.time() - start < timeout and proc.poll() is None:
+ time.sleep(0.1)
+ if proc.poll() is None:
+ proc.kill()
+ raise Exception("Timed out: " + note)
+ return proc.communicate()[0]
+
+class T(unittest.TestCase):
+ def do_test(self, src, expected_output, args=[], output_nicerizer=None, no_python=False, no_build=False):
+ global DEBUG
+ dirname = TEMP_DIR + '/tmp' # tempfile.mkdtemp(dir=TEMP_DIR)
+ if not os.path.exists(dirname):
+ os.makedirs(dirname)
+ filename = os.path.join(dirname, 'src.cpp')
+ if not no_build:
+ f = open(filename, 'w')
+ f.write(src)
+ f.close()
+ if DEBUG: print "[[C++ => LLVM]]"
+ output = Popen([LLVM_GCC, '-emit-llvm', '-c', filename, '-o', filename + '.o'], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ if DEBUG: print output
+ if DEBUG: print "[[LLVM => JS]]"
+ if False:
+ # Use an llc backend, written in C++, to generate JS
+ output = Popen([LLC, '-march='+LLVM_BACKEND, filename + '.o', '-o=' + filename + '.o.cpp'], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ elif False:
+ # Use python parser to generate JS from disassembled llvm
+ output = Popen([LLVM_DIS, filename + '.o', '-o=' + filename + '.o.llvm'], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ if DEBUG: print output
+ output = Popen(['python', PY_PARSER, filename + '.o.llvm'], stdout=open(filename + '.o.js', 'w'), stderr=STDOUT).communicate()[0]
+ else:
+ # JS parser/compiler
+ output = Popen([LLVM_DIS, filename + '.o', '-o=' + filename + '.o.llvm'], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ if DEBUG: print output
+ cwd = os.getcwd()
+ os.chdir(path_from_root(['src']))
+ output = timeout_run(Popen([PARSER_ENGINE] + PARSER_OPTS + [JS_COMPILER], stdin=open(filename + '.o.llvm', 'r'), stdout=open(filename + '.o.js', 'w'), stderr=STDOUT), 20, 'Parser')
+ os.chdir(cwd)
+ # return
+ if DEBUG: print output
+ output = open(filename + '.o.js').read()
+ if output is not None and 'Traceback' in output: print output; assert (0) # 'generating JavaScript failed'
+ if DEBUG: print "\nGenerated JavaScript:\n\n===\n\n%s\n\n===\n\n" % output
+ # if not DEBUG:
+ js_output = timeout_run(Popen([JS_ENGINE] + JS_ENGINE_OPTS + [filename + '.o.js'] + args, stdout=PIPE, stderr=STDOUT), 20, 'Execution')
+ if output_nicerizer is not None:
+ js_output = output_nicerizer(js_output)
+ # else:
+ # print "[[JS output]]"
+ # ret = "Output shown on screen, test not actually run!"
+ # Popen([JS_ENGINE, filename + '.o.js'] + args, stderr=STDOUT).communicate()[0]
+ self.assertContained(expected_output, js_output)
+ self.assertNotContained('ERROR', js_output)
+ return
+
+ if not no_python:
+ #DEBUG = True
+ SPIDERMONKEY = True
+ if SPIDERMONKEY:
+ if DEBUG: print "[[RJS ==> SpiderMonkey parsed tree]]"
+ args = [SPIDERMONKEY_SHELL, '-e', 'parse(snarf(\"%s\"))' % (filename + '.o.js')]
+ output = Popen(args, stdout=PIPE, stderr=STDOUT).communicate()[0]
+ f = open(filename + 'o.js.sm', 'w')
+ f.write(output)
+ f.close()
+ else:
+ if DEBUG: print "[[RJS ==> RPython]]"
+ output = Popen(['python', RJS_RPYTHON, filename + '.o.js', filename + '.o.js.py'], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ if DEBUG: print output
+
+ py_output = Popen(['python', filename + '.o.js.py'] + args, stdout=PIPE, stderr=STDOUT).communicate()[0]
+ if output_nicerizer is not None:
+ py_output = output_nicerizer(py_output)
+ self.assertContained(expected_output, py_output)
+ if js_output != py_output:
+ print "WARNING: js and py outputs not identical (but each is similar enough to the expected_output)"
+
+ PYPY = True
+# PYPY = False
+ if PYPY:
+ pypy_source = filename.replace('.', '_') + '_o_js_py.py'
+ if DEBUG: print "[[RPython ==> PyPy]]"
+ output = Popen(['python', RJS_PYPY, filename + '.o.js.py', pypy_source], stdout=PIPE, stderr=STDOUT).communicate()[0]
+ print output
+
+# # Python on pypy-ready source
+ # pypy_output = Popen(['python', pypy_source] + args, stdout=PIPE, stderr=STDOUT).communicate()[0]
+ # if output_nicerizer is not None:
+ # pypy_output = output_nicerizer(pypy_output)
+ # self.assertContained(expected_output, pypy_output)
+ # if js_output != pypy_output:
+ # print "WARNING: js and PYpy outputs not identical (but each is similar enough to the expected_output)"
+
+ # PyPy compilation of source to binary
+
+# shutil.rmtree(dirname)
+
+ def assertContained(self, value, string):
+ if value not in string:
+ print "Expected to find '%s' in '%s'" % (value, string)
+ self.assertTrue(value in string)
+
+ def assertNotContained(self, value, string):
+ if value in string:
+ print "Expected to NOT find '%s' in '%s'" % (value, string)
+ self.assertTrue(value not in string)
+
+ def test_hello_world(self):
+ src = '''
+ #include <stdio.h>
+ int main()
+ {
+ printf("hello, world!\\n");
+ return 0;
+ }
+ '''
+ self.do_test(src, 'hello, world!')
+
+ def test_intvars(self):
+ src = '''
+ #include <stdio.h>
+ int main()
+ {
+ int x = 5;
+ int y = x+17;
+ int z = (y-1)/2; // Should stay an integer after division!
+ y += 1;
+ int w = x*3+4;
+ int k = w < 15 ? 99 : 101;
+ int i = k > 100; // Should be an int, not a bool!
+ printf("*%d,%d,%d,%d,%d,%d*\\n", x, y, z, w, k,i);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*5,23,10,19,101,1*')
+
+ def test_if(self):
+ src = '''
+ #include <stdio.h>
+ int main()
+ {
+ int x = 5;
+ if (x > 3) {
+ printf("*yes*\\n");
+ }
+ return 0;
+ }
+ '''
+ self.do_test(src, '*yes*')
+
+ def test_loop(self):
+ src = '''
+ #include <stdio.h>
+ int main()
+ {
+ int x = 5;
+ for (int i = 0; i < 6; i++)
+ x += x*i;
+ printf("*%d*\\n", x);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*3600*')
+
+ def test_strings(self):
+ src = '''
+ #include <stdio.h>
+ #include <stdlib.h>
+ int main(int argc, char **argv)
+ {
+ printf("*%d", argc);
+ puts(argv[1]);
+ puts(argv[2]);
+ printf("%d*", atoi(argv[3])+2);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*4*wowie*too*76*', ['wowie', 'too', '74'], lambda x: x.replace('\n', '*'))
+
+ def test_funcs(self):
+ src = '''
+ #include <stdio.h>
+ int funcy(int x)
+ {
+ return x*9;
+ }
+ int main()
+ {
+ printf("*%d,%d*\\n", funcy(8), funcy(10));
+ return 0;
+ }
+ '''
+ self.do_test(src, '*72,90*')
+
+ def test_structs(self):
+ src = '''
+ #include <stdio.h>
+ struct S
+ {
+ int x, y;
+ };
+ int main()
+ {
+ S a, b;
+ a.x = 5; a.y = 6;
+ b.x = 101; b.y = 7009;
+ S *c, *d;
+ c = &a;
+ c->x *= 2;
+ c = &b;
+ c->y -= 1;
+ d = c;
+ d->y += 10;
+ printf("*%d,%d,%d,%d,%d,%d,%d,%d*\\n", a.x, a.y, b.x, b.y, c->x, c->y, d->x, d->y);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*10,6,101,7018,101,7018,101,7018*')
+
+ gen_struct_src = '''
+ #include <stdio.h>
+ #include <stdlib.h>
+ struct S
+ {
+ int x, y;
+ };
+ int main()
+ {
+ S* a = {{gen_struct}};
+ a->x = 51; a->y = 62;
+ printf("*%d,%d*\\n", a->x, a->y);
+ {{del_struct}}(a);
+ return 0;
+ }
+ '''
+
+ def test_mallocstruct(self):
+ self.do_test(self.gen_struct_src.replace('{{gen_struct}}', '(S*)malloc(sizeof(S))').replace('{{del_struct}}', 'free'), '*51,62*')
+
+ def test_newstruct(self):
+ self.do_test(self.gen_struct_src.replace('{{gen_struct}}', 'new S').replace('{{del_struct}}', 'delete'), '*51,62*')
+
+ def test_addr_of_stacked(self):
+ src = '''
+ #include <stdio.h>
+ void alter(int *y)
+ {
+ *y += 5;
+ }
+ int main()
+ {
+ int x = 2;
+ alter(&x);
+ printf("*%d*\\n", x);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*7*')
+
+ def test_linked_list(self):
+ src = '''
+ #include <stdio.h>
+ struct worker_args {
+ int value;
+ struct worker_args *next;
+ };
+ int main()
+ {
+ worker_args a;
+ worker_args b;
+ a.value = 60;
+ a.next = &b;
+ b.value = 900;
+ b.next = NULL;
+ worker_args* c = &a;
+ int total = 0;
+ while (c) {
+ total += c->value;
+ c = c->next;
+ }
+ printf("*%d*\\n", total);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*960*')
+
+ def test_class(self):
+ src = '''
+ #include <stdio.h>
+ struct Random {
+ enum { IM = 139968, IA = 3877, IC = 29573 };
+ Random() : last(42) {}
+ float get( float max = 1.0f ) {
+ last = ( last * IA + IC ) % IM;
+ return max * last / IM;
+ }
+ protected:
+ unsigned int last;
+ } rng1;
+ int main()
+ {
+ Random rng2;
+ int count = 0;
+ for (int i = 0; i < 100; i++) {
+ float x1 = rng1.get();
+ float x2 = rng2.get();
+ printf("%f, %f\\n", x1, x2);
+ if (x1 != x2) count += 1;
+ }
+ printf("*%d*\\n", count);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*0*')
+
+ def test_inherit(self):
+ src = '''
+ #include <stdio.h>
+ struct Parent {
+ int x1, x2;
+ };
+ struct Child : Parent {
+ int y;
+ };
+ int main()
+ {
+ Parent a;
+ a.x1 = 50;
+ a.x2 = 87;
+ Child b;
+ b.x1 = 78;
+ b.x2 = 550;
+ b.y = 101;
+ Child* c = (Child*)&a;
+ c->x1 ++;
+ c = &b;
+ c->y --;
+ printf("*%d,%d,%d,%d,%d,%d,%d*\\n", a.x1, a.x2, b.x1, b.x2, b.y, c->x1, c->x2);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*51,87,78,550,100,78,550*')
+
+ def test_polymorph(self):
+ src = '''
+ #include <stdio.h>
+ struct Parent {
+ virtual int getit() { return 11; };
+ };
+ struct Child : Parent {
+ int getit() { return 74; }
+ };
+ int main()
+ {
+ Parent *x = new Parent();
+ Parent *y = new Child();
+ printf("*%d,%d*\\n", x->getit(), y->getit());
+ return 0;
+ }
+ '''
+ self.do_test(src, '*11,74*')
+
+ def zzzzzzzzzzzzzzztest_constglobalstructs(self): # TODO: make this work
+ src = '''
+ #include <stdio.h>
+ struct IUB {
+ int c;
+ double p;
+ unsigned int pi;
+ };
+
+ IUB iub[] = {
+ { 'a', 0.27, 5 },
+ { 'c', 0.15, 4 },
+ { 'g', 0.12, 3 },
+ { 't', 0.27, 2 },
+ };
+
+ int main( int argc, const char *argv[] ) {
+ printf("*%d,%d,%d*\\n", iub[0].c, int(iub[1].p*100), iub[2].pi);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*97,15,3*')
+
+ def test_conststructs(self):
+ src = '''
+ #include <stdio.h>
+ struct IUB {
+ int c;
+ double p;
+ unsigned int pi;
+ };
+
+ int main( int argc, const char *argv[] ) {
+ IUB iub[] = {
+ { 'a', 0.27, 5 },
+ { 'c', 0.15, 4 },
+ { 'g', 0.12, 3 },
+ { 't', 0.27, 2 },
+ };
+ printf("*%d,%d,%d*\\n", iub[0].c, int(iub[1].p*100), iub[2].pi);
+// printf("*%d*\\n", int(iub[1].p*100));
+ return 0;
+ }
+ '''
+ self.do_test(src, '*97,15,3*')
+
+
+ def test_memcpy(self):
+ src = '''
+ #include <stdio.h>
+ #include <string.h>
+
+ int main( int argc, const char *argv[] ) {
+ int *a = new int[10];
+ int *b = new int[1];
+ int *c = new int[10];
+ for (int i = 0; i < 10; i++)
+ a[i] = 2;
+ *b = 5;
+ for (int i = 0; i < 10; i++)
+ c[i] = 8;
+ printf("*%d,%d,%d,%d,%d*\\n", a[0], a[9], *b, c[0], c[9]);
+ // Should overwrite a, but not touch b!
+ memcpy(a, c, 10*sizeof(int));
+ printf("*%d,%d,%d,%d,%d*\\n", a[0], a[9], *b, c[0], c[9]);
+ return 0;
+ }
+ '''
+ self.do_test(src, '*2,2,5,8,8*\n*8,8,5,8,8*')
+
+ def test_fannkuch(self):
+ results = [ (1,0), (2,1), (3,2), (4,4), (5,7), (6,10), (7, 16), (8,22) ]
+ for i, j in results:
+ src = open(path_from_root(['tests', 'fannkuch.cpp']), 'r').read()
+ self.do_test(src, 'Pfannkuchen(%d) = %d.' % (i,j), [str(i)], no_build=i>1)
+
+ def zzztest_fasta(self):
+ results = [ (1,'''GG*ctt**tgagc**'''), (20,'''GGCCGGGCGCGGTGGCTCACGCCTGTAATCCCAGCACTTT*cttBtatcatatgctaKggNcataaaSatgtaaaDcDRtBggDtctttataattcBgtcg**tacgtgtagcctagtgtttgtgttgcgttatagtctatttgtggacacagtatggtcaaa**tgacgtcttttgatctgacggcgttaacaaagatactctg**'''),
+(50,'''GGCCGGGCGCGGTGGCTCACGCCTGTAATCCCAGCACTTTGGGAGGCCGAGGCGGGCGGA*TCACCTGAGGTCAGGAGTTCGAGACCAGCCTGGCCAACAT*cttBtatcatatgctaKggNcataaaSatgtaaaDcDRtBggDtctttataattcBgtcg**tactDtDagcctatttSVHtHttKtgtHMaSattgWaHKHttttagacatWatgtRgaaa**NtactMcSMtYtcMgRtacttctWBacgaa**agatactctgggcaacacacatacttctctcatgttgtttcttcggacctttcataacct**ttcctggcacatggttagctgcacatcacaggattgtaagggtctagtggttcagtgagc**ggaatatcattcgtcggtggtgttaatctatctcggtgtagcttataaatgcatccgtaa**gaatattatgtttatttgtcggtacgttcatggtagtggtgtcgccgatttagacgtaaa**ggcatgtatg**''') ]
+ for i, j in results:
+ src = open(path_from_root(['tests', 'fasta.cpp']), 'r').read()
+ self.do_test(src, j, [str(i)], lambda x: x.replace('\n', '*'), no_python=True, no_build=i>1)
+
+if __name__ == '__main__':
+ if DEBUG: print "LLVM_GCC:", LLVM_GCC
+ if DEBUG: print "LLC:", LLC
+ if DEBUG: print "PARSER:", PARSER
+ if DEBUG: print "JS_ENGINE:", JS_ENGINE
+ for cmd in [LLVM_GCC, JS_ENGINE]:
+ if DEBUG: print "Checking for existence of", cmd
+ assert(os.path.exists(cmd))
+ unittest.main()
+