430 lines
12 KiB
JavaScript
Raw Normal View History

2018-02-01 17:35:51 +01:00
#!/usr/bin/env node
const fs = require("fs");
const path = require("path");
const minimist = require("minimist");
const glob = require("glob");
const { SourceMapConsumer, SourceMapGenerator } = require("source-map");
2017-12-05 15:06:44 +01:00
var assemblyscript;
var isDev = true;
try {
assemblyscript = require("../dist/assemblyscript.js");
try { require("source-map-support").install(); } catch (e) {} // optional
2017-12-05 15:06:44 +01:00
isDev = false;
} catch (e) {
require("ts-node").register({ project: require("path").join(__dirname, "..", "src") });
require("../src/glue/js");
assemblyscript = require("../src");
}
const SOURCEMAP_ROOT = "assemblyscript:///";
const LIBRARY_PREFIX = assemblyscript.LIBRARY_PREFIX;
2018-02-01 17:35:51 +01:00
const conf = require("./asc.json");
const opts = {};
2017-12-05 15:06:44 +01:00
Object.keys(conf).forEach(key => {
var opt = conf[key];
if (opt.aliases)
(opts.alias || (opts.alias = {}))[key] = opt.aliases;
if (opt.default !== undefined)
(opts.default || (opts.default = {}))[key] = opt.default;
if (opt.type === "string")
(opts.string || (opts.string = [])).push(key);
else if (opt.type === "boolean")
(opts.boolean || (opts.boolean = [])).push(key);
});
2018-02-01 17:35:51 +01:00
const args = minimist(process.argv.slice(2), opts);
const indent = 24;
2017-12-05 15:06:44 +01:00
var version = require("../package.json").version;
if (isDev) version += "-dev";
if (args.version) {
console.log([
"Version " + version
].join("\n"));
process.exit(0);
}
if (args.help || args._.length < 1) {
2018-02-01 17:35:51 +01:00
const options = [];
2017-12-05 15:06:44 +01:00
Object.keys(conf).forEach(name => {
var option = conf[name];
var text = " ";
2017-12-05 15:06:44 +01:00
text += "--" + name;
if (option.aliases && option.aliases[0].length === 1)
text += ", -" + option.aliases[0];
while (text.length < indent)
2017-12-05 15:06:44 +01:00
text += " ";
if (Array.isArray(option.desc)) {
options.push(text + option.desc[0] + option.desc.slice(1).map(line => {
for (var i = 0; i < indent; ++i)
line = " " + line;
return "\n" + line;
}).join(""));
} else
options.push(text + option.desc);
2017-12-05 15:06:44 +01:00
});
console.log([
"Version " + version,
"Syntax: asc [entryFile ...] [options]",
2017-12-05 15:06:44 +01:00
"",
"Examples: asc hello.ts",
" asc hello.ts -b hello.wasm -t hello.wast",
" asc hello1.ts hello2.ts -b -O > hello.wasm",
2017-12-05 15:06:44 +01:00
"",
"Options:"
].concat(options).join("\n"));
process.exit(args.help ? 0 : 1);
}
2017-12-09 02:52:20 +01:00
var parser = null;
var readTime = 0;
var readCount = 0;
var writeTime = 0;
2018-02-01 17:35:51 +01:00
var writeCount = 0;
var parseTime = 0;
var compileTime = 0;
var validateTime = 0;
var optimizeTime = 0;
function measure(fn) {
var start = process.hrtime();
fn();
var times = process.hrtime(start);
return times[0] * 1e9 + times[1];
}
2017-12-09 02:52:20 +01:00
function checkDiagnostics(parser) {
var diagnostic;
var hasErrors = false;
while ((diagnostic = assemblyscript.nextDiagnostic(parser)) != null) {
console.error(assemblyscript.formatDiagnostic(diagnostic, process.stderr.isTTY, true));
if (assemblyscript.isError(diagnostic))
hasErrors = true;
}
2017-12-09 02:52:20 +01:00
if (hasErrors)
process.exit(1);
}
2017-12-05 15:06:44 +01:00
// Include standard library if --noLib isn't set
var libDirs = args.noLib ? [] : [ path.join(__dirname, "..", "std", "assembly") ];
// Include custom library components (with or without stdlib)
if (args.lib) {
if (Array.isArray(args.lib))
Array.prototype.push.apply(libDirs, args.lib.map(dir));
else
libDirs.push(args.lib);
}
libDirs.forEach(libDir => {
var notIoTime = 0;
readTime += measure(() => {
glob.sync("*.ts", { cwd: libDir }).forEach(file => {
var nextText = fs.readFileSync(path.join(libDir, file), { encoding: "utf8" });
++readCount;
var time = measure(() => {
parser = assemblyscript.parseFile(nextText, LIBRARY_PREFIX + file, parser, false);
});
parseTime += time;
notIoTime += time;
});
}) - notIoTime;
});
2017-12-16 02:27:39 +01:00
// Include entry files
2017-12-09 02:52:20 +01:00
args._.forEach(filename => {
var entryPath = filename.replace(/\\/g, "/").replace(/(\.ts|\/)$/, "");
var entryDir = path.dirname(entryPath);
var entryText;
2017-12-05 15:06:44 +01:00
try {
readTime += measure(() => {
entryText = fs.readFileSync(entryPath + ".ts", { encoding: "utf8" });
2018-02-01 17:35:51 +01:00
entryPath += ".ts";
});
++readCount;
2017-12-05 15:06:44 +01:00
} catch (e) {
try {
readTime += measure(() => {
entryText = fs.readFileSync(entryPath + "/index.ts", { encoding: "utf8" });
2018-02-01 17:35:51 +01:00
entryPath += "/index.ts";
});
++readCount;
2017-12-09 02:52:20 +01:00
entryPath = entryPath + "/index";
} catch (e) {
2017-12-09 02:52:20 +01:00
console.error("File '" + entryPath + ".ts' not found.");
process.exit(1);
}
2017-12-05 15:06:44 +01:00
}
2018-02-01 17:35:51 +01:00
var nextFile;
2017-12-09 02:52:20 +01:00
var nextText;
2017-12-05 15:06:44 +01:00
2017-12-16 02:27:39 +01:00
// Load entry text
parseTime += measure(() => {
parser = assemblyscript.parseFile(entryText, entryPath, parser, true);
});
2017-12-16 02:27:39 +01:00
2018-02-01 17:35:51 +01:00
while ((nextFile = parser.nextFile()) != null) {
var found = false;
if (nextFile.startsWith(LIBRARY_PREFIX)) {
for (var i = 0; i < libDirs.length; ++i) {
readTime += measure(() => {
try {
2018-02-01 17:35:51 +01:00
nextText = fs.readFileSync(path.join(libDirs[i], nextFile.substring(4) + ".ts"), { encoding: "utf8" });
nextFile = nextFile + ".ts";
found = true;
} catch (e) {}
});
++readCount;
if (found)
break;
}
} else {
readTime += measure(() => {
try {
2018-02-01 17:35:51 +01:00
nextText = fs.readFileSync(nextFile + ".ts", { encoding: "utf8" });
nextFile = nextFile + ".ts";
found = true;
} catch (e) {}
});
++readCount;
if (!found) {
readTime += measure(() => {
try {
2018-02-01 17:35:51 +01:00
nextText = fs.readFileSync(nextFile + "/index.ts", { encoding: "utf8" });
nextFile = nextFile + "/index.ts";
found = true;
} catch (e) {}
});
++readCount;
2017-12-09 02:52:20 +01:00
}
}
if (!found) {
2018-02-01 17:35:51 +01:00
console.error("Imported file '" + nextFile + ".ts' not found.");
process.exit(1);
}
parseTime += measure(() => {
2018-02-01 17:35:51 +01:00
assemblyscript.parseFile(nextText, nextFile, parser);
});
2017-12-09 02:52:20 +01:00
}
checkDiagnostics(parser);
});
2017-12-05 15:06:44 +01:00
var options = assemblyscript.createOptions();
assemblyscript.setTarget(options, 0);
assemblyscript.setNoTreeShaking(options, args.noTreeShaking);
2017-12-13 23:24:13 +01:00
assemblyscript.setNoAssert(options, args.noAssert);
assemblyscript.setNoMemory(options, args.noMemory);
2018-02-01 17:35:51 +01:00
assemblyscript.setSourceMap(options, args.sourceMap != null);
var module;
compileTime += measure(() => {
module = assemblyscript.compile(parser, options);
});
2017-12-09 02:52:20 +01:00
checkDiagnostics(parser);
2017-12-05 15:06:44 +01:00
if (args.validate)
validateTime += measure(() => {
if (!module.validate()) {
module.dispose();
2018-02-01 17:35:51 +01:00
console.error("Validation failed");
process.exit(1);
}
});
2017-12-05 15:06:44 +01:00
if (args.trapMode === "clamp")
optimizeTime += measure(() => {
module.runPasses([ "trap-mode-clamp" ]);
});
else if (args.trapMode === "js")
optimizeTime += measure(() => {
module.runPasses([ "trap-mode-js" ]);
});
else if (args.trapMode !== "allow") {
2018-02-01 17:35:51 +01:00
console.error("Unsupported trap mode: " + args.trapMode);
process.exit(1);
}
2018-01-18 06:55:30 +01:00
var defaultOptimizeLevel = 2;
var defaultShrinkLevel = 1;
var optimizeLevel = -1;
var shrinkLevel = 0;
var debugInfo = !args.noDebug;
var runPasses = [];
2018-01-18 06:55:30 +01:00
if (args.optimize !== false) {
if (typeof args.optimize === "number")
optimizeLevel = args.optimize;
else if (args["0"])
optimizeLevel = 0;
else if (args["1"])
optimizeLevel = 1;
else if (args["2"])
optimizeLevel = 2;
else if (args["3"])
optimizeLevel = 3;
2018-01-18 06:55:30 +01:00
else if (args.optimize === true) {
optimizeLevel = defaultOptimizeLevel;
shrinkLevel = defaultShrinkLevel;
} else
optimizeLevel = 0;
}
if (args["s"])
shrinkLevel = 1;
else if (args["z"])
shrinkLevel = 2;
// Check explicit levels
if (typeof args.optimizeLevel === "number")
optimizeLevel = args.optimizeLevel;
if (typeof args.shrinkLevel === "number")
shrinkLevel = args.shrinkLevel;
// Check additional passes
2018-01-16 17:52:48 +01:00
if (args.runPasses) {
if (typeof args.runPasses === "string")
args.runPasses = args.runPasses.split(",");
if (args.runPasses.length)
args.runPasses.forEach(pass => {
if (runPasses.indexOf(pass) < 0)
runPasses.push(pass);
});
2018-01-16 17:52:48 +01:00
}
module.setOptimizeLevel(optimizeLevel);
module.setShrinkLevel(shrinkLevel);
module.setDebugInfo(debugInfo);
2018-01-18 06:55:30 +01:00
if (optimizeLevel >= 0)
optimizeTime += measure(() => {
module.optimize();
});
if (runPasses.length)
optimizeTime += measure(() => {
module.runPasses(runPasses.map(pass => pass.trim()));
});
2018-02-01 17:35:51 +01:00
function processSourceMap(sourceMap, sourceMapURL) {
var json = JSON.parse(sourceMap);
json.sources = json.sources.map(name => SOURCEMAP_ROOT + name);
var libPrefix = SOURCEMAP_ROOT + LIBRARY_PREFIX;
return SourceMapConsumer.with(json, sourceMapURL, consumer => {
2018-02-01 17:35:51 +01:00
var generator = SourceMapGenerator.fromSourceMap(consumer);
json.sources.forEach(name => {
var text, found = false;
if (name.startsWith(libPrefix)) {
2018-02-01 17:35:51 +01:00
for (var i = 0, k = libDirs.length; i < k; ++i) {
readTime += measure(() => {
try {
text = fs.readFileSync(path.join(libDirs[i], name.substring(libPrefix.length)), { encoding: "utf8" });
2018-02-01 17:35:51 +01:00
found = true;
} catch (e) {}
});
++readCount;
}
} else {
readTime += measure(() => {
try {
text = fs.readFileSync(name.substring(SOURCEMAP_ROOT.length), { encoding: "utf8" });
2018-02-01 17:35:51 +01:00
found = true;
} catch (e) {}
});
++readCount;
}
if (found) {
2018-02-01 17:35:51 +01:00
generator.setSourceContent(name, text);
console.log(name + " ???");
} else
2018-02-01 17:35:51 +01:00
console.error("No source content found for file '" + name + "'.");
});
return generator.toString();
});
}
if (!args.noEmit) {
var hasOutput = false;
2017-12-05 15:06:44 +01:00
if (args.outFile != null) {
if (/\.wast$/.test(args.outFile) && args.textFile == null)
args.textFile = args.outFile;
else if (/\.js$/.test(args.outFile) && args.asmjsFile == null)
args.asmjsFile = args.outFile;
else if (args.binaryFile == null)
args.binaryFile = args.outFile;
}
if (args.binaryFile != null && args.binaryFile.length) {
2018-02-01 17:35:51 +01:00
var sourceMapURL = args.sourceMap != null
? args.sourceMap.length
? args.sourceMap
: path.basename(args.binaryFile) + ".map"
: null;
var binary;
writeTime += measure(() => {
binary = module.toBinary("http://127.0.0.1:8080/"+sourceMapURL); // FIXME: 'not a valid URL' in FF
2018-02-01 17:35:51 +01:00
fs.writeFileSync(args.binaryFile, binary.output);
});
2018-02-01 17:35:51 +01:00
++writeCount;
if (binary.sourceMap != null)
processSourceMap(binary.sourceMap).then(sourceMap => {
writeTime += measure(() => {
fs.writeFileSync(path.join(path.dirname(args.binaryFile), path.basename(sourceMapURL)), sourceMap, { encoding: "utf8" });
}, err => {
throw err;
});
++writeCount;
});
hasOutput = true;
}
if (args.textFile != null && args.textFile.length) {
writeTime += measure(() => {
fs.writeFileSync(args.textFile, module.toText(), { encoding: "utf8" });
});
2018-02-01 17:35:51 +01:00
++writeCount;
hasOutput = true;
}
if (args.asmjsFile != null && args.asmjsFile.length) {
writeTime += measure(() => {
fs.writeFileSync(args.asmjsFile, module.toAsmjs(), { encoding: "utf8" });
});
2018-02-01 17:35:51 +01:00
++writeCount;
hasOutput = true;
}
if (!hasOutput) {
2018-02-01 17:35:51 +01:00
if (args.binaryFile === "") {
writeTime += measure(() => {
process.stdout.write(Buffer.from(module.toBinary()));
});
2018-02-01 17:35:51 +01:00
++writeCount;
} else if (args.asmjsFile === "") {
writeTime += measure(() => {
module.printAsmjs();
});
2018-02-01 17:35:51 +01:00
++writeCount;
} else {
writeTime += measure(() => {
module.print();
});
2018-02-01 17:35:51 +01:00
++writeCount;
}
}
2017-12-09 00:45:12 +01:00
}
2017-12-05 15:06:44 +01:00
module.dispose();
2018-02-01 17:35:51 +01:00
if (args.measure) process.on("beforeExit", () => console.error([
"I/O Read : " + (readTime ? (readTime / 1e6).toFixed(3) + " ms (" + readCount + " files)" : "N/A"),
"I/O Write : " + (writeTime ? (writeTime / 1e6).toFixed(3) + " ms (" + writeCount + " files)" : "N/A"),
"Parse : " + (parseTime ? (parseTime / 1e6).toFixed(3) + " ms" : "N/A"),
"Compile : " + (compileTime ? (compileTime / 1e6).toFixed(3) + " ms" : "N/A"),
"Validate : " + (validateTime ? (validateTime / 1e6).toFixed(3) + " ms" : "N/A"),
"Optimize : " + (optimizeTime ? (optimizeTime / 1e6).toFixed(3) + " ms" : "N/A")
].join("\n")));