diff --git a/doc/compilers.txt b/doc/compilers.txt index 03d1e44d..5009707a 100644 --- a/doc/compilers.txt +++ b/doc/compilers.txt @@ -57,6 +57,8 @@ https://github.com/EtchedPixels/FUZIX/wiki gcc6809 - need to check this out +SmallerC - https://github.com/alexfru/SmallerC + OTHER COMPILERS diff --git a/doc/notes.txt b/doc/notes.txt index 766294b6..8975f087 100644 --- a/doc/notes.txt +++ b/doc/notes.txt @@ -75,6 +75,7 @@ debugger disasm view memory profile +errors separate into files @@ -92,3 +93,13 @@ window methods: - update debug window (200 ms) file store mirrors that on worker + +worker result: +{ + output: + sources:{ + :{lines, asmlines, listing} + } +} + +add file to errors diff --git a/src/ui.js b/src/ui.js index fbf04060..0617d0e4 100644 --- a/src/ui.js +++ b/src/ui.js @@ -43,6 +43,7 @@ var toolbar = $("#controls_top"); var SourceFile = function(lines, text) { lines = lines || []; + this.lines = lines; this.text = text; this.offset2line = {}; this.line2offset = {}; @@ -441,10 +442,23 @@ function setCode(text) { function setCompileOutput(data) { if (data.unchanged) return; // TODO: kills current selection - sourcefile = new SourceFile(data.lines); - if (data.asmlines) { - assemblyfile = new SourceFile(data.asmlines, data.intermediate.listing); + // choose first listing (TODO:support multiple source files) + sourcefile = null; + assemblyfile = null; + if (data.listings) { + var lst; + for (var lstname in data.listings) { + lst = data.listings[lstname]; + break; + } + if (lst) { + sourcefile = new SourceFile(lst.lines); + if (lst.asmlines) { + assemblyfile = new SourceFile(lst.asmlines, lst.text); + } + } } + if (!sourcefile) sourcefile = new SourceFile(); symbolmap = data.symbolmap; addr2symbol = invertMap(symbolmap); addr2symbol[0x10000] = '__END__'; // TODO? @@ -510,7 +524,9 @@ function setCompileOutput(data) { editor.clearGutter("gutter-bytes"); editor.clearGutter("gutter-offset"); editor.clearGutter("gutter-clock"); - for (var info of data.lines) { + // TODO: support multiple files + var lstlines = sourcefile.lines || []; + for (var info of lstlines) { if (info.offset >= 0) { var textel = document.createTextNode(hex(info.offset,4)); editor.setGutterMarker(info.line-1, "gutter-offset", textel); diff --git a/src/worker/unfinished.js b/src/worker/unfinished.js new file mode 100644 index 00000000..a36f6067 --- /dev/null +++ b/src/worker/unfinished.js @@ -0,0 +1,321 @@ +"use strict"; + +// TODO: not quite done +function assembleACME(code) { + load("acme"); + // stderr + var re_err2 = /(Error|Warning) - File (.+?), line (\d+) ([^:]+) (.*)/; + var errors = []; + var errline = 0; + function match_fn(s) { + var matches = re_err2.exec(s); + if (matches) { + errors.push({ + line:1, // TODO: parseInt(matches[3]), + msg:matches[0] // TODO: matches[5] + }); + } + } + var Module = ACME({ + noInitialRun:true, + print:match_fn, + printErr:match_fn + }); + var FS = Module['FS']; + FS.writeFile("main.a", code); + // TODO: --msvc + Module.callMain(["-o", "a.out", "-r", "a.rpt", "-l", "a.sym", "--setpc", "24576", "main.a"]); + if (errors.length) { + return {errors:errors}; + } + var aout = FS.readFile("a.out"); + var alst = FS.readFile("a.rpt", {'encoding':'utf8'}); + var asym = FS.readFile("a.sym", {'encoding':'utf8'}); + var listing = parseDASMListing(alst, {}); // TODO + return { + output:aout, + lines:listing.lines, + errors:listing.errors, + intermediate:{listing:alst, symbols:asym}, + }; +} + +function assembleZ80ASM(step) { + load("z80asm"); + var Module = z80asm({ + noInitialRun:true, + //logReadFiles:true, + print:print_fn, + printErr:function() {}, + TOTAL_MEMORY:256*1024*1024, + }); + var FS = Module['FS']; + //setupFS(FS); + // changes for dialect + populateFiles(step, FS, { + mainFilePath:"main.asm", + transform:function(code) { + code = code.replace(".optsdcc -mz80",""); + code = code.replace(/^(\w+)\s*=/gim,"DEFC $1 ="); + code = code.replace(/\tXREF /gi,"\tEXTERN "); + code = code.replace(/\tXDEF /gi,"\tPUBLIC "); + return code; + } + }); + try { + execMain(step, Module, ["-b", "-s", "-l", "-m", "-g", "--origin=" + origin.toString(16), step.path]); + try { + var aerr = FS.readFile(step.prefix+".err", {'encoding':'utf8'}); // TODO + if (aerr.length) { + return {errors:extractErrors(/.+? line (\d+): (.+)/, aerr.split("\n"))}; + } + // Warning at file 'test.asm' line 9: 'XREF' is deprecated, use 'EXTERN' instead + } catch (e) { + } +/* +77 0000 ;test.c:5: return 0; +78 0000 21 00 00 ld hl,$0000 +*/ + var alst = FS.readFile(step.prefix+".lst", {'encoding':'utf8'}); // TODO +/* +_main = 0000, G: test +l_main00101 = 0003, L: test +*/ + var amap = FS.readFile(step.prefix+".map", {'encoding':'utf8'}); // TODO + var aout = FS.readFile(step.prefix+".bin", {'encoding':'binary'}); + var asmlines = parseListing(alst, /^(\d+)\s+([0-9A-F]+)\s+([0-9A-F][0-9A-F ]*[0-9A-F])\s+/i, 1, 2, 3); + var srclines = parseListing(alst, /^(\d+)\s+([0-9A-F]+)\s+;[(]null[)]:(\d+)/i, 3, 2, 1); + // TODO; multiple listing files + return { + output:aout, + errors:[], + lines:asmlines, + srclines:srclines, + intermediate:{listing:alst, mapfile:amap}, + }; + } catch (e) { + throw (e); + } +} + +function compileCASPR(code, platform, options) { + loadNative("caspr"); + var errors = []; + var match_fn = makeErrorMatcher(errors, /(ERROR|FATAL) - (.+)/, 2, 2); + var caspr_mod = caspr({ + wasmBinary:wasmBlob['caspr'], + noInitialRun:true, + print:print_fn, + printErr:match_fn, + }); + var FS = caspr_mod['FS']; + FS.writeFile("main.asm", code); + var arch = code.match(/^[.]arch\s+(\w+)/m); + var deps = [{prefix:'verilog',filename:arch[1]+'.cfg'}]; // TODO: parse file for ".arch femto8" + writeDependencies(deps, FS, errors); + starttime(); + caspr_mod.callMain(["main.asm"]); + endtime("compile"); + var miffile = FS.readFile("main.mif", {encoding:'utf8'}); + // TODO + return { + errors:errors, + output:parseMIF(miffile), + intermediate:{listing:miffile}, + lines:[]}; +} + +function parseMIF(s) { + var lines = s.split('\n'); + var words = []; + for (var i=0; i:(\d+):/i, /^\s*([0-9A-F]{4})/i); + var listings = {}; + for (var fn of step.files) { + if (fn.endsWith('.lst')) { + var rstout = FS.readFile(fn.replace('.lst','.rst'), {encoding:'utf8'}); + // 0000 21 02 00 [10] 52 ld hl, #2 + var asmlines = parseListing(rstout, /^\s*([0-9A-F]+)\s+([0-9A-F][0-9A-F r]*[0-9A-F])\s+\[([0-9 ]+)\]\s+(\d+) (.*)/i, 4, 1, 2); + var srclines = parseSourceLines(rstout, /^\s+\d+ ;:(\d+):/i, /^\s*([0-9A-F]{4})/i); + putWorkFile(fn, rstout); + listings[fn] = { + asmlines:srclines.length ? asmlines : null, + lines:srclines.length ? srclines : asmlines, + text:rstout + }; + } + } // parse symbol map var symbolmap = {}; for (var s of mapout.split("\n")) { @@ -1023,11 +837,10 @@ function linkSDLDZ80(step) putWorkFile("main.ihx", hexout); return { output:parseIHX(hexout, params.rom_start?params.rom_start:params.code_start, params.rom_size), - asmlines:srclines.length?asmlines:null, - lines:srclines.length?srclines:asmlines, - errors:errors, // TODO? + listings:listings, + errors:errors, symbolmap:symbolmap, - intermediate:{listing:rstout}, + //TODO intermediate:{listing:rstout}, }; } } @@ -1093,50 +906,6 @@ function compileSDCC(step) { }; } -function assembleXASM6809(code, platform) { - load("xasm6809"); - var origin = 0; // TODO: configurable - var alst = ""; - var lasterror = null; - var errors = []; - function match_fn(s) { - alst += s; - alst += "\n"; - if (lasterror) { - var line = parseInt(s.slice(0,5)); - errors.push({ - line:line, - msg:lasterror - }); - lasterror = null; - } - else if (s.startsWith("***** ")) { - lasterror = s.slice(6); - } - } - var Module = xasm6809({ - noInitialRun:true, - //logReadFiles:true, - print:match_fn, - printErr:print_fn - }); - var FS = Module['FS']; - //setupFS(FS); - FS.writeFile("main.asm", code); - Module.callMain(["-c", "-l", "-s", "-y", "-o=main.bin", "main.asm"]); - if (errors.length) - return {errors:errors}; - var aout = FS.readFile("main.bin", {encoding:'binary'}); - // 00001 0000 [ 2] 1048 asld - var asmlines = parseListing(alst, /^\s*([0-9A-F]+)\s+([0-9A-F]+)\s+\[([0-9 ]+)\]\s+(\d+) (.*)/i, 1, 2, 4); - return { - output:aout, - errors:errors, - lines:asmlines, - intermediate:{listing:alst}, - }; -} - function preprocessMCPP(code, platform, toolname) { load("mcpp"); var params = PLATFORM_PARAMS[platform]; @@ -1185,34 +954,7 @@ function preprocessMCPP(code, platform, toolname) { return {code:iout}; } -function assembleNAKEN(code, platform) { - load("naken_asm"); - var errors = []; - var match_fn = makeErrorMatcher(errors, /Error: (.+) at (.+):(\d+)/, 3, 1); - var Module = naken_asm({ - noInitialRun:true, - //logReadFiles:true, - print:match_fn, - printErr:print_fn - }); - var FS = Module['FS']; - //setupFS(FS); - FS.writeFile("main.asm", code); - Module.callMain(["-l", "-b", "main.asm"]); - if (errors.length) - return {errors:errors}; - var aout = FS.readFile("out.bin", {encoding:'binary'}); - var alst = FS.readFile("out.lst", {encoding:'utf8'}); - //console.log(alst); - // 0x0000: 77 ld (hl),a cycles: 4 - var asmlines = parseListing(alst, /^0x([0-9a-f]+):\s+([0-9a-f]+)\s+(.+)cycles: (\d+)/i, 0, 1, 2); - return { - output:aout, - errors:errors, - lines:asmlines, - intermediate:{listing:alst}, - }; -} +// TODO: must be a better way to do all this function detectModuleName(code) { var m = /\bmodule\s+(\w+_top)\b/.exec(code) @@ -1258,47 +1000,6 @@ function writeDependencies(depends, FS, errors, callback) { } } -function parseMIF(s) { - var lines = s.split('\n'); - var words = []; - for (var i=0; i