Telegram Web, preconfigured for usage in I2P.
http://web.telegram.i2p/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
103 lines
4.6 KiB
103 lines
4.6 KiB
// Pthread Web Worker startup routine: |
|
// This is the entry point file that is loaded first by each Web Worker |
|
// that executes pthreads on the Emscripten application. |
|
|
|
// Thread-local, communicated via globals: |
|
var threadInfoStruct = 0; // Info area for this thread in Emscripten HEAP (shared). If zero, this worker is not currently hosting an executing pthread. |
|
var selfThreadId = 0; // The ID of this thread. 0 if not hosting a pthread. |
|
var parentThreadId = 0; // The ID of the parent pthread that launched this thread. |
|
|
|
// Send the pthreads mode and other params in through Module object settings |
|
var Module = { |
|
ENVIRONMENT: 'PTHREAD' |
|
}; |
|
|
|
// Cannot use console.log or console.error in a web worker, since that would risk a browser deadlock! https://bugzilla.mozilla.org/show_bug.cgi?id=1049091 |
|
// Therefore implement custom logging facility for threads running in a worker, which queue the messages to main thread to print. |
|
function threadPrint() { |
|
var text = Array.prototype.slice.call(arguments).join(' '); |
|
console.log(text); |
|
} |
|
function threadPrintErr() { |
|
var text = Array.prototype.slice.call(arguments).join(' '); |
|
console.error(text); |
|
} |
|
function threadAlert() { |
|
var text = Array.prototype.slice.call(arguments).join(' '); |
|
postMessage({cmd: 'alert', text: text, threadId: selfThreadId}); |
|
} |
|
Module['print'] = threadPrint; |
|
Module['printErr'] = threadPrintErr; |
|
this.alert = threadAlert; |
|
|
|
// If modularized, we can't reuse the module's assert() function. |
|
function assert(condition, text) { |
|
if (!condition) { |
|
abort('Assertion failed: ' + text); |
|
} |
|
} |
|
|
|
this.onmessage = function(e) { |
|
if (e.data.cmd === 'load') { // Preload command that is called once per worker to parse and load the Emscripten code. |
|
// Initialize the thread-local field(s): |
|
Module['tempDoublePtr'] = e.data.tempDoublePtr; |
|
|
|
// Initialize the global "process"-wide fields: |
|
Module['buffer'] = e.data.buffer; |
|
Module['TOTAL_MEMORY'] = e.data.TOTAL_MEMORY; |
|
Module['STATICTOP'] = e.data.STATICTOP; |
|
Module['DYNAMIC_BASE'] = e.data.DYNAMIC_BASE; |
|
Module['DYNAMICTOP_PTR'] = e.data.DYNAMICTOP_PTR; |
|
|
|
Module['pthreadWorkerInit'] = e.data.PthreadWorkerInit; |
|
importScripts(e.data.url); |
|
if (e.data.modularize) { |
|
// Feed input options into the modularized constructor... |
|
// 'this' is the Worker, which is also global scope. |
|
Module = new this[e.data.moduleExportName](Module); |
|
} |
|
if (typeof FS !== 'undefined') FS.createStandardStreams(); |
|
postMessage({ cmd: 'loaded' }); |
|
} else if (e.data.cmd === 'objectTransfer') { |
|
Module.PThread.receiveObjectTransfer(e.data); |
|
} else if (e.data.cmd === 'run') { // This worker was idle, and now should start executing its pthread entry point. |
|
threadInfoStruct = e.data.threadInfoStruct; |
|
Module.PThread.registerPthreadPtr(threadInfoStruct, /*isMainBrowserThread=*/0, /*isMainRuntimeThread=*/0); // Pass the thread address inside the asm.js scope to store it for fast access that avoids the need for a FFI out. |
|
assert(threadInfoStruct); |
|
selfThreadId = e.data.selfThreadId; |
|
parentThreadId = e.data.parentThreadId; |
|
assert(selfThreadId); |
|
assert(parentThreadId); |
|
Module.PThread.setStackSpace(e.data.stackBase, e.data.stackBase + e.data.stackSize); |
|
var result = 0; |
|
|
|
Module.PThread.receiveObjectTransfer(e.data); |
|
|
|
Module.PThread.setThreadStatus(threadInfoStruct, 1/*EM_THREAD_STATUS_RUNNING*/); |
|
|
|
try { |
|
Module.PThread.runThreadFunc(e.data.start_routine, e.data.arg); |
|
} catch(e) { |
|
if (e === 'Canceled!') { |
|
Module.PThread.threadCancel(); |
|
return; |
|
} else { |
|
Atomics.store(Module.HEAPU32, (threadInfoStruct + 4 /*{{{ C_STRUCTS.pthread.threadExitCode }}}*/ ) >> 2, -2 /*A custom entry specific to Emscripten denoting that the thread crashed.*/); |
|
Atomics.store(Module.HEAPU32, (threadInfoStruct + 0 /*{{{ C_STRUCTS.pthread.threadStatus }}}*/ ) >> 2, 1); // Mark the thread as no longer running. |
|
Module.PThread.wakeAllThreads(); |
|
throw e; |
|
} |
|
} |
|
// The thread might have finished without calling pthread_exit(). If so, then perform the exit operation ourselves. |
|
// (This is a no-op if explicit pthread_exit() had been called prior.) |
|
if (!Module['noExitRuntime']) Module.PThread.threadExit(result); |
|
else console.log('pthread noExitRuntime: not quitting.'); |
|
} else if (e.data.cmd === 'cancel') { // Main thread is asking for a pthread_cancel() on this thread. |
|
if (threadInfoStruct && Module.PThread.thisThreadCancelState == 0/*PTHREAD_CANCEL_ENABLE*/) { |
|
Module.PThread.threadCancel(); |
|
} |
|
} else { |
|
// Module['printErr']('pthread-main.js received unknown command ' + e.data.cmd); |
|
// console.error(e.data) |
|
} |
|
}
|
|
|