mirror of
https://github.com/sehugg/8bitworkshop.git
synced 2024-10-31 23:09:49 +00:00
318 lines
13 KiB
JavaScript
318 lines
13 KiB
JavaScript
|
|
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<msgs.length; i++) {
|
|
global.onmessage({data:msgs[i]});
|
|
}
|
|
}
|
|
|
|
describe('Worker', function() {
|
|
it('should assemble DASM', function(done) {
|
|
compile('dasm', '\tprocessor 6502\n\torg $f000\nfoo lda #0\n', 'vcs', done, 2, 1);
|
|
});
|
|
it('should NOT assemble DASM', function(done) {
|
|
compile('dasm', '\tprocessor 6502\n\torg $f000 ; this is a comment\nfoo asl a\n', 'vcs', done, 0, 0, 1);
|
|
});
|
|
/*
|
|
it('should assemble ACME', function(done) {
|
|
compile('acme', 'foo: lda #0\n', 'vcs', done, 2, 0); // TODO
|
|
});
|
|
it('should NOT assemble ACME', function(done) {
|
|
compile('acme', 'foo: xxx #0\n', 'vcs', done, 0, 0, 2); // TODO
|
|
});
|
|
it('should compile PLASMA', function(done) {
|
|
compile('plasm', 'word x = 0', 'apple2', done, 5, 0);
|
|
});
|
|
it('should NOT compile PLASMA', function(done) {
|
|
compile('plasm', 'word x = ', 'apple2', done, 0, 0, 1);
|
|
});
|
|
*/
|
|
it('should compile CC65', function(done) {
|
|
compile('cc65', 'int main() {\nint x=1;\nreturn x+2;\n}', 'nes', done, 40976, 3);
|
|
});
|
|
it('should NOT compile CC65 (compile error)', function(done) {
|
|
compile('cc65', 'int main() {\nint x=1;\nprintf("%d",x);\nreturn x+2;\n}', 'nes', done, 0, 0, 1);
|
|
});
|
|
it('should NOT compile CC65 (link error)', function(done) {
|
|
compile('cc65', 'extern void bad();\nint main() {\nbad();\nreturn 0;\n}', 'nes', done, 0, 0, 3, {ignoreErrorPath:true});
|
|
});
|
|
it('should NOT compile CC65 (preproc error)', function(done) {
|
|
compile('cc65', '#include "NOSUCH.file"\n', 'nes', done, 0, 0, 1, {ignoreErrorPath:true});
|
|
});
|
|
/*
|
|
it('should assemble Z80ASM', function(done) {
|
|
compile('z80asm', '\tMODULE test\n\tEXTERN _puts\n\tld hl,$0000\n\tret\n', 'mw8080bw', done, 4, 2, 0);
|
|
});
|
|
it('should NOT assemble Z80ASM', function(done) {
|
|
compile('z80asm', 'ddwiuweq', 'mw8080bw', done, 0, 0, 1);
|
|
});
|
|
*/
|
|
it('should assemble SDASZ80', function(done) {
|
|
compile('sdasz80', '\tld hl,#0\n\tret\n', 'mw8080bw', done, 8192, 2);
|
|
});
|
|
it('should NOT assemble SDASZ80', function(done) {
|
|
compile('sdasz80', '\txxx hl,#0\n\tret\n', 'mw8080bw', done, 0, 0, 1);
|
|
});
|
|
it('should NOT link SDASZ80', function(done) {
|
|
compile('sdasz80', '\tcall divxxx\n', 'mw8080bw', done, 0, 0, 1, {ignoreErrorPath:true});
|
|
});
|
|
it('should compile SDCC', function(done) {
|
|
compile('sdcc', 'int foo=0; // comment\nint main(int argc) {\nint x=1;\nint y=2+argc;\nreturn x+y+argc;\n}\n', 'mw8080bw', done, 8192, 3, 0);
|
|
});
|
|
it('should compile SDCC w/ include', function(done) {
|
|
compile('sdcc', '#include <string.h>\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);
|
|
});
|
|
|
|
});
|