var assert = require('assert'); var fs = require('fs'); var wtu = require('./workertestutils.js'); //var heapdump = require('heapdump'); // TODO: why memory leak? CACHE_WASM_MODULES = false; global.onmessage({data:{preload:'cc65', platform:'nes'}}); global.onmessage({data:{preload:'ca65', platform:'nes'}}); global.onmessage({data:{preload:'cc65', platform:'apple2'}}); global.onmessage({data:{preload:'ca65', platform:'apple2'}}); global.onmessage({data:{preload:'sdcc'}}); // TODO: check msg against spec function compile(tool, code, platform, callback, outlen, nlines, nerrors, options) { var msgs = [{code:code, platform:platform, tool:tool, path:'src.'+tool}]; doBuild(msgs, callback, outlen, nlines, nerrors, options); } function compileFiles(tool, files, platform, callback, outlen, nlines, nerrors, options) { var msg = {updates:[], buildsteps:[]}; for (var fn of files) { var text = ab2str(fs.readFileSync('presets/'+platform+'/'+fn)); msg.updates.push({path:fn, data:text}); msg.buildsteps.push({path:fn, platform:platform, tool:tool}); } doBuild([msg], callback, outlen, nlines, nerrors, options); } function doBuild(msgs, callback, outlen, nlines, nerrors, options) { var msgcount = msgs.length; global.postMessage = function(msg) { if (!msg.unchanged) { if (msg.errors && msg.errors.length) { for (var err of msg.errors) { console.log(err); assert.ok(err.line >= 0); if (options && !options.ignoreErrorPath) { assert.equal(msgs[0].path, err.path); } assert.ok(err.msg); } assert.equal(nerrors, msg.errors.length, "errors"); } else { assert.equal(nerrors||0, 0, "errors"); assert.equal(msg.output.code?msg.output.code.length:msg.output.length, outlen, "output binary"); assert.ok(msg.output.code || msg.output instanceof Uint8Array); if (nlines) { if (typeof nlines === 'number') nlines = [nlines]; //console.log(msg.listings, nlines); var i = 0; var lstkeys = Object.keys(msg.listings); lstkeys.sort(); for (var key of lstkeys) { var listing = msg.listings[key]; assert.equal(listing.lines.length, nlines[i++], "listing lines"); } } } } if (--msgcount == 0) { callback(null, msg); //heapdump.writeSnapshot(); } else console.log(msgcount + ' msgs left'); }; global.onmessage({data:{reset:true}}); for (var i=0; i\nvoid main() {\nstrlen(0);\n}\n', 'mw8080bw', done, 8192, 2, 0); }); it('should compile mw8080 skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/mw8080bw/skeleton.sdcc')); compile('sdcc', csource, 'mw8080bw', done, 8192, 84, 0); }); it('should compile galaxian skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/galaxian-scramble/skeleton.sdcc')); compile('sdcc', csource, 'galaxian-scramble', done, 20512, 29, 0); }); it('should compile vector skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/vector-z80color/skeleton.sdcc')); compile('sdcc', csource, 'vector-z80color', done, 32768, 23, 0); }); it('should compile williams skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/williams-z80/skeleton.sdcc')); compile('sdcc', csource, 'williams-z80', done, 38912, 38, 0); }); it('should compile williams_sound skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/sound_williams-z80/skeleton.sdcc')); compile('sdcc', csource, 'sound_williams-z80', done, 16384, 6, 0); }); it('should compile coleco skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/coleco/text.c')); compile('sdcc', csource, 'coleco', done, 32768, 15, 0); }); it('should compile sg1000 skeleton', function(done) { var csource = ab2str(fs.readFileSync('presets/sms-sg1000-libcv/text.c')); compile('sdcc', csource, 'sms-sg1000-libcv', done, 49152, 25, 0); }); it('should compile verilog example', function(done) { var csource = ab2str(fs.readFileSync('presets/verilog/lfsr.v')); var msgs = [{code:csource, platform:"verilog", tool:"verilator", path:'main.v'}]; var done2 = function(err, msg) { var jscode = msg.output.code; var fn = new Function(jscode); assert.ok(fn); done(err, msg); }; doBuild(msgs, done2, 2764, 0, 0); }); it('should NOT compile verilog example', function(done) { var csource = "foobar"; var msgs = [{code:csource, platform:"verilog", tool:"verilator", path:'foomain.v'}]; doBuild(msgs, done, 0, 0, 1); }); it('should compile verilog inline assembler (JSASM)', function(done) { var dependfiles = ["racing_game_cpu.v", "hvsync_generator.v", "sprite_bitmap.v", "sprite_renderer.v", "cpu8.v", "femto8.json"]; var depends = []; for (var dfile of dependfiles) { var code = ab2str(fs.readFileSync('presets/verilog/' + dfile)); depends.push({path:dfile, data:code}); } var msgs = [{ updates:depends, buildsteps:[{platform:"verilog", tool:"verilator", path:'racing_game_cpu.v', files:dependfiles}] }]; var done2 = function(err, msg) { var jscode = msg.output.code; var fn = new Function(jscode); assert.ok(fn); done(err, msg); }; doBuild(msgs, done2, 51459, 0, 0); }); it('should compile verilog assembler file (JSASM)', function(done) { var dependfiles = ["test2.asm", "hvsync_generator.v", "font_cp437_8x8.v", "ram.v", "tile_renderer.v", "sprite_scanline_renderer.v", "lfsr.v", "sound_generator.v", "cpu16.v", "cpu_platform.v", "femto16.json"]; var depends = []; for (var dfile of dependfiles) { var code = ab2str(fs.readFileSync('presets/verilog/' + dfile)); depends.push({path:dfile, data:code}); } var msgs = [{ updates:depends, buildsteps:[{platform:"verilog", tool:"jsasm", path:'test2.asm', files:dependfiles}] }]; var done2 = function(err, msg) { var jscode = msg.output.code; var fn = new Function(jscode); assert.ok(fn); done(err, msg); }; doBuild(msgs, done2, 1997608, 0, 0); }); it('should NOT preprocess SDCC', function(done) { compile('sdcc', 'int x=0\n#bah\n', 'mw8080bw', done, 0, 0, 1); }); it('should compile XASM6809', function(done) { compile('xasm6809', '\tasld\n\tasld\n', 'williams', done, 4, 2, 0); }); it('should link two files with SDCC', function(done) { var msgs = [ { "updates":[ {"path":"main.c", "data":"extern int mul2(int x);\nint main() { return mul2(2); }\n"}, {"path":"fn.c", "data":"int mul2(int x) { return x*x; }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"mw8080bw", "tool":"sdcc"}, {"path":"fn.c", "platform":"mw8080bw", "tool":"sdcc"} ] } ]; doBuild(msgs, done, 8192, [1,1], 0); }); // TODO: tests don't fail if too many compile steps it('should not build unchanged files with CC65', function(done) { var m = { "updates":[ {"path":"main.c", "data":"extern int mul2(int x);\n int main() { return mul2(2); }\n"}, {"path":"fn.c", "data":"int mul2(int x) { return x*x; }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"nes", "tool":"cc65"}, {"path":"fn.c", "platform":"nes", "tool":"cc65"} ] }; var m2 = { "updates":[ {"path":"main.c", "data":"extern int mul2(int x); \nint main() { return mul2(2); }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"nes", "tool":"cc65"}, {"path":"fn.c", "platform":"nes", "tool":"cc65"} ] }; var msgs = [m, m, m2]; doBuild(msgs, done, 40976, [1,1], 0); }); it('should not build unchanged files with SDCC', function(done) { var m = { "updates":[ {"path":"main.c", "data":"extern int mul2(int x);\n int main() { return mul2(2); }\n"}, {"path":"fn.c", "data":"int mul2(int x) { return x*x; }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"mw8080bw", "tool":"sdcc"}, {"path":"fn.c", "platform":"mw8080bw", "tool":"sdcc"} ] }; var m2 = { "updates":[ {"path":"main.c", "data":"extern int mul2(int x); \nint main() { return mul2(2); }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"mw8080bw", "tool":"sdcc"}, {"path":"fn.c", "platform":"mw8080bw", "tool":"sdcc"} ] }; var msgs = [m, m, m2]; doBuild(msgs, done, 8192, [1,1], 0); }); it('should include filename in compile errors', function(done) { var m = { "updates":[ {"path":"main.c", "data":"extern int mul2(int x);\n int main() { return mul2(2); }\n"}, {"path":"fn.c", "data":"void int mul2(int x) { return x*x; }\n"} ], "buildsteps":[ {"path":"main.c", "platform":"mw8080bw", "tool":"sdcc"}, {"path":"fn.c", "platform":"mw8080bw", "tool":"sdcc"} ], "path":"fn.c" }; var msgs = [m]; doBuild(msgs, done, 8192, [1,1], 2); // TODO: check error file }); it('should compile vicdual skeleton', function(done) { var files = ['skeleton.sdcc', 'cp437.c']; compileFiles('sdcc', files, 'vicdual', done, 16416, [0,45], 0); // TODO? }); it('should compile apple2 skeleton with CC65', function(done) { var csource = ab2str(fs.readFileSync('presets/apple2/skeleton.cc65')); compile('cc65', csource, 'apple2', done, 17349, 4, 0); }); // TODO: test if compile, errors, then compile same file // TODO: params persist because of fixParamsWithDefines() it('should compile CC65 banked', function(done) { compile('cc65', '#define NES_MAPPER 4\nint main() {\nint x=1;\nreturn x+2;\n}', 'nes', done, 131088, 3); }); it('should assemble CA65', function(done) { compile('ca65', ';#define LIBARGS ,\n\t.segment "HEADER"\n\t.segment "STARTUP"\n\t.segment "CHARS"\n\t.segment "VECTORS"\n\t.segment "SAMPLES"\n\t.segment "CODE"\n\tlda #0\n\tsta $1\n', 'nes', done, 131088, 2); }); });