mirror of
https://github.com/fluencelabs/assemblyscript
synced 2025-06-13 23:11:41 +00:00
Add CLI definitions; CLI restructure
This commit is contained in:
@ -1,48 +0,0 @@
|
||||
Compiler frontend for node.js
|
||||
=============================
|
||||
|
||||
Usage
|
||||
-----
|
||||
|
||||
For an up to date list of available command line options, see:
|
||||
|
||||
```
|
||||
$> asc --help
|
||||
```
|
||||
|
||||
API
|
||||
---
|
||||
|
||||
The API accepts the same options as the CLI but also lets you override stdout and stderr and/or provide a callback. Example:
|
||||
|
||||
```js
|
||||
const asc = require("assemblyscript/bin/asc");
|
||||
asc.main([
|
||||
"myModule.ts",
|
||||
"--binaryFile", "myModule.wasm",
|
||||
"--optimize",
|
||||
"--sourceMap",
|
||||
"--measure"
|
||||
], {
|
||||
stdout: process.stdout,
|
||||
stderr: process.stderr
|
||||
}, function(err) {
|
||||
if (err)
|
||||
throw err;
|
||||
...
|
||||
});
|
||||
```
|
||||
|
||||
Available command line options can also be obtained programmatically:
|
||||
|
||||
```js
|
||||
const options = require("assemblyscript/bin/asc.json");
|
||||
...
|
||||
```
|
||||
|
||||
You can also compile a source string directly, for example in a browser environment:
|
||||
|
||||
```js
|
||||
const { binary, text, stdout, stderr } = asc.compileString(`...`, { optimize: 2 });
|
||||
...
|
||||
```
|
6
bin/asc
6
bin/asc
@ -1,5 +1,3 @@
|
||||
#!/usr/bin/env node
|
||||
const asc = module.exports = require("./asc.js");
|
||||
if (/\basc$/.test(process.argv[1])) {
|
||||
process.exitCode = asc.main(process.argv.slice(2));
|
||||
}
|
||||
const asc = module.exports = require("../cli/asc.js");
|
||||
if (/\basc$/.test(process.argv[1])) process.exitCode = asc.main(process.argv.slice(2));
|
||||
|
892
bin/asc.js
892
bin/asc.js
@ -1,892 +0,0 @@
|
||||
/**
|
||||
* Compiler frontend for node.js
|
||||
*
|
||||
* Uses the low-level API exported from src/index.ts so it works with the compiler compiled to
|
||||
* JavaScript as well as the compiler compiled to WebAssembly (eventually). Runs the sources
|
||||
* directly through ts-node if distribution files are not present (indicated by a `-dev` version).
|
||||
*
|
||||
* Can also be packaged as a bundle suitable for in-browser use with the standard library injected
|
||||
* in the build step. See dist/asc.js for the bundle and webpack.config.js for building details.
|
||||
*
|
||||
* @module asc
|
||||
*/
|
||||
|
||||
const fs = require("fs");
|
||||
const path = require("path");
|
||||
const utf8 = require("@protobufjs/utf8");
|
||||
const EOL = process.platform === "win32" ? "\r\n" : "\n";
|
||||
|
||||
// Use distribution files if present, otherwise run the sources directly
|
||||
var assemblyscript, isDev;
|
||||
(() => {
|
||||
try {
|
||||
assemblyscript = require("../dist/assemblyscript.js");
|
||||
isDev = false;
|
||||
try { require("source-map-support").install(); } catch (e) {/* optional */}
|
||||
} catch (e) {
|
||||
try {
|
||||
require("ts-node").register({ project: path.join(__dirname, "..", "src", "tsconfig.json") });
|
||||
require("../src/glue/js");
|
||||
assemblyscript = require("../src");
|
||||
isDev = true;
|
||||
} catch (e) {
|
||||
// last resort: same directory CommonJS
|
||||
assemblyscript = eval("require('./assemblyscript')");
|
||||
isDev = false;
|
||||
}
|
||||
}
|
||||
})();
|
||||
|
||||
/** Whether this is a webpack bundle or not. */
|
||||
exports.isBundle = typeof BUNDLE_VERSION === "string";
|
||||
|
||||
/** Whether asc runs the sources directly or not. */
|
||||
exports.isDev = isDev;
|
||||
|
||||
/** AssemblyScript veresion. */
|
||||
exports.version = exports.isBundle ? BUNDLE_VERSION : require("../package.json").version;
|
||||
|
||||
/** Available options. */
|
||||
exports.options = require("./asc.json");
|
||||
|
||||
/** Common root used in source maps. */
|
||||
exports.sourceMapRoot = "assemblyscript:///";
|
||||
|
||||
/** Prefix used for library files. */
|
||||
exports.libraryPrefix = assemblyscript.LIBRARY_PREFIX;
|
||||
|
||||
/** Default Binaryen optimization level. */
|
||||
exports.defaultOptimizeLevel = 2;
|
||||
|
||||
/** Default Binaryen shrink level. */
|
||||
exports.defaultShrinkLevel = 1;
|
||||
|
||||
/** Bundled library files. */
|
||||
exports.libraryFiles = exports.isBundle ? BUNDLE_LIBRARY : (() => { // set up if not a bundle
|
||||
const libDir = path.join(__dirname, "..", "std", "assembly");
|
||||
const libFiles = require("glob").sync("**/*.ts", { cwd: libDir });
|
||||
const bundled = {};
|
||||
libFiles.forEach(file => bundled[file.replace(/\.ts$/, "")] = fs.readFileSync(path.join(libDir, file), "utf8" ));
|
||||
return bundled;
|
||||
})();
|
||||
|
||||
/** Bundled definition files. */
|
||||
exports.definitionFiles = exports.isBundle ? BUNDLE_DEFINITIONS : (() => { // set up if not a bundle
|
||||
const stdDir = path.join(__dirname, "..", "std");
|
||||
return {
|
||||
"assembly": fs.readFileSync(path.join(stdDir, "assembly.d.ts"), "utf8"),
|
||||
"portable": fs.readFileSync(path.join(stdDir, "portable.d.ts"), "utf8")
|
||||
};
|
||||
})();
|
||||
|
||||
/** Convenience function that parses and compiles source strings directly. */
|
||||
exports.compileString = (sources, options) => {
|
||||
if (typeof sources === "string") sources = { "input.ts": sources };
|
||||
const output = Object.create({
|
||||
stdout: createMemoryStream(),
|
||||
stderr: createMemoryStream(),
|
||||
binary: null,
|
||||
text: null
|
||||
});
|
||||
exports.main([
|
||||
"--binaryFile", "binary",
|
||||
"--textFile", "text",
|
||||
...Object.keys(options || {}).map(arg => `--${arg}=${options[arg]}`),
|
||||
...Object.keys(sources),
|
||||
], {
|
||||
stdout: output.stdout,
|
||||
stderr: output.stderr,
|
||||
readFile: name => sources.hasOwnProperty(name) ? sources[name] : null,
|
||||
writeFile: (name, contents) => output[name] = contents,
|
||||
listFiles: () => []
|
||||
});
|
||||
return output;
|
||||
}
|
||||
|
||||
/** Runs the command line utility using the specified arguments array. */
|
||||
exports.main = function main(argv, options, callback) {
|
||||
if (typeof options === "function") {
|
||||
callback = options;
|
||||
options = {};
|
||||
} else if (!options) {
|
||||
options = {};
|
||||
}
|
||||
|
||||
const stdout = options.stdout || process.stdout;
|
||||
const stderr = options.stderr || process.stderr;
|
||||
const readFile = options.readFile || readFileNode;
|
||||
const writeFile = options.writeFile || writeFileNode;
|
||||
const listFiles = options.listFiles || listFilesNode;
|
||||
const stats = options.stats || createStats();
|
||||
|
||||
// All of the above must be specified in browser environments
|
||||
if (!stdout) throw Error("'options.stdout' must be specified");
|
||||
if (!stderr) throw Error("'options.stderr' must be specified");
|
||||
if (!fs.readFileSync) {
|
||||
if (readFile === readFileNode) throw Error("'options.readFile' must be specified");
|
||||
if (writeFile === writeFileNode) throw Error("'options.writeFile' must be specified");
|
||||
if (listFiles === listFilesNode) throw Error("'options.listFiles' must be specified");
|
||||
}
|
||||
|
||||
const args = parseArguments(argv);
|
||||
const indent = 24;
|
||||
|
||||
// Use default callback if none is provided
|
||||
if (!callback) callback = function defaultCallback(err) {
|
||||
var code = 0;
|
||||
if (err) {
|
||||
stderr.write(err.stack + EOL);
|
||||
code = 1;
|
||||
}
|
||||
return code;
|
||||
};
|
||||
|
||||
// Just print the version if requested
|
||||
if (args.version) {
|
||||
stdout.write("Version " + exports.version + (isDev ? "-dev" : "") + EOL);
|
||||
return callback(null);
|
||||
}
|
||||
// Print the help message if requested or no source files are provided
|
||||
if (args.help || args._.length < 1) {
|
||||
const opts = [];
|
||||
Object.keys(exports.options).forEach(name => {
|
||||
var option = exports.options[name];
|
||||
var text = " ";
|
||||
text += "--" + name;
|
||||
if (option.aliases && option.aliases[0].length === 1) {
|
||||
text += ", -" + option.aliases[0];
|
||||
}
|
||||
while (text.length < indent) {
|
||||
text += " ";
|
||||
}
|
||||
if (Array.isArray(option.desc)) {
|
||||
opts.push(text + option.desc[0] + option.desc.slice(1).map(line => {
|
||||
for (let i = 0; i < indent; ++i) {
|
||||
line = " " + line;
|
||||
}
|
||||
return EOL + line;
|
||||
}).join(""));
|
||||
} else {
|
||||
opts.push(text + option.desc);
|
||||
}
|
||||
});
|
||||
|
||||
(args.help ? stdout : stderr).write([
|
||||
"Version " + exports.version + (isDev ? "-dev" : ""),
|
||||
"Syntax: asc [entryFile ...] [options]",
|
||||
"",
|
||||
"Examples: asc hello.ts",
|
||||
" asc hello.ts -b hello.wasm -t hello.wat",
|
||||
" asc hello1.ts hello2.ts -b -O > hello.wasm",
|
||||
"",
|
||||
"Options:"
|
||||
].concat(opts).join(EOL) + EOL);
|
||||
return callback(null);
|
||||
}
|
||||
|
||||
// Set up base directory
|
||||
const baseDir = args.baseDir ? path.resolve(args.baseDir) : ".";
|
||||
|
||||
// Set up transforms
|
||||
const transforms = [];
|
||||
if (args.transform) {
|
||||
if (typeof args.transform === "string") args.transform = args.transform.split(",");
|
||||
args.transform.forEach(transform =>
|
||||
transforms.push(
|
||||
require(
|
||||
path.isAbsolute(transform = transform.trim())
|
||||
? transform
|
||||
: path.join(process.cwd(), transform)
|
||||
)
|
||||
)
|
||||
);
|
||||
}
|
||||
function applyTransform(name, ...args) {
|
||||
transforms.forEach(transform => {
|
||||
if (typeof transform[name] === "function") transform[name](...args);
|
||||
});
|
||||
}
|
||||
|
||||
// Begin parsing
|
||||
var parser = null;
|
||||
|
||||
// Include library files
|
||||
if (!args.noLib) { // bundled
|
||||
Object.keys(exports.libraryFiles).forEach(libPath => {
|
||||
if (libPath.indexOf("/") >= 0) return; // in sub-directory: imported on demand
|
||||
stats.parseCount++;
|
||||
stats.parseTime += measure(() => {
|
||||
parser = assemblyscript.parseFile(
|
||||
exports.libraryFiles[libPath],
|
||||
exports.libraryPrefix + libPath + ".ts",
|
||||
false,
|
||||
parser
|
||||
);
|
||||
});
|
||||
});
|
||||
}
|
||||
const customLibDirs = [];
|
||||
if (args.lib) {
|
||||
if (typeof args.lib === "string") args.lib = args.lib.split(",");
|
||||
Array.prototype.push.apply(customLibDirs, args.lib.map(lib => lib.trim()));
|
||||
for (let i = 0, k = customLibDirs.length; i < k; ++i) { // custom
|
||||
let libDir = customLibDirs[i];
|
||||
let libFiles;
|
||||
if (libDir.endsWith(".ts")) {
|
||||
libFiles = [ path.basename(libDir) ];
|
||||
libDir = path.dirname(libDir);
|
||||
} else {
|
||||
libFiles = listFiles(libDir);
|
||||
}
|
||||
for (let j = 0, l = libFiles.length; j < l; ++j) {
|
||||
let libPath = libFiles[j];
|
||||
let libText = readFile(path.join(libDir, libPath));
|
||||
if (libText === null) return callback(Error("Library file '" + libPath + "' not found."));
|
||||
stats.parseCount++;
|
||||
stats.parseTime += measure(() => {
|
||||
parser = assemblyscript.parseFile(
|
||||
libText,
|
||||
exports.libraryPrefix + libPath,
|
||||
false,
|
||||
parser
|
||||
);
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Include entry files
|
||||
for (let i = 0, k = args._.length; i < k; ++i) {
|
||||
const filename = args._[i];
|
||||
|
||||
let sourcePath = filename.replace(/\\/g, "/").replace(/(\.ts|\/)$/, "");
|
||||
|
||||
// Try entryPath.ts, then entryPath/index.ts
|
||||
let sourceText = readFile(path.join(baseDir, sourcePath) + ".ts");
|
||||
if (sourceText === null) {
|
||||
sourceText = readFile(path.join(baseDir, sourcePath, "index.ts"));
|
||||
if (sourceText === null) {
|
||||
return callback(Error("Entry file '" + sourcePath + ".ts' not found."));
|
||||
} else {
|
||||
sourcePath += "/index.ts";
|
||||
}
|
||||
} else {
|
||||
sourcePath += ".ts";
|
||||
}
|
||||
|
||||
stats.parseCount++;
|
||||
stats.parseTime += measure(() => {
|
||||
parser = assemblyscript.parseFile(sourceText, sourcePath, true, parser);
|
||||
});
|
||||
|
||||
// Process backlog
|
||||
while ((sourcePath = parser.nextFile()) != null) {
|
||||
let found = false;
|
||||
|
||||
// Load library file if explicitly requested
|
||||
if (sourcePath.startsWith(exports.libraryPrefix)) {
|
||||
const plainName = sourcePath.substring(exports.libraryPrefix.length);
|
||||
const indexName = sourcePath.substring(exports.libraryPrefix.length) + "/index";
|
||||
if (exports.libraryFiles.hasOwnProperty(plainName)) {
|
||||
sourceText = exports.libraryFiles[plainName];
|
||||
sourcePath = exports.libraryPrefix + plainName + ".ts";
|
||||
} else if (exports.libraryFiles.hasOwnProperty(indexName)) {
|
||||
sourceText = exports.libraryFiles[indexName];
|
||||
sourcePath = exports.libraryPrefix + indexName + ".ts";
|
||||
} else {
|
||||
for (let i = 0, k = customLibDirs.length; i < k; ++i) {
|
||||
const dir = customLibDirs[i];
|
||||
sourceText = readFile(path.join(dir, plainName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = exports.libraryPrefix + plainName + ".ts";
|
||||
break;
|
||||
} else {
|
||||
sourceText = readFile(path.join(dir, indexName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = exports.libraryPrefix + indexName + ".ts";
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Otherwise try nextFile.ts, nextFile/index.ts, ~lib/nextFile.ts, ~lib/nextFile/index.ts
|
||||
} else {
|
||||
const plainName = sourcePath;
|
||||
const indexName = sourcePath + "/index";
|
||||
sourceText = readFile(path.join(baseDir, plainName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = plainName + ".ts";
|
||||
} else {
|
||||
sourceText = readFile(path.join(baseDir, indexName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = indexName + ".ts";
|
||||
} else if (!plainName.startsWith(".")) {
|
||||
if (exports.libraryFiles.hasOwnProperty(plainName)) {
|
||||
sourceText = exports.libraryFiles[plainName];
|
||||
sourcePath = exports.libraryPrefix + plainName + ".ts";
|
||||
} else if (exports.libraryFiles.hasOwnProperty(indexName)) {
|
||||
sourceText = exports.libraryFiles[indexName];
|
||||
sourcePath = exports.libraryPrefix + indexName + ".ts";
|
||||
} else {
|
||||
for (let i = 0, k = customLibDirs.length; i < k; ++i) {
|
||||
const dir = customLibDirs[i];
|
||||
sourceText = readFile(path.join(dir, plainName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = exports.libraryPrefix + plainName + ".ts";
|
||||
break;
|
||||
} else {
|
||||
sourceText = readFile(path.join(dir, indexName + ".ts"));
|
||||
if (sourceText !== null) {
|
||||
sourcePath = exports.libraryPrefix + indexName + ".ts";
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (sourceText == null) {
|
||||
return callback(Error("Import file '" + sourcePath + ".ts' not found."));
|
||||
}
|
||||
stats.parseCount++;
|
||||
stats.parseTime += measure(() => {
|
||||
assemblyscript.parseFile(sourceText, sourcePath, false, parser);
|
||||
});
|
||||
}
|
||||
if (checkDiagnostics(parser, stderr)) {
|
||||
return callback(Error("Parse error"));
|
||||
}
|
||||
}
|
||||
|
||||
applyTransform("afterParse", parser);
|
||||
|
||||
// Finish parsing
|
||||
const program = assemblyscript.finishParsing(parser);
|
||||
|
||||
// Begin compilation
|
||||
const compilerOptions = assemblyscript.createOptions();
|
||||
assemblyscript.setTarget(compilerOptions, 0);
|
||||
assemblyscript.setNoTreeShaking(compilerOptions, !!args.noTreeShaking);
|
||||
assemblyscript.setNoAssert(compilerOptions, !!args.noAssert);
|
||||
assemblyscript.setNoMemory(compilerOptions, !!args.noMemory);
|
||||
assemblyscript.setImportMemory(compilerOptions, !!args.importMemory);
|
||||
assemblyscript.setImportTable(compilerOptions, !!args.importTable);
|
||||
assemblyscript.setMemoryBase(compilerOptions, args.memoryBase >>> 0);
|
||||
assemblyscript.setSourceMap(compilerOptions, args.sourceMap != null);
|
||||
|
||||
// Initialize default aliases
|
||||
assemblyscript.setGlobalAlias(compilerOptions, "Math", "NativeMath");
|
||||
assemblyscript.setGlobalAlias(compilerOptions, "Mathf", "NativeMathf");
|
||||
assemblyscript.setGlobalAlias(compilerOptions, "abort", "~lib/env/abort"); // to disable: --use abort=
|
||||
|
||||
// Add or override aliases if specified
|
||||
var aliases = args.use;
|
||||
if (aliases != null) {
|
||||
if (typeof aliases === "string") aliases = aliases.split(",");
|
||||
for (let i = 0, k = aliases.length; i < k; ++i) {
|
||||
let part = aliases[i];
|
||||
let p = part.indexOf("=");
|
||||
if (p < 0) return callback(Error("Global alias '" + part + "' is invalid."));
|
||||
let name = part.substring(0, p).trim();
|
||||
let alias = part.substring(p + 1).trim();
|
||||
if (!name.length) return callback(Error("Global alias '" + part + "' is invalid."));
|
||||
assemblyscript.setGlobalAlias(compilerOptions, name, alias);
|
||||
}
|
||||
}
|
||||
|
||||
// Enable additional features if specified
|
||||
var features = args.enable;
|
||||
if (features != null) {
|
||||
if (typeof features === "string") features = features.split(",");
|
||||
for (let i = 0, k = features.length; i < k; ++i) {
|
||||
let name = features[i].trim();
|
||||
let flag = assemblyscript["FEATURE_" + name.replace(/\-/g, "_").toUpperCase()];
|
||||
if (!flag) return callback(Error("Feature '" + name + "' is unknown."));
|
||||
assemblyscript.enableFeature(compilerOptions, flag);
|
||||
}
|
||||
}
|
||||
|
||||
var module;
|
||||
stats.compileCount++;
|
||||
(() => {
|
||||
try {
|
||||
stats.compileTime += measure(() => {
|
||||
module = assemblyscript.compileProgram(program, compilerOptions);
|
||||
});
|
||||
} catch (e) {
|
||||
return callback(e);
|
||||
}
|
||||
})();
|
||||
if (checkDiagnostics(parser, stderr)) {
|
||||
if (module) module.dispose();
|
||||
return callback(Error("Compile error"));
|
||||
}
|
||||
|
||||
// Validate the module if requested
|
||||
if (args.validate) {
|
||||
stats.validateCount++;
|
||||
stats.validateTime += measure(() => {
|
||||
if (!module.validate()) {
|
||||
module.dispose();
|
||||
return callback(Error("Validate error"));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Set Binaryen-specific options
|
||||
if (args.trapMode === "clamp") {
|
||||
stats.optimizeCount++;
|
||||
stats.optimizeTime += measure(() => {
|
||||
module.runPasses([ "trap-mode-clamp" ]);
|
||||
});
|
||||
} else if (args.trapMode === "js") {
|
||||
stats.optimizeCount++;
|
||||
stats.optimizeTime += measure(() => {
|
||||
module.runPasses([ "trap-mode-js" ]);
|
||||
});
|
||||
} else if (args.trapMode !== "allow") {
|
||||
module.dispose();
|
||||
return callback(Error("Unsupported trap mode"));
|
||||
}
|
||||
|
||||
var optimizeLevel = -1;
|
||||
var shrinkLevel = 0;
|
||||
var debugInfo = !args.noDebug;
|
||||
|
||||
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;
|
||||
} else if (args.optimize === true) {
|
||||
optimizeLevel = exports.defaultOptimizeLevel;
|
||||
shrinkLevel = exports.defaultShrinkLevel;
|
||||
} else
|
||||
optimizeLevel = 0;
|
||||
}
|
||||
|
||||
if (args["s"]) {
|
||||
shrinkLevel = 1;
|
||||
} else if (args["z"]) {
|
||||
shrinkLevel = 2;
|
||||
}
|
||||
|
||||
if (typeof args.optimizeLevel === "number") {
|
||||
optimizeLevel = args.optimizeLevel;
|
||||
}
|
||||
|
||||
if (typeof args.shrinkLevel === "number") {
|
||||
shrinkLevel = args.shrinkLevel;
|
||||
} else if (args.shrinkLevel === "s") {
|
||||
shrinkLevel = 1;
|
||||
} else if (args.shrinkLevel === "z") {
|
||||
shrinkLevel = 2;
|
||||
}
|
||||
|
||||
module.setOptimizeLevel(optimizeLevel > 0 ? optimizeLevel : 0);
|
||||
module.setShrinkLevel(shrinkLevel);
|
||||
module.setDebugInfo(debugInfo);
|
||||
|
||||
var runPasses = [];
|
||||
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);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// Optimize the module if requested
|
||||
if (optimizeLevel >= 0) {
|
||||
stats.optimizeCount++;
|
||||
stats.optimizeTime += measure(() => {
|
||||
module.optimize();
|
||||
});
|
||||
}
|
||||
|
||||
// Run additional passes if requested
|
||||
if (runPasses.length) {
|
||||
stats.optimizeCount++;
|
||||
stats.optimizeTime += measure(() => {
|
||||
module.runPasses(runPasses.map(pass => pass.trim()));
|
||||
});
|
||||
}
|
||||
|
||||
// Prepare output
|
||||
if (!args.noEmit) {
|
||||
let hasStdout = false;
|
||||
let hasOutput = false;
|
||||
|
||||
if (args.outFile != null) {
|
||||
if (/\.was?t$/.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;
|
||||
}
|
||||
}
|
||||
|
||||
// Write binary
|
||||
if (args.binaryFile != null) {
|
||||
let sourceMapURL = args.sourceMap != null
|
||||
? args.sourceMap.length
|
||||
? args.sourceMap
|
||||
: path.basename(args.binaryFile) + ".map"
|
||||
: null;
|
||||
|
||||
let wasm;
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
wasm = module.toBinary(sourceMapURL)
|
||||
});
|
||||
|
||||
if (args.binaryFile.length) {
|
||||
writeFile(path.join(baseDir, args.binaryFile), wasm.output);
|
||||
} else {
|
||||
writeStdout(wasm.output);
|
||||
hasStdout = true;
|
||||
}
|
||||
hasOutput = true;
|
||||
|
||||
// Post-process source map
|
||||
if (wasm.sourceMap != null) {
|
||||
if (args.binaryFile.length) {
|
||||
let sourceMap = JSON.parse(wasm.sourceMap);
|
||||
sourceMap.sourceRoot = exports.sourceMapRoot;
|
||||
sourceMap.sources.forEach((name, index) => {
|
||||
let text = null;
|
||||
if (name.startsWith(exports.libraryPrefix)) {
|
||||
let stdName = name.substring(exports.libraryPrefix.length).replace(/\.ts$/, "");
|
||||
if (exports.libraryFiles.hasOwnProperty(stdName)) {
|
||||
text = exports.libraryFiles[stdName];
|
||||
} else {
|
||||
for (let i = 0, k = customLibDirs.length; i < k; ++i) {
|
||||
text = readFile(path.join(
|
||||
customLibDirs[i],
|
||||
name.substring(exports.libraryPrefix.length))
|
||||
);
|
||||
if (text !== null) break;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
text = readFile(path.join(baseDir, name));
|
||||
}
|
||||
if (text === null) {
|
||||
return callback(Error("Source file '" + name + "' not found."));
|
||||
}
|
||||
if (!sourceMap.sourceContents) sourceMap.sourceContents = [];
|
||||
sourceMap.sourceContents[index] = text;
|
||||
});
|
||||
writeFile(path.join(
|
||||
baseDir,
|
||||
path.dirname(args.binaryFile),
|
||||
path.basename(sourceMapURL)
|
||||
), JSON.stringify(sourceMap));
|
||||
} else {
|
||||
stderr.write("Skipped source map (stdout already occupied)" + EOL);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Write asm.js
|
||||
if (args.asmjsFile != null) {
|
||||
let asm;
|
||||
if (args.asmjsFile.length) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
asm = module.toAsmjs();
|
||||
});
|
||||
writeFile(path.join(baseDir, args.asmjsFile), asm);
|
||||
} else if (!hasStdout) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
asm = module.toAsmjs();
|
||||
});
|
||||
writeStdout(asm);
|
||||
hasStdout = true;
|
||||
}
|
||||
hasOutput = true;
|
||||
}
|
||||
|
||||
// Write WebIDL
|
||||
if (args.idlFile != null) {
|
||||
let idl;
|
||||
if (args.idlFile.length) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
idl = assemblyscript.buildIDL(program);
|
||||
});
|
||||
writeFile(path.join(baseDir, args.idlFile), idl);
|
||||
} else if (!hasStdout) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
idl = assemblyscript.buildIDL(program);
|
||||
});
|
||||
writeStdout(idl);
|
||||
hasStdout = true;
|
||||
}
|
||||
hasOutput = true;
|
||||
}
|
||||
|
||||
// Write TypeScript definition
|
||||
if (args.tsdFile != null) {
|
||||
let tsd;
|
||||
if (args.tsdFile.length) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
tsd = assemblyscript.buildTSD(program);
|
||||
});
|
||||
writeFile(path.join(baseDir, args.tsdFile), tsd);
|
||||
} else if (!hasStdout) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
tsd = assemblyscript.buildTSD(program);
|
||||
});
|
||||
writeStdout(tsd);
|
||||
hasStdout = true;
|
||||
}
|
||||
hasOutput = true;
|
||||
}
|
||||
|
||||
// Write text (must be last)
|
||||
if (args.textFile != null || !hasOutput) {
|
||||
let wat;
|
||||
if (args.textFile && args.textFile.length) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
wat = module.toText();
|
||||
});
|
||||
writeFile(path.join(baseDir, args.textFile), wat);
|
||||
} else if (!hasStdout) {
|
||||
stats.emitCount++;
|
||||
stats.emitTime += measure(() => {
|
||||
wat = module.toText()
|
||||
});
|
||||
writeStdout(wat);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
module.dispose();
|
||||
if (args.measure) {
|
||||
printStats(stats, stderr);
|
||||
}
|
||||
return callback(null);
|
||||
|
||||
function readFileNode(filename) {
|
||||
try {
|
||||
let text;
|
||||
stats.readCount++;
|
||||
stats.readTime += measure(() => {
|
||||
text = fs.readFileSync(filename, { encoding: "utf8" });
|
||||
});
|
||||
return text;
|
||||
} catch (e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
function writeFileNode(filename, contents) {
|
||||
try {
|
||||
stats.writeCount++;
|
||||
stats.writeTime += measure(() => {
|
||||
if (typeof contents === "string") {
|
||||
fs.writeFileSync(filename, contents, { encoding: "utf8" } );
|
||||
} else {
|
||||
fs.writeFileSync(filename, contents);
|
||||
}
|
||||
});
|
||||
return true;
|
||||
} catch (e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
function listFilesNode(dirname) {
|
||||
var files;
|
||||
try {
|
||||
stats.readTime += measure(() => {
|
||||
files = require("glob").sync("*.ts", { cwd: dirname });
|
||||
});
|
||||
return files;
|
||||
} catch (e) {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
function writeStdout(contents) {
|
||||
if (!writeStdout.used) {
|
||||
stats.writeCount++;
|
||||
writeStdout.used = true;
|
||||
}
|
||||
stats.writeTime += measure(() => {
|
||||
if (typeof contents === "string") {
|
||||
stdout.write(contents, { encoding: "utf8" });
|
||||
} else {
|
||||
stdout.write(contents);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/** Parses the specified command line arguments. */
|
||||
function parseArguments(argv) {
|
||||
const opts = {};
|
||||
Object.keys(exports.options).forEach(key => {
|
||||
const opt = exports.options[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);
|
||||
}
|
||||
});
|
||||
return require("minimist")(argv, opts);
|
||||
}
|
||||
|
||||
exports.parseArguments = parseArguments;
|
||||
|
||||
/** Checks diagnostics emitted so far for errors. */
|
||||
function checkDiagnostics(emitter, stderr) {
|
||||
var diagnostic;
|
||||
var hasErrors = false;
|
||||
while ((diagnostic = assemblyscript.nextDiagnostic(emitter)) != null) {
|
||||
stderr.write(
|
||||
assemblyscript.formatDiagnostic(diagnostic, stderr.isTTY, true) +
|
||||
EOL + EOL
|
||||
);
|
||||
if (assemblyscript.isError(diagnostic)) hasErrors = true;
|
||||
}
|
||||
return hasErrors;
|
||||
}
|
||||
|
||||
exports.checkDiagnostics = checkDiagnostics;
|
||||
|
||||
/** Creates an empty set of stats. */
|
||||
function createStats() {
|
||||
return {
|
||||
readTime: 0,
|
||||
readCount: 0,
|
||||
writeTime: 0,
|
||||
writeCount: 0,
|
||||
parseTime: 0,
|
||||
parseCount: 0,
|
||||
compileTime: 0,
|
||||
compileCount: 0,
|
||||
emitTime: 0,
|
||||
emitCount: 0,
|
||||
validateTime: 0,
|
||||
validateCount: 0,
|
||||
optimizeTime: 0,
|
||||
optimizeCount: 0
|
||||
};
|
||||
}
|
||||
|
||||
exports.createStats = createStats;
|
||||
|
||||
if (!process.hrtime) process.hrtime = require("browser-process-hrtime");
|
||||
|
||||
/** Measures the execution time of the specified function. */
|
||||
function measure(fn) {
|
||||
const start = process.hrtime();
|
||||
fn();
|
||||
const times = process.hrtime(start);
|
||||
return times[0] * 1e9 + times[1];
|
||||
}
|
||||
|
||||
exports.measure = measure;
|
||||
|
||||
function formatTime(time) {
|
||||
return time ? (time / 1e6).toFixed(3) + " ms" : "N/A";
|
||||
}
|
||||
|
||||
exports.formatTime = formatTime;
|
||||
|
||||
/** Formats and prints out the contents of a set of stats. */
|
||||
function printStats(stats, output) {
|
||||
function format(time, count) {
|
||||
return formatTime(time);
|
||||
}
|
||||
(output || process.stdout).write([
|
||||
"I/O Read : " + format(stats.readTime, stats.readCount),
|
||||
"I/O Write : " + format(stats.writeTime, stats.writeCount),
|
||||
"Parse : " + format(stats.parseTime, stats.parseCount),
|
||||
"Compile : " + format(stats.compileTime, stats.compileCount),
|
||||
"Emit : " + format(stats.emitTime, stats.emitCount),
|
||||
"Validate : " + format(stats.validateTime, stats.validateCount),
|
||||
"Optimize : " + format(stats.optimizeTime, stats.optimizeCount)
|
||||
].join(EOL) + EOL);
|
||||
}
|
||||
|
||||
exports.printStats = printStats;
|
||||
|
||||
var allocBuffer = typeof global !== "undefined" && global.Buffer
|
||||
? global.Buffer.allocUnsafe || function(len) { return new global.Buffer(len); }
|
||||
: function(len) { return new Uint8Array(len) };
|
||||
|
||||
/** Creates a memory stream that can be used in place of stdout/stderr. */
|
||||
function createMemoryStream(fn) {
|
||||
var stream = [];
|
||||
stream.write = function(chunk) {
|
||||
if (fn) fn(chunk);
|
||||
if (typeof chunk === "string") {
|
||||
let buffer = allocBuffer(utf8.length(chunk));
|
||||
utf8.write(chunk, buffer, 0);
|
||||
chunk = buffer;
|
||||
}
|
||||
this.push(chunk);
|
||||
};
|
||||
stream.toBuffer = function() {
|
||||
var offset = 0, i = 0, k = this.length;
|
||||
while (i < k) offset += this[i++].length;
|
||||
var buffer = allocBuffer(offset);
|
||||
offset = i = 0;
|
||||
while (i < k) {
|
||||
buffer.set(this[i], offset);
|
||||
offset += this[i].length;
|
||||
++i;
|
||||
}
|
||||
return buffer;
|
||||
};
|
||||
stream.toString = function() {
|
||||
var buffer = this.toBuffer();
|
||||
return utf8.read(buffer, 0, buffer.length);
|
||||
};
|
||||
return stream;
|
||||
}
|
||||
|
||||
exports.createMemoryStream = createMemoryStream;
|
||||
|
||||
/** Compatible TypeScript compiler options. */
|
||||
exports.tscOptions = {
|
||||
alwaysStrict: true,
|
||||
noImplicitAny: true,
|
||||
noImplicitReturns: true,
|
||||
noImplicitThis: true,
|
||||
noEmitOnError: true,
|
||||
strictNullChecks: true,
|
||||
experimentalDecorators: true,
|
||||
target: "esnext",
|
||||
module: "commonjs",
|
||||
noLib: true,
|
||||
types: [],
|
||||
allowJs: false
|
||||
};
|
171
bin/asc.json
171
bin/asc.json
@ -1,171 +0,0 @@
|
||||
{
|
||||
"version": {
|
||||
"desc": "Prints just the compiler's version and exits.",
|
||||
"type": "boolean",
|
||||
"aliases": [ "v" ]
|
||||
},
|
||||
"help": {
|
||||
"desc": "Prints this message and exits.",
|
||||
"type": "boolean",
|
||||
"aliases": [ "h" ]
|
||||
},
|
||||
"optimize": {
|
||||
"desc": [
|
||||
"Optimizes the module. Also accepts the optimize level:",
|
||||
"",
|
||||
" -O Uses defaults. Equivalent to -O2s",
|
||||
" -O0 Equivalent to --optimizeLevel 0",
|
||||
" -O1 Equivalent to --optimizeLevel 1",
|
||||
" -O2 Equivalent to --optimizeLevel 2",
|
||||
" -O3 Equivalent to --optimizeLevel 3",
|
||||
" -Oz Equivalent to -O but with --shrinkLevel 2",
|
||||
" -O3s Equivalent to -O3 with --shrinkLevel 1 etc.",
|
||||
""
|
||||
],
|
||||
"type": "boolean",
|
||||
"aliases": [ "O" ]
|
||||
},
|
||||
"optimizeLevel": {
|
||||
"desc": "How much to focus on optimizing code. [0-3]",
|
||||
"type": "number"
|
||||
},
|
||||
"shrinkLevel": {
|
||||
"desc": "How much to focus on shrinking code size. [0-2, s=1, z=2]",
|
||||
"type": "number"
|
||||
},
|
||||
"validate": {
|
||||
"desc": "Validates the module using Binaryen. Exits if invalid.",
|
||||
"type": "boolean",
|
||||
"aliases": [ "c", "check" ]
|
||||
},
|
||||
"baseDir": {
|
||||
"desc": "Specifies the base directory of input and output files.",
|
||||
"type": "string"
|
||||
},
|
||||
"outFile": {
|
||||
"desc": "Specifies the output file. File extension indicates format.",
|
||||
"type": "string",
|
||||
"aliases": [ "o" ]
|
||||
},
|
||||
"binaryFile": {
|
||||
"desc": "Specifies the binary output file (.wasm).",
|
||||
"type": "string",
|
||||
"aliases": [ "b" ]
|
||||
},
|
||||
"textFile": {
|
||||
"desc": "Specifies the text output file (.wat).",
|
||||
"type": "string",
|
||||
"aliases": [ "t" ]
|
||||
},
|
||||
"asmjsFile": {
|
||||
"desc": "Specifies the asm.js output file (.js).",
|
||||
"type": "string",
|
||||
"aliases": [ "a" ]
|
||||
},
|
||||
"idlFile": {
|
||||
"desc": "Specifies the WebIDL output file (.webidl).",
|
||||
"type": "string",
|
||||
"aliases": [ "i" ]
|
||||
},
|
||||
"tsdFile": {
|
||||
"desc": "Specifies the TypeScript definition output file (.d.ts).",
|
||||
"type": "string",
|
||||
"aliases": [ "d", "dtsFile" ]
|
||||
},
|
||||
"sourceMap": {
|
||||
"desc": [
|
||||
"Enables source map generation. Optionally takes the URL",
|
||||
"used to reference the source map from the binary file."
|
||||
],
|
||||
"type": "string"
|
||||
},
|
||||
"noTreeShaking": {
|
||||
"desc": "Disables compiler-level tree-shaking, compiling everything.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"noDebug": {
|
||||
"desc": "Disables maintaining of debug information in binaries.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"noAssert": {
|
||||
"desc": "Replaces assertions with just their value without trapping.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"noEmit": {
|
||||
"desc": "Performs compilation as usual but does not emit code.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"noMemory": {
|
||||
"desc": "Does not set up a memory. Useful for low-level WebAssembly.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"importMemory": {
|
||||
"desc": "Imports the memory instance provided by the embedder.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"memoryBase": {
|
||||
"desc": "Sets the start offset of compiler-generated static memory.",
|
||||
"type": "number"
|
||||
},
|
||||
"importTable": {
|
||||
"desc": "Imports the function table instance provided by the embedder.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"noLib": {
|
||||
"desc": "Does not include the shipped standard library.",
|
||||
"type": "boolean"
|
||||
},
|
||||
"lib": {
|
||||
"desc": [
|
||||
"Adds one or multiple paths to custom library components and",
|
||||
"uses exports of all top-level files at this path as globals."
|
||||
],
|
||||
"type": "string"
|
||||
},
|
||||
"use": {
|
||||
"desc": [
|
||||
"Aliases a global object under another name, e.g., to switch",
|
||||
"the default 'Math' implementation used: --use Math=JSMath"
|
||||
],
|
||||
"type": "string",
|
||||
"aliases": [ "u" ]
|
||||
},
|
||||
"trapMode": {
|
||||
"desc": [
|
||||
"Sets the trap mode to use.",
|
||||
"",
|
||||
" allow Allow trapping operations. This is the default.",
|
||||
" clamp Replace trapping operations with clamping semantics.",
|
||||
" js Replace trapping operations with JS semantics.",
|
||||
""
|
||||
],
|
||||
"type": "string",
|
||||
"default": "allow"
|
||||
},
|
||||
"runPasses": {
|
||||
"desc": [
|
||||
"Specifies additional Binaryen passes to run after other",
|
||||
"optimizations, if any. See: Binaryen/src/passes/pass.cpp"
|
||||
],
|
||||
"type": "string"
|
||||
},
|
||||
"enable": {
|
||||
"desc": [
|
||||
"Enables additional (experimental) WebAssembly features.",
|
||||
"",
|
||||
" sign-extension Enables sign-extension operations",
|
||||
" mutable-global Enables mutable global imports and exports",
|
||||
""
|
||||
],
|
||||
"type": "string",
|
||||
"aliases": [ "feature" ]
|
||||
},
|
||||
"transform": {
|
||||
"desc": "Specifies the path to a custom transform to 'require'.",
|
||||
"type": "string"
|
||||
},
|
||||
"measure": {
|
||||
"desc": "Prints measuring information on I/O and compile times.",
|
||||
"type": "boolean"
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user