wasm_exec.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. // Copyright 2018 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. (() => {
  5. // Map multiple JavaScript environments to a single common API,
  6. // preferring web standards over Node.js API.
  7. //
  8. // Environments considered:
  9. // - Browsers
  10. // - Node.js
  11. // - Electron
  12. // - Parcel
  13. // - Webpack
  14. if (typeof global !== "undefined") {
  15. // global already exists
  16. } else if (typeof window !== "undefined") {
  17. window.global = window;
  18. } else if (typeof self !== "undefined") {
  19. self.global = self;
  20. } else {
  21. throw new Error("cannot export Go (neither global, window nor self is defined)");
  22. }
  23. if (!global.require && typeof require !== "undefined") {
  24. global.require = require;
  25. }
  26. if (!global.fs && global.require) {
  27. const fs = require("fs");
  28. if (typeof fs === "object" && fs !== null && Object.keys(fs).length !== 0) {
  29. global.fs = fs;
  30. }
  31. }
  32. const enosys = () => {
  33. const err = new Error("not implemented");
  34. err.code = "ENOSYS";
  35. return err;
  36. };
  37. if (!global.fs) {
  38. let outputBuf = "";
  39. global.fs = {
  40. constants: { O_WRONLY: -1, O_RDWR: -1, O_CREAT: -1, O_TRUNC: -1, O_APPEND: -1, O_EXCL: -1 }, // unused
  41. writeSync(fd, buf) {
  42. outputBuf += decoder.decode(buf);
  43. const nl = outputBuf.lastIndexOf("\n");
  44. if (nl != -1) {
  45. console.log(outputBuf.substr(0, nl));
  46. outputBuf = outputBuf.substr(nl + 1);
  47. }
  48. return buf.length;
  49. },
  50. write(fd, buf, offset, length, position, callback) {
  51. if (offset !== 0 || length !== buf.length || position !== null) {
  52. callback(enosys());
  53. return;
  54. }
  55. const n = this.writeSync(fd, buf);
  56. callback(null, n);
  57. },
  58. chmod(path, mode, callback) { callback(enosys()); },
  59. chown(path, uid, gid, callback) { callback(enosys()); },
  60. close(fd, callback) { callback(enosys()); },
  61. fchmod(fd, mode, callback) { callback(enosys()); },
  62. fchown(fd, uid, gid, callback) { callback(enosys()); },
  63. fstat(fd, callback) { callback(enosys()); },
  64. fsync(fd, callback) { callback(null); },
  65. ftruncate(fd, length, callback) { callback(enosys()); },
  66. lchown(path, uid, gid, callback) { callback(enosys()); },
  67. link(path, link, callback) { callback(enosys()); },
  68. lstat(path, callback) { callback(enosys()); },
  69. mkdir(path, perm, callback) { callback(enosys()); },
  70. open(path, flags, mode, callback) { callback(enosys()); },
  71. read(fd, buffer, offset, length, position, callback) { callback(enosys()); },
  72. readdir(path, callback) { callback(enosys()); },
  73. readlink(path, callback) { callback(enosys()); },
  74. rename(from, to, callback) { callback(enosys()); },
  75. rmdir(path, callback) { callback(enosys()); },
  76. stat(path, callback) { callback(enosys()); },
  77. symlink(path, link, callback) { callback(enosys()); },
  78. truncate(path, length, callback) { callback(enosys()); },
  79. unlink(path, callback) { callback(enosys()); },
  80. utimes(path, atime, mtime, callback) { callback(enosys()); },
  81. };
  82. }
  83. if (!global.process) {
  84. global.process = {
  85. getuid() { return -1; },
  86. getgid() { return -1; },
  87. geteuid() { return -1; },
  88. getegid() { return -1; },
  89. getgroups() { throw enosys(); },
  90. pid: -1,
  91. ppid: -1,
  92. umask() { throw enosys(); },
  93. cwd() { throw enosys(); },
  94. chdir() { throw enosys(); },
  95. }
  96. }
  97. if (!global.crypto && global.require) {
  98. const nodeCrypto = require("crypto");
  99. global.crypto = {
  100. getRandomValues(b) {
  101. nodeCrypto.randomFillSync(b);
  102. },
  103. };
  104. }
  105. if (!global.crypto) {
  106. throw new Error("global.crypto is not available, polyfill required (getRandomValues only)");
  107. }
  108. if (!global.performance) {
  109. global.performance = {
  110. now() {
  111. const [sec, nsec] = process.hrtime();
  112. return sec * 1000 + nsec / 1000000;
  113. },
  114. };
  115. }
  116. if (!global.TextEncoder && global.require) {
  117. global.TextEncoder = require("util").TextEncoder;
  118. }
  119. if (!global.TextEncoder) {
  120. throw new Error("global.TextEncoder is not available, polyfill required");
  121. }
  122. if (!global.TextDecoder && global.require) {
  123. global.TextDecoder = require("util").TextDecoder;
  124. }
  125. if (!global.TextDecoder) {
  126. throw new Error("global.TextDecoder is not available, polyfill required");
  127. }
  128. // End of polyfills for common API.
  129. const encoder = new TextEncoder("utf-8");
  130. const decoder = new TextDecoder("utf-8");
  131. global.Go = class {
  132. constructor() {
  133. this.argv = ["js"];
  134. this.env = {};
  135. this.exit = (code) => {
  136. if (code !== 0) {
  137. console.warn("exit code:", code);
  138. }
  139. };
  140. this._exitPromise = new Promise((resolve) => {
  141. this._resolveExitPromise = resolve;
  142. });
  143. this._pendingEvent = null;
  144. this._scheduledTimeouts = new Map();
  145. this._nextCallbackTimeoutID = 1;
  146. const setInt64 = (addr, v) => {
  147. this.mem.setUint32(addr + 0, v, true);
  148. this.mem.setUint32(addr + 4, Math.floor(v / 4294967296), true);
  149. }
  150. const getInt64 = (addr) => {
  151. const low = this.mem.getUint32(addr + 0, true);
  152. const high = this.mem.getInt32(addr + 4, true);
  153. return low + high * 4294967296;
  154. }
  155. const loadValue = (addr) => {
  156. const f = this.mem.getFloat64(addr, true);
  157. if (f === 0) {
  158. return undefined;
  159. }
  160. if (!isNaN(f)) {
  161. return f;
  162. }
  163. const id = this.mem.getUint32(addr, true);
  164. return this._values[id];
  165. }
  166. const storeValue = (addr, v) => {
  167. const nanHead = 0x7FF80000;
  168. if (typeof v === "number" && v !== 0) {
  169. if (isNaN(v)) {
  170. this.mem.setUint32(addr + 4, nanHead, true);
  171. this.mem.setUint32(addr, 0, true);
  172. return;
  173. }
  174. this.mem.setFloat64(addr, v, true);
  175. return;
  176. }
  177. if (v === undefined) {
  178. this.mem.setFloat64(addr, 0, true);
  179. return;
  180. }
  181. let id = this._ids.get(v);
  182. if (id === undefined) {
  183. id = this._idPool.pop();
  184. if (id === undefined) {
  185. id = this._values.length;
  186. }
  187. this._values[id] = v;
  188. this._goRefCounts[id] = 0;
  189. this._ids.set(v, id);
  190. }
  191. this._goRefCounts[id]++;
  192. let typeFlag = 0;
  193. switch (typeof v) {
  194. case "object":
  195. if (v !== null) {
  196. typeFlag = 1;
  197. }
  198. break;
  199. case "string":
  200. typeFlag = 2;
  201. break;
  202. case "symbol":
  203. typeFlag = 3;
  204. break;
  205. case "function":
  206. typeFlag = 4;
  207. break;
  208. }
  209. this.mem.setUint32(addr + 4, nanHead | typeFlag, true);
  210. this.mem.setUint32(addr, id, true);
  211. }
  212. const loadSlice = (addr) => {
  213. const array = getInt64(addr + 0);
  214. const len = getInt64(addr + 8);
  215. return new Uint8Array(this._inst.exports.mem.buffer, array, len);
  216. }
  217. const loadSliceOfValues = (addr) => {
  218. const array = getInt64(addr + 0);
  219. const len = getInt64(addr + 8);
  220. const a = new Array(len);
  221. for (let i = 0; i < len; i++) {
  222. a[i] = loadValue(array + i * 8);
  223. }
  224. return a;
  225. }
  226. const loadString = (addr) => {
  227. const saddr = getInt64(addr + 0);
  228. const len = getInt64(addr + 8);
  229. return decoder.decode(new DataView(this._inst.exports.mem.buffer, saddr, len));
  230. }
  231. const timeOrigin = Date.now() - performance.now();
  232. this.importObject = {
  233. go: {
  234. // Go's SP does not change as long as no Go code is running. Some operations (e.g. calls, getters and setters)
  235. // may synchronously trigger a Go event handler. This makes Go code get executed in the middle of the imported
  236. // function. A goroutine can switch to a new stack if the current stack is too small (see morestack function).
  237. // This changes the SP, thus we have to update the SP used by the imported function.
  238. // func wasmExit(code int32)
  239. "runtime.wasmExit": (sp) => {
  240. sp >>>= 0;
  241. const code = this.mem.getInt32(sp + 8, true);
  242. this.exited = true;
  243. delete this._inst;
  244. delete this._values;
  245. delete this._goRefCounts;
  246. delete this._ids;
  247. delete this._idPool;
  248. this.exit(code);
  249. },
  250. // func wasmWrite(fd uintptr, p unsafe.Pointer, n int32)
  251. "runtime.wasmWrite": (sp) => {
  252. sp >>>= 0;
  253. const fd = getInt64(sp + 8);
  254. const p = getInt64(sp + 16);
  255. const n = this.mem.getInt32(sp + 24, true);
  256. fs.writeSync(fd, new Uint8Array(this._inst.exports.mem.buffer, p, n));
  257. },
  258. // func resetMemoryDataView()
  259. "runtime.resetMemoryDataView": (sp) => {
  260. sp >>>= 0;
  261. this.mem = new DataView(this._inst.exports.mem.buffer);
  262. },
  263. // func nanotime1() int64
  264. "runtime.nanotime1": (sp) => {
  265. sp >>>= 0;
  266. setInt64(sp + 8, (timeOrigin + performance.now()) * 1000000);
  267. },
  268. // func walltime() (sec int64, nsec int32)
  269. "runtime.walltime": (sp) => {
  270. sp >>>= 0;
  271. const msec = (new Date).getTime();
  272. setInt64(sp + 8, msec / 1000);
  273. this.mem.setInt32(sp + 16, (msec % 1000) * 1000000, true);
  274. },
  275. // func scheduleTimeoutEvent(delay int64) int32
  276. "runtime.scheduleTimeoutEvent": (sp) => {
  277. sp >>>= 0;
  278. const id = this._nextCallbackTimeoutID;
  279. this._nextCallbackTimeoutID++;
  280. this._scheduledTimeouts.set(id, setTimeout(
  281. () => {
  282. this._resume();
  283. while (this._scheduledTimeouts.has(id)) {
  284. // for some reason Go failed to register the timeout event, log and try again
  285. // (temporary workaround for https://github.com/golang/go/issues/28975)
  286. console.warn("scheduleTimeoutEvent: missed timeout event");
  287. this._resume();
  288. }
  289. },
  290. getInt64(sp + 8) + 1, // setTimeout has been seen to fire up to 1 millisecond early
  291. ));
  292. this.mem.setInt32(sp + 16, id, true);
  293. },
  294. // func clearTimeoutEvent(id int32)
  295. "runtime.clearTimeoutEvent": (sp) => {
  296. sp >>>= 0;
  297. const id = this.mem.getInt32(sp + 8, true);
  298. clearTimeout(this._scheduledTimeouts.get(id));
  299. this._scheduledTimeouts.delete(id);
  300. },
  301. // func getRandomData(r []byte)
  302. "runtime.getRandomData": (sp) => {
  303. sp >>>= 0;
  304. crypto.getRandomValues(loadSlice(sp + 8));
  305. },
  306. // func finalizeRef(v ref)
  307. "syscall/js.finalizeRef": (sp) => {
  308. sp >>>= 0;
  309. const id = this.mem.getUint32(sp + 8, true);
  310. this._goRefCounts[id]--;
  311. if (this._goRefCounts[id] === 0) {
  312. const v = this._values[id];
  313. this._values[id] = null;
  314. this._ids.delete(v);
  315. this._idPool.push(id);
  316. }
  317. },
  318. // func stringVal(value string) ref
  319. "syscall/js.stringVal": (sp) => {
  320. sp >>>= 0;
  321. storeValue(sp + 24, loadString(sp + 8));
  322. },
  323. // func valueGet(v ref, p string) ref
  324. "syscall/js.valueGet": (sp) => {
  325. sp >>>= 0;
  326. const result = Reflect.get(loadValue(sp + 8), loadString(sp + 16));
  327. sp = this._inst.exports.getsp() >>> 0; // see comment above
  328. storeValue(sp + 32, result);
  329. },
  330. // func valueSet(v ref, p string, x ref)
  331. "syscall/js.valueSet": (sp) => {
  332. sp >>>= 0;
  333. Reflect.set(loadValue(sp + 8), loadString(sp + 16), loadValue(sp + 32));
  334. },
  335. // func valueDelete(v ref, p string)
  336. "syscall/js.valueDelete": (sp) => {
  337. sp >>>= 0;
  338. Reflect.deleteProperty(loadValue(sp + 8), loadString(sp + 16));
  339. },
  340. // func valueIndex(v ref, i int) ref
  341. "syscall/js.valueIndex": (sp) => {
  342. sp >>>= 0;
  343. storeValue(sp + 24, Reflect.get(loadValue(sp + 8), getInt64(sp + 16)));
  344. },
  345. // valueSetIndex(v ref, i int, x ref)
  346. "syscall/js.valueSetIndex": (sp) => {
  347. sp >>>= 0;
  348. Reflect.set(loadValue(sp + 8), getInt64(sp + 16), loadValue(sp + 24));
  349. },
  350. // func valueCall(v ref, m string, args []ref) (ref, bool)
  351. "syscall/js.valueCall": (sp) => {
  352. sp >>>= 0;
  353. try {
  354. const v = loadValue(sp + 8);
  355. const m = Reflect.get(v, loadString(sp + 16));
  356. const args = loadSliceOfValues(sp + 32);
  357. const result = Reflect.apply(m, v, args);
  358. sp = this._inst.exports.getsp() >>> 0; // see comment above
  359. storeValue(sp + 56, result);
  360. this.mem.setUint8(sp + 64, 1);
  361. } catch (err) {
  362. sp = this._inst.exports.getsp() >>> 0; // see comment above
  363. storeValue(sp + 56, err);
  364. this.mem.setUint8(sp + 64, 0);
  365. }
  366. },
  367. // func valueInvoke(v ref, args []ref) (ref, bool)
  368. "syscall/js.valueInvoke": (sp) => {
  369. sp >>>= 0;
  370. try {
  371. const v = loadValue(sp + 8);
  372. const args = loadSliceOfValues(sp + 16);
  373. const result = Reflect.apply(v, undefined, args);
  374. sp = this._inst.exports.getsp() >>> 0; // see comment above
  375. storeValue(sp + 40, result);
  376. this.mem.setUint8(sp + 48, 1);
  377. } catch (err) {
  378. sp = this._inst.exports.getsp() >>> 0; // see comment above
  379. storeValue(sp + 40, err);
  380. this.mem.setUint8(sp + 48, 0);
  381. }
  382. },
  383. // func valueNew(v ref, args []ref) (ref, bool)
  384. "syscall/js.valueNew": (sp) => {
  385. sp >>>= 0;
  386. try {
  387. const v = loadValue(sp + 8);
  388. const args = loadSliceOfValues(sp + 16);
  389. const result = Reflect.construct(v, args);
  390. sp = this._inst.exports.getsp() >>> 0; // see comment above
  391. storeValue(sp + 40, result);
  392. this.mem.setUint8(sp + 48, 1);
  393. } catch (err) {
  394. sp = this._inst.exports.getsp() >>> 0; // see comment above
  395. storeValue(sp + 40, err);
  396. this.mem.setUint8(sp + 48, 0);
  397. }
  398. },
  399. // func valueLength(v ref) int
  400. "syscall/js.valueLength": (sp) => {
  401. sp >>>= 0;
  402. setInt64(sp + 16, parseInt(loadValue(sp + 8).length));
  403. },
  404. // valuePrepareString(v ref) (ref, int)
  405. "syscall/js.valuePrepareString": (sp) => {
  406. sp >>>= 0;
  407. const str = encoder.encode(String(loadValue(sp + 8)));
  408. storeValue(sp + 16, str);
  409. setInt64(sp + 24, str.length);
  410. },
  411. // valueLoadString(v ref, b []byte)
  412. "syscall/js.valueLoadString": (sp) => {
  413. sp >>>= 0;
  414. const str = loadValue(sp + 8);
  415. loadSlice(sp + 16).set(str);
  416. },
  417. // func valueInstanceOf(v ref, t ref) bool
  418. "syscall/js.valueInstanceOf": (sp) => {
  419. sp >>>= 0;
  420. this.mem.setUint8(sp + 24, (loadValue(sp + 8) instanceof loadValue(sp + 16)) ? 1 : 0);
  421. },
  422. // func copyBytesToGo(dst []byte, src ref) (int, bool)
  423. "syscall/js.copyBytesToGo": (sp) => {
  424. sp >>>= 0;
  425. const dst = loadSlice(sp + 8);
  426. const src = loadValue(sp + 32);
  427. if (!(src instanceof Uint8Array || src instanceof Uint8ClampedArray)) {
  428. this.mem.setUint8(sp + 48, 0);
  429. return;
  430. }
  431. const toCopy = src.subarray(0, dst.length);
  432. dst.set(toCopy);
  433. setInt64(sp + 40, toCopy.length);
  434. this.mem.setUint8(sp + 48, 1);
  435. },
  436. // func copyBytesToJS(dst ref, src []byte) (int, bool)
  437. "syscall/js.copyBytesToJS": (sp) => {
  438. sp >>>= 0;
  439. const dst = loadValue(sp + 8);
  440. const src = loadSlice(sp + 16);
  441. if (!(dst instanceof Uint8Array || dst instanceof Uint8ClampedArray)) {
  442. this.mem.setUint8(sp + 48, 0);
  443. return;
  444. }
  445. const toCopy = src.subarray(0, dst.length);
  446. dst.set(toCopy);
  447. setInt64(sp + 40, toCopy.length);
  448. this.mem.setUint8(sp + 48, 1);
  449. },
  450. "debug": (value) => {
  451. console.log(value);
  452. },
  453. }
  454. };
  455. }
  456. async run(instance) {
  457. if (!(instance instanceof WebAssembly.Instance)) {
  458. throw new Error("Go.run: WebAssembly.Instance expected");
  459. }
  460. this._inst = instance;
  461. this.mem = new DataView(this._inst.exports.mem.buffer);
  462. this._values = [ // JS values that Go currently has references to, indexed by reference id
  463. NaN,
  464. 0,
  465. null,
  466. true,
  467. false,
  468. global,
  469. this,
  470. ];
  471. this._goRefCounts = new Array(this._values.length).fill(Infinity); // number of references that Go has to a JS value, indexed by reference id
  472. this._ids = new Map([ // mapping from JS values to reference ids
  473. [0, 1],
  474. [null, 2],
  475. [true, 3],
  476. [false, 4],
  477. [global, 5],
  478. [this, 6],
  479. ]);
  480. this._idPool = []; // unused ids that have been garbage collected
  481. this.exited = false; // whether the Go program has exited
  482. // Pass command line arguments and environment variables to WebAssembly by writing them to the linear memory.
  483. let offset = 4096;
  484. const strPtr = (str) => {
  485. const ptr = offset;
  486. const bytes = encoder.encode(str + "\0");
  487. new Uint8Array(this.mem.buffer, offset, bytes.length).set(bytes);
  488. offset += bytes.length;
  489. if (offset % 8 !== 0) {
  490. offset += 8 - (offset % 8);
  491. }
  492. return ptr;
  493. };
  494. const argc = this.argv.length;
  495. const argvPtrs = [];
  496. this.argv.forEach((arg) => {
  497. argvPtrs.push(strPtr(arg));
  498. });
  499. argvPtrs.push(0);
  500. const keys = Object.keys(this.env).sort();
  501. keys.forEach((key) => {
  502. argvPtrs.push(strPtr(`${key}=${this.env[key]}`));
  503. });
  504. argvPtrs.push(0);
  505. const argv = offset;
  506. argvPtrs.forEach((ptr) => {
  507. this.mem.setUint32(offset, ptr, true);
  508. this.mem.setUint32(offset + 4, 0, true);
  509. offset += 8;
  510. });
  511. // The linker guarantees global data starts from at least wasmMinDataAddr.
  512. // Keep in sync with cmd/link/internal/ld/data.go:wasmMinDataAddr.
  513. const wasmMinDataAddr = 4096 + 8192;
  514. if (offset >= wasmMinDataAddr) {
  515. throw new Error("total length of command line and environment variables exceeds limit");
  516. }
  517. this._inst.exports.run(argc, argv);
  518. if (this.exited) {
  519. this._resolveExitPromise();
  520. }
  521. await this._exitPromise;
  522. }
  523. _resume() {
  524. if (this.exited) {
  525. throw new Error("Go program has already exited");
  526. }
  527. this._inst.exports.resume();
  528. if (this.exited) {
  529. this._resolveExitPromise();
  530. }
  531. }
  532. _makeFuncWrapper(id) {
  533. const go = this;
  534. return function () {
  535. const event = { id: id, this: this, args: arguments };
  536. go._pendingEvent = event;
  537. go._resume();
  538. return event.result;
  539. };
  540. }
  541. }
  542. if (
  543. typeof module !== "undefined" &&
  544. global.require &&
  545. global.require.main === module &&
  546. global.process &&
  547. global.process.versions &&
  548. !global.process.versions.electron
  549. ) {
  550. if (process.argv.length < 3) {
  551. console.error("usage: go_js_wasm_exec [wasm binary] [arguments]");
  552. process.exit(1);
  553. }
  554. const go = new Go();
  555. go.argv = process.argv.slice(2);
  556. go.env = Object.assign({ TMPDIR: require("os").tmpdir() }, process.env);
  557. go.exit = process.exit;
  558. WebAssembly.instantiate(fs.readFileSync(process.argv[2]), go.importObject).then((result) => {
  559. process.on("exit", (code) => { // Node.js exits if no event handler is pending
  560. if (code === 0 && !go.exited) {
  561. // deadlock, make Go print error and stack traces
  562. go._pendingEvent = { id: 0 };
  563. go._resume();
  564. }
  565. });
  566. return go.run(result.instance);
  567. }).catch((err) => {
  568. console.error(err);
  569. process.exit(1);
  570. });
  571. }
  572. })();