sh buildall.sh
diff --git a/azure/docsite/search/search_index.json b/azure/docsite/search/search_index.json
index bf8e163c..4d12092c 100644
--- a/azure/docsite/search/search_index.json
+++ b/azure/docsite/search/search_index.json
@@ -1 +1 @@
-{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"]},"docs":[{"location":"","title":"Easier WebAssembly with twr-wasmDocumentation and Examples","text":"Version 2.5.0
twr-wasm is a simple, lightweight and easy to use library for building C/C++ WebAssembly code directly with clang. Run C/C++ code in a web browser. Legacy code, libraries, full applications, or single functions can be integrated with JavaScript and TypeScript. twr-wam solves some common use cases with less work than the more feature rich emscripten.
Key Features:
- build
.wasm
modules using C/C++ using clang directly (no wrapper) - from JavaScript load
.wasm
modules, call C/C++ functions, and access wasm memory -
comprehensive console support for stdin
, stdio
, and stderr
.
- in C/C++, print and get characters to/from
<div>
tags in your HTML page - in C/C++, print and get characters to/from a
<canvas>
based \"terminal\" - localization support, UTF-8, and windows-1252 support
-
the optional TypeScript class twrWasmModuleAsync
can be used to:
- integrate CLI C/C++ code with JavaScript
- In JavaScript
await
on blocking/synchronous C/C++ functions.
-
2D drawing API for C/C++ compatible with JavaScript Canvas
- audio playback APIs for C/C++
- create your own C/C++ APIs using TypeScript by extending
class twrLibrary
- standard C library optimized for WebAssembly
- libc++ built for WebAssembly
- comprehensive examples and documentation
"},{"location":"#live-webassembly-examples-and-source","title":"Live WebAssembly Examples and Source","text":"Name View Live Link Source Link Bouncing Balls (C++) View bouncing balls Source for balls Pong (C++) Pong Source Input/Output with <div>
View square demo Source I/O to terminal with <canvas>
View demo Source CLI using libc++ and <canvas>
) View console Source"},{"location":"#hello-world","title":"Hello World","text":"Here is the simplest twr-wasm
example.
helloworld.c#include <stdio.h>\n\nvoid hello() {\n printf(\"hello, world!\\n\");\n}\n
index.html<head>\n <title>Hello World</title>\n</head>\n<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n </script>\n</body>\n
"},{"location":"#on-github","title":"On Github","text":"https://github.com/twiddlingbits/twr-wasm
"},{"location":"#why","title":"Why?","text":"The Wasm Runtime Limitations section explains why a library like twr-wasm is needed to use WebAssembly.
"},{"location":"#limitations","title":"Limitations","text":" - libc++ not built with exceptions enabled
- some standard C library functions are not 100% implemented
- Designed to work with a browser. Not tested with or designed to work with node.js
- Not all of compile-rt is ported (but most bits you need are)
"},{"location":"#post-feedback","title":"Post Feedback","text":"Please post feedback (it worked for you, didn't work, requests, questions, etc) at https://github.com/twiddlingbits/twr-wasm/
"},{"location":"api/api-c-audio/","title":"Audio API for WebAssembly","text":"This section describes twr-wasm's C Audio API, which allows audio API functions to be called using C/C++ from WebAssembly.
"},{"location":"api/api-c-audio/#examples","title":"Examples","text":"Name View Live Link Source Link Pong (C++) View Pong Source for Pong tests-audio View tests-audio Source for tests-audio"},{"location":"api/api-c-audio/#code-example","title":"Code Example","text":"Play Audio#include \"twr-audio.h\"\n#include <math.h>\n#include <stdlib.h>\n\n#define M_PI 3.14159265358979323846\n\nvoid play() {\n twr_audio_play_file(\"example.mp3\"); //plays audio from specified URL\n\n const long SAMPLE_RATE = 48000; //48,000 samples per second\n const double DURATION = 10.0; //play for 10 seconds\n const double freq = 493.883; //Middle B (B4)\n\n long length = (long)ceil(SAMPLE_RATE*DURATION);\n //PCM audio data in the form of -1.0 to 1.0\n float* wave = (float*)malloc(sizeof(float) * length);\n\n //generate square wave at specified frequency and duration\n for (long i = 0; i < length; i++) {\n wave[i] = cos(2*M_PI*freq*(i/(float)sample_rate)) > 0 ? 1 : -1;\n }\n\n //creates a mon-audio channel buffer at our given SAMPLE_RATE\n // and square-wave data we generated\n long node_id = twr_audio_from_float_pcm(1, SAMPLE_RATE, wave, length);\n\n //plays the square wave\n // Can be played multiple times, and is only freed on twr_audio_free\n twr_audio_play(node_id);\n}\n
"},{"location":"api/api-c-audio/#overview","title":"Overview","text":"The Audio API is part a twr-wasm library that can be accessed via #include \"twr-audio.h\"
. It has two main methods to play audio: from raw PCM data and from a URL.
Raw PCM data can be initialized via twr_audio_from_<type>_pcm
or twr_audio_load
. There are multiple types for twr_audio_from_<type>_pcm
. These types include Float, 8bit, 16bit, and 32bit. Float takes in values between -1.0 and 1.0. Meanwhile, the 8bit, 16bit, and 32bit versions take them in as signed numbers between their minimum and maximum values. twr_audio_load
, on the other hand, reads the PCM data from an audio file that is specified by a URL. This method does not stream the audio, so it might take some time to read the file (depending how long the file is). Each function returns an integer node_id
that identifies the loaded PCM data. Once the PCM data is initialized, it can be played via functions like twr_audio_play
, twr_audio_play_range
, twr_audio_play_sync
, and twr_audio_play_range_sync
. The play functions can be called multiple times for each audio ID.
You can also play audio directly from a URL. Unlike twr_audio_load
, the url is initialized directly into an HTMLAudioElement
which streams the audio and starts playback immediately.
In addition to playing audio, there are functions that allow you to query and modify an ongoing playback. These include stopping the playback, getting how long it's been playing, and modifying the pan; volume; or playback rate of the audio.
"},{"location":"api/api-c-audio/#notes","title":"Notes","text":"When playing audio, a playback_id
is returned to query or modify the playback. However, once playback completes, the playback_id
becomes invalid. Most functions that take in a playback_id
will simply return a warning and return without error if the playback_id
is invalid. An exception is twr_audio_query_playback_position
which will return -1 when given an invalid playback_id
.
Functions that end in _sync
are for use with twrWamModuleAsync
, and are synchronous. Meaning they don't return until the operation, such as playback, is complete.
Note that on some platforms, sounds that are too short might not play correctly. This is true in JavaScript as well.
"},{"location":"api/api-c-audio/#functions","title":"Functions","text":"These are the current Audio APIs available in C/C++:
long twr_audio_from_float_pcm(long num_channels, long sample_rate, float* data, long singleChannelDataLen);\nlong twr_audio_from_8bit_pcm(long number_channels, long sample_rate, char* data, long singleChannelDataLen);\nlong twr_audio_from_16bit_pcm(long number_channels, long sample_rate, short* data, long singleChannelDataLen);\nlong twr_audio_from_32bit_pcm(long number_channels, long sample_rate, int* data, long singleChannelDataLen);\n\nfloat* twr_audio_get_float_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nchar* twr_audio_get_8bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nshort* twr_audio_get_16bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nint* twr_audio_get_32bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\n\nlong twr_audio_play(long node_id);\nlong twr_audio_play_volume(long node_id, double volume, double pan);\nlong twr_audio_play_callback(long node_id, double volume, double pan, int finish_callback);\n\nstruct PlayRangeFields {\n double pan, volume;\n int loop, finish_callback;\n long sample_rate;\n};\nstruct PlayRangeFields twr_audio_default_play_range();\nlong twr_audio_play_range(long node_id, long start_sample, long end_sample);\nlong twr_audio_play_range_ex(long node_id, long start_sample, long end_sample, struct PlayRangeFields* fields);\n\nlong twr_audio_play_sync(long node_id);\nlong twr_audio_play_sync_ex(long node_id, double volume, double pan);\n\n\nstruct PlayRangeSyncFields {\n double pan, volume;\n int loop;\n long sample_rate;\n};\n\nstruct PlayRangeSyncFields twr_audio_default_play_range_sync();\nlong twr_audio_play_range_sync(long node_id, long start_sample, long end_sample);\nlong twr_audio_play_range_sync_ex(long node_id, long start_sample, long end_sample, struct PlayRangeSyncFields* fields);\n\nlong twr_audio_load_sync(char* url);\nlong twr_audio_load(int event_id, char* url);\nlong twr_audio_query_playback_position(long playback_id);\nvoid twr_audio_free_id(long node_id);\n\nvoid twr_audio_stop_playback(long playback_id);\n\nvoid twr_audio_modify_playback_volume(long playback_id, double volume);\nvoid twr_audio_modify_playback_pan(long playback_id, double pan);\nvoid twr_audio_modify_playback_rate(long playback_id, double sample_rate);\n\nlong twr_audio_play_file(char* file_url);\nlong twr_audio_play_file_ex(char* file_url, double volume, double playback_rate, int loop);\n\nstruct AudioMetadata {\n long length;\n long sample_rate;\n long channels;\n};\n\nvoid twr_audio_get_metadata(long node_id, struct AudioMetadata* metadata);\n
"},{"location":"api/api-c-con/","title":"WebAssembly Character Console API","text":"twr-wasm for WebAssembly provides Consoles for interactive user I/O. Character and graphic 2D draw consoles exist. This section covers the streaming and addressable character APIs that can be used with an instance of twrConsoleDebug, twrConsoleTerminal, twrConsoleDiv. This API works with stdin, stdout, stderr and custom named consoles.
Also see the Consoles section in Getting Started
"},{"location":"api/api-c-con/#examples","title":"Examples","text":"Name View Live Link Source Link \"terminal\" in/out with a <canvas>
View mini-term demo Source"},{"location":"api/api-c-con/#getting-a-console","title":"Getting a Console","text":""},{"location":"api/api-c-con/#stdin-stdout-stderr","title":"stdin, stdout, stderr","text":"stdin
, stdout
, stderr
are defined in <stdio.h>
.
This section describes how to configure stdio
In C, consoles are represented by a twr_ioconsole_t
.
stdio.h also defines FILE
like this:
typedef twr_ioconsole_t FILE; \n
from <stdio.h>
:
#define stderr (FILE *)(twr_get_stderr_con())\n#define stdin (FILE *)(twr_get_stdio_con())\n#define stdout (FILE *)(twr_get_stdio_con())\n
"},{"location":"api/api-c-con/#twr_get_console","title":"twr_get_console","text":"This function will retrieve a console by its name. The standard names are stdio
, stderr
, and std2d
. In addition, any named console that was passed to a module using the io
option can be retrieved with this function.
See io doc.
See the multi-io example.
#include \"twr-crt.h\"\n\ntwr_ioconsole_t* twr_get_console(const char* name)\n
"},{"location":"api/api-c-con/#io_nullcon","title":"io_nullcon","text":"Returns an IoConsole that goes to the bit bucket. io_getc32 will return 0.
#include \"twr-io.h\"\n\ntwr_ioconsole_t* io_nullcon(void);\n
"},{"location":"api/api-c-con/#io-console-functions","title":"IO Console Functions","text":""},{"location":"api/api-c-con/#io_cls","title":"io_cls","text":"For addressable display consoles only.
Clears the screen. That is, all character cells in the console are set to a space, their colors are reset to the current default colors (see io_set_colors
).
#include <twr_io.h>\n\nvoid io_cls(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_getc32","title":"io_getc32","text":"Waits for the user to press a key and then returns a unicode code point.
To return characters encoded with the current locale, see io_mbgetc
#include <twr_io.h>\n\nint io_getc32(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_colors","title":"io_get_colors","text":"For addressable display consoles only.
Gets the current default foreground and background colors. These colors are used by an new text updates.
The color format is a 24 bit int as RGB.
#include <twr_io.h>\n\nvoid io_get_colors(twr_ioconsole_t* io, unsigned long *foreground, unsigned long *background);\n
"},{"location":"api/api-c-con/#io_get_cursor","title":"io_get_cursor","text":"Returns an integer of the current cursor position. The cursor is where the next io_putc
is going to go.
For addressable display consoles, the cursor position ranges from [0, width*height-1], inclusive.
#include <twr_io.h>\n\nint io_get_cursor(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_prop","title":"io_get_prop","text":"Given a string key (name) of a property, returns its integer value. The available properties varies by console type.
#include <twr_io.h>\n\nint io_get_prop(twr_ioconsole_t* io, const char* key)\n
All consoles support: \"type\". Addressable consoles also support: \"cursorPos\", \"charWidth\", \"charHeight\", \"foreColorAsRGB\", \"backColorAsRGB\", \"widthInChars\", \"heightInChars\", \"fontSize\", \"canvasWidth\", \"canvasHeight\"
You can do a bitwise &
on type with the following C defines to determine a console capabilities:
IO_TYPE_CHARREAD
IO_TYPE_CHARWRITE
IO_TYPE_ADDRESSABLE_DISPLAY
IO_TYPE_CANVAS2D
For example:
if (io_get_prop(stdin, \"type\")&IO_TYPE_CHARREAD) {\n printf (\"okay to read from stdin);\n}\n
"},{"location":"api/api-c-con/#io_get_width","title":"io_get_width","text":"Returns the width in characters of an addressable console.
#include <twr_io.h>\n\nint io_get_width(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_height","title":"io_get_height","text":"Returns the height in characters of an addressable console.
#include <twr_io.h>\n\nint io_get_height(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_set_colors","title":"io_set_colors","text":"For addressable display consoles only.
Sets a 24 bit RGB default color for the foreground and background. The prior default colors are changed (lost). For example, if you set the default colors when you created the console (see twrConsoleTerminal Options), the defaults will no longer be active. Use io_get_colors
to save existing colors for later restoration using io_set_colors
.
A call to io_set_colors
doesn't actually cause any on screen changes. Instead, these new default colors are used in future draw and text calls. A foreground and background color is set for each cell in the console window. The cell's colors are set to these default foreground/background colors when a call to io_setc
, io_setreset
, etc is made.
#include <twr_io.h>\n\nvoid io_set_colors(twr_ioconsole_t* io, unsigned long foreground, unsigned long background);\n
"},{"location":"api/api-c-con/#io_setc","title":"io_setc","text":"For addressable display consoles only.
Sets a console cell to the specified character. Sends a byte to an console and supports the current locale's character encoding. This function will \"stream\" using the current code page. In other words, if you are in the \"C\" locale io_setc
it will set ASCII characters. If the current locale is set to 1252, then you can send windows-1252 encoded characters. If the current locale is UTF-8, then you can stream UTF-8 (that is, call io_setc
once for each byte of the multi-byte UTF-8 character).
#include <twr_io.h>\n\nbool io_setc(twr_ioconsole_t* io, int location, unsigned char c);\n
"},{"location":"api/api-c-con/#io_setc32","title":"io_setc32","text":"For addressable display consoles only.
Sets a console cell to a unicode code point. The colors are set to the defaults (see io_set_colors
).
#include <twr_io.h>\n\nvoid io_setc32(twr_ioconsole_t* io, int location, int c);\n
"},{"location":"api/api-c-con/#io_set_cursor","title":"io_set_cursor","text":"Moves the cursor. See io_get_cursor
.
#include <twr_io.h>\n\nvoid io_set_cursor(twr_ioconsole_t* io, int loc);\n
"},{"location":"api/api-c-con/#io_set_cursorxy","title":"io_set_cursorxy","text":"Set's the cursor's x,y position in an addressable console.
#include <twr_io.h>\n\nvoid io_set_cursorxy(twr_ioconsole_t* io, int x, int y);\n
"},{"location":"api/api-c-con/#io_setfocus","title":"io_setfocus","text":"Sets the input focus to the indicated console.
#include <twr_io.h>\n\nvoid io_setfocus(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_set_range","title":"io_set_range","text":"Sets a range of characters in an addressable display.
#include <twr_io.h>\n\nvoid io_set_range(twr_ioconsole_t* io, int *chars32, int start, int len)\n
"},{"location":"api/api-c-con/#io_setreset","title":"io_setreset","text":"For addressable display consoles only.
Sets or resets (clears) a chunky graphics \"pixel\". Each character cell can also be a 2x3 grid of graphic \"pixels\". In other words, the terminal window has pixel dimensions of width2 x height3.
The color will be set to the defaults if the impacted cell is not a graphics cell. If it is an existing graphics cell, the colors don't change.
See the terminal
example.
#include <twr_io.h>\n\nbool io_setreset(twr_ioconsole_t* io, int x, int y, bool isset);\n
"},{"location":"api/api-c-con/#io_mbgetc","title":"io_mbgetc","text":"io_mbgetc
will get a character from stdin and encode it using the character encoding of the LC_CTYPE category of the current locale. \"C\" will use ASCII. UTF-8 and windows-1252 are also supported.
#include <twr_io.h>\n\nvoid io_mbgetc(twr_ioconsole_t* io, char* strout);\n
"},{"location":"api/api-c-con/#io_mbgets","title":"io_mbgets","text":"Gets a string from a Console. Returns when the user presses \"Enter\". Displays a cursor character and echos the inputted characters, at the current cursor position. Uses character encoding of LC_TYPE of current locale. If the encoding is UTF-8, then the result will be multibyte.
This function is commonly used with stdin
.
This function requires that you use twrWasmModuleAsync
.
#include <twr_io.h>\n\nchar *io_mbgets(twr_ioconsole_t* io, char *buffer );\n
"},{"location":"api/api-c-con/#io_point","title":"io_point","text":"For addressable display consoles only.
Checks if a chunky graphics \"pixel\" is set or clear. See io_setreset
.
#include <twr_io.h>\n\nbool io_point(twr_ioconsole_t* io, int x, int y);\n
"},{"location":"api/api-c-con/#io_putc","title":"io_putc","text":"Sends a byte to an IoConsole and supports the current locale's character encoding. This function will \"stream\" using the current code page. In other words, if you io_putc
ASCII, it will work as \"normal\". If the current locale is set to 1252, then you can send windows-1252 encoded characters. If the current locale is UTF-8, then you can stream UTF-8 (that is, call io_putc
once for each byte of the multi-byte UTF-8 character).
Note that when characters are sent to the browser console using stderr
they will not render to the console until a newline or return is sent.
#include \"twr-io.h\"\n\nvoid io_putc(twr_ioconsole_t* io, unsigned char c);\n
"},{"location":"api/api-c-con/#io_putstr","title":"io_putstr","text":"Calls io_putc
for each byte in the passed string.
#include \"twr-io.h\"\n\nvoid io_putstr(twr_ioconsole_t* io, const char* s);\n
"},{"location":"api/api-c-con/#io_printf","title":"io_printf","text":"Identical to fprintf
, however io_printf will call io_begin_draw
and io_end_draw
around its drawing activities -- resulting in snapper performance.
For example:
#include \"twr-io.h\"\n\nio_printf(twr_debugcon(), \"hello over there in browser debug console land\\n\");\n
or
#include <stdio.h>\n#include <twr_io.h>\n\nio_printf(stdout, \"hello world\\n\");\n
#include <twr_io.h>\n\nvoid io_printf(twr_ioconsole_t *io, const char *format, ...);\n
"},{"location":"api/api-c-con/#io_begin_draw","title":"io_begin_draw","text":"For addressable display consoles only.
This call (and its matching io_end_draw) are not required. But if you bracket any call sequence that draws to the terminal window with an io_begin_draw
and io_end_draw
, the updates will be batched into one update. This will increase performance and usually prevents the user from seeing partial updates.
io_begin_draw
can be nested.
See the terminal example.
#include <twr_io.h>\n\nvoid io_begin_draw(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_end_draw","title":"io_end_draw","text":"For addressable display consoles only.
See io_begin_draw
.
#include <twr_io.h>\n\nvoid io_end_draw(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#deprecated-functions","title":"Deprecated Functions","text":""},{"location":"api/api-c-con/#twr_debugcon","title":"twr_debugcon","text":"This function has been removed. Use stderr
or twr_conlog
.
#include \"twr-crt.h\"\n\ntwr_conlog(\"hello 99 in hex: %x\", 99);\n
or
#include <stdio.h>\n\nfprintf(stderr, \"hello over there in browser debug console land\\n\");\n
"},{"location":"api/api-c-con/#twr_divcon","title":"twr_divcon","text":"This function has been removed.
"},{"location":"api/api-c-con/#twr_windowcon","title":"twr_windowcon","text":"This function has been removed.
"},{"location":"api/api-c-d2d/","title":"2D Draw C API for WebAssembly","text":"This section describes twr-wasm's C D2D API, which allows your WebAssembly module to call many of the JavaScript Canvas APIs.
"},{"location":"api/api-c-d2d/#examples","title":"Examples","text":"Name View Live Link Source Link Bouncing Balls (C++) View bouncing balls Source for balls Pong (C++) View Pong Source for Pong Maze (Win32 C Port) View live maze here Source for maze"},{"location":"api/api-c-d2d/#code-example","title":"Code Example","text":"Draw A Rectangle#include \"twr-draw2d.h\"\n\nvoid square() {\n // batch draw commands, with a maximum of 100 commands before render\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n // set color using CSS color string\n d2d_setfillstyle(ds, \"blue\");\n // draw a the rect\n d2d_fillrect(ds, 10, 10, 100, 100);\n // this will cause the JavaScript thread to render\n d2d_end_draw_sequence(ds);\n}\n
"},{"location":"api/api-c-d2d/#overview","title":"Overview","text":"The Draw 2D APIs are C APIs and are part of the twr-wasm library that you access with #include \"twr-draw2d.h\"
. There is also a C++ canvas wrapper class in examples/twr-cpp
used by the balls and pong examples.
To create a canvas surface, that you can draw to using the twr-wasm 2D C drawing APIs, use the twrConsoleCanvas
class in your JavaScript/HTML (see Consoles Section). Or more simply, if you add a canvas tag to your HTML named twr_d2dcanvas
, the needed twrConsoleCanvas
will be created automatically.
<canvas id=\"twr_d2dcanvas\" width=\"600\" height=\"600\"></canvas>\n//Feel free to change the `width=\"600` and/or `height=\"600` attributes.\n
To draw using the C 2D Draw API:
- call
d2d_start_draw_sequence
(or alternately d2d_start_draw_sequence_with_con
) - call one or more (a sequence) of 2D draw commands, like
d2d_fillrect
- call
d2d_end_draw_sequence
- repeat as desired
d2d_start_draw_sequence
will draw to the default twrConsoleCanvas
, as explained at the start of this section. d2d_start_draw_sequence_with_con
is optional, and allows you to specify the twrConsoleCanvas
to draw to. You would typically get this console in C using the twr_get_console
function (which retrieves a named console that you specified in the io
module option.)
Commands are queued until flushed -- which will take the batch of queued draw commands, and execute them. The 2D draw APIs will work with either twrWasmModule
or twrWasmModuleAsync
. With twrWasmModuleAsync
, the batch of commands is sent from the worker thread over to the JavaScript main thread for execution. By batching the calls between calls to d2d_start_draw_sequence
and d2d_end_draw_sequence
, performance is improved.
d2d_flush
waits for the commands to finish execution before returning. d2d_flush
is called automatically by d2d_end_draw_sequence
and so you generally don't need to call it manually.
You pass an argument to d2d_start_draw_sequence
specifying how many instructions will trigger an automatic call to d2d_flush
. You can make this larger for efficiency, or smaller if you want to see the render progress more frequently. There is no limit on the size of the queue, except memory used in the Wasm module. The d2d_flush
function can be called manually, but this is not normally needed, unless you would like to ensure a sequence renders before your d2d_end_draw_sequence
is called, or before the count passed d2d_start_draw_sequence
is met.
If you are using twrWasmModuleAsync
, or if you are re-rendering the entire frame for each animation update, you should ensure that all of your draws for a complete frame are made without an explicit or implicit call to d2d_flush
in the middle of the draw sequence, as this may cause flashing.
"},{"location":"api/api-c-d2d/#possible-pitfalls","title":"Possible Pitfalls","text":"Some commands have extra details that you need to be aware of to avoid performance loss or bugs.
- Getters, like d2d_measuretext, will flush the queue in order to retrieve the requested data. If your program relies on not flushing early (for example, to avoid flashes), then getters should be avoided in your main render loops.
- putImageData references the provided pointer, so the given image data needs to stay valid on the caller's stack or heap until flush is called.
- getLineDash takes in a buffer_length, double * array (the buffer), and returns the amount of the buffer filled. If there are more line segments than can fit in the buffer_length, a warning is printed and the excess is voided. If you want to know the size before hand for allocation, the getLineDashLength function is available.
"},{"location":"api/api-c-d2d/#notes","title":"Notes","text":"The functions listed below are based on the JavaScript Canvas 2D API (found here). However, there are some slight differences since these APIs are made for C rather than JavaScript. For example some items keep resources stored on the JavaScript side (such as d2d_createlineargradient) which are referenced by a numeric ID , rather than an actual object reference.
Additionally, there are alternative functions like d2d_setstrokestylergba, which calls the same underlying function as d2d_setstrokestyle, but takes in a color as a number rather than CSS style string.
As noted above, putImageData requires that the image data be valid until flush is called.
Other functions that take a string, like d2d_filltext, don't have this same issue because they make a copy of the string argument. These string copies will be automatically freed.
getCanvasPropDouble, getCanvasPropString, setCanvasPropDouble, and setCanvasPropString allow you to change canvas properties by name. If the previous values type is either undefined, a string rather than a number, etc. then it will throw an error so ensure that you have your property names correct.
d2d_load_image is not called like other instructions which rely on d2d_start_draw_sequence. This means it always gets called immediately and doesn't queue up in or flush the instruction queue. This can cause some issues such as the example below. Load Image Pitfall
#include \"twr-draw2d.h\"\nbool has_background = false;\nconst long BACKGROUND_ID = 1;\n//draws the background\nvoid draw_background(struct d2d_draw_seq* ds) {\n assert(has_background);\n d2d_drawimage(ds, BACKGROUND_ID, x, y);\n}\n//loads a new background image\nvoid load_background_image(struct d2d_draw_seq* ds, const char * url) {\n if (has_background) {\n //free previous background\n //this isn't called until the buffer in ds get's flushed.\n // For this program, that doesn't happen until d2d_end_draw_sequence is called,\n // so d2d_load_image processes before d2d_releasid throws a warning and then is deleted when d2d_releaseid\n // is eventually called.\n d2d_releaseid(ds, BACKGROUND_ID);\n //d2d_flush(ds) //by adding a flush like so, it ensures releaseid is called before d2d_load_image\n } else {\n has_background = true;\n }\n d2d_load_image(url, BACKGROUND_ID);\n}\nvoid render() {\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n\n //load background\n load_background_image(ds, \"example_image.com\");\n\n draw_background(ds); //draw it\n\n d2d_end_draw_sequence(ds);\n\n\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n\n //load new background image\n load_background_image(ds, \"example_image2.com\");\n draw_background(ds);\n\n d2d_end_draw_sequence(ds);\n}\n
"},{"location":"api/api-c-d2d/#functions","title":"Functions","text":"These are the Canvas APIs currently available in C:
struct d2d_draw_seq* d2d_start_draw_sequence(int flush_at_ins_count);\nstruct d2d_draw_seq* d2d_start_draw_sequence_with_con(int flush_at_ins_count, twr_ioconsole_t * con);\nvoid d2d_end_draw_sequence(struct d2d_draw_seq* ds);\nvoid d2d_flush(struct d2d_draw_seq* ds);\nint d2d_get_canvas_prop(const char* prop);\n\nvoid d2d_fillrect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_strokerect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_filltext(struct d2d_draw_seq* ds, const char* str, double x, double y);\nvoid d2d_fillcodepoint(struct d2d_draw_seq* ds, unsigned long c, double x, double y);\nvoid d2d_stroketext(struct d2d_draw_seq* ds, const char* text, double x, double y);\n\nvoid d2d_measuretext(struct d2d_draw_seq* ds, const char* str, struct d2d_text_metrics *tm);\nvoid d2d_save(struct d2d_draw_seq* ds);\nvoid d2d_restore(struct d2d_draw_seq* ds);\n\nvoid d2d_setlinewidth(struct d2d_draw_seq* ds, double width);\nvoid d2d_setstrokestylergba(struct d2d_draw_seq* ds, unsigned long color);\nvoid d2d_setfillstylergba(struct d2d_draw_seq* ds, unsigned long color);\nvoid d2d_setstrokestyle(struct d2d_draw_seq* ds, const char* css_color);\nvoid d2d_setfillstyle(struct d2d_draw_seq* ds, const char* css_color);\nvoid d2d_setfont(struct d2d_draw_seq* ds, const char* font);\nvoid d2d_setlinecap(struct d2d_draw_seq* ds, const char* line_cap);\nvoid d2d_setlinejoin(struct d2d_draw_seq* ds, const char* line_join);\nvoid d2d_setlinedash(struct d2d_draw_seq* ds, unsigned long len, const double* segments);\nunsigned long d2d_getlinedash(struct d2d_draw_seq* ds, unsigned long length, double* buffer);\nunsigned long d2d_getlinedashlength(struct d2d_draw_seq* ds);\nvoid d2d_setlinedashoffset(struct d2d_draw_seq* ds, double line_dash_offset);\n\nvoid d2d_createlineargradient(struct d2d_draw_seq* ds, long id, double x0, double y0, double x1, double y1);\nvoid d2d_createradialgradient(struct d2d_draw_seq* ds, long id, double x0, double y0, double radius0, double x1, double y1, double radius1);\nvoid d2d_addcolorstop(struct d2d_draw_seq* ds, long gradID, long position, const char* csscolor);\nvoid d2d_setfillstylegradient(struct d2d_draw_seq* ds, long gradID);\nvoid d2d_releaseid(struct d2d_draw_seq* ds, long id);\n\nvoid d2d_beginpath(struct d2d_draw_seq* ds);\nvoid d2d_fill(struct d2d_draw_seq* ds);\nvoid d2d_stroke(struct d2d_draw_seq* ds);\nvoid d2d_moveto(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_lineto(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_arc(struct d2d_draw_seq* ds, double x, double y, double radius, double start_angle, double end_angle, bool counterclockwise);\nvoid d2d_arcto(struct d2d_draw_seq* ds, double x1, double y1, double x2, double y2, double radius);\nvoid d2d_bezierto(struct d2d_draw_seq* ds, double cp1x, double cp1y, double cp2x, double cp2y, double x, double y);\nvoid d2d_roundrect(struct d2d_draw_seq* ds, double x, double y, double width, double height, double radii);\nvoid d2d_ellipse(struct d2d_draw_seq* ds, double x, double y, double radiusX, double radiusY, double rotation, double startAngle, double endAngle, bool counterclockwise);\nvoid d2d_quadraticcurveto(struct d2d_draw_seq* ds, double cpx, double cpy, double x, double y);\nvoid d2d_rect(struct d2d_draw_seq* ds, double x, double y, double width, double height);\nvoid d2d_closepath(struct d2d_draw_seq* ds);\n\n//deprecated, use d2d_ctoimagedata instead\nvoid d2d_imagedata(struct d2d_draw_seq* ds, long id, void* mem, unsigned long length, unsigned long width, unsigned long height);\n\nvoid d2d_ctoimagedata(struct d2d_draw_seq* ds, long id, void* mem, unsigned long length, unsigned long width, unsigned long height);\nvoid d2d_putimagedata(struct d2d_draw_seq* ds, long id, unsigned long dx, unsigned long dy);\nvoid d2d_putimagedatadirty(struct d2d_draw_seq* ds, long id, unsigned long dx, unsigned long dy, unsigned long dirtyX, unsigned long dirtyY, unsigned long dirtyWidth, unsigned long dirtyHeight);\n\nvoid d2d_reset(struct d2d_draw_seq* ds);\nvoid d2d_clearrect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_scale(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_translate(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_rotate(struct d2d_draw_seq* ds, double angle);\nvoid d2d_gettransform(struct d2d_draw_seq* ds, struct d2d_2d_matrix *transform);\nvoid d2d_settransform(struct d2d_draw_seq* ds, double a, double b, double c, double d, double e, double f);\nvoid d2d_settransformmatrix(struct d2d_draw_seq* ds, const struct d2d_2d_matrix * transform);\nvoid d2d_transform(struct d2d_draw_seq* ds, double a, double b, double c, double d, double e, double f);\nvoid d2d_transformmatrix(struct d2d_draw_seq* ds, const struct d2d_2d_matrix * transform);\nvoid d2d_resettransform(struct d2d_draw_seq* ds);\n\nbool d2d_load_image(const char* url, long id);\nbool d2d_load_image_with_con(const char* url, long id, twr_ioconsole_t * con);\nvoid d2d_drawimage(struct d2d_draw_seq* ds, long id, double dx, double dy);\nvoid d2d_drawimage_ex(struct d2d_draw_seq* ds, long id, double sx, double sy, double sWidth, double sHeight, double dx, double dy, double dWidth, double dHeight);\nvoid d2d_getimagedata(struct d2d_draw_seq* ds, long id, double x, double y, double width, double height);\nunsigned long d2d_getimagedatasize(double width, double height);\nvoid d2d_imagedatatoc(struct d2d_draw_seq* ds, long id, void* buffer, unsigned long buffer_len);\n\ndouble d2d_getcanvaspropdouble(struct d2d_draw_seq* ds, const char* prop_name);\nvoid d2d_getcanvaspropstring(struct d2d_draw_seq* ds, const char* prop_name, char* buffer, unsigned long buffer_len);\nvoid d2d_setcanvaspropdouble(struct d2d_draw_seq* ds, const char* prop_name, double val);\nvoid d2d_setcanvaspropstring(struct d2d_draw_seq* ds, const char* prop_name, const char* val);\n
d2d_measuretext() returns this structure:
struct d2d_text_metrics {\n double actualBoundingBoxAscent;\n double actualBoundingBoxDescent;\n double actualBoundingBoxLeft;\n double actualBoundingBoxRight;\n double fontBoundingBoxAscent;\n double fontBoundingBoxDescent;\n double width;\n};\n
d2d_get_canvas_prop() returns a value of:
export interface ICanvasProps {\n charWidth: number,\n charHeight: number,\n foreColor: number,\n backColor: number,\n widthInChars: number,\n heightInChars: number,\n canvasWidth:number,\n canvasHeight:number\n}\n
d2d_gettransform() returns this structure:
struct d2d_2d_matrix {\n double a, b, c, d, e, f;\n};\n
d2d_getlinedash() returns this structure:
struct d2d_line_segments {\n long len;\n double *segments;\n};\n
"},{"location":"api/api-c-general/","title":"General C API for Wasm","text":""},{"location":"api/api-c-general/#overview","title":"Overview","text":"This sections describes the \"general\" twr-wasm functions available that don't fit neatly into another category (such as standard C library functions, Draw 2D functions, etc.)
These functions often start with \"twr_\" and are generally found in this include file:
\\twr-wasm\\include\\twr-crt.h
"},{"location":"api/api-c-general/#bzero","title":"bzero","text":"Set a block of memory to zeros. Calls memset(to, 0, count)
.
#include <string.h>\n\nvoid bzero (void *to, size_t count);\n
"},{"location":"api/api-c-general/#getc","title":"getc","text":"This is the standard c library function (see the the standard library docs available on the internet).
Of note this function will return extended ASCII (128-255 inclusive). The extend ASCII are always encoded with Windows-1252 encoding.
See twr_getc32
for a list of related functions.
Note that C character input is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
"},{"location":"api/api-c-general/#twr_atod","title":"twr_atod","text":"Similar to stdlib atof
.
#include \"twr-crt.h\"\n\ndouble twr_atod(const char* str);\n
"},{"location":"api/api-c-general/#twr_atou64","title":"twr_atou64","text":"Convert a string to a 64 bit unsigned integer, stopping when the first non-valid character is encountered. If len is provided, it will be set to the number of characters read. Radix should be >=2 and <=36 -- for example, 10 is a normal base 10 number and 16 is hexadecimal.
#include \"twr-crt.h\"\n\nint64_t twr_atou64(const char *str, int* len, int radix);\n
"},{"location":"api/api-c-general/#twr_dtoa","title":"twr_dtoa","text":"The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
#include \"twr-crt.h\"\n\nvoid twr_dtoa(char* buffer, int sizeInBytes, double value, int max_precision);\n
"},{"location":"api/api-c-general/#twr_cache_mallocfree","title":"twr_cache_malloc/free","text":"These functions keep allocated memory in a cache for much faster re-access than the standard malloc/free.
#include \"twr-crt.h\"\n\nvoid *twr_cache_malloc(twr_size_t size);\nvoid twr_cache_free(void* mem);\n
"},{"location":"api/api-c-general/#twr_code_page_to_utf32_streamed","title":"twr_code_page_to_utf32_streamed","text":"Return a unicode code point (aka utf-32 value) when passed a byte stream that represents an encoded character using the current local's LC_CTYPE code page. A zero is returned if the byte stream has not yet completed a decode.
For example:
int cp\n\nsetlocale(LC_ALL, \"\"); // set to default locale, which will be UTF-8 encoding with local language/region\n\n// turn a UTF-8 Euro into a UTF-32 value\ncp==twr_code_page_to_utf32_streamed(0xE2);\nassert (cp==0);\ncp=twr_code_page_to_utf32_streamed(0x82);\nassert (cp==0);\ncp=twr_code_page_to_utf32_streamed(0xAC);\nassert (cp==0x000020AC); // Euro Code points\n
#include <locale.h>\n\nint twr_code_page_to_utf32_streamed(unsigned char byte) \n
"},{"location":"api/api-c-general/#twr_conlog","title":"twr_conlog","text":"twr_conlog
prints debug messages to stderr
(usually your browser console) from your C code.
#include \"twr-crt.h\"\n\nvoid twr_conlog(char* format, ...);\n
This call is identical to fprintf(stderr, ...)
, except that it adds a newline. When stderr
is set to twrConsoleDebug
each call to twr_conlog() will generate a single call to console.log() in JavaScript to ensure that you see debug prints.
The current implementation does not wait for the debug string to output to the console before returning from twr_conlog, when using twrWasmModuleAsync. In this case, it can take a small bit of time for the string to make its way across the Worker Thread boundary. This is normally not a problem and results in faster performance. But if your code crashes soon after the debug print, the print might not appear. If you think this is an issue, you can call twr_sleep(1)
after your twr_conlog call. This will force a blocking wait for the print to print.
"},{"location":"api/api-c-general/#twr_epoch_timems","title":"twr_epoch_timems","text":"Returns the number of milliseconds since the start of the epoch.
#include \"twr-crt.h\"\n\nuint64_t twr_epoch_timems();\n
"},{"location":"api/api-c-general/#twr_getc32","title":"twr_getc32","text":"Gets a 32 bit unicode code point character from stdin. Unlike the standard C library function getchar
, twr_getc32
does not buffer a line (that is, twr_getc32
will return a character before the user presses Enter).
twr_getc32
is implemented as:
int twr_getc32() {\n return io_getc32(twr_get_stdio_con());\n}\n
Note that stdlib getchar
and ungetc
are not currently implemented.
Note that C character input with these functions is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
Also see:
io_mbgets
- get a multibyte string from a console using the current locale character encoding. Console must support IO_TYPE_CHARREAD. twr_mbgets
- the same as io_mbgets
with the console set to stdin
. io_mbgetc
- get a multibyte character from an IoConsole (like stdin
) using the current locale character encoding getc
(sames as fgetc
) - get a single byte from a FILE * (IoConsole) -- returning ASCII or extended ASCII (window-1252 encoding) io_getc32
- gets a 32 bit unicode code point from an IoConsole (which must support IO_TYPE_CHARREAD)
#include \"twr-crt.h\"\n\nint twr_getc32();\n
"},{"location":"api/api-c-general/#twr_get_navlang","title":"twr_get_navlang","text":"Returns the BCP 47 language tag as found in javacript navigator.language
. If len is not null, it will be filled in with the string length of the language tag.
#include \"twr-crt.h\"\n\nconst char* twr_get_navlang(int *len);\n
"},{"location":"api/api-c-general/#twr_get_current_locale","title":"twr_get_current_locale","text":"extern inline locale_t twr_get_current_locale(void);\n
twr_get_current_locale
will return the locale that has been set by setlocale
. It can be used to pass to a function that takes a locale_t.
"},{"location":"api/api-c-general/#twr_localize_numeric_string","title":"twr_localize_numeric_string","text":"Functions like twr_dtoa
do not localize the decimal point. To get a localized decimal point, you can use printf
, or alternately twr_localize_numeric_string
to post process a string. For example:
char b[10];\nstrcpy(b, \"1.23\");\ntwr_localize_numeric_string(b, twr_get_current_locale());\n// if locale was set to french, then b is now 1,23\n
#include <locale.h>\n\nvoid twr_localize_numeric_string(char* str, locale_t locale);\n
"},{"location":"api/api-c-general/#twr_mem_debug_stats","title":"twr_mem_debug_stats","text":"Print memory map and malloc stats to stderr or stdout.
(note FILE * is the same as twr_ioconsole_t*)
#include <stdio.h>\n\nvoid twr_mem_debug_stats(twr_ioconsole_t* outcon);\n
"},{"location":"api/api-c-general/#twr_mbgets","title":"twr_mbgets","text":"Gets a string from stdin. The string will be in the current locale's character encoding -- ASCII for \"C\", and either UTF-8 or windows-1252 for \"\". See Character Encoding Support with twr-wasm.
#include \"twr-crt.h\"\n\nchar* twr_mbgets(char* buffer);\n
Internally this function uses the stdio IoConsole -- see the IoConsole section for more advanced input/output.
This function will encode characters as specified by the LC_CTYPE category of the current locale. ASCII is used for \"C\", and UTF-8 and Windows-1252 are also supported (see localization)
Note that C character input is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
"},{"location":"api/api-c-general/#twr_mbslen_l","title":"twr_mbslen_l","text":"Returns the number of characters in a string using the character encoding of the passed locale (ASCII for \"C\", UTF-8, or windows-1252 for \"\"). You can use twr_get_current_locale
to find the current locale.
#include <string.h>\n\nsize_t twr_mbslen_l(const char *str, locale_t locale);\n
"},{"location":"api/api-c-general/#twr_sleep","title":"twr_sleep","text":"twr_sleep
is a traditional blocking sleep function. This function is blocking, and so is only available if you use twrWasmModuleAsync
.
#include \"twr-crt.h\"\n\nvoid twr_sleep(int ms);\n
"},{"location":"api/api-c-general/#twr_register_callback","title":"twr_register_callback","text":"Returns a new event ID that is paired with the specified C function. This event ID can be passed to functions that accept an event ID. When the event is triggered, the specified callback is called.
The callback function's first argument will be the event ID. Subsequent arguments are event specific. It is legal to register the same callback for multiple event IDs.
#include \"twr-crt.h\"\n\nint twr_register_callback(const char* func_name);\n
For example:
// timer event callback (called once)\n__attribute__((export_name(\"on_timer1\")))\nvoid on_timer1(int event_id) {\n printf(\"timer callback 1 entered (event id=%d) !\\n\", event_id);\n}\n\n// entry point\n__attribute__((export_name(\"twr_main\")))\nvoid twr_main() {\n int timer1=twr_register_callback(\"on_timer1\");\n twr_timer_single_shot(2000, timer1);\n}\n
"},{"location":"api/api-c-general/#twr_timer_single_shot","title":"twr_timer_single_shot","text":"Triggers the specified event (callback) once after milliSeconds
. Returns a timerID
which can be used with twr_timer_cancel
.
int twr_timer_single_shot(int milliSeconds, int eventID);\n
"},{"location":"api/api-c-general/#twr_timer_repeat","title":"twr_timer_repeat","text":"Triggers the specified event (callback) repeatedly after milliSeconds
. Returns a timerID
which can be used with twr_timer_cancel
.
int twr_timer_repeat(int milliSeconds, int eventID);\n
"},{"location":"api/api-c-general/#twr_timer_cancel","title":"twr_timer_cancel","text":"Cancels the specfied timer.
void twr_timer_cancel(int timerID);\n
"},{"location":"api/api-c-general/#twr_tofixed","title":"twr_tofixed","text":"This function is identical to its JavaScript version.
#include \"twr-crt.h\"\n\nvoid twr_tofixed(char* buffer, int buffer_size, double value, int dec_digits);\n
The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
"},{"location":"api/api-c-general/#twr_toexponential","title":"twr_toexponential","text":"This function is identical to its JavaScript version.
#include \"twr-crt.h\"\n\nvoid twr_toexponential(char* buffer, int buffer_size, double value, int dec_digits);\n
The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
"},{"location":"api/api-c-general/#twr_strhorizflip","title":"twr_strhorizflip","text":"Mirror image the passed in string.
#include \"twr-crt.h\"\n\nvoid twr_strhorizflip(char * buffer, int n);\n
"},{"location":"api/api-c-general/#twr_utf8_char_len","title":"twr_utf8_char_len","text":"Returns the number of bytes in a UTF-8 character (passed as a string pointer). UTF-8 characters can be 1 to 4 bytes in length.
#include <string.h>\n\nint twr_utf8_char_len(const char *str);\n
"},{"location":"api/api-c-general/#twr_utf32_to_code_page","title":"twr_utf32_to_code_page","text":"Takes a utf32 value (aka unicode code point value), and fills in the passed character array buffer with the character encoding of the utf32 value, using the current locale's LC_CTYPE code page. The buffer is 0 terminated.
Also see c32rtomb
and c16rtomb
.
For example:
char strbuf[6]; // max size of utf-8 is 4+terminating zero. Max size of ASCII or windows 1252 is 1 + terminating zero\nsetlocale(LC_ALL, \"\"); // set to default locale, which will be UTF-8 encoding with local language/region\ntwr_utf32_to_code_page(strbuf, 0x000020AC); // encode a Euro code point \nprintf(\"%s\", strbuf); \nassert ( strcmp(strbuf,\"\\xE2\\x82\\xAC\")==0 ); // utf-8 encoding of euro\nassert ( strcmp(strbuf,\"\u20ac\")==0 ); // clang string literals default to utf-8 encoding\n
include <locale.h>\n\nvoid twr_utf32_to_code_page(char* out, int utf32)\n
"},{"location":"api/api-c-general/#twr_vprintf","title":"twr_vprintf","text":"Performs a printf by calling the callback with cbdata for each character.
#include \"twr-crt.h\"\n\nvoid twr_vprintf(twr_cbprintf_callback out, void* cbdata, const char *format, va_list* args);\n
"},{"location":"api/api-c-general/#floating-math-helpers","title":"floating math helpers","text":"int twr_isnan(double v);\nint twr_isinf(double v);\ndouble twr_nanval();\ndouble twr_infval();\n
"},{"location":"api/api-c-localization/","title":"Localization Reference for twr-wasm","text":"This section details twr-wasm's WebAssembly localization support.
Also see Introduction to Character Encoding Support with twr-wasm
"},{"location":"api/api-c-localization/#using-c","title":"Using C:","text":"Standard C locale functions are supported by twr-wasm. ASCII, UTF-8 and windows-1252 encoding is supported by the twr-wasm standard C library locale. twr-wasm also includes C functions for UTF-32 support.
"},{"location":"api/api-c-localization/#using-c_1","title":"Using C++:","text":" - libc++ locale and unicode functions are supported by twr-wasm.
- libc++ unicode support includes utf-16 and utf-32 strings.
"},{"location":"api/api-c-localization/#character-encodings","title":"Character Encodings","text":"twr-wasm C locales support ASCII, UTF-8 or windows-1252 encoding. UTF-16/32 are not supported as a std c lib locale setting, but functions are provided to convert utf-32 (unicode code points) to and from ASCII, UTF-8, and windows-1252 \"code pages\" (there are other miscellaneous utf-32 based functions as well.)
"},{"location":"api/api-c-localization/#locales-standard-c-library","title":"Locales (Standard C Library)","text":""},{"location":"api/api-c-localization/#c","title":"\"C\"","text":"\"C\" is the default locale, as usual. When \"C\" is selected, the functions operate as usual. One subtly is that console i/o functions (such as printf
) will generally function as expected with UTF-8, since the div
and window
consoles correctly handle UTF-8 character encoding. This is normal on some OSs, such as linux, but not the default on Windows (which often defaults to windows-1252 for backward compatibility).
isgraph
style functions will only recognize ASCII characters, as is normal. Functions such as strcmp
operate on the byte sequence, which will typically results in UTF-8 codes being compared lexically. strcoll
will use lexical ordering.
"},{"location":"api/api-c-localization/#posix","title":"\"POSIX\"","text":"\"POSIX\" is the same as \"C\"
"},{"location":"api/api-c-localization/#_1","title":"\"\"","text":"\"\" is the locale to specify the users default setting (this selects the setting used by the browser). This will also enable UTF-8 in functions such as strcoll
. For example, if your browser is set to \"en-US\" as its default locale, setlocale(LC_ALL, \"\")
will return en-US.UTF-8
.
isgraph
style functions will still only recognize ASCII characters (since UTF-8 doesn't encode any single bytes greater than 127). strcoll
uses locale specific ordering, and printf
will use locale specific decimal points. strcmp
still compares two strings lexicographically (byte-by-byte) without considering locale-specific rules, per the spec.
"},{"location":"api/api-c-localization/#utf-8","title":"\".UTF-8\"","text":"\".UTF-8\" is the same as \"\" with twr-wasm.
"},{"location":"api/api-c-localization/#1252","title":"\".1252\"","text":"\".1252\" will select the current default locale, but use windows-1252 character encoding (instead of UTF-8). Windows-1252 is a super set of ISO-8859-1 and is the most commonly used encoding for many european languages when unicode is not used. This mode is primarily for legacy software, backwards compatibly, and windows compatibility.
"},{"location":"api/api-c-localization/#others","title":"Others","text":"Setting arbitrary locales, such as \"fr-FR\" when the browser is defaulted to another locale, is not supported.
"},{"location":"api/api-c-localization/#select-the-default-locale","title":"Select the default locale","text":"To select the user's browser's default locale using the C language, and enable consistent utf-8 support, use a call like this:
setlocale(LC_ALL, \"\")\n
"},{"location":"api/api-c-localization/#c-and-libc-functions","title":"C and libc++ functions","text":"If you are using twr-wasm's build of libc++, libc++ locale and unicode functions work as normal.
The usual standard C library locale support is available, along with some POSIX extensions. In addition, some locale useful twr-wasm specific functions are documented in C API, such as twr_get_current_locale
,twr_mbgets
, twr_getc32
, twr_utf8_char_len
, twr_mbslen_l
, twr_utf32_to_code_page
, twr_code_page_to_utf32_streamed
, twr_get_navlang
, twr_localize_numeric_string
.
Note that io_getc32
, getc(stdin)
, fgetc(stdin)
do not look at the current locale. io_getc32
returns a 32 bit unicode code point, and getc
/fgetc
return extended ASCII.
For a locale aware character input, use io_mbgetc()
or twr_mbgets()
. Both use the locale category LC_CTYPE. See C API.
Note that when the locale is not set (or whenever the \"C\" locale is set) functions that get character(s) from stdin that are locale aware, like twr_mbgets()
, behave different than functions that output characters to stdout (like puts
, io_putstr
, io_putc
, putchar
). Characters to stdout in \"C\" locale will handle UTF-8 characters. For stdin, \"C\" locale uses ASCII.
For consistent UTF-8 (or windows-1252) behavior, set the locale as discussed above ( use setlocale
)
The primary standard C library locale functions are:
char* setlocale(int category, const char* locale);\nstruct lconv *localeconv(void);\n
As well as the two standard library functions above, appropriate functions take into account the current locale (printf, strcoll, etc).
Note that setlocale
returns a string using BCP 47 format (like a web browser). Locale strings look like \"en-US.UTF-8\", instead of \"en_US.UTF-8\". A dash, not an underscore, is used as a separator.
POSIX functions These are the extended POSIX style functions provided that are related to locale:
locale_t newlocale(int category_mask, const char *locale, locale_t base);\nlocale_t uselocale(locale_t);\nvoid freelocale(locale_t);\nlocale_t duplocale(locale_t);\n\nint isalnum_l(int c, locale_t loc);\nint isalpha_l(int c, locale_t loc);\nint isblank_l(int c, locale_t loc);\nint iscntrl_l(int c, locale_t loc);\nint isdigit_l(int c, locale_t loc);\nint isgraph_l(int c, locale_t loc);\nint islower_l(int c, locale_t loc);\nint isprint_l(int c, locale_t loc);\nint ispunct_l(int c, locale_t loc);\nint isspace_l(int c, locale_t loc);\nint isupper_l(int c, locale_t loc);\nint isxdigit_l(int c, locale_t loc);\nint tolower_l(int c, locale_t loc);\nint toupper_l(int c, locale_t loc);\n\nlong long strtoll_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long long strtoull_l(const char *str, char **str_end, int base, locale_t loc);\nfloat strtof_l(const char *str, char ** str_end, locale_t locale);\ndouble strtod_l(const char *str, char **str_end, locale_t locale);\nlong double strtold_l(const char *str, char **str_end, locale_t locale);\n\nint strcoll_l(const char* lhs, const char* rhs, locale_t loc);\n\nsize_t strftime_l(char *s, size_t maxsize, const char *format, const struct tm *timeptr, locale_t locale);\n
"},{"location":"api/api-c-stdlib/","title":"Standard C library for WebAssembly","text":"This section describes twr-wasm's support for the Standard C Library. twr-wasm includes its own implementation of the standard C library optimized for WebAssembly and Wasm running in a web browser. This is a core feature of twr-wasm.
For documentation of these functions, see the many standard C library documentation web sites.
The following subset of the standard C library is available. Also see twr-wasm/include
folder for include files.
"},{"location":"api/api-c-stdlib/#stdioh","title":"stdio.h","text":"* fprintf will only work with these -- stderr, stdin, stdout */\n/* these return 'twr_ioconsole_t *' which is same as 'FILE *' */\n#define stderr (FILE *)(twr_get_stderr_con())\n#define stdin (FILE *)(twr_get_stdio_con())\n#define stdout (FILE *)(twr_get_stdio_con())\n\nint snprintf(char *buffer, size_t bufsz, const char *format, ... );\nint sprintf( char *buffer, const char *format, ... );\nint vsnprintf(char *buffer, size_t bufsz, const char *format, va_list vlist);\nint vasprintf(char **strp, const char* format, va_list vlist );\nint printf(const char* format, ...);\nint vprintf(const char* format, va_list vlist );\nint puts(const char *str);\nint putchar(int c);\n\ntypedef twr_ioconsole_t FILE; \nint vfprintf(FILE *stream, const char *format, va_list vlist);\nint fprintf(FILE *stream, const char* format, ...);\nsize_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);\nint ferror(FILE *stream);\nint feof(FILE *stream);\nint fflush(FILE *stream);\nint is_terminal(FILE *stream);\nint fputc(int ch, FILE* stream);\nint putc(int ch, FILE* stream);\nint fgetc(FILE *stream );\nint getc(FILE *stream);\n
"},{"location":"api/api-c-stdlib/#stdlibh","title":"stdlib.h","text":"void *malloc(size_t size);\nvoid free(void *mem);\nsize_t avail(void);\nvoid *realloc( void *ptr, size_t new_size );\nvoid* calloc( size_t num, size_t size );\nvoid *aligned_alloc( size_t alignment, size_t size );\n\nint rand(void);\nvoid srand(int seed);\n\n#define __min(a,b) (((a) < (b)) ? (a) : (b))\n#define __max(a,b) (((a) > (b)) ? (a) : (b))\n\nint abs(int n);\n\nint _fcvt_s(\n char* buffer,\n size_t sizeInBytes,\n double value,\n int fracpart_numdigits,\n int *dec,\n int *sign\n);\ndouble atof(const char* str);\nint atoi(const char *str);\nlong atol( const char *str );\nlong long atoll( const char *str );\nlong strtol(const char *str, char **str_end, int base);\nlong long strtoll(const char *str, char **str_end, int base);\nlong long strtoll_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long long strtoull(const char *str, char **str_end, int base);\nunsigned long long strtoull_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long strtoul(const char *str, char ** str_end, int base);\nfloat strtof(const char *str, char ** str_end);\nfloat strtof_l(const char *str, char ** str_end, locale_t locale);\ndouble strtod(const char *str, char **str_end);\ndouble strtod_l(const char *str, char **str_end, locale_t locale);\nlong double strtold(const char *str, char **str_end);\nlong double strtold_l(const char *str, char **str_end, locale_t locale);\nint _itoa_s(int64_t value, char * buffer, size_t size, int radix);\n\ndiv_t div( int x, int y );\nldiv_t ldiv( long x, long y );\nlldiv_t lldiv( long long x, long long y );\n\n_Noreturn void abort(void);\nint atexit(void (*func)(void));\n
Note that _fcvt_s as currently enabled has these limitations: - fractional digits <=100 - values must be less than 1e+21 - values negative exponents must be smaller than 1e-99
There is a full featured version of _fcvt_s in the source code, but it is not currently enabled, since the version enabled is smaller and works in most use cases.
"},{"location":"api/api-c-stdlib/#asserth","title":"assert.h","text":"void assert(int expression);\n
"},{"location":"api/api-c-stdlib/#mathh","title":"math.h","text":"double acos(double arg);\ndouble asin(double arg);\ndouble atan(double arg);\ndouble atan2(double y, double x);\ndouble ceil(double arg);\ndouble cos(double arg);\ndouble exp(double arg);\ndouble fabs(double arg);\ndouble floor(double arg);\ndouble fmod(double x, double y);\ndouble log(double arg);\ndouble pow(double base, double exp);\ndouble sin(double arg);\ndouble sqrt(double arg);\ndouble tan(double arg);\ndouble trunc(double arg);\n
"},{"location":"api/api-c-stdlib/#stdargh","title":"stdarg.h","text":"#define va_start(v,l) __builtin_va_start(v,l)\n#define va_end(v) __builtin_va_end(v)\n#define va_arg(v,l) __builtin_va_arg(v,l)\n#define va_copy(d,s) __builtin_va_copy(d,s)\ntypedef __builtin_va_list va_list;\n
"},{"location":"api/api-c-stdlib/#ctypeh","title":"ctype.h","text":"int isascii(int);\nint toascii(int);\nint isalnum(int c);\nint isalpha(int c);\nint isblank(int);\nint iscntrl(int);\nint isdigit(int c);\nint isgraph(int c);\nint islower(int);\nint isprint(int);\nint ispunct(int);\nint isspace(int c);\nint isupper(int);\nint isxdigit(int);\nint tolower(int c);\nint toupper(int c);\n\nint isalnum_l(int c, locale_t loc);\nint isalpha_l(int c, locale_t loc);\nint isblank_l(int c, locale_t loc);\nint iscntrl_l(int c, locale_t loc);\nint isdigit_l(int c, locale_t loc);\nint isgraph_l(int c, locale_t loc);\nint islower_l(int c, locale_t loc);\nint isprint_l(int c, locale_t loc);\nint ispunct_l(int c, locale_t loc);\nint isspace_l(int c, locale_t loc);\nint isupper_l(int c, locale_t loc);\nint isxdigit_l(int c, locale_t loc);\nint tolower_l(int c, locale_t loc);\nint toupper_l(int c, locale_t loc);\n
"},{"location":"api/api-c-stdlib/#stddefh","title":"stddef.h","text":"#define offsetof(TYPE, MEMBER) __builtin_offsetof (TYPE, MEMBER)\ntypedef __PTRDIFF_TYPE__ ptrdiff_t;\ntypedef double max_align_t;\n
"},{"location":"api/api-c-stdlib/#stringh","title":"string.h","text":"size_t strlen(const char * str);\nchar *strdup(const char * source);\nchar *strcpy(char *dest, const char *source);\nint strcat_s(char *dest, size_t destsz, const char *src);\nchar* strcat(char *dest, const char *src);\nchar *strncpy(char *dest, const char *source, size_t count);\nint strcmp(const char* string1, const char* string2);\nint strncmp(const char* lhs, const char* rhs, size_t count);\nint stricmp(const char* string1, const char* string2);\nint strnicmp(const char* string1, const char* string2, size_t count);\nint strcoll(const char* lhs, const char* rhs);\nint strcoll_l(const char* lhs, const char* rhs, locale_t loc);\nchar *strchr(const char *str, int ch);\nvoid *memchr(const void *ptr, int ch, size_t count);\nchar *strstr(const char *haystack, const char *needle);\nchar * strerror(int errnum );\nchar * _strerror(const char *strErrMsg);\nvoid *memmove(void *dest, const void *src, size_t n);\nint memcmp( const void* lhs, const void* rhs, size_t count );\nvoid bzero (void *to, size_t count);\n\n// implemented in memcpy.wat\nvoid *memcpy(void *dest, const void * src, size_t n);\nvoid *memset(void *mem, int c, size_t n);\n
"},{"location":"api/api-c-stdlib/#timeh","title":"time.h","text":"typedef unsigned long time_t;\nunsigned long time(unsigned long *time);\nsize_t strftime(char *s, size_t maxsize, const char *format, const struct tm *timeptr);\nsize_t strftime_l(char *s, size_t maxsize, const char *format, const struct tm *timeptr, locale_t locale);\nstruct tm *localtime(const time_t *timer);\nint gettimeofday(struct timeval *tv, void* notused);\n#define timerisset(tvp) ((tvp)->tv_sec || (tvp)->tv_usec)\n#define timercmp(tvp,uvp,cmp) \\\n ((tvp)->tv_sec cmp (uvp)->tv_sec || \\\n ((tvp)->tv_sec == (uvp)->tv_sec && (tvp)->tv_usec cmp (uvp)->tv_usec))\n#define timerclear(tvp) (tvp)->tv_sec = (tvp)->tv_usec = 0\n
"},{"location":"api/api-c-stdlib/#localeh","title":"locale.h","text":"#define LC_GLOBAL_LOCALE twr_get_current_locale()\nchar* setlocale(int category, const char* locale);\nstruct lconv *localeconv(void);\nlocale_t newlocale(int category_mask, const char *locale, locale_t base);\nlocale_t uselocale(locale_t);\nvoid freelocale(locale_t);\nlocale_t duplocale(locale_t);\nextern inline locale_t twr_get_current_locale(void);\n
"},{"location":"api/api-c-stdlib/#ucharh","title":"uchar.h","text":"typedef uint_least32_t char32_t;\ntypedef uint_least16_t char16_t;\n\nsize_t c32rtomb( char* s, char32_t c32, mbstate_t* ps );\n
"},{"location":"api/api-c-stdlib/#errnoh","title":"errno.h","text":"typedef int errno_t;\n\nextern int * _errno(void);\n#define errno (*_errno())\n\nerrno_t _set_errno(int _Value);\nerrno_t _get_errno(int *_Value);\n
"},{"location":"api/api-c-stdlib/#_stdtypesh","title":"_stdtypes.h","text":"// don't include directly -- included by various .h files
typedef unsigned long size_t;\n#define MAX_SIZE_T 2147483647 \n\n#ifdef __cplusplus\n#define NULL __null\n#else\n#define NULL ((void*)0)\n#endif\n\ntypedef struct __locale_t_struct * locale_t;\n
"},{"location":"api/api-c-stdlib/#other-include-files-available","title":"Other include files available","text":"float.h\nlimits.h\nstdbool.h\nstdint.h\n
"},{"location":"api/api-libcpp/","title":"libc++ for WebAssembly","text":"This section describes twr-wasm's support for using the standard c++ library libc++ with WebAssembly.
twr-wasm includes libc++ built for WebAssembly in the twr-wasm/lib-c
folder.
For C++ the use of libc++ is optional. That is you can build twr-wasm projects in C++ with or without libc++.
See the examples tests-libcx and tests-user for examples of using libc++.
See the balls example for how to create a C++ WebAssembly program without the standard C++ library. The primary advantage to this approach is a bit smaller code size. You don't need to staticly link libc++.
Some of the key options twr-wasm's libc++ for WebAssembly was built with are these:
DLIBCXX_ENABLE_LOCALIZATION=ON \nDLIBCXX_ENABLE_UNICODE=ON \nDLIBCXX_ENABLE_RTTI=ON \nDLIBCXX_ENABLE_STATIC_ABI_LIBRARY=ON \n\nDCMAKE_BUILD_TYPE=Release \nDCMAKE_CXX_STANDARD=20 \n\nDLIBCXX_ENABLE_EXCEPTIONS=OFF \nDLIBCXX_ENABLE_THREADS=OFF \nDLIBCXX_ENABLE_SHARED=OFF \nDLIBCXX_ENABLE_WIDE_CHARACTERS=OFF \nDLIBCXX_ENABLE_FILESYSTEM=OFF \nDLIBCXX_ENABLE_TIME_ZONE_DATABASE=OFF \nDLIBCXX_ENABLE_MONOTONIC_CLOCK=OFF \nDLIBCXX_ENABLE_RANDOM_DEVICE=OFF\n
"},{"location":"api/api-ts-consoles/","title":"Console Classes","text":"This section describes the twr-wasm TypeScript/JavaScript classes that you use to create I/O Consoles for character streaming, a terminal, or 2D Canvas Drawing
The classes twrConsoleDiv
, twrConsoleTerminal
, twrConsoleDebug
, and twrConsoleCanvas
create consoles that enable user i/o. Your C/C++ can direct user interactive i/o to these consoles.
"},{"location":"api/api-ts-consoles/#related-console-documentation","title":"Related Console Documentation","text":" - Console Introduction
- Console C APIs
"},{"location":"api/api-ts-consoles/#class-twrconsoledebug","title":"class twrConsoleDebug","text":"twrConsoleDebug
streamings characters to the browser debug console.
C type: IO_TYPE_CHARWRITE
There are no constructor parameters.
"},{"location":"api/api-ts-consoles/#class-twrconsolediv","title":"class twrConsoleDiv","text":"twrConsoleDiv
streams character input and output to a div tag .
C type: IO_TYPE_CHARREAD
and IO_TYPE_CHARWRITE
The div tag will expand as you add more text (via printf, etc).
You pass a <div>
element to use to render the Console to to the twrConsoleDiv
constructor. For example:
<div id=\"div1\" tabindex=\"0\"></div>\n\n<script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleDiv} from \"twr-wasm\";\n\n const stream1Element=document.getElementById(\"div1\");\n\n // adding keyDown events is needed if the console will accept key input\n // don't forget to set \"tabindex\" in your tag, otherwise it won't get key events\n stream1Element.addEventListener(\"keydown\",(ev)=>{stream1.keyDown(ev)});\n\n const stream1 = new twrConsoleDiv(stream1Element);\n const mod = new twrWasmModuleAsync( {stdio: stream1} );\n // mod.callC would go here...\n</script>\n
There are constructor options to set the color and font size. You can also set these directly in the HTML for your <div>
tag. If you wish to change the default font, set the font in the div
tag with the normal HTML tag options.
twrConsoleDiv constructor optionsconstructor(element:HTMLDivElement, params:IConsoleDivParams)\n\nexport interface IConsoleDivParams {\n foreColor?: string,\n backColor?: string,\n fontSize?: number,\n}\n
You can use the putStr
member function to print a string to the div console in JavaScript.
"},{"location":"api/api-ts-consoles/#class-twrconsoleterminal","title":"class twrConsoleTerminal","text":"twrConsoleTerminal
provides streaming and addressable character input and output. A <canvas>
tag is used to render into.
C types: IO_TYPE_CHARREAD
, IO_TYPE_CHARWRITE
, IO_TYPE_ADDRESSABLE_DISPLAY
twrConsoleTerminal is a simple windowed terminal and supports the same streamed output and input features as a does twrConsoleDiv
, but also supports x,y coordinates, colors, and other features. The window console supports chunky (low res) graphics (each character cell can be used as a 2x3 graphic array).
The canvas width and height, in pixels, will be set based on your selected font size and the width and height (in characters) of the terminal. These are passed as constructor options when you instantiate the twrConsoleTerminal
.
You can use the putStr
member function on twrConsoleTerminal to print a string to the terminal in JavaScript.
As you add more text (via printf, etc), the twrConsoleTerminal
will scroll if it becomes full (unlike twrConsoleDiv
, which expands)
A list of C functions that operate on twrConsoleTerminal
are available.
Here is an example:
<body>\n\n <canvas id=\"canvas1forterm\" tabindex=\"0\"></canvas>\n\n <script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleTerminal} from \"twr-wasm\";\n\n // find the HTML elements that we will use for our console to render into\n const term1Element=document.getElementById(\"canvas1forterm\");\n\n // adding keyDown events is needed if the console will accept key input\n // don't forget to set \"tabindex\" in your tag, otherwise it won't get key events\n term1Element.addEventListener(\"keydown\",(ev)=>{term1.keyDown(ev)});\n\n // create the console\n const term1 = new twrConsoleTerminal(term1Element, {widthInChars: 50, heightInChars: 20});\n\n const amod = new twrWasmModuleAsync( \n {io:{\n stdio: debug, stderr: debug, stream1: stream1, stream2: stream2, term1: term1, term2: term2, draw1: draw1, draw2: draw2\n }} );\n\n // set the input focus so user doesn't have to click\n stream1Element.focus();\n\n // load the wasm code and call the multi C function\n await amod.loadWasm(\"./multi-io.wasm\");\n await amod.callC([\"multi\"]);\n\n // example of using a console in in JavaScript\n stream1.putStr(`Hello stream1 of type ${stream1.getProp(\"type\")} from JavaScript!\\n`);\n\n </script>\n</body>\n
twrConsoleTerminal constructor optionsconstructor (canvasElement:HTMLCanvasElement, params:IConsoleTerminalParams)\n\n// see twrConsoleDiv options elsewhere, which are also supported\nexport interface IConsoleTerminalParams extends IConsoleDivParams {\n widthInChars?: number,\n heightInChars?: number,\n}\n
"},{"location":"api/api-ts-consoles/#class-twrconsolecanvas","title":"class twrConsoleCanvas","text":"twrConsoleCanvas
creates a 2D drawing surface that the Canvas compatible 2d drawing APIs can be used with.
C type: IO_TYPE_CANVAS2D
.
constructor(element:HTMLCanvasElement)\n
twrConsoleCanvas Example<body>\n canvas id=\"canvas1for2d\"></canvas>\n\n <script type=\"module\">\n import {twrWasmModule, twrConsoleCanvas} from \"twr-wasm\";\n\n // find the HTML elements that we will \n // use for our console to render into\n const draw1Element=document.getElementById(\"canvas1for2d\");\n\n // create the console\n const draw1 = new twrConsoleCanvas(draw1Element);\n\n const mod = new twrWasmModule( {io: {std2d: draw1} }} );\n\n // callC here...\n </script>\n
"},{"location":"api/api-ts-library/","title":"twr-wasm Libraries","text":"twr-wasm Libraries are used to expose TypeScript code to C/C++ as C APIs. All of the twr-wasm C APIs are implemented with twr-wasm libraries. You can also use a library to implement your own C APIs using TypeScript.
There are two kinds of Libraries:
- Those that have only once instance (such as the math library)
- Those that can have multiple instances across one more more library types, where each library type implements the same interface. Consoles are an example of this (see interfaceName).
"},{"location":"api/api-ts-library/#basic-steps","title":"Basic Steps","text":"twr-wasm Libraries support both twrWasmModule
and twrWasmModuleAsync
. That is, when you create a twrLibrary, it will function with either type of module. In many cases no extra work is needed for the twrWasmModuleAsync
, but in some cases, extra code is needed.
The class twrLibrary
provides the core functionality:
- Support for functions to be imported into the WebAssembly Module
- Support for
twrWasmModuleAsync
proxy Web Worker thread - An event framework, allowing you to post events to WebAssembly C code.
To implement a twr-wasm library you:
- create a new TypeScript class that extends
class twrLibrary
. - create a C .h file for your new functions (with function signatures)
- add one or more functions to your TypeScript class
- add the functions to the
import
object (that is part of your class) - consider if special handling is needed for
twrWasmModuleAsync
(more on this below)
"},{"location":"api/api-ts-library/#lib-example","title":"Lib Example","text":"See the lib
example here for a more complete example which shows how each of the different use cases can be handled. In addition, you can look in /source/twr-ts
for files that start with twrlib*
or twrcon*
for examples.
"},{"location":"api/api-ts-library/#example-twrlibtimer","title":"Example twrLibTimer","text":"The following code is from the twr-wasm source for twrlibtimer.
twr_timer_single_shot
- sends an event to C after the timer times out. twr_sleep
- blocks C execution for a period of time.
import {IWasmModule,} from \"./twrmod.js\"\nimport {IWasmModuleAsync} from \"./twrmodasync.js\"\nimport {twrLibrary, TLibImports, twrLibraryInstanceRegistry} from \"./twrlibrary.js\"\n\n// Libraries use default export\nexport default class twrLibTimer extends twrLibrary {\n id: number;\n imports:TLibImports = {\n twr_timer_single_shot:{},\n twr_sleep:{isAsyncFunction: true, isModuleAsyncOnly: true},\n };\n\n libSourcePath = new URL(import.meta.url).pathname;\n\n constructor() {\n // all library constructors should start with these two lines\n super();\n this.id=twrLibraryInstanceRegistry.register(this);\n }\n\n twr_timer_single_shot(callingMod:IWasmModule|IWasmModuleAsync, milliSeconds:number, eventID:number) {\n setTimeout(()=>{\n callingMod.postEvent(eventID)\n }, milliSeconds); \n }\n\n async twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number) {\n const p = new Promise<void>( (resolve)=>{\n setTimeout(()=>{ resolve() }, milliSeconds); \n });\n\n return p;\n }\n\n}\n
"},{"location":"api/api-ts-library/#c-header-files","title":"C Header Files","text":"You need to create a .h file that provides signatures to the C users of your new API. For example, for this library your .h file would be this:
__attribute__((import_name(\"twr_timer_single_shot\"))) void twr_timer_single_shot(int ms, int event_id);\n__attribute__((import_name(\"twr_sleep\"))) void twr_sleep(int ms);\n
The purpose of import_name
code is to export your functions from WebAssembly to JavaScript. These are an equivalent alternative to adding the functions to an wasm-ld
-export
option.
"},{"location":"api/api-ts-library/#registering-your-api","title":"Registering your API","text":"To register you class so that the APIs are available to C code, you use code akin to this in your index.html
(or similar):
import twrLibTimerMod from \"./twrlibtimer.js\" // default export\n\nnew twrLibTimerMod(); // will register itself\n
If you are a contributor to twr-wasm and plan to add your library as new built-in APIs, add the registration to twrLibBultins.ts
"},{"location":"api/api-ts-library/#example-function-explained","title":"Example Function Explained","text":"Here is what is happening in this code:
imports:TLibImports = {\n twr_timer_single_shot:{},\n}\n\n// this function will work in both twrWasmModule and twrWasmModuleAsync\ntwr_timer_single_shot(callingMod:IWasmModule|IWasmModuleAsync, milliSeconds:number, eventID:number) {\n setTimeout(()=>{\n callingMod.postEvent(eventID)\n }, milliSeconds); \n}\n
twr_timer_single_shot
is listed in the imports
class member variable which causes the function twr_timer_single_shot
to be added to the WebAssembly.ModuleImports imports.
The argument for callingMod:IWasmModule|IWasmModuleAsync
is filled in by the twrLibrary
code -- the calling C function does not include this as an argument. All Parameters following callingMod
are passed by the C code calling the function.
twr_timer_single_shot
creates a JavaScript timer. When the timer completes, an event is posted, which will trigger a callback in the C code.
"},{"location":"api/api-ts-library/#events","title":"Events","text":"To receive an Event, the C code needs to register an event callback, and then pass the event ID to the function that will generate events. Like this:
__attribute__((export_name(\"on_timer\")))\nvoid on_timer(int event_id) {\n printf(\"timer callback 2 entered (event id=%d)\\n\", event_id);\n}\n\n__attribute__((export_name(\"twr_main\")))\nvoid twr_main() {\n int timer1=twr_register_callback(\"on_timer\");\n twr_timer_single_shot(2000, timer);\n
In this example, on_timer
will be called after 2 seconds.
__attribute__((export_name(\"on_timer\")))
is needed because this C function is exported out of the WebAssembly module and into the JavaScript code. JavaScript needs to call this function.
In this example, the event does not have any arguments. But it may -- integers (which includes pointers) can be accepted as arguments to the event callback. These arguments are event specific.
"},{"location":"api/api-ts-library/#imports","title":"imports","text":"All TypeScript functions that you wish to import into a WebAssembly module as C APIs, should be listed in the imports
object.
Each function in the imports
object has optional options, that are primarily for use with twrWasmModuleAsync
modules.
imports
are added to WebAssembly.ModuleImports imports.
"},{"location":"api/api-ts-library/#callingmod","title":"callingMod","text":"Each function listed in the import
section will be passed a module as the first parameter. In general, a function should be written to handle being called either with IWasmModule
or IWasmModuleAsync
as the calling module interface ( callingMod:IWasmModule|IWasmModuleAsync
). This is generally straight forward.
Examples that might cause some extra work, and that are covered below, are:
- Implementing blocking functions like
twr_sleep
- Allocating memory, for example, to return a string
"},{"location":"api/api-ts-library/#numbers-only","title":"Numbers Only","text":"All of the parameters received by an import
function need to be numbers. These functions interface directly with the WebAssembly module with no conversion. If you are passing or returning strings, or accessing structures, you will need to use the data access functions that are provided in callingMod.memWasm
(more on this below). The general issue and approach is explained in this document..
"},{"location":"api/api-ts-library/#memwasm","title":"memWasm","text":"A callingMod
member function that you may need to use is memWasm
(callingMod.memWasm
). memWasm
is used to access data in the WebAssembly Memory. This will happen when you need to dereference a pointer, access strings, or access structures. See wasmMem
documentation here.
memWasm
is exposed by both IWasmModule
and IWasmModuleAsync
.
IWasmModule
exposes wasmMem: IWasmMemory
IWasmModuleAsync
exposes wasmMem: IWasmMemoryAsync
.
If you wish to write a function that accesses the async
PutXXX
functions, you should use the isAsyncFunction: true
option.
"},{"location":"api/api-ts-library/#twrwasmmodule-and-twrwasmmoduleasync","title":"twrWasmModule
and twrWasmModuleAsync
.","text":"twrLibrary's are designed to work with either twrWasmModule
and twrWasmModuleAsync
. (Recall that twr-wasm has two different module types: twrWasmModule
and twrWasmModuleAsync
).
twrWasmModule
runs in the JavaScript main thread, and thus all functions that it exposes are asynchronous -- meaning that they should return relatively quickly and not block execution. twrWasmModuleAsync
runs in a JavaScript Worker thread , and thus all functions that it exposes are synchronous -- meaning they can block C execution. The \"Async\" in twrWasmModuleAsync
refers to the fact that javaScript can await
on twrWasmModuleAsync
blocking APIs. It takes blocking APIs and makes them \"asynchronous\" to the JavaScript main thread.
Although many functions can be listed in imports
and written without worrying about which type of module is using them, this isn't always true. Some tomes extra code or thought is needed to have optimal APIs for twrWasmModuleAsync
.
In the above example, twr_timer_single_shot
will work correctly with both twrWasmModule
and twrWasmModuleAsync
. It is an asynchronous function -- meaning that it returns quickly.
However, the function twr_sleep
blocks C code, and will only work with twrWasmModuleAsync
.
"},{"location":"api/api-ts-library/#twrwasmmoduleasync-thread-structure","title":"twrWasmModuleAsync
thread structure","text":"To understand more clearly why twrWasmModuleAsync
might need more attention, it is helpful to understand how it is allocates task between its two threads: the JavaScript main thread, and a worker thread.
"},{"location":"api/api-ts-library/#the-twr_sleep-function-is-used-to-illustrate-thread-structure","title":"The twr_sleep
function is used to illustrate thread structure","text":"The function twr_sleep_async
will only function with twrWasmModuleAsync
because it causes the C code to block. For example, twr_sleep
can be used like this:
printf(\"going to sleep...\");\ntwr_sleep(1000);\nprintf(\"awake!\\n\");\n
"},{"location":"api/api-ts-library/#twrwasmmoduleasync-uses-two-threads","title":"twrWasmModuleAsync
uses Two Threads","text":"ThetwrWasmModuleAsync
consists of two threads: The JavaScript main thread, and the Web Worker. By default the code for your library functions is always executed in the JavaScript main thread. In the Web Worker, an internal class called twrWasmModuleAsyncProxy
executes. The default execution (unless isCommonCode
is specified -- which is explained later), happens like this:
- in C: twr_sleep() is called
- in
twrWasmModuleAsyncProxy
, a message is sent to the JavaScript main thread, requesting execution of the twrLibTimer.twr_sleep
function. twrWasmModuleAsyncProxy
is paused (thus twr_sleep
is blocking), waiting for a response to the message sent ins step 2. - The JavaScript main thread receives the message, and
awaits
on twrLibTimer.twr_sleep
- When the Promise that is being awaited on resolves, the JavaScript main threads sends a reply back to
twrWasmModuleAsyncProxy
indicating that execution is complete. If there are any return codes they are also sent (twr_sleep does not have a return code) twr_sleep
returns to the C caller
The above sequence actually happens for all import
functions by default when using twrWasmModuleAsync
, irregardless if or how long they block for. This is because certain JavaScript code can only execute in the JavaScript main thread. Import function options exists to modify this behavior, in the cases where it is not desired.
The above steps also glosses over an important point -- the method that the twrWasmModuleAsyncProxy
uses to wait for a response from the main JavaScript thread. In step 3 above (Worker thread is blocking from twr_sleep
call), the worker thread is blocking on a call to Atomics.wait
. Communication from the JavaScript main thread to the Worker is through shared memory and a circular buffer. This is how twrWasmModuleAsync is able to block execution of the C code. This means that the Worker Thead main event loop can block for long periods of time -- perhaps indefinitely. And this means common JavaScript code can not run reliably in the Worker thread. For example, a setTimeout callback may not happen (because it is dispatched by the main JavaScript event loop). Likewise, Animations won't work since they are often executed inside the JavaScript event loop. This is another important reason that all the import
code generally runs inside the JavaScript main thread.
"},{"location":"api/api-ts-library/#blocking-function-explained","title":"Blocking Function Explained","text":"twr-wasm supports blocking functions like sleep when the API user is using twrWasmModuleAsync
. This section explains the sleep
function which causes C code to block. In other words, in C, code like this will work:
printf(\"going to sleep...\");\ntwr_sleep(1000);\nprintf(\"awake!\\n\");\n
The TypeScript twrLibrary derived class implementation looks like this:
// this function will only work in twrWasmModuleAsync since it blocks the C caller.\nasync twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number) {\n const p = new Promise<void>( (resolve)=>{\n setTimeout(()=>{ resolve() }, milliSeconds); \n });\n\n return p;\n}\n
And has these import options set:
imports:TLibImports = {\n twr_sleep:{isAsyncFunction: true, isModuleAsyncOnly: true},\n};\n
isAsyncFunction: true
is telling twrLibrary to call the twr_sleep_async
function with await
and so allow the function being called to await
. isModuleAsyncOnly: true
is telling twrLibrary that this function only exists when twrWasmModuleAsync
is used.
This code will execute in the JavaScript main thread. The Calling C code (twr_sleep
) will block in a Worker thread while waiting for this code in the JavaScrit main thread to complete. The function twr_sleep_async
creates a JavaScript promise that the calling code will await
on. Once the promise resolves, the calling function will unblock the C twr_sleep
function that is in the Worker Thread.
"},{"location":"api/api-ts-library/#import-options","title":"import
options","text":"The various import
options are used to handle different cases for twrWasmModuleAsync
.
The import options are:
isAsyncFunction?:boolean;\nisModuleAsyncOnly?:boolean;\nisCommonCode?:boolean;\n
"},{"location":"api/api-ts-library/#isasyncfunction","title":"isAsyncFunction
","text":"This option is used when you wish to await
inside the implementation of your function. This option also specifies that the function will be called with await
.
This option will only modify behavior this way when your function is called from twrWasmModuleAsync
.
- If this option is not specified, the same
import
function will be used for both module types, and the function can not use the await
keyword. - if this option is specified, then when
twrWasmModuleAsync
calls the indicated import
function, it will call a version of the function that has _async
append to the function name. This means that you will create two versions of the import
funcA
- funcA
and funcA_async
. - If, however, you also specify the option
isModuleAsyncOnly
, then only the _async
function is expected.
Here is an example of how declare a function with the import
option isAsyncFunction
:
async twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number)\n
Note that:
- the function declaration starts with the async keyword
- the function has the suffix
_async
appended to its import
name - that the function is passed an
IWasmModuleAsync
as the callingMod.
By using this option, you may need to create two versions of the import
function -- one that is async
(for use by twrWasmModuleAsync
), and one that does not use the async
keyword--for use by twrWasmModule
.
In a case like twr_sleep
, there is only one function implemented for sleep - the async function. But this is not always the case. For example, if your import
function uses the wasmMem.PutXX
functions, you will need to create two functions for the import
. Here is an example:
imports:TLibImports = {\n ex_append_two_strings:{isAsyncFunction: true},\n };\n\n ex_append_two_strings(callingMod:IWasmModule, str1Idx:number, str2Idx:number) {\n const newStr=callingMod.wasmMem.getString(str1Idx)+callingMod.wasmMem.getString(str2Idx);\n const rv=callingMod.wasmMem.putString(newStr);\n return rv;\n }\n\n async ex_append_two_strings_async(callingMod:IWasmModuleAsync, str1Idx:number, str2Idx:number) {\n const newStr=callingMod.wasmMem.getString(str1Idx)+callingMod.wasmMem.getString(str2Idx);\n const rv=await callingMod.wasmMem.putString(newStr);\n return rv;\n }\n
"},{"location":"api/api-ts-library/#ismoduleasynconly","title":"isModuleAsyncOnly
","text":"This option specifies that the indicated function is only available to twrWasmModuleAsync
. This option should be used for functions that block C execution. The twr_sleep
is an example.
"},{"location":"api/api-ts-library/#iscommoncode","title":"isCommonCode
","text":"This option is used to specify a function that should be used directly by the twrWasmModuleAsync
Web Worker. Without this option, the behavior is that code running in the Web Worker will send a message to the JavaScript Main thread requesting that the function be executed in the context of the JavaScript main thread. The Web Worker will then wait for a reply to the message before continuing C execution. However, in certain cases, it is possible, and might be more performant, to have the code execute directly in the WorkerThread.
There are limitations on the code that will work correctly with isCommonCode
:
- The functions must be available to a Worker thread
- The function can not be
async
(that is, it can not use await
) - The function can not call
PostEvent
- The functions must not depend on the Worker's main event loop running (this event loop often doesn't execute with the
twrWasmModuleAsync
Worker thread.) - The function can not use a callback (the callback won't get called because callbacks are often dispatched in the JavaScript main event loop)
Here is an Example of using isCommonCode
:
export default class twrLibMath extends twrLibrary {\n imports:TLibImports = {\n twrSin:{isCommonCode: true},\n }\n\n libSourcePath = new URL(import.meta.url).pathname;\n\n twrSin(callingMod:IWasmModule|twrWasmBase, angle:number ) {return Math.sin(angle)}\n}\n
In this case the Math.sin
function is available in both a Web Worker and the JavaScript main thread. It is a simple function, that works fine without the JavaScript event loop operating.
"},{"location":"api/api-ts-library/#noblock","title":"noBlock","text":"noBlock
will cause a function call in an twrWasmModuleAsync
to send the message from the Worker proxy thread to the JS main thread to execute the function, but not to wait for the result. This should only be used for functions with a void
return value. This has the advantage that (a) the C code will not block waiting for a void return value (so it returns faster), and (b) it takes advantage of multiple cores by allowing the JS Main thread and the Worker thread to execute in parallel.
Note that the messages sent from the proxy thread to the JS main thread (for function execution) will cause execution of function calls to serialize, and so if a function that blocks (waits for results from JS main thread) is called after a call with noBlock
, everything should work as expected.
Do not use noBlock
if:
- the function returns a value
- the C code should not continue executing until the function completes execution.
- if the following scenario could arise:
- funcA (with noBlock) called
- funcB called and returns a value or otherwise depends on funcA completing execution, and funcA uses async keyword.
Use noBlock
carefully.
"},{"location":"api/api-ts-library/#libsourcepath","title":"libSourcePath","text":"Always set this as follows:
libSourcePath = new URL(import.meta.url).pathname;\n
libSourcePath
is used to uniquely identify the library class, as well as to dynamically import the library when isCommonCode
is used.
"},{"location":"api/api-ts-library/#interfacename","title":"interfaceName","text":"In a twrLibrary,
- An \"interface\" refers to the set of functions that the library exposes to C. Ie, the functions in the
import
object. - The name of the interface is anonymous, unless
interfaceName
is set. - An undefined interfaceName (anonymous interface) means that only one instance of that class is allowed (for example
twrLibMath
) - Set
interfaceName
to a unique name when multiple instances that support the same interface are allowed (for example the twr-wasm Consoles). - Multiple classes may have the same interfaceName (a class is identified by its libSourcePath). For example
twrConDiv
, twrConDebug
, twrConTerminal
all have the same interface.
When multiple instances of classes with the same interface are enabled (by setting interfaceName
), the first argument in every C function call is expected to be the twrLibrary id
(a member variable of the twrLibrary derived class). The twrLibrary will use this id
to route the function call to the correct instance of the library. The id
is not passed to the twrLibrary function (even though it is required to be the first C arg).
The twrLibrary instance should be created in the JavaScript main thread, and passed to the module in the io
option. The C code can discover the id
, by using the twr_get_console
.
exampleinterfaceName = \"twrConsole\";\n
"},{"location":"api/api-ts-library/#the-twrwasmmoduleasync-event-loop","title":"The twrWasmModuleAsync
Event Loop","text":"TODO
"},{"location":"api/api-ts-memory/","title":"Accessing Data in WebAssembly Memory","text":"There are situations where you may need to access WebAssembly memory from your TypeScript code. For example, if you need to allocate a new string, de-reference a pointer, or examine or modify a structure. A TwrLibrary
in particular may need to do this. ( For background on the issues involved in using WebAssembly Memory with C and TypeScript see Passing Function Arguments from JavaScript to C/C++ with WebAssembly.).
To access WebAssembly memory you will use the wasmMem
public member variable:
twrWasmModule
has the public member variable wasmMem:IWasmMemory
twrWasmModuleAsync
has the public member variable wasmMem:IWasmMemoryAsync
If you are writing a twrLibrary
, the appropriate wasmMem
is the first parameter of your import functions.
Both versions of wasmMem extend IWasmMemoryBase
which has common functions for retrieving or setting values from WebAssembly memory. With IWasmMemoryAsync
, for functions that call malloc
internally, await
must be used. This situation arises in the IWasmMemoryAsync
versions of the PutXXX
functions -- they return a Promise. PutXX
makes a call to malloc
, and in twrWasmModuleAsync
, malloc
needs to message the Worker thread and await
for a response.
mem8
, mem32
, memF
, and memD
can be used to access the WebAssembly memory directly. They are different views on the same memory.
getU8Arr
and getU32Arr
expect that idx
(a pointer) points to a:
struct {\n int size;\n void* data;\n}\n
Note: In prior versions of twr-wasm, these functions were available directly on the module instance. For example, mod.GetString
. These functions have been deprecated. Now you should use mod.wasmMem.getString
(for example).
// IWasmMemoryBase operate on shared memory, so they will function in any WasmModule \nexport interface IWasmMemoryBase {\n memory:WebAssembly.Memory;\n mem8:Uint8Array;\n mem32:Uint32Array;\n memF:Float32Array;\n memD:Float64Array;\n stringToU8(sin:string, codePage?:number):Uint8Array;\n copyString(buffer:number, buffer_size:number, sin:string, codePage?:number):void;\n getLong(idx:number): number;\n setLong(idx:number, value:number):void;\n getDouble(idx:number): number;\n setDouble(idx:number, value:number):void;\n getShort(idx:number): number;\n getString(strIndex:number, len?:number, codePage?:number): string;\n getU8Arr(idx:number): Uint8Array;\n getU32Arr(idx:number): Uint32Array;\n}\n\n// IWasmMemory does not support await, and so will only work in a thread that has the module loaded\n// That would be twrWasmModule, twrWasmModuleAsyncProxy\nexport interface IWasmMemory extends IWasmMemoryBase {\n malloc:(size:number)=>number;\n free:(size:number)=>void;\n putString(sin:string, codePage?:number):number;\n putU8(u8a:Uint8Array):number;\n putArrayBuffer(ab:ArrayBuffer):number;\n}\n\n// IWasmMemoryAsync must be used from an async function since await is needed\nexport interface IWasmMemoryAsync extends IWasmMemoryBase {\n malloc:(size:number)=>Promise<number>;\n free:(size:number)=>Promise<void>;\n putString(sin:string, codePage?:number):Promise<number>;\n putU8(u8a:Uint8Array):Promise<number>;\n putArrayBuffer(ab:ArrayBuffer):Promise<number>;\n}\n
"},{"location":"api/api-ts-modules/","title":"Wasm Modules","text":"This section describes the twr-wasm TypeScript/JavaScript classes twrWasmModule
and twrWasmModuleAsync
that are used to load .wasm
modules, call their C functions, and access wasm memory. Both classes have similar APIs.
"},{"location":"api/api-ts-modules/#about-twrwasmmodule","title":"About twrWasmModule
","text":"class twrWasmModule
allows you to integrate WebAssembly C/C++ code into your Web Page. You can call C/C++ functions, and read and write WebAssembly memory. Function calls are asynchronous, as is normal for a JavaScript function. That is C/C++ functions should not block - they should return quickly (just as happens in JavaScript).
The constructor accepts an optional object (type IModOpts
), which is explained further down.
import {twrWasmModule} from \"twr-wasm\";\n\nconst mod = new twrWasmModule();\n
"},{"location":"api/api-ts-modules/#about-twrwasmmoduleasync","title":"About twrWasmModuleAsync
","text":"class twrWasmModuleAsync
allows you to integrate WebAssembly C/C++ code into your Web Page that uses a CLI pattern or code that blocks. For example, with twrWasmModuleAsync
your C/C++ code can call a synchronous function for keyboard input (that blocks until the user has entered the keyboard input). Or your C/C++ code can sleep
or otherwise block. This is the pattern that is used by many standard C library functions - fread
, etc.
class twrWasmModuleAsync
creates a WorkerThread that runs in parallel to the JavaScript main thread. This Worker thread executes your C/C++ code, and proxies functionality that needs to execute in the JavaScript main thread via remote procedure calls. This allows the JavaScript main thread to await
on a blocking callC
in your JavaScript main thread.
The Async
part of the twrWasmModuleAsync
name refers to the property of twrWasmModuleAsync
that makes your synchronous C/C++ code asynchronous.
The APIs in class twrWasmModuleAsync
are identical to class twrWasmModule
, except that certain functions use the async
keyword and thus need to be called with await
. This happens whenever the function needs to cross the JavaScript main thread and the Worker thread boundary. For example: callC
or malloc
.
The constructor accepts an optional object (type IModOpts
), which is explained further down.
import {twrWasmModuleAsync} from \"twr-wasm\";\n\nconst amod = new twrWasmModuleAsync();\n
"},{"location":"api/api-ts-modules/#loadwasm","title":"loadWasm","text":"This function is available on both class twrWasmModule
and class twrWasmModuleAsync
.
Use loadWasm
to load your compiled C/C++ code (the .wasm
file).
await mod.loadWasm(\"./mycode.wasm\")\n
"},{"location":"api/api-ts-modules/#callc","title":"callC","text":"This function is available on both class twrWasmModule
and class twrWasmModuleAsync
. twrWasmModuleAsync
returns a Promise, twrWasmModule
does not.
After your .wasm
module is loaded with loadWasm
, you call functions in your C/C++ from TypeScript/JavaScript like this:
twrWasmModulelet result=mod.callC([\"function_name\", ...params])\n
twrWasmModuleAsynclet result=await mod.callC([\"function_name\", ...params])\n
If you are calling into C++, you need to use extern \"C\"
like this in your C++ function:
extern \"C\" int function_name() {}\n
Each C/C++ function that you wish to call from TypeScript/JavaScript needs to be exported in your wasm-ld
command line with an option like this:
--export=function_name\n
Or like this in your source file: __attribute__((export_name(\"function_name\")))\nvoid function_name() {\n ...\n}\n
Fo more details, see the Compiler Options.
callC
takes an array where:
callC
returns the value returned by the C function. long
, int32_t
, int
, float
or double
and the like are returned as a number
. int64_t
is returned as a bigint
, and pointers are returned as a number
. The contents of the pointer will need to be extracted using the functions listed below. More details can be found in this article: Passing Function Arguments to WebAssembly and in this example. The FFT example demonstrates passing and modifying a Float32Array
view of an ArrayBuffer
.
Although you can always use await
on a callC
, it is only strictly necessary if the module is of class twrWasmModuleAsync
.
CallC
is mapped to wasmCall.callC
. wasmCall
also exposes callCImpl
, which can be used if no argument conversion is needed.That is if the arguments are all numbers).
"},{"location":"api/api-ts-modules/#fetchandputurl","title":"fetchAndPutURL","text":"fetchAndPutURL
is available as a member function of both twrWasmModule
and twrWasmModuleAsync
.
fetchAndPutURL(fnin:URL) : Promise<[number, number]>;\n
The returned array contains the index of where the URL contents have been loaded into wasm memory (in index 0), and the length (in index 1).
In prior versions of twr-wasm, callC
could accept an argument type of URL. This is no longer supported, and instead fetchAndPutURL
should be used.
"},{"location":"api/api-ts-modules/#log","title":"log","text":"log
is available as a member function of both twrWasmModule
and twrWasmModuleAsync
.
log
is similar to the JavaScript console.log
, except that the output is sent to the stdio
console.
log(...params: string[]):void;\n
Also note that most consoles have a putStr
function.
"},{"location":"api/api-ts-modules/#twrwasmmoduleasync-details","title":"twrWasmModuleAsync Details","text":"twrWasmModuleAsync
implements all of the same functions as twrWasmModule
, plus allows blocking inputs, and blocking code generally. This is achieved by proxying all the calls through a Web Worker thread.
For example, with this C function in your Wasm module:
void mysleep() {\n twr_sleep(5000); // sleep 5 seconds\n}\n
can be called from your JavaScript main loop like this:
await amod.callC([\"mysleep\"]);\n
You must use twrWasmModuleAsync
in order to:
- call any blocking C function (meaning it takes \"a long time\") to return
- use blocking input from a div or canvas ( eg.
twr_mbgets
) - use
twr_sleep
"},{"location":"api/api-ts-modules/#linking-requirements","title":"Linking Requirements","text":"When linking your C/C++ code, twrWasmModule
and twrWasmModuleAsync
use slightly different wasm-ld
options since twrWasmModuleAsync
uses shared memory. twrWasmModule
will operate with shared memory, so technically you could just use the same share memory options with either module, but you don't need the overhead of shared memory when using twrWasmModule, and so better to not enable it.
See wasm-ld Linker Options.
"},{"location":"api/api-ts-modules/#javascript-needed-for-char-input","title":"JavaScript Needed for Char Input","text":"When a console will handle key input, you need to add a line to your JavaScript to send key events to the console. There are two options for this: You can send the key events directly to the console, or if the key events are always directed to stdio
, you cam send the key events to the module. This latter case is primarily for when you are using tag shortcuts.
To send key events to the console, you add a line like this:
yourDivOrCanvasElement.addEventListener(\"keydown\",(ev)=>{yourConsoleClassInstance.keyDown(ev)});\n
To send key events to the module's stdio
, you add a line like this:
yourDivOrCanvasElement.addEventListener(\"keydown\",(ev)=>{yourModuleClassInstance.keyDown(ev)});\n
You likely want a line like this to automatically set the focus to the div or canvas element (so the user doesn't have to click on the element to manually set focus. Key events are sent to the element with focus.):
yourDivOrCanvasElement.focus();\n
You will also need to set the tabindex
attribute in your tag like this to enable key events:
<div id=\"twr_iodiv\" tabindex=\"0\"></div>\n<canvas id=\"twr_iocanvas\" tabindex=\"0\"></canvas>\n
See this example on character input.
Note that this section describes blocking input. As an alternative, you can send events (keyboard, mouse, timer, etc) to a non-blocking C function from JavaScript using callC
. See the balls
or pong
examples.
"},{"location":"api/api-ts-modules/#sharedarraybuffers","title":"SharedArrayBuffers","text":"twrWasmModuleAsync
uses SharedArrayBuffers which require certain CORS HTTP headers to be set. Note that twrWasmModule
does not use SharedArrayBuffers. If you limit yourself to twrWasmModule
you will not need to worry about configuring the CORS http headers on your web server.
See this note on enabling CORS HTTP headers for SharedArrayBuffers.
"},{"location":"api/api-ts-modules/#module-options","title":"Module Options","text":"The twrWasmModule
and twrWasmModuleAsync
constructor both take optional options.
For example:
let amod=new twrWasmModuleAsync();\n\nlet amod=new twrWasmModuleAsync({\n stdio: new twrConsoleDebug(); // send stdio to debug console\n });\n
These are the options: twrWasmModule & twrWasmModuleAsync Options
export interface IModOpts {\n stdio?: IConsoleStream&IConsoleBase,\n d2dcanvas?: IConsoleCanvas&IConsoleBase,\n io?: {[key:string]: IConsole},\n}\n
"},{"location":"api/api-ts-modules/#stdio-option","title":"stdio
Option","text":"Set this to a Console class instance. If you leave it undefined, twrConsoleDebug
will be used (or a tag shortcut, if set)
This option is a shortcut to setting stdio
using the io
option.
"},{"location":"api/api-ts-modules/#d2dcanvas-option","title":"d2dcanvas
Option","text":"Set this to a twrConsoleCanvas
instance to configure a 2D drawing surface. If you leave it undefined, a tag shortcut will be used.
This option is a shortcut to setting std2d
using the io
option (note the different names).
"},{"location":"api/api-ts-modules/#io-option-multiple-consoles-with-names","title":"io
Option: Multiple Consoles with Names","text":"This option allows you to assign names to consoles. The C/C++ code can then retrieve a console by name.
When using the io
object to specify named consoles:
- You can use the attribute
stdio
to set stdio. - You can use the attribute
stderr
to set stderr - You can use the attribute
std2d
to set the default 2D Drawing Surfaces -- used by twr-wasm 2D APIs. - all other attribute names are available for your consoles. Use this to access consoles in C/C++ beyond (or instead of) stdio, etc.
Alternately, you can specify stdio
and std2d
directly as module attributes (outside of io
) as a shortcut (see above).
There is a twr-wasm C API to access named consoles: twr_get_console
.
This code snippet shows how to use the io
option to pass in an object containing named console attributes:
const stream1Element=document.getElementById(\"stream1\");\nconst stream2Element=document.getElementById(\"stream2\");\n\nconst debug = new twrConsoleDebug();\nconst stream1 = new twrConsoleDiv(stream1Element);\nconst stream2 = new twrConsoleDiv(stream2Element);\n\nstream1Element.addEventListener(\"keydown\",(ev)=>{stream1.keyDown(ev)});\nstream2Element.addEventListener(\"keydown\",(ev)=>{stream2.keyDown(ev)});\n\n// setting stdio and/or stderr to a debug console isn't necessary since that will be the default if stdio or stderr is not set.\n// but here to show how to set stdio and/or stderr. They can be set to any console.\nconst amod = new twrWasmModuleAsync( {io:{stdio: debug, stderr: debug, stream1: stream1, stream2: stream2}} );\nconst mod = new twrWasmModule( {io:{stdio: debug, stderr: debug, stream1: stream1, stream2: stream2}} );\n
In this case, as well as setting stdio and stderr, consoles named \"stream1\" and \"stream2\" are made available to the C/C++ code.
Using a Named Consoletwr_ioconsole_t * stream1=twr_get_console(\"stream1\");\nfprintf(stream1, \"Hello Stream One!\\n\");\n
A complete example multi-io is provided.
"},{"location":"api/api-ts-modules/#deprecated-options","title":"Deprecated Options","text":"The following options are deprecated. Instead of these, use options available to twrConsoleDiv
and twrConsoleTerminal
constructors.
deprecatedexport interface IModOpts {\n windim?:[number, number],\n forecolor?:string,\n backcolor?:string,\n fontsize?:number,\n}\n
Note:
windim
- if stdio is set to a twrConsoleTerminal
, this will set the width and height, in characters. Instead, use constructor options on twrConsoleTerminal. forecolor
and backcolor
- if stdio is set to twrConsoleDiv
or twrConsoleTerminal
, these can be set to a CSS color (like '#FFFFFF' or 'white') to change the default background and foreground colors. However, these are deprecated, and instead, use the twrConsoleDiv
or twrConsoleTerminal
constructor options. fonsize
- Changes the default fontsize if stdio is set to twrConsoleDiv
or twrConsoleTerminal
. Deprecated, instead use twrConsoleDiv
or twrConsoleTerminal
constructor options. TStdioVals
have been removed (they were a not too useful option in prior versions of twr-wasm) divLog
has been renamed log
. Or use the putStr
member function on most consoles.
"},{"location":"examples/examples-balls/","title":"Bouncing Balls - 2D Draw API Wasm Example","text":"This example uses twr-wasm's 2D Draw API and a C++ Canvas class with WebAssembly and C++ to bounce balls around your HTML page.
- View bouncing balls
- Source for balls
The bouncing balls example demonstrates
- C++
- Using the twr-wasm draw 2D APIs that match Javascript Canvas APIs.
- Using the twr-wasm canvas.cpp wrapper class.
This example does not use libc++, which results in smaller code size. For an example that uses libc++ see tests-libcxx.
"},{"location":"examples/examples-balls/#screen-grab-of-balls-example","title":"Screen Grab of Balls Example","text":""},{"location":"examples/examples-callc/","title":"callC - Calling WebAssembly Functions Example","text":"This example demonstrates how to pass and return values between TypeScript/JavaScript and C/C++ when you are using WebAssembly with twr-wasm.
This article explains the key concepts to pass arguments between JavaScript/TypeScript and Wasm C/C++.
- View callC example running live
- View callC example source
"},{"location":"examples/examples-divcon/","title":"divcon - Printf and Input Using a div Tag","text":"This simple WebAssembly C program demos inputting and printing characters with a div
tag.
- view divcon example running live
- View divcon source code
"},{"location":"examples/examples-divcon/#screen-grab-of-square-calculator","title":"Screen Grab of Square Calculator","text":""},{"location":"examples/examples-divcon/#c-code","title":"C Code","text":"divcon.c#include <stdio.h>\n#include <stdlib.h>\n#include \"twr-crt.h\"\n\nvoid stdio_div() {\n char inbuf[64];\n char *r;\n int i;\n\n printf(\"Square Calculator\\n\");\n\n while (1) {\n printf(\"Enter an integer: \");\n r=twr_mbgets(inbuf); // r is NULL if esc entered. Otherwise r == inbuf\n if (r) { \n i=atoi(inbuf);\n printf(\"%d squared is %d\\n\\n\",i,i*i);\n }\n else {\n printf(\"\\n\");\n }\n }\n}\n
"},{"location":"examples/examples-divcon/#html-code","title":"HTML Code","text":"We are using twrWasmModuleAsync
which integrates blocking C code into JavaScript. twrWasmModuleAsync
can also be used to receive key input from a <div>
or <canvas>
tag.
index.html<body>\n <div id=\"stdioDiv\" \n tabindex=\"0\" \n style=\"color: DarkGreen; background-color: LightGray; font-size: 18px;font-family: Arial, sans-serif;\" >\n Loading... <br>\n </div>\n\n <script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleDiv} from \"twr-wasm\";\n\n const con = new twrConsoleDiv(document.getElementById(\"stdioDiv\"));\n const amod = new twrWasmModuleAsync({stdio: con});\n\n // remove 'Loading...'\n document.getElementById(\"stdioDiv\").innerHTML =\"<br>\"; \n // send key events to twrConsoleDiv\n document.getElementById(\"stdioDiv\").addEventListener(\"keydown\",(ev)=>{con.keyDown(ev)});\n\n await amod.loadWasm(\"./divcon.wasm\");\n await amod.callC([\"stdio_div\"]);\n\n </script>\n</body>\n
"},{"location":"examples/examples-fft/","title":"FFT - Example of using C FFT with HTML/JavaScript","text":"This example is a demo of integrating the popular KISS FFT C library with TypeScript/JavaScript/HTML using WebAssembly. The FFT C library is compiled into a Wasm (WebAssembly) module using clang, with the help of twr-wasm. The FFT Wasm module is used by the HTML page to calculate the FFT. The FFT input and output is drawn to the web page using JavaScript canvas functions.
The FFT library exposes APIs to process data, and doesn't use stdio.
The FFT APIs use float32 arrays for complex-number input and output data, and a configuration C struct. In the example I generate the input data by adding a 1K and 5K sine waves, call the kiss FFT API to perform the FFT on the generated sine waves, and then graph the input and output data using a JavaScript Canvas.
- View example running on the web
- View example source code
"},{"location":"examples/examples-fft/#screen-grab-of-output","title":"Screen Grab of Output","text":""},{"location":"examples/examples-fft/#code","title":"Code","text":"Here is part of the code. The rest can be found on github.
index.html
<head>\n <title>Fast Fourier transform (FFT)</title>\n</head>\n<body style=\"background-color:white\">\n\n <br>\n\n <div style=\"font:24px arial\">Input Signal</div>\n <canvas id=\"c-input\" width=\"1024\" height=\"300\" style=\"background-color:lightgray\"></canvas>\n\n <br><br><br>\n\n <div style=\"font:24px arial\">FFT Output</div>\n <canvas id=\"c-output\" width=\"1024\" height=\"300\" style=\"background-color:lightgray\"></canvas>\n\n <script type=\"module\">\n import {fftDemo} from \"./fft-script.js\";\n\n fftDemo();\n\n </script>\n</body>\n
fft-script.jsimport {twrWasmModule} from \"twr-wasm\";\n\nexport async function fftDemo() {\n\n const mod=new twrWasmModule();\n\n // load the kiss_fft C code as is, unmodified\n await mod.loadWasm('kiss_fft.wasm');\n\n // kissFFTData stores and graphs the input and output data\n // in this example the fft has 1024 bins, and I am using a 48K sampling rate\n let fft=new kissFFTData(1024, 48000);\n fft.genSin(1000)\n fft.addSin(5000)\n fft.graphIn(\"c-input\");\n\n // see kiss_fft README, but in summary you: (a) alloc config, (b) compute the FFT, (c) free the config\n // kiss_fft_alloc() returns a malloced structure. Pointers are numbers (index into Wasm module memory) in JS land \n //\n //kiss_fft_cfg cfg = kiss_fft_alloc( nfft ,is_inverse_fft ,0,0 );\n let cfg:number = await mod.callC([\"kiss_fft_alloc\", fft.nfft, 0, 0, 0 ]);\n\n // The FFT input and output data are C arrays of complex numbers.\n // typedef struct {\n // kiss_fft_scalar r;\n // kiss_fft_scalar i;\n // } kiss_fft_cpx;\n //\n // /* default is float */\n // define kiss_fft_scalar float\n\n // So if the FFT data has 1024 bins, then 1024 * 2 floats (r & i) * 4 bytes per float are needed.\n // I use a JS Float32Array view on the ArrayBuffer to access the floats\n\n // When an arrayBuffer is passed in as an argument to mod.callC,\n // callC will malloc memory in the Wasm module of a size that matches the array buffer, then\n // copy the arraybuffer into the malloc'd memory prior to the function call, \n // then copy the malloc'd memory contents back into the arrayBuffer post call.\n // The malloc'd memory is free'd post call. \n\n // void kiss_fft(kiss_fft_cfg cfg,const kiss_fft_cpx *fin,kiss_fft_cpx *fout);\n await mod.callC([\"kiss_fft\", cfg, fft.inArrayBuf, fft.outArrayBuf]);\n\n fft.graphOut(\"c-output\");\n\n await mod.callC([\"free\", cfg]); // not much point to this since all the module memory is about to disappear\n}\n
"},{"location":"examples/examples-helloworld/","title":"Hello World - WebAssembly C Example","text":"This example is a very simple twr-wasm program. It uses WebAssembly and C to print \"hello, world!\" to an HTML <div>
tag.
Also see: Hello World - Step-by-Step C to Wasm.
- View helloworld example running live
- View helloworld source code
"},{"location":"examples/examples-lib/","title":"class twrLibrary Example","text":""},{"location":"examples/examples-lib/#what-it-does","title":"What It Does","text":"This example is a twr-wasm Library that implements functions that can be called by your C/C++ code. A twr-wasm Library is written in TypeScript, and derives from the class twrLibrary.
The lib example demos:
- Creating functions in TypeScript that can be called from C/C++
- Posting Events to C
- Implementing blocking as well as non-blocking functions
"},{"location":"examples/examples-lib/#running-examples-and-source","title":"Running Examples and Source:","text":" - View lib output
- Source for lib
Also see twr-wasm Libraries Documentation
"},{"location":"examples/examples-libcxx/","title":"tests-libcxx - WebAssembly libc++ Smoke Test","text":"This is a simple test of various libc++ functions using WebAssembly with twr-wasm. The C++ program links with libc++. An example makefile is provided.
- view tests-libcxx example running live
- View tests-libcxx source code
Also see this WebAssembly program that uses libc++ with twr-wasm to implement a CLI console.
- tests-user Live
- tests-user Source
"},{"location":"examples/examples-maze/","title":"Maze Generator/Solver","text":"This example is a port to Wasm of a 20 year old Win32 C Maze creator, with the help of twr-wasm 2D Draw APIs.
- View live maze here
- Source for maze
"},{"location":"examples/examples-maze/#screen-grab-of-output","title":"Screen Grab of Output","text":""},{"location":"examples/examples-maze/#overview","title":"Overview","text":"This Maze generator uses the twr-wasm \"d2d\" (Draw 2D) C APIs. These allow drawing onto an HTML canvas from C/C++. (Also see the balls C++ example).
This C code is interesting in that it is a combination of blocking and non blocking functions. The CalcMaze
function is blocking when the \"slow draw\" flag is set. It uses Sleep
in this case. For this reason, I use twrWasmModuleAsync. The solve section uses repeated calls to SolveStep
, which works well with a JavaScript main loop. I used a javascript interval timer to make repeated calls to the C SolveStep
function. If all the C code was structured this way, twrWasmModule
could have been used (instead of the Async version)
To port this code to twr-wasm I wrote a (very tiny) Win32 compatible API (in winemu.c/winemu.h). It only implements the features needed to port Maze, but it might be useful to use as a starting point for porting your Win32 code to the web.
index.html<head>\n <title>Maze</title>\n</head>\n<body style=\"background-color:powderblue\">\n <canvas id=\"twr_d2dcanvas\" width=\"600\" height=\"600\"></canvas>\n\n <script type=\"module\">\n import {mazeRunner} from \"./maze-script.js\";\n\n mazeRunner();\n </script>\n</body>\n
maze-script.jsimport {twrWasmModuleAsync} from \"twr-wasm\";\n\nexport async function mazeRunner() {\n\n const amod=new twrWasmModuleAsync();\n\n await amod.loadWasm('maze.wasm');\n\n //void CalcMaze(HWND hWnd, LONG cell_size, LONG is_black_bg, LONG isd - slow draw)\n await amod.callC([\"CalcMaze\", 0, 7, 0, 1]);\n await amod.callC([\"SolveBegin\"]);\n\n let timer = setInterval(async ()=>{\n let isdone=await amod.callC([\"SolveStep\", 0]); //SolveStep(hwnd))\n if (isdone) clearInterval(timer);\n }, 50);\n}\n
"},{"location":"examples/examples-multi-io/","title":"Multi-io -Multiple Console Example","text":""},{"location":"examples/examples-multi-io/#what-it-does","title":"What It Does","text":"This example demos six simultaneous consoles:
- Two character streaming consoles directed to a
<div>
tag - Two character terminal consoles directed to a
<canvas>
tag - Two 2D draw surface consoles directed to a
<canvas>
tag
The multi-io example also demos:
- Creating consoles in JavaScript/TypeScript
- Using consoles in C/C++ and JavaScript/TypeScript
- Mixing multiple Consoles
- Using multiple .wasm modules using the same console
- Inputting, printing, and drawing to consoles
"},{"location":"examples/examples-multi-io/#running-examples-and-source","title":"Running Examples and Source:","text":" - View multi-io
- Source for multi-io
Also see Console Introduction
"},{"location":"examples/examples-overview/","title":"WebAssembly C/C++ Examples","text":""},{"location":"examples/examples-overview/#overview","title":"Overview","text":"These C and C++ examples demonstrate how to create different types of WebAssembly (wasm) programs with the twr-wasm library.
These are good examples to use as starting points for your own Wasm projects.
These examples are a good place to learn how to configure clang and wasm-ld to compile and link C/C++ code for use with WebAssembly (wasm).
"},{"location":"examples/examples-overview/#example-quick-links","title":"Example Quick Links","text":" - Click here to view C/C++ WebAssembly twr-wasm examples running live
- Click here to view source code and make files
"},{"location":"examples/examples-overview/#hello-world","title":"Hello World","text":"Name Description Link helloworld A very simple C Wasm example to get you started helloworld"},{"location":"examples/examples-overview/#console-examples","title":"Console Examples","text":"Name Description Link divcon A simple C program demos inputting and printing characters to a div
tag divcon terminal A simple C program demos writing and inputting from a <canvas>
tagthat twr-wasm configures as a windowed \"terminal\" terminal multi-io Demo 6 simultaneous consoles: stream i/o, terminal, and 2D Drawing. multi-io"},{"location":"examples/examples-overview/#draw-2d-examples","title":"Draw 2D Examples","text":"Name Description Link balls These fun Bouncing Balls are written in C++ and demo the 2D drawingAPIs with a C++ Canvas wrapper class balls pong A simple game of Pong written in C++ to demo 2D drawing APIs with aC++ canvas wrapper class and taking user input from JS pong maze This is an old Win32 program ported to wasm and demos 2D Draw APIs maze"},{"location":"examples/examples-overview/#call-argument-examples","title":"Call Argument Examples","text":"Name Description Link callC A demo of passing and returning values between JavaScript and Wasm module callc fft A demo of calling a C library to perform an FFT that is graphed in TypeScript fft"},{"location":"examples/examples-overview/#twrlibrary-examples","title":"twrLibrary examples","text":"Name Description Link lib A demo of createing a twrLibrary (use TypeScript to create C/C++ APIs) library"},{"location":"examples/examples-overview/#unit-tests","title":"Unit Tests","text":"Name Description Link tests twr-wasm unit tests tests tests-user \"cli\" for tests using libc++ and <canvas>
tests-user tests-libcxx Smoke test for libc++. Shows how to use libc++. tests-libcxx tests-d2d Unit tests for Draw 2D canvas console tests-d2d tests-audio Unit tests for the Audio Library tests-audio"},{"location":"examples/examples-overview/#running-or-building-the-examples-locally","title":"Running or Building the examples locally","text":"Online versions of the examples can be viewed here.
You can also run the examples locally, or build them..
"},{"location":"examples/examples-overview/#copying-examples-to-start-your-own-project","title":"Copying Examples to Start your own Project","text":"All of the examples have makefiles that use a relative path for twr.a
and includes
. These paths will work fine if your code is in an examples sub-folder as a peer to the other examples. But assuming your code is in your own project folder elsewhere, you will need to determine the correct path to twr.a
and includes
for your project's makefile. Details on how to do this can be found in the following sections: Hello World walk through and the Compiler and Linker Options section.
Also see the section on Import Resolution if you installed with git clone.
"},{"location":"examples/examples-pong/","title":"Pong - 2D Game Example","text":"Similar to the balls example, this example uses twr-wasm's 2D Draw API and a C++ canvas class to implement a simple game of 2 player and single player Pong with WebAssembly.
- View Pong
- Source for Pong
The Pong example demonstrates
- C++
- Using twr-wasm draw 2D APIs that match Javascript Canvas APIs.
- Using the twr-wasm canvas.cpp class.
- A custom typescript library
- User mouse and keyboard input via events
- Using the Audio Library to play simple sounds
This example does not use libc++, which results in smaller code size. For an example that uses libc++ see tests-libcxx.
"},{"location":"examples/examples-pong/#screen-grab-of-pong-example","title":"Screen Grab of Pong Example","text":""},{"location":"examples/examples-terminal/","title":"Terminal Console Demo","text":"A simple WebAssembly C \"terminal\" is demoed with input and output directed to an HTML <canvas>
tag.
"},{"location":"examples/examples-terminal/#what-it-does","title":"What it Does","text":" - uses class twrConsoleTerminal, a Console
- moves a string up or down in the terminal window when you press the u or d or arrow keys.
- shows basic color usage
- draws a graphic box around the terminal window.
"},{"location":"examples/examples-terminal/#run-and-view-the-code","title":"Run and View the Code","text":" - View terminal example running live
- View terminal source code
- For another 'terminal' demo View tests-user
"},{"location":"examples/examples-terminal/#screen-grab-of-terminal","title":"Screen Grab of Terminal","text":""},{"location":"examples/examples-tests-audio/","title":"tests-audio - Unit tests for Audio Library","text":"This is a simple set of Unit Tests for testing the Audio API functions. It includes a test for each Audio function. WARNING: Some of the audio played can be loud, so turn down your volume before playing.
- view tests-audio example running live
- View tests-audio source code
Also see these WebAssembly programs that use this API
"},{"location":"examples/examples-tests-d2d/","title":"tests-d2d - Unit tests for Draw 2D canvas console","text":"This is a simple set of Unit Tests for testing the D2D API functions. It includes a test for each D2D function as well as a test for memory leaks within the API itself.
- view tests-d2d example running live
- View tests-d2d source code
Also see these WebAssembly programs that use this API
"},{"location":"gettingstarted/basicsteps/","title":"Basic Steps To Create Your Wasm Project","text":"This section describes the basic steps to integrate your TypeScript/JavaScript with C/C++ WebAssembly code.
"},{"location":"gettingstarted/basicsteps/#overview-of-webassembly-project","title":"Overview of WebAssembly Project","text":"Your C/C++ WebAssembly project consists of HTML (and related JavaScript or TypeScript) and C/C++ source. The C/C++ is compiled using clang
into a .wasm
binary module. The .wasm
module is loaded as a WebAssembly module by your JavaScript using twr-wasm
.
"},{"location":"gettingstarted/basicsteps/#javascripttypescript-part-of-wasm-project","title":"JavaScript/TypeScript Part of Wasm Project","text":"On the JavaScript side of your WebAssembly project you will use the twr-wasm JavaScript/TypeScript class twrWasmModule
or twrWasmModuleAsync
to load the .wasm
module, and then call C functions in it using callC
(more details are in the TypeScript/Javascript API section).
"},{"location":"gettingstarted/basicsteps/#cc-part-of-wasm-project","title":"C/C++ Part of Wasm Project","text":"You will call C functions (or C++ with ' extern \"C\" ' linkage) in the .wasm
module from your JavaScript. You can also call JavaScript functions from your C/C++ code, but this is less common.
There is no direct equivalent to a C \"main\". Instead, a Wasm module provides exported C functions that you can call from JavaScript/TypeScript. A Wasm module is more like a runtime loaded dynamic library.
twr-wasm
supports C/C++ code that is either asynchronous (non-blocking) or syncronous (blocking). A CLI app is an example of typical blocking code. A CLI app typically blocks waiting for keyboard input (blocking means that it \"takes a long time\" to return). If your C code is a big loop that never returns, that would be very blocking. Alternately,if you send mouse events to C code, have the code process them then return, this would be non-blocking. You can use the twr-wasm class twrWasmModuleAsync
to execute blocking code from JavaScript or twrWasmModule
to integrate asynchronous C/C++ code. The example maze demonstrates both non-blocking and blocking C calls.
See the examples of different types of C/C++ apps.
"},{"location":"gettingstarted/basicsteps/#steps-to-integrate-c-code-with-javascript-code","title":"Steps to integrate C code with JavaScript code","text":"Here are the general steps to integrate your C with your JavaScript:
- Compile your C code with
clang
and link with wasm-ld
to create the .wasm
file. - On the JavaScript side you:
- Access
twr-wasm
\"ES\" modules in the normal way with import
. - Add a
<div id=twr_iodiv>
or <canvas id=twr_iocanvas>
to your HTML (see stdio) - Use
new twrWasmModule
, followed by a call to loadWasm
, then one or more callC
. - Alternately, use
twrWasmModuleAsync
-- which is interchangeable with twrWasmModule
, but proxies through a worker thread, which allows you to call blocking functions from the asynchronous JavaScript main thread. - For more details, see the remainder of this documentation, or see the hello world or other exampes.
"},{"location":"gettingstarted/charencoding/","title":"Character Encoding Support with twr-wasm","text":"This section explains twr-wasm's WebAssembly support for ASCII, UTF-8, windows-1252, and UTF-32 character encoding.
"},{"location":"gettingstarted/charencoding/#getting-started","title":"Getting Started","text":"When using C with twr-wasm, you will likely want to add this line to the start of your code:
setlocale(LC_ALL, \"\")\n
This will change the C locale language to the one selected in the browser, and will enable consistent UTF-8 character encoding support.
Without this line, the standard C runtime will default character encoding to ASCII, as per the std c lib standard. However, just as with gcc, twr-wasm consoles support outputting UTF-8, even in the default setting.
"},{"location":"gettingstarted/charencoding/#character-encodings","title":"Character Encodings","text":"twr-wasm supports ASCII, UNICODE, and extended-ASCII (in the form of Windows-1252).
"},{"location":"gettingstarted/charencoding/#utf-8","title":"UTF-8","text":"These days UNICODE with UTF-8 encoding is the most popular method of displaying and encoding text. UTF-8 is popular because it has the deep character glyph definitions of UNICODE with an encoding that provides (a) the best backwards compatibility with ASCII, and (b) a compact memory footprint. It does this at the expense of some multibyte complexities.
UTF-8 is variable length, and uses between one to four bytes to represent any unicode code point, with ASCII compatibility in the first 128 characters. It is also the standard for the web, and the default for clang. But because UTF-8 uses a variable number of bytes per character it can make string manipulation in C a bit harder than ASCII, Windows-1252 or UTF-32.
"},{"location":"gettingstarted/charencoding/#locale","title":"Locale","text":"In this document you will see the term \"locale\". This term originated (at least as its commonly used in programming) in the standard C library, and is also used in the standard C++ library (libc++ in twr-wasm). A locale refers to a region of the world, along with a specific character encoding. The twr-wasm standard c runtime uses a label akin to this to define a locale: en-US.UTF-8
. Of note is that libc++ and the standard C runtime have different domains for their locales (ie, they don't directly impact each other). You can learn more about locales by searching the internet.
twr-wasm C locales support ASCII, UTF-8 or windows-1252 character encoding.
"},{"location":"gettingstarted/charencoding/#utf-32","title":"UTF-32","text":"UTF-16/32 are not supported as a std c lib locale setting, but functions are provided to convert utf-32 (unicode code points) to and from ASCII, UTF-8, and windows-1252 \"code pages\" (there are other miscellaneous utf-32 based functions as well.)
You can also use libc++, which has classes that directly support utf-16 and utf-32.
"},{"location":"gettingstarted/charencoding/#windows-compatibility-with-windows-1252","title":"Windows Compatibility with Windows-1252","text":"Windows-1252 is the default character encoding on Windows computers in many countries - particularly the Americas and western Europe -- and particularly when using MSVC. Linux, clang, gcc, and the web commonly default in some way to UTF-8 character encoding. Windows-1252 is an extension of ASCII that uses a single byte per character. This makes it easier than UTF-8 from a programmers perspective, but it doesn't represent as many characters. It is supported by twr-wasm to make it easier to port legacy C code, windows code, as well as a simpler alternative to UTF-8.
twr-wasm supports Windows-1252, and you can enable it like this:
setlocale(LC_ALL, \".1252\")\n
This will set the locale to the default browser language, and character encoding to Windows-1252.
1252 String Literals These days text editors generally default to UTF-8. In order to use windows-1252 source code and/or string literals, such as const char * str=\"\u20ac100\"
you may need to:
- Configure your text editor to save in Windows-1252/ISO-8859-1 format (instead of UTF-8)
- use compiler flags like
--finput-charset
and -fexec-charset
By default, the Microsoft Visual Studio C compiler (MSVC) does not treat string literals as UTF-8. Instead, it treats them as being encoded in the current code page of the system, which is typically Windows-1252 on western european language Windows systems. twr-wasm is designed to work with clang, which does default to utf-8, so if you are compiling code written for MSVC, and you use extend character sets (non ASCII), you may need to adjust your compiler settings with the flags mentioned above.
"},{"location":"gettingstarted/charencoding/#more","title":"More","text":"For more details see Localization Reference for twr-wasm
"},{"location":"gettingstarted/compiler-opts/","title":"Compiling, Linking, and Memory Options","text":"This section describes how to use clang
to compile C/C++ code for WebAssembly, and how to use wasm-ld
to link your files into a .wasm module, when using twr-wasm.
twr-wasm lets you use clang directly, without a wrapper. This section describes the needed clang compile options and the wasm-ld link options. You can also take a look at the example makefiles.
"},{"location":"gettingstarted/compiler-opts/#compiler-notes","title":"Compiler Notes","text":"twr-wasm has been tested with clang 17.0.6 and wasm-ld 17.0.6.
If you are using nix, the default clang packages are wrapped with flags that break compilation. The following packages don't have this issue:
- llvmPackages_18.clang-unwrapped (clang 18.1.7)
- llvmPackages_17.clang-unwrapped (clang 17.0.6)
"},{"location":"gettingstarted/compiler-opts/#c-clang-compiler-options","title":"C clang Compiler Options","text":"When compiling C code with clang for use with Wasm and twr-wasm, use these clang options:
--target=wasm32 -nostdinc -nostdlib -isystem ../../include\n
Here is an example of a compile command:
clang --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include -c helloworld.c -o helloworld.o\n
-isystem
should be adjusted to point to where the folder twr-wasm/include
is installed. For example:
../../include
is a relative link to include
that works if your project is a sub folder in the examples
folder. ./node_modules/twr-wasm/include
assumes you installed with npm
into your project folder. (see the Hello World Walk Through).
"},{"location":"gettingstarted/compiler-opts/#c-clang-compiler-options_1","title":"C++ clang Compiler Options","text":"When compiling C++ code with clang for use with Wasm and twr-wasm, use these clang options:
--target=wasm32 -fno-exceptions -nostdlibinc -nostdinc -nostdlib -isystem ../../include\n
"},{"location":"gettingstarted/compiler-opts/#wasm-ld-linker-options","title":"wasm-ld Linker Options","text":"Use the wasm-ld linker directly with twr-wasm.
For example:
wasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm --no-entry --initial-memory=131072 --max-memory=131072 --export=hello \n
For C and C++ link to twr.a
to link to the twr-wasm library.
For C++ link to libc++.a
if you are using libc++. (see the tests-libcxx example makefile).
Be sure to adjust the path to twr.a
and libc++.a
as needed to the location where twr-wasm/lib-c/
is installed.
All of the twr-wasm functions are staticly linked from the library lib-c/twr.a
. There is also a version ( lib-c/twrd.a
) of twr-wasm library available with debug symbols. One of these two static libraries should be added to the list of files to link (normally this is twr.a
). Both versions are built with asserts enabled. twr.a
is built with -O3
. twrd.a
is built with -g -O0
.
C functions that you wish to call from JavaScript should either have an -export
option passed to wasm-ld
, or you can use the __attribute__((export_name(\"function_name\")))
option in your C function definition.
All exported functions to JavaScript should be C linkage (extern \"C\"
if using C++).
wasm-ld should be passed the following options:
If Using twrWasmModule:
--no-entry --initial-memory=<size> --max-memory=<size>\n
If Using twrWasmModuleAsync:
--no-entry --shared-memory --no-check-features --initial-memory=<size> --max-memory=<size>\n
"},{"location":"gettingstarted/compiler-opts/#memory-options-memory-size-stack-size-etc","title":"Memory Options (Memory Size, Stack Size, etc)","text":"WebAssembly.Memory
contains all the data used by your code (including the data needs of staticly linked libraries such as twr-wasm or libc++), but it does not store your actual code. It provides a contiguous, mutable array of raw bytes. Code execution and storage in WebAssembly are handled separately using the WebAssembly.Module
and WebAssembly.Instance
objects. The code (compiled WebAssembly instructions) is stored in the WebAssembly.Module
, while WebAssembly.Memory
is used to manage the linear memory accessible to the WebAssembly instance for storing data. Examples of data include your static data (.bss section or the .data section), the heap (used by malloc
and free
), and the stack (used for function calls and local variables).
The memory size should be a multiple of 64*1024 (64K) chunks. \"initial-memory\" and \"max-memory\" should be set to the same number since there is no support for automatically growing memory in twr-wasm. The memory is an export out of the .wasm
into the JavaScript code -- you should not create or set the size of WebAssembly.Memory
in JavaScript when using twr-wasm.
You set the memory size for your module (WebAssembly.Memory
) using wasm-ld
options as follows (this examples sets your Wasm memory to 1MB).
"},{"location":"gettingstarted/compiler-opts/#twrwasmmodule","title":"twrWasmModule","text":"if using twrWasmModule
:
--initial-memory=1048576 --max-memory=1048576\n
"},{"location":"gettingstarted/compiler-opts/#twrwasmmoduleasync","title":"twrWasmModuleAsync","text":"If you are using twrWasmModuleAsync
, shared memory must also be enabled. Like this:
--shared-memory --no-check-features --initial-memory=1048576 --max-memory=1048576\n
See this note on CORS headers with shared memory.
"},{"location":"gettingstarted/compiler-opts/#stack-size","title":"Stack Size","text":"You can change your C/C++ stack size from the default 64K with the following wasm-ld
option. This example sets the stack at 128K
-z stack-size=131072\n
"},{"location":"gettingstarted/compiler-opts/#print-memory-map","title":"Print Memory Map","text":"You can print your module memory map, heap stats, and stack size using the function from C:
void twr_mem_debug_stats(twr_ioconsole_t* outcon);\n
You can call it from Javascript with the output sent to the debug console (stderr) like this: twrWasmModule/Async.callC([\"twr_wasm_print_mem_debug_stats\"])\n
"},{"location":"gettingstarted/compiler-opts/#typescriptjavascript-malloc-and-memory-access","title":"TypeScript/JavaScript malloc and Memory Access","text":"twrWasmModule
and twrWasmModuleAsync
expose malloc
as an async function, as well as the WebAssembly Module memory as:
async malloc(size:number);\n\nmemory?:WebAssembly.Memory;\nmem8:Uint8Array;\nmem32:Uint32Array;\nmemD:Float64Array;\n
to call free
from JavaScript (you probably won't need to), you can use: twrWasmModule/Async.callC([\"twr_free\", index]); // index to memory to free, as returned by malloc\n
more information on these functions and module public variables can be found in the examples in this section: Passing Function Arguments to WebAssembly.
"},{"location":"gettingstarted/debugging/","title":"Debugging WebAssembly","text":"This section describes tips for debugging your WebAssembly (Wasm) program. Some of these techniques are WebAssembly generic, some are specific to using twr-wasm.
"},{"location":"gettingstarted/debugging/#debug-and-release-libraries","title":"Debug and Release libraries","text":"There are release (twr.a) and debug (twrd.a) versions of the twr-wasm C library. The \"debug\" version has debug symbols enabled with -g
and is built with optimizations disabled via -O0
. The \"release\" version has no debug symbols and optimization is set to -O3
. Both have asserts enabled. In general, you should use the \"release\" version unless you wish to step through the twr-wasm source -- in which case use the \"debug\" version.
libc++.a is not built with debug symbols.
"},{"location":"gettingstarted/debugging/#source-level-debugging-webassembly-cc","title":"Source Level Debugging WebAssembly C/C++","text":"In order to enable C/C++ source debugging with Wasm and clang, do the following:
- Use Chrome
- Install the Chrome extension: C/C++ DevTools Support (DWARF)
- Use the clang compile flag -g to add debug annotation to your object files
- You may want to turn off optimization to allow the debugger to have a bit more logical behavior (remove the
-O
flag or set to -O0
) - You may want to use the version of the twr-wasm C library that has debug symbols enabled (twrd.a). Only if you want to step into the twrd.a source.
- You need to serve your files with a (likely local) web server.
- For example, 'python server.py' is provided. 'server.py' can be found in the examples root folder. Note that your local server needs to enable SharedArrayBuffers if you are using
twrWasmModuleAsync
-- see these CORS notes. - your code can be bundled or unbundled, but
- you need to ensure that the web server/browser can find the source code
- also see Example Readme
"},{"location":"gettingstarted/debugging/#resolving-imports","title":"Resolving Imports","text":"If you are having issues with import resolution, see this section.
"},{"location":"gettingstarted/debugging/#useful-twr-wasm-debug-functions","title":"Useful twr-wasm Debug Functions","text":"Use twr_conlog
to print to the JavaScript console from C (see API ref section).
#include \"twr-crt.h\"\n\ntwr_conlog(\"hello 99 in hex: %x\",99);\n
Inside JavaScript, you can print to a console using the putStr
console member function that is available on all consoles.
For example:
const stream1 = new twrConsoleDiv(stream1Element);\nstream1.putStr(`Hello stream1 of type ${stream1.getProp(\"type\")} from JavaScript!\\n`);\n
"},{"location":"gettingstarted/debugging/#testing-webassembly-without-a-web-server","title":"Testing WebAssembly Without a Web Server","text":"Note: If you use this technique, you will not be able to get the C/C++ DevTool chrome extension to run, and so source level debugging won't work. (If you know how to fix this, please contact me on github.)
You can execute and debug JavaScript with Wasm from local files without an HTTP server. It might be helpful to download the twr-wasm source code from github when you do this (so you can step through the twr-wasm typescript code as needed).
See the examples and Example Readme for more detail on how this works.
In general, you will need to add a clip of code similar to this to your HTML:
<script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"./../../lib-js/index.js\"\n }\n }\n</script>\n
Make sure the paths to twr-wasm/lib-js/index.js
are correct for where your source is located. The above is correct for the provided examples.
You will need to set the following flags when running chrome from the shell or VS Code (the first is only strictly required if using twrWasmModuleAsync).
--enable-features=SharedArrayBuffer\n--allow-file-access-from-files\n
If you are using VS Code, You can create a launch.json entry similar to this:
launch.json{\n \"configurations\": [\n {\n \"name\": \"Launch Chrome\",\n \"request\": \"launch\",\n \"type\": \"chrome\",\n \"runtimeArgs\": [\n \"--allow-file-access-from-files\",\n \"--autoplay-policy=no-user-gesture-required\",\n \"--enable-features=SharedArrayBuffer\"\n ],\n \"file\": \"${workspaceFolder}/index.html\",\n \"cwd\": \"${workspaceFolder}/\",\n }\n ]\n}\n
"},{"location":"gettingstarted/events/","title":"Overview of Events","text":"This section describes how to use twr-wasm to:
- register event callbacks in C/C++
- use events in C/C++
"},{"location":"gettingstarted/events/#quick-example","title":"Quick Example","text":"timer events#include <stdio.h>\n#include \"twr-crt.h\"\n\nint t2_count=0;\nint t2_id;\n\n// timer2 event callback (called multiple times)\n__attribute__((export_name(\"on_timer2\")))\nvoid on_timer2(int event_id) {\n t2_count++;\n printf(\"timer callback 2 entered (event id=%d, count=%d)\\n\", event_id, t2_count);\n\n if (t2_count==5) {\n twr_timer_cancel(t2_id);\n printf(\"timer example complete\\n\")\n }\n}\n\n// C entry point to call from JavaScript\nint timer_main() {\n printf(\"the timer will trigger 5 times...\\n\");\n\n int t2_eventid=twr_register_callback(\"on_timer2\");\n t2_id=twr_timer_repeat(500, t2_eventid);\n}\n
"},{"location":"gettingstarted/events/#examples","title":"Examples","text":"Name View Live Link Source Link timer example View timer test Source library example View library example Source"},{"location":"gettingstarted/events/#events","title":"Events","text":"In twr-wasm, certain APIs can trigger events. For example a timer can trigger a \"timer complete\" event, or an audio api my trigger a \"file has finished playing\" event. An event had an id
and an associated callback. The id
is an integer that identifies the event (int event_id
). In order to receive an event call back:
- Write your callback in C/C++. It must be C linkage, and you should be exported from your C code to JavaScript/TypeScript using the
export_name
clang __attribute__
like this: __attribute__((export_name(\"on_timer2\")))
. Replace on_timer2
with your callback function name. - Register your callback. This will also allocate the event ID paired to this callback. For example:
int t2_event_id=twr_register_callback(\"on_timer2\");
- Call an API that takes an event
id
. For example: twr_timer_repeat(500, t2_eventid);
. This will call the t2_event
callback every 500ms.
You can use the same event/callback with multiple APIs if you wish. When the event callback is called, the first argument will be the event id
triggering the callback. There may then be optional parameters. These are event specific.
As in JavaScript, twr-wasm event callbacks only occur when your C/C++ code is not running.
"},{"location":"gettingstarted/events/#when-using-twrwasmmoduleasync","title":"When using twrWasmModuleAsync","text":"With a twrWasmModuleAsync
module, various blocking APIs are available. For example: twr_sleep
. When these functions are blocking (waiting), event callbacks are queued and not processed until your C functions return back to JavaScript.
With a twrWasmModuleAsync
module, events are sent from the JavaScript main thread to the worker thread that is running the C/C++ code.
"},{"location":"gettingstarted/events/#twr_register_callback","title":"twr_register_callback","text":"See twr_register_callback
"},{"location":"gettingstarted/helloworld/","title":"Create and Run WebAssembly Hello World","text":"This section shows you, step by step, how to to create a C \"hello world\" program for WebAssembly (Wasm) with twr-wasm, C, HTML, and JavaScript.
You will learn how to:
- Create the helloworld.c file
- Create the index.html file
- Compile the helloworld.c code with
clang
- Link the helloworld.o and twr.a files with
wasm-ld
to create a helloworld.wasm file - Set the needed library and include paths to allow the twr-wasm libraries to be discovered
- Create an optional Makefile
- Execute the \"hello world\" program using a local web server or directly with VS Code and Chrome
You can find code for a hello world example in the folder examples\\helloworld. It is similar, but not identical to this walk through. The primary differences are the paths for lib-c, lib-js, and include.
"},{"location":"gettingstarted/helloworld/#step-0-installation","title":"Step 0: Installation","text":""},{"location":"gettingstarted/helloworld/#step-1-create-the-c-code","title":"Step 1: Create the C code","text":"Create a file helloworld.c
in hello-proj
helloworld.c
#include <stdio.h>\n\nvoid hello() {\n printf(\"hello world\\n\");\n}\n
"},{"location":"gettingstarted/helloworld/#step-2-create-the-html","title":"Step 2: Create the HTML","text":"Create a file index.html
in hello-proj
index.html
<!doctype html>\n<html>\n<head>\n <title>Hello World</title>\n\n <script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"./node_modules/twr-wasm/lib-js/index.js\"\n }\n }\n </script>\n\n</head>\n<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n </script>\n</body>\n</html>\n
This example uses Import Maps, which are used when not using a bundler like WebPack or Parcel. For smaller projects, this can be simpler with a more clear debugging and development environment. This is the approach we will use for this example (no bundler).
The path in the importmap
section of index.html
should point to the location where you installed twr-wasm/lib-js
. The path above is correct for this project example with the indicated folder structure.
For more detail on import resolution see this section.
"},{"location":"gettingstarted/helloworld/#step-3-compile-your-c-code-to-create-your-wasm-file","title":"Step 3: Compile your C code to create your .wasm file","text":"cd hello-proj\nclang --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include -c helloworld.c -o helloworld.o\nwasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm --no-entry --initial-memory=131072 --max-memory=131072 --export=hello \n
The path to twr.a
and to include
should match your installation. The above path is correct for this example.
As an alternate to executing clang and wasm-ld from the shell, here is a Makefile that will work for this example:
MakefileCC := clang\nTWRCFLAGS := --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include\nCFLAGS := -c -Wall -O3 $(TWRCFLAGS)\nCFLAGS_DEBUG := -c -Wall -g -O0 $(TWRCFLAGS)\n\n.PHONY: default\n\ndefault: helloworld.wasm\n\nhelloworld.o: helloworld.c\n $(CC) $(CFLAGS) $< -o $@\n\nhelloworld.wasm: helloworld.o \n wasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm \\\n --no-entry --initial-memory=131072 --max-memory=131072 \\\n --export=hello \n
Copy the above into a file named Makefile
and execute with make
(or mingw32-make
in windows).
"},{"location":"gettingstarted/helloworld/#step-4-load-and-execute-your-web-page","title":"Step 4: Load and execute your web page","text":"The two easiest ways to load and execute your index.html
web page locally are:
"},{"location":"gettingstarted/helloworld/#option-a-run-a-local-web-server","title":"Option A: Run a local web Server","text":"You can run a local server to view your helloworld program.
- Copy the file server.py from the examples folder to your
hello-proj
folder (where your index.html
resides). - Execute with the shell command
python server.py
. - Open your web browser and browse to
http://localhost:8000/index.html
- You should see 'Hello World' in the browser window!
At this pont your folder structure should look like this:
hello-proj\\\n\u2514\u2500\u2500node_modules\\\n\u2514\u2500\u2500helloworld.c\n\u2514\u2500\u2500helloworld.o\n\u2514\u2500\u2500helloworld.wasm\n\u2514\u2500\u2500index.html\n\u2514\u2500\u2500Makefile\n\u2514\u2500\u2500package.json\n\u2514\u2500\u2500server.py\n
"},{"location":"gettingstarted/helloworld/#option-b-vs-code-launchjson","title":"Option B: VS Code launch.json","text":"Alternately, you can launch chrome without a local web server. Add an entry similar to the following to hello-proj\\.vscode\\launch.json
. This assumes your workspaceFolder is hello-proj
.
launch.json{\n \"configurations\": [\n {\n \"name\": \"Launch Chrome Hello, World!\",\n \"request\": \"launch\",\n \"type\": \"chrome\",\n \"runtimeArgs\": [\n \"--allow-file-access-from-files\",\n \"--autoplay-policy=no-user-gesture-required\",\n \"--enable-features=SharedArrayBuffer\"\n ],\n \"file\": \"${workspaceFolder}/index.html\",\n \"cwd\": \"${workspaceFolder}/\",\n }\n ]\n}\n
Once you have created this file, you:
- select the Run and Debug icon on left
- Select the green play icon at the top, with \"Launch Chrome Hello, World!\" selected
- Chrome should launch, and you should see 'Hello World' in the browser window!
--autoplay-policy=no-user-gesture-required
and --enable-features=SharedArrayBuffer
are not required for this simple \"hello world\" example, but will be needed if you request user input or you are using twrWasModuleAsync
.
"},{"location":"gettingstarted/helloworld/#see-live-version","title":"See live version","text":"You can find a live link to hello world on this page.
"},{"location":"gettingstarted/helloworld/#next-steps-after-hello-world","title":"Next steps after hello world","text":"A good way to get your own code up and running is to copy one of the examples, get it to build and run, then start modifying it. Note you will need to modify the paths for include
, lib-js
, lib-c
, etc. based on your project structure. The examples are all setup with relative paths assuming the folder structure twr-wasm\\examples\\<example>
The examples include Makefiles.
\"Hello World\" uses the twr-wasm class twrWasmModule
. If you wish to use C blocking functions, such as twr_getc32
or twr_sleep
, you should use twrWasmModuleAsync
. This square calculator example shows how to do this.
If you wish to build an app that makes non-block calls into C, the balls example shows how to do this. The maze example uses a combination of blocking and non-blocking C functions.
"},{"location":"gettingstarted/helloworld/#debugging","title":"Debugging","text":"See the debugging section for debugging tips, including setting up Wasm source level debugging.
"},{"location":"gettingstarted/installation/","title":"Installing twr-wasm","text":"A simple way to install twr-wasm is:
npm install twr-wasm\n
See the \"Hello World walk through\" in the following section for more specifics.
There are actually two methods of installation with different pros and cons:
npm install
will install everything necessary to build your software: built libraries (lib-js, lib-c) and includes. In addition the examples are installed. git clone
will copy the above as well as the source and VS Code settings.
When using twr-wasm
your applications needs to access both JavaScript and C twr-wasm libraries. This is explained in the installation sections below, as well as in the Hello World walk through.
"},{"location":"gettingstarted/installation/#npm-install","title":"npm install","text":"npm install twr-wasm\n
After installation from npm, you will have a folder structure like this: node_modules\\\n twr-wasm\\\n examples\\\n include\\\n lib-c\\\n lib-js\\\n LICENSE\n package.json\n readme.md\n
The JavaScript and TypeScript exports are in lib-js
and should be found by VS Code, TypeScript or your bundler as usual when using a statement like import {twrWasmModule} from \"twr-wasm\"
. The C library (twr.a
) that you will need to link your C/C++ program to is found in the libs-c
folder, and the C/C++ include files that you will need to use in your C/C++ program are found in the include
folder. You will need to use paths to to these folders in your makefile. See the Hello World walk through for details.
There is no real downside to this installation method, except possibly: (1) it does not include source code (use git clone for that), and (b) the C libraries are buried inside your node_modules.
"},{"location":"gettingstarted/installation/#git-install","title":"git install","text":" git clone https://github.com/twiddlingbits/twr-wasm\n
This method of installation installs the complete code base, including source and built binaries.
The primary downside to this method is that the JavaScript side of twr-wasm will not be placed in a node_modules folder. This will create a little extra work to configure a bundler, TypeScript or VS Code to find the location of imports.
There are a few solutions to this. For example, in the provided Hello World example, a package.json
file with an alias
entry is used. This syntax is supported by the Parcel bundler:
{\n \"@parcel/resolver-default\": {\n \"packageExports\": true\n },\n \"alias\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n },\n \"dependencies\": {\n \"twr-wasm\": \"^2.0.0\"\n }\n}\n
The FFT example uses the paths
entry in the tsconfig.json
file. This is found by TypeScript, VS Code and the Parcel bundler. This is probably the best solution if you are using TypeScript.
\"paths\": {\n \"twr-wasm\": [\"./../../lib-js/index\"]\n}\n
The paths for alias
and paths
shown above are correct for the included examples, but will likely need to be adjust for your project.
For more details see this section on Import Resolution
"},{"location":"gettingstarted/installation/#clang-and-wasm-ld","title":"clang and wasm-ld","text":"To build C/C++ code for use in your Wasm project, you will need to install clang and the wasm-ld linker. If you are using Windows, more details can be found at the end of the Building Source section.
"},{"location":"gettingstarted/installation/#python-and-more","title":"python and more","text":"To use the included examples\\server.py
to execute your project you will need to install python. server.py
is a simple HTTP server for local testing that sets the correct CORS headers for twrWasmModuleAsync
. As explained in the Hello World walk through, you can alternately execute HTML files directly using VS Code and Chrome.
You will likely want these tools installed to use twr-wasm:
- gnu make (all the examples use make)
- VS Code (not required, but the repo includes VS Code launch.json, etc)
- TypeScript (not required, but the twr-wasm source code is TypeScript)
"},{"location":"gettingstarted/installation/#note-on-examples","title":"Note on Examples","text":"You can run the examples either locally or with the online versions.
"},{"location":"gettingstarted/parameters/","title":"Passing Function Arguments to WebAssembly","text":"This article describes techniques to transfer data between JavaScript/TypeScript and C/C++ when using WebAssembly. It delves a bit \u201cunder the covers\u201d to explain how this works when you use a library like twr-wasm or Emscripten. In this article, I am using twr-wasm for the examples. Emscripten does something similar.
For an example that illustrates the concepts discussed here, see: the callC example.
"},{"location":"gettingstarted/parameters/#webassembly-virtual-machine-intrinsic-capabilities","title":"WebAssembly Virtual Machine Intrinsic Capabilities","text":"The WebAssembly VM (often referred to as a Wasm \u201cRuntime\u201d) is limited to passing numbers between C functions and the Wasm host (I\u2019ll assume that\u2019s JavaScript for this document). In other words, if you are using the most basic WebAssembly capabilities provided by JavaScript, such as WebAssembly.Module
, WebAssembly.Instance
, and instance.exports
, your function calls and return types can only be:
- Integer 32 or 64 bit
- Floating point 32 or 64 bit
These correspond to the WebAssembly spec support for: i32, i64, f32, and f64.
Note that a JavaScript number
is of type Float 64 (known as a double
in C/C++.). If you are storing an integer into a JavaScript number
, it is converted to a Float 64, and its maximum \"integer\" precision is significantly less than 64 bits (its about 52 bits, but this is a simplification). As a result, to use a 64-bit integers with JavaScript the bigint
type is used.
When using 32-bit WebAssembly (by far the most common default), and you call a C function from JavaScript without using any \u201chelper\u201d libraries (like twr-wasm), the following argument types can be passed:
- Integer 32: JavaScript
number
type is converted to an Integer 32 and passed to C when the C function prototype specifies a signed or unsigned int
, long
, int32_t
, or a pointer type. All of these are 32 bits in length in wasm32. - Integer 64: JavaScript
bigint
type is converted to an Integer 64 and passed to C when the C function prototype specifies signed or unsigned int64_t
(or equivalent). Attempting to pass a JavaScript number
to a C int64_t
will fail with a JavaScript runtime error. - Float 32: JavaScript
number
type is converted to a Float 32 when the C function prototype specifies a float
. - Float 64: JavaScript
number
type is passed as a Float 64 when the C function prototype specifies a double
.
The same rules apply to the return types.
"},{"location":"gettingstarted/parameters/#c-structs-javascript-c","title":"C Structs: JavaScript <--> C","text":"This section shows how to create a C struct
in JavaScript, then pass it to a C function, and then read the modified C struct
in JavaScript.
Although the techniques described here are explained with a struct
example, the basic techniques are used with other data types as well (such as strings). For common data types, like a string, libraries like twr-wasm will handle these details for you automatically.
To create and pass a C struct
from JavaScript to C, the technique is to call the WebAssembly C malloc
from JavaScript to allocate WebAssembly memory and then manipulating the memory in JavaScript. One complexity is that each struct entry\u2019s memory address needs to be calculated. And when calculating the WebAssembly Memory indices for the struct entries, C structure padding must be accounted for.
"},{"location":"gettingstarted/parameters/#struct-entry-padding","title":"struct Entry Padding","text":"Before we delve into the actual code, lets review C struct entry padding.
In clang, if you declare this structure in your C code:
struct test_struct {\n int a;\n char b;\n int *c;\n};\n
- The first entry,
int a
, will be at offset 0 in memory (from the start of the struct
in memory). - The second entry,
char b
, will be at offset 4 in memory. This is expected since the length of an int is 4 bytes. - The third entry,
int *c
, will be at offset 8 in memory, not at offset 5 as you might expect. The compiler adds three bytes of padding to align the pointer to a 4-byte boundary.
This behavior is dependent on your compiler, cpu, and whether you are using 32 or 64-bit architecture. For wasm32 with clang:
- char is 1 byte aligned
- short is 2 byte aligned
- pointers are 4 byte aligned
- int, long, int32_t are 4 byte aligned
- double (Float 64) is 8-byte aligned
If you are not familiar with structure padding, there are many articles on the web.
Alignment requirements are why twr-wasm malloc
(and GCC malloc
for that matter) aligns new memory allocations on an 8-byte boundary.
"},{"location":"gettingstarted/parameters/#creating-a-struct-in-javascript","title":"Creating a struct in JavaScript","text":"We can create and initialize the above struct test_struct
like this in JavaScript:
//...\nconst mod = new twrWasmModule();\n//...\nconst structSize=12;\nconst structIndexA=0;\nconst structIndexB=4;\nconst structIndexC=8; // compiler allocates pointer on 4 byte boundaries\nlet structMem=mod.wasmMem.malloc(structSize);\nlet intMem=mod.wasmMem.malloc(4);\nmod.wasmMem.setLong(structMem+structIndexA, 1);\nmod.wasmMem.mem8[structMem+structIndexB]=2; // you can access the memory directly with the mem8, mem32, and memD (float64 aka double) byte arrays.\nmod.wasmMem.setLong(structMem+structIndexC, intMem);\nmod.wasmMem.setLong(intMem, 200000);\n
note that:
mod.wasmMem.malloc(structSize)
is a shortcut for: mod.callC([\"malloc\", structSize])
mod.wasmMem.malloc
returns a C pointer as a number
. This pointer is also an index into WebAssembly.Memory
-- which is exposed as the byte array (Uint8Array
) via mod.wasmMem.mem8
by twr-wasm. - When accessing a C
struct
in JavaScript/TypeScript, you have to do a bit of arithmetic to find the correct structure entry. - The entry
int *c
is a pointer to an int
. So a separate malloc
to hold the int
is needed. - In twr-wasm there is no function like
setLong
to set a byte. Instead you access the byte array view of the WebAssembly memory with mod.wasmMem.mem8
. Functions like mod.wasmMem.setLong
manipulate this byte array for you. - As well as
mod.wasmMem.mem8
(Uint8Array), you can also access WebAssembly.Memory directly via mod.wasmMem.mem32
(Uint32Array), and mod.wasmMem.memD
(Float64Array). - The list of functions available to access WebAssembly.Memory can be found at the end of this page.
"},{"location":"gettingstarted/parameters/#passing-struct-to-c-from-javascript","title":"Passing struct to C from JavaScript","text":"Assume we have C code that adds 2 to each entry of the test_struct
:
__attribute__((export_name(\"do_struct\")))\nvoid do_struct(struct test_struct *p) {\n p->a=p->a+2;\n p->b=p->b+2;\n (*p->c)++;\n (*p->c)++;\n}\n
Once the struct
has been created in JavaScript, you can call the C function do_struct
that adds 2 to each entry like this in twr-wasm:
await mod.callC([\"do_struct\", structMem]); // will add two to each value\n
"},{"location":"gettingstarted/parameters/#reading-c-struct-in-javascript","title":"Reading C struct in JavaScript","text":"You read the modified elements like this using JavaScript:
success=mod.wasmMem.getLong(structMem+structIndexA)==3;\nsuccess=success && mod.wasmMem.mem8[structMem+structIndexB]==4;\nconst intValPtr=mod.wasmMem.getLong(structMem+structIndexC);\nsuccess=success && intValPtr==intMem;\nsuccess=success && mod.wasmMem.getLong(intValPtr)==200002;\n
You can see the additional complexity of de-referencing the int *
.
"},{"location":"gettingstarted/parameters/#cleanup","title":"Cleanup","text":"You can free the malloced memory like this:
mod.wasmMem.free(intMem);\nmod.wasmMem.free(structMem);\n
The complete code for this example is here.
"},{"location":"gettingstarted/parameters/#passing-strings-from-javascript-to-cc-webassembly","title":"Passing Strings from JavaScript to C/C++ WebAssembly","text":"Although you can use the technique I am about to describe here directly (by writing your own code), it is generally accomplished by using a third-party library such as twr-wasm or Emscripten. These libraries handle the nitty-gritty for you.
To pass a string from JavaScript/TypeScript to a WebAssembly module, the general approach is to:
- Allocate memory for the string inside the WebAssembly memory. This is typically done by calling the C
malloc
from JavaScript. malloc
returns a pointer, which is an index into the WebAssembly Memory. - Copy the JavaScript string to this malloc'd Wasm memory. In the case of twr-wasm, this copying also converts the character encoding as necessary, for example, to UTF-8.
- Pass the malloc'd memory index to your function as an integer (which is accepted as a pointer by C code).
In the case of twr-wasm, the above steps are handled automatically for you by the callC
function:
mod.callC([\"my_function\", \"this is my string\"]); // mod is instance of twrWasmModule\n
Under the covers, to pass \"this is my string\" from JavaScript to the C Web Assembly function, callC
will execute code like this:
// twrWasmMemory member function\nputString(sin:string, codePage = codePageUTF8) {\n const ru8 = this.stringToU8(sin, codePage); // convert a string to UTF8 encoded characters stored in a Uint8Array\n const strIndex = this.malloc(ru8.length + 1); // shortcut for: await this.callC([\"malloc\", ru8.length + 1]);\n this.mem8.set(ru8, strIndex); // mem8 is of type Uint8Array and is the Wasm Module\u2019s Memory\n this.mem8[strIndex + ru8.length] = 0;\n return strIndex;\n}\n
this.malloc
is the standard C runtime malloc
function, provided by twr-wasm, and linked into your .wasm
code that is loaded into the WebAssembly Module. Likewise, twr-wasm will call free
after the function call is executed."},{"location":"gettingstarted/parameters/#returning-a-string-from-cc-webassembly-to-javascript","title":"Returning a String from C/C++ WebAssembly to JavaScript","text":"Returning a string from C to JavaScript is the reverse of passing in a string from JavaScript to C. When the \u201craw\u201d WebAssembly capabilities are used (WebAssembly.Module
, etc.) and your C code looks like this:
return(\"my string\");\n
The WebAssembly VM and JavaScript host will cause your JavaScript to receive an unsigned 32-bit integer. This is the pointer to the string, cast to an unsigned 32-bit integer. This integer is an index into the WebAssembly Memory.
twr-wasm provides a function to pull the string out of WebAssembly Memory and convert the character encoding to a JavaScript string. JavaScript strings are Unicode 16, but twr-wasm supports ASCII, UTF-8, and windows-1252 string encoding. When extracted and converted, a copy of the string is made.
const retStringPtr = await mod.callC([\"ret_string_function\"]);\nconsole.log(mod.wasmMem.getString(retStringPtr));\n
The retStringPtr
is an integer 32 (but converted to a JavaScript number
, which is Float 64). This integer is an index into the WebAssembly Memory.
"},{"location":"gettingstarted/parameters/#passing-arraybuffers-from-javascript-to-cc-webassembly","title":"Passing ArrayBuffers from JavaScript to C/C++ WebAssembly","text":"When callC
in twr-wasm is used to pass an ArrayBuffer to and from C/C++, some details are handled for you. The technique is similar to that used for a string
or as performed manually for a struct
above, with the following differences:
ArrayBuffers
have entries of all the same length, so the index math is straight forward and no struct
padding is needed. - When an
ArrayBuffer
is passed to a function, the function receives a pointer to the malloc
memory. If the length is not known by the function, the length needs to be passed as a separate argument. - Before
callC
returns, any modifications made to the memory by the C code are reflected back into the ArrayBuffer
. - the malloced copy of the ArrayBuffer is freed.
Here is an example:
let ba = new Uint8Array(4);\nba[0] = 99; ba[1] = 98; ba[2] = 97; ba[3] = 6;\nconst ret_sum = await mod.callC([\"param_bytearray\", ba.buffer, ba.length]);\n
See this example for the complete example.
"},{"location":"gettingstarted/parameters/#passing-a-javascript-object-to-webassembly","title":"Passing a JavaScript Object to WebAssembly","text":""},{"location":"gettingstarted/parameters/#simple-case-use-c-struct","title":"Simple Case - use C struct","text":"For a simple object like this:
const a = 'foo';\nconst b = 42;\n\nconst obj = {\n a: a,\n b: b\n};\n
It is straightforward to convert to a C struct like this:
struct obj {\n const char* a;\n int b;\n};\n
To pass this JavaScript object to WebAssembly, a C struct is created (using the struct
techniques described above). Each object entry is then copied into the corresponding C struct
entry (using the struct
and string techniques described above)."},{"location":"gettingstarted/parameters/#more-complicated-object","title":"More Complicated Object","text":"A JavaScript object can contain entries that are of more complexity than simple C data types. For example:
const a = 'foo';\nconst b = 42;\nconst map = new Map();\nmap1.set('a', 1);\nmap1.set('b', 2);\nmap1.set('c', 3);\nconst object2 = { a: a, b: b, c: map };\n
In this case, you are going to have to do more work. An approach is to use the libc++ map
class, which is similar to the JavaScript Map
. You could also perhaps use the libc++ vector
.
To handle this more complicated JavaScript object with a Map
entry, an approach is to export functions from WebAssembly to create and add entries to the libc++ map
(you need to use extern 'C'
to export these C++ access functions as C functions). In otherworld, you might export from your Wasm Module C functions like this:
void* createMap(); // return an unsigned long Map ID\nvoid addIntToMap(void* mapID, int newInt);\n
You would then use these functions in JavaScript to build your C++ map
. JavaScript would access this map
using the unsigned long
identifier (the void *
returned by createMap
). After creating and adding entries to the map
, you would set this MapID to object2.c
.
There are alternative approaches. For example, you could convert the JavaScript Map
to a C struct, by enumerating every entry in the Map
. Your C struct might look like: `
struct entry {\n char* name;\n int value;\n};\n\nstruct mapUnroll {\n int MapLen;\n struct entry* entries[];\n};\n
This approach is probably even more work, less general purpose, and less efficient.
"},{"location":"gettingstarted/parameters/#summary","title":"Summary","text":"I hope this has demystified how JavaScript values are passed to and from WebAssembly. In many cases, functions like twr-wasm's mod.callC
will handle the work for you. But in more bespoke cases, you will have to handle some of the work yourself.
"},{"location":"gettingstarted/stdio/","title":"Overview of Consoles","text":"This section describes how to use twr-wasm to:
- create input/output consoles for use by C/C++ with WebAssembly
- direct stdin, stdout and stderr to a console
- use addressable display and canvas 2D consoles
- use multiple consoles at once
"},{"location":"gettingstarted/stdio/#quick-example","title":"Quick Example","text":"Hello World#include <stdio.h>\n\nvoid hello() {\n printf(\"hello world\\n\");\n}\n
Using twrConsoleDiv<body>\n <div id=\"console-tag\"></div>\n\n <script type=\"module\">\n import {twrConsoleDiv, twrWasmModule} from \"twr-wasm\";\n\n const tag=document.getElementById(\"console-tag\");\n const streamConsole=new twrConsoleDiv(tag); \n const mod = new twrWasmModule({stdio: streamConsole});\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n\n </script>\n</body>\n
Using twr_iodiv Shortcut<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n\n </script>\n</body>\n
"},{"location":"gettingstarted/stdio/#running-examples","title":"Running Examples","text":"Name View Live Link Source Link stdin and stdout to <div>
View square demo Source simple \"terminal\" via <canvas>
View hello world demo Source \"cli\" with a <canvas>
stdio View CLI demo using libc++ Source Multiple Consoles, including Canvas2D View multi-io demo Source"},{"location":"gettingstarted/stdio/#capabilities","title":"Capabilities","text":"With a Console you can:
- read character streams (Use C statements like
getc
or io_mbgets
) - write character streams (use C statements like
printf
or cout
) - position characters, graphics, colors with an addressable display (Use C statements like
io_setc32
or io_set_cursor
). - draw to a Canvas compatible 2D surface (Use C statements like
d2d_fillrect
).
Consoles are primarily designed for use by twr-wasm C/C++ modules, but they can also be used by JavaScript/TypeScript.
Although it is common to have a single console, an arbitrary number of consoles can be created, and they can be used by an arbitrary number of twr-wasm C/C++ modules.
Unicode characters are supported by consoles (see Character Encoding Support with twr-wasm).
"},{"location":"gettingstarted/stdio/#tag-shortcuts","title":"Tag Shortcuts","text":"If you add a <div id=\"twr_iodiv\">
, a <canvas id=\"twr_iocanvas\">
, or a <canvas id=\"twr_d2dcanvas\">
tag to your HTML, twr-wasm will create the appropriate class for you when you instantiate the class twrWasmModule
or twrWasmModuleAsync
. Use these tag shortcuts as an aternative to instantiating the console classes in your JavaScript/TypeScript.
<div id=\"twr_iodiv\">
will be used to create a twrConsoleDiv
as stdio
<canvas id=\"twr_iocanvas\">
will be used to create a twrConsoleTerminal
as stdio
. <canvas id=\"twr_d2dcanvas\">
will be used to create a twrConsoleCanvas
as std2d
-- the default 2D drawing surface. See 2D drawing APIs.
If neither of the above <div>
or <canvas>
is defined in your HTML, and if you have not set stdio
via the io
or stdio
module options, then stdout
is sent to the debug console in your browser. And stdin
is not available.
"},{"location":"gettingstarted/stdio/#console-classes","title":"Console Classes","text":"Consoles are implemented in TypeScript and run in the JavaScript main thread. This allows consoles to be shared by multiple wasm modules.
For simple cases, when you use the tag shortcuts, you won't need to use these console classes directly. For more bespoke cases, they will come in handy. For details on console classes, see the TypeScript/JavaScript API reference
These conosle classes are available in twr-wasm:
twrConsoleDiv
streams character input and output to a div tag twrConsoleTerminal
provides streaming or addressable character input and output using a canvas tag. twrConsoleDebug
streamings characters to the browser debug console. twrConsoleCanvas
creates a 2D drawing surface that the Canvas compatible 2d drawing APIs can be used with.
"},{"location":"gettingstarted/stdio/#multiple-consoles-with-names","title":"Multiple Consoles with Names","text":"When you instantiate a class twrWasmModule
or twrWasmModuleAsync
, you can pass it the module option io
-- a javascript object containing name-console attributes. Your C/C++ code can then retrieve a console by name. This is described in more detail the TypeScript/JavaScript API Reference.
Also see the multi-io example.
"},{"location":"gettingstarted/stdio/#setting-stdio-and-stderr","title":"Setting stdio and stderr","text":"stdio
can be defined automatically if you use a Tag Shortcut. stderr
streams to the browser's debug console by default. Both can be set to a specific console with the module io
option.
For example, given:
const tag=document.getElementById(\"console-tag\");\nconst streamConsole=new twrConsoleDiv(tag);\n
Either of these will set stdio
to a streaming div
console:
const mod = new twrWasmModule({stdio: streamConsole});\n
const mod = new twrWasmModule({ io: {stdio: streamConsole} });\n
This option would send stderr and stdio to the same console:
const mod = new twrWasmModule({ io: \n {stdio: streamConsole, stderr: streamConsole} \n});\n
"},{"location":"gettingstarted/stdio/#utf-8-or-windows-1252","title":"UTF-8 or Windows-1252","text":"Consoles can support UTF-8 or Windows-1252 character encodings (see Character Encoding Support with twr-wasm).
"},{"location":"gettingstarted/stdio/#c-access-to-consoles","title":"C Access To Consoles","text":""},{"location":"gettingstarted/stdio/#io_functions","title":"io_functions","text":"io_functions
are available to operate on all character based Consoles.
"},{"location":"gettingstarted/stdio/#d2d_functions","title":"d2d_functions","text":"d2d_functions
are available to operate on Canvas 2D Consoles.
"},{"location":"gettingstarted/stdio/#reading-from-a-console","title":"Reading from a Console","text":"Reading from a console is blocking, and so twrWasmModuleAsync
must be used to receive keys. There are some specific requirements to note in the twrWasmModuleAsync
API docs.
You can get characters with any of these functions:
io_mbgets
- get a multibyte string from a console using the current locale character encoding. Console must support IO_TYPE_CHARREAD. twr_mbgets
- the same as io_mbgets
with the console set to stdin
. io_mbgetc
- get a multibyte character from an twr_ioconsole_t *
(aka FILE *
) like stdin
using the current locale character encoding getc
(same as fgetc
) - get a single byte from a FILE *
(aka twr_ioconsole_t *
) -- returning ASCII or extended ASCII (window-1252 encoding) io_getc32
- gets a 32 bit unicode code point from an twr_ioconsole_t *
(which must support IO_TYPE_CHARREAD)
"},{"location":"gettingstarted/stdio/#standard-c-library-functions","title":"Standard C Library Functions","text":"Many of the common standard C library functions, plus twr-wasm specific functions, are available to stream characters to and from the standard input and output console that supports character streaming (most do).
In C, a console is represented by twr_ioconsole_t
. In addition, FILE
is the same as a twr_ioconsole_t
(typedef twr_ioconsole_t FILE
). stdout
, stdin
, stderr
are all consoles.
#include <stdio.h>
to access stdout
, stdin
, stderr
, and FILE
.
FILE
is supported for user input and output, and for stderr. FILE
as filesystem I/O is not currently supported.
"},{"location":"gettingstarted/stdio/#stdout-and-stderr-functions","title":"stdout and stderr functions","text":"You can use these functions to output to the standard library defines stderr
or stdout
:
fputc, putc, vfprintf, fprintf, fwrite\n
These functions go to stdout
:
printf, vprintf, puts, putchar\n
Note that when characters are sent to the browser console using stderr
they will not render to the console until a newline, return, or ASCII 03 (End-of-Text) is sent.
For example:
#include <stdio.h>\n\nfprintf(stderr, \"hello over there in browser debug console land\\n\");\n
A more common method to send output to the debug console is to use twr_conlog
.
"},{"location":"more/building/","title":"Building the twr-wasm Source","text":""},{"location":"more/building/#source-for-twr-wasm","title":"Source for twr-wasm","text":"The source can be found at:
https://github.com/twiddlingbits/twr-wasm\n
The main
branch contains the latest release. The dev
branch is work in progress.
"},{"location":"more/building/#tools-needed-to-build-twr-wasm-source","title":"Tools Needed to Build twr-wasm Source","text":"You will need these core tools, versions used in release are in ():
- TypeScript (5.4.5)
- clang tool chain (17.0.6) - for C/C++ code
- wasm-ld (17.0.6) - to link the .wasm files
- wat2wasm (1.0.34) - to compile WebAssembly (.wat) files of which I have a few
- GNU make (4.4.1)
- git - to clone twr-wasm source, or to clone llvm, if you want to build libc++
In addition, you might need:
- VS Code - to use the debug launcher and build tasks
- NPM - package manager
- Parcel v2 - to bundle the examples
- mkdocs, material theme, meta-descriptions plugin - to build the documentation static web site
- python - mkdocs is built with python, and you need python to run server.py in examples
- CMake and ninja - to build llvm libc++
There is a deprecated gcc build that I used to use for testing, but now the tests are executed in wasm.
"},{"location":"more/building/#to-build-the-libraries-lib-c-lib-js","title":"To Build the Libraries (lib-c, lib-js)","text":"cd source\nmake\n
or on windows cd source\nmingw32-make\n
"},{"location":"more/building/#to-build-the-examples","title":"To Build the Examples","text":"See examples/readme.md for more information.
To build the examples, but not bundle them.
cd examples\nsh buildall.sh\n
To build bundles:
sh buildbundles.sh\n
"},{"location":"more/building/#to-build-the-docs","title":"To Build the docs","text":"The docs are created using the material theme for mkdocs.
In twr-wasm root folder:
mkdocs build\n
The destination of the build is found in the mkdocs.yml
file (site_dir: azure/docsite/
).
Usually the docs are built as part of building the static web site that hosts the docs and examples. This is accomplished using this shell script (found in examples folder):
buildazure.sh\n
"},{"location":"more/building/#to-build-libc-for-wasm-and-twr-wasm","title":"To Build libc++ for Wasm and twr-wasm","text":"See the instructions in the comments in the shell script source\\libcxx\\buildlibcxx.sh
"},{"location":"more/building/#installing-clang-and-wasm-ld-on-windows","title":"Installing clang and wasm-ld on Windows","text":"Here is how I installed the tools for windows:
install MSYS2 \n 1. https://www.msys2.org/\n 2. After the install completes, run UCRT64 terminal by clicking on the MSYS2 UCRT64 in the Start menu\n 3. pacman -Syuu\n\n install gcc using MSYS2 UCRT64\n 1. Use MSYS2 UCRT64 terminal (per above)\n 1. pacman -S mingw-w64-ucrt-x86_64-toolchain\n\n install clang and wasm-ld using MSYS2 UCRT64\n 2. Use MSYS2 UCRT64 (per above)\n 1. pacman -S mingw-w64-ucrt-x86_64-clang\n 2. pacman -S mingw-w64-x86_64-lld\n\nupdate PATH env variable using the windows control panel (search for path)\n 2. added C:\\msys64\\ucrt64\\bin \n 3. added C:\\msys64\\mingw64\\bin \n 4. added C:\\msys64\\usr\\bin (for sh.exe used by mingw32-make)\n
wabt tools: can be found here https://github.com/WebAssembly/wabt/releases
"},{"location":"more/imports/","title":"twr-wasm Import Resolution","text":"This section covers path resolution for statements like this:
import {twrWasmModule} from \"twr-wasm\";\n
It can be confusing to determine how tools like VS Code, a bundler, or a Web Browser resolve imports like \"twr-wasm\". This section explains how it works, using the included examples as examples.
Two situations are addressed in this document: (1) browser import resolution when not using a bundler, and (2) if you installed using git clone
.
"},{"location":"more/imports/#if-you-installed-using-git-clone","title":"If you installed using git clone
","text":"If you have installed twr-wasm
using npm
, most tools will automatically resolve imports by finding the node_modules
folder.
However, if you installed using git clone
, then other steps will need to be taken to help a bundler, VS Code, and tsc resolve imports. This situation can apply to the examples, which are in the git repo tree, and as a result there may be no node_modules
for the twr-wasm libraries. This will also be the case for your project, if you installed with git clone
.
"},{"location":"more/imports/#import-path-resolution-by-the-bundler","title":"Import path resolution by the bundler","text":"A bundler will find the twr-wasm library using one of these methods:
- If twr-wasm has been installed with npm install, the bundler will find the
node_modules
folder - Alternately, If all your scripts are in TypeScript, use
paths
entry in tsconfig.json
(see maze example) - Alternately, use alias option in package.json as in the helloworld example
\"alias\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n },\n
In the examples, the alias entry in the package.json
exists so that the parcel bundler can find twr-wasm.
If you are using a bundler, you don't need to add a <script type=\"importmap\">
tag.
"},{"location":"more/imports/#import-resolution-by-vs-code-and-tsc","title":"Import resolution by VS Code and tsc","text":"VS Code Intellisense and the typescript compiler need to find modules. If twr-wasm is installed using npm
into a node_modules
folder, this is probably automatic. But if this is not the case, you can add a line to the tsconfig.json
as follows (this example assumes the tsconfig.json
is in a examples/example folder). See the maze example.
\"paths\": {\n \"twr-wasm\": [\"./../../lib-js/index\"]\n}\n
"},{"location":"more/imports/#import-path-resolution-by-the-browser","title":"Import path resolution by the browser","text":"This section apples to executing your javascript without first \"bundling\" it. Whether execution is from the filesystem directly in a browser or using a web server.
In order for the browser to locate the twr-wasm path when import is used, you can add code like this to your HTML prior to the import. You should make sure the path for twr-wasm is correct for your project (this is correct for the examples).
<script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n }\n }\n</script>\n
"},{"location":"more/production/","title":"HTTP CORS headers needed to use twrWasmModuleAsync","text":"If you are using twr-wasm with web pages served by an http server, you may need to enable certain CORS headers. This applies whether using a remote server or using your local machine for development.
twr-wasm class twrWasmModuleAsync
uses SharedArrayBuffers
, and there are special CORS headers that need to be configured to use SharedArrayBuffers
, that are not widely enabled by default on web servers. It may be helpful to also see the SharedArrayBuffer
documentation online.
Github pages doesn't support the needed CORS headers for SharedArrayBuffers. But other web serving sites do have options to enable the needed CORS headers.
Here are two provided examples of how to enable the necessary headers:
- server.py
- staticwebapp.config.json
The azure static web site config file staticwebapp.config.json
looks like this:
{\n \"globalHeaders\": {\n \"Access-Control-Allow-Origin\": \"*\",\n \"Cross-Origin-Embedder-Policy\": \"require-corp\",\n \"Cross-Origin-Opener-Policy\": \"same-origin\"\n }\n}\n
server.py in the examples folder will launch a local server with the correct headers. To use Chrome without a web server, see the Hello World walk through.
"},{"location":"more/production/#using-twrwasmmoduleasync-with-file","title":"Using twrWasmModuleAsync with file:","text":"If you are loading html with chrome from files (not using an https server), you will need to set these command line args:
--enable-features=SharedArrayBuffer\n--allow-file-access-from-files\n
More detail is found in the debugging section.
"},{"location":"more/wasm-problem/","title":"Wasm Runtime Limitations","text":"HTML browsers can load a WebAssembly module, and execute it's bytecode in a virtual machine. To create this bytecode (.wasm file) from C/C++, you compile your code using clang with the target code format being WebAssembly (aka Wasm) byte code. If you are not using a support library like twr-wasm or emscripten, there are a few issues that one immediately encounters trying to execute code that is more complicated than squaring a number.
The Wasm virtual machine simply executes the instructions that are generated by the clang compiler and linked by the linker into the .wasm file. The first issue encountered is that some code that is generated by a compiler assumes a compiler support library will be linked to your code. That is, clang code generation will produce calls to compiler support routines for floating point, memcpy
, and the like. In clang, these support routines are in the \"compile-rt\" support library. Typically clang handles this behind to scenes for you. But support for a WebAssembly version of this compiler support library is not (as of this writing) included in a clang distribution.
The next level up the library stack is the standard C runtime library. This library provides functions like malloc
and printf
. And then built on the standard C runtime library is the standard c++ library - like libc++. WebAssembly versions of these libraries are also not part of a clang distribution.
To get access to WebAssembly versions of these libraries you need to use emscripten or twr-wasm.
The second problem is that all the function calls between your Wasm module and your javascript are limited to parameters and return values that are numbers (integer and float). No strings, arrays, struct pointers, etc. (for more on this see this doc).
The third problem is that legacy C code or games often block, and when written this way they don't naturally integrate with the JavaScript asynchronous programming model.
twr-wasm is a static C library (twr.a) that you can link to your clang C/C++ Wasm code, as well as a set of JavaScript/TypeScript modules that solve these issues.
In addition, twr-wasm provides APIs that you can use in your WebAssembly code - such as Canvas compatible 2D drawing APIs, a simple terminal emulator, character encoding support, and more.
"}]}
\ No newline at end of file
+{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"]},"docs":[{"location":"","title":"Easier WebAssembly with twr-wasmDocumentation and Examples","text":"Version 2.5.0
twr-wasm is a simple, lightweight and easy to use library for building C/C++ WebAssembly code directly with clang. Run C/C++ code in a web browser. Legacy code, libraries, full applications, or single functions can be integrated with JavaScript and TypeScript. twr-wam solves some common use cases with less work than the more feature rich emscripten.
Key Features:
- build
.wasm
modules using C/C++ using clang directly (no wrapper) - from JavaScript load
.wasm
modules, call C/C++ functions, and access wasm memory -
comprehensive console support for stdin
, stdio
, and stderr
.
- in C/C++, print and get characters to/from
<div>
tags in your HTML page - in C/C++, print and get characters to/from a
<canvas>
based \"terminal\" - localization support, UTF-8, and windows-1252 support
-
the optional TypeScript class twrWasmModuleAsync
can be used to:
- integrate a C/C++ Read-Eval-Print Loop (REPL) with JavaScript
- integrate a C/C++ CLI or Shell with JavaScript
- In JavaScript
await
on blocking/synchronous C/C++ functions.
-
2D drawing API for C/C++ compatible with JavaScript Canvas
- audio playback APIs for C/C++
- create your own C/C++ APIs using TypeScript by extending
class twrLibrary
- standard C library optimized for WebAssembly
- libc++ built for WebAssembly
- comprehensive examples and documentation
- TypeScript and JavaScript support
"},{"location":"#live-webassembly-examples-and-source","title":"Live WebAssembly Examples and Source","text":"Name View Live Link Source Link Bouncing Balls (C++) View bouncing balls Source for balls Pong (C++) Pong Source Input/Output with <div>
View square demo Source I/O to terminal with <canvas>
View demo Source CLI using libc++ and <canvas>
) View console Source"},{"location":"#hello-world","title":"Hello World","text":"Here is the simplest twr-wasm
example.
helloworld.c#include <stdio.h>\n\nvoid hello() {\n printf(\"hello, world!\\n\");\n}\n
index.html<head>\n <title>Hello World</title>\n</head>\n<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n </script>\n</body>\n
"},{"location":"#on-github","title":"On Github","text":"https://github.com/twiddlingbits/twr-wasm
"},{"location":"#why","title":"Why?","text":"The Wasm Runtime Limitations section explains why a library like twr-wasm is needed to use WebAssembly.
"},{"location":"#limitations","title":"Limitations","text":" - libc++ not built with exceptions enabled
- some standard C library functions are not 100% implemented
- Designed to work with a browser. Not tested with or designed to work with node.js
- Not all of compile-rt is ported (but most bits you need are)
"},{"location":"#post-feedback","title":"Post Feedback","text":"Please post feedback (it worked for you, didn't work, requests, questions, etc) at https://github.com/twiddlingbits/twr-wasm/
"},{"location":"api/api-c-audio/","title":"Audio API for WebAssembly","text":"This section describes twr-wasm's C Audio API, which allows audio API functions to be called using C/C++ from WebAssembly.
"},{"location":"api/api-c-audio/#examples","title":"Examples","text":"Name View Live Link Source Link Pong (C++) View Pong Source for Pong tests-audio View tests-audio Source for tests-audio"},{"location":"api/api-c-audio/#code-example","title":"Code Example","text":"Play Audio#include \"twr-audio.h\"\n#include <math.h>\n#include <stdlib.h>\n\n#define M_PI 3.14159265358979323846\n\nvoid play() {\n twr_audio_play_file(\"example.mp3\"); //plays audio from specified URL\n\n const long SAMPLE_RATE = 48000; //48,000 samples per second\n const double DURATION = 10.0; //play for 10 seconds\n const double freq = 493.883; //Middle B (B4)\n\n long length = (long)ceil(SAMPLE_RATE*DURATION);\n //PCM audio data in the form of -1.0 to 1.0\n float* wave = (float*)malloc(sizeof(float) * length);\n\n //generate square wave at specified frequency and duration\n for (long i = 0; i < length; i++) {\n wave[i] = cos(2*M_PI*freq*(i/(float)sample_rate)) > 0 ? 1 : -1;\n }\n\n //creates a mon-audio channel buffer at our given SAMPLE_RATE\n // and square-wave data we generated\n long node_id = twr_audio_from_float_pcm(1, SAMPLE_RATE, wave, length);\n\n //plays the square wave\n // Can be played multiple times, and is only freed on twr_audio_free\n twr_audio_play(node_id);\n}\n
"},{"location":"api/api-c-audio/#overview","title":"Overview","text":"The Audio API is part a twr-wasm library that can be accessed via #include \"twr-audio.h\"
. It has two main methods to play audio: from raw PCM data and from a URL.
Raw PCM data can be initialized via twr_audio_from_<type>_pcm
or twr_audio_load
. There are multiple types for twr_audio_from_<type>_pcm
. These types include Float, 8bit, 16bit, and 32bit. Float takes in values between -1.0 and 1.0. Meanwhile, the 8bit, 16bit, and 32bit versions take them in as signed numbers between their minimum and maximum values. twr_audio_load
, on the other hand, reads the PCM data from an audio file that is specified by a URL. This method does not stream the audio, so it might take some time to read the file (depending how long the file is). Each function returns an integer node_id
that identifies the loaded PCM data. Once the PCM data is initialized, it can be played via functions like twr_audio_play
, twr_audio_play_range
, twr_audio_play_sync
, and twr_audio_play_range_sync
. The play functions can be called multiple times for each audio ID.
You can also play audio directly from a URL. Unlike twr_audio_load
, the url is initialized directly into an HTMLAudioElement
which streams the audio and starts playback immediately.
In addition to playing audio, there are functions that allow you to query and modify an ongoing playback. These include stopping the playback, getting how long it's been playing, and modifying the pan; volume; or playback rate of the audio.
"},{"location":"api/api-c-audio/#notes","title":"Notes","text":"When playing audio, a playback_id
is returned to query or modify the playback. However, once playback completes, the playback_id
becomes invalid. Most functions that take in a playback_id
will simply return a warning and return without error if the playback_id
is invalid. An exception is twr_audio_query_playback_position
which will return -1 when given an invalid playback_id
.
Functions that end in _sync
are for use with twrWamModuleAsync
, and are synchronous. Meaning they don't return until the operation, such as playback, is complete.
Note that on some platforms, sounds that are too short might not play correctly. This is true in JavaScript as well.
"},{"location":"api/api-c-audio/#functions","title":"Functions","text":"These are the current Audio APIs available in C/C++:
long twr_audio_from_float_pcm(long num_channels, long sample_rate, float* data, long singleChannelDataLen);\nlong twr_audio_from_8bit_pcm(long number_channels, long sample_rate, char* data, long singleChannelDataLen);\nlong twr_audio_from_16bit_pcm(long number_channels, long sample_rate, short* data, long singleChannelDataLen);\nlong twr_audio_from_32bit_pcm(long number_channels, long sample_rate, int* data, long singleChannelDataLen);\n\nfloat* twr_audio_get_float_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nchar* twr_audio_get_8bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nshort* twr_audio_get_16bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\nint* twr_audio_get_32bit_pcm(long node_id, long* singleChannelDataLenPtr, long* numChannelsPtr);\n\nlong twr_audio_play(long node_id);\nlong twr_audio_play_volume(long node_id, double volume, double pan);\nlong twr_audio_play_callback(long node_id, double volume, double pan, int finish_callback);\n\nstruct PlayRangeFields {\n double pan, volume;\n int loop, finish_callback;\n long sample_rate;\n};\nstruct PlayRangeFields twr_audio_default_play_range();\nlong twr_audio_play_range(long node_id, long start_sample, long end_sample);\nlong twr_audio_play_range_ex(long node_id, long start_sample, long end_sample, struct PlayRangeFields* fields);\n\nlong twr_audio_play_sync(long node_id);\nlong twr_audio_play_sync_ex(long node_id, double volume, double pan);\n\n\nstruct PlayRangeSyncFields {\n double pan, volume;\n int loop;\n long sample_rate;\n};\n\nstruct PlayRangeSyncFields twr_audio_default_play_range_sync();\nlong twr_audio_play_range_sync(long node_id, long start_sample, long end_sample);\nlong twr_audio_play_range_sync_ex(long node_id, long start_sample, long end_sample, struct PlayRangeSyncFields* fields);\n\nlong twr_audio_load_sync(char* url);\nlong twr_audio_load(int event_id, char* url);\nlong twr_audio_query_playback_position(long playback_id);\nvoid twr_audio_free_id(long node_id);\n\nvoid twr_audio_stop_playback(long playback_id);\n\nvoid twr_audio_modify_playback_volume(long playback_id, double volume);\nvoid twr_audio_modify_playback_pan(long playback_id, double pan);\nvoid twr_audio_modify_playback_rate(long playback_id, double sample_rate);\n\nlong twr_audio_play_file(char* file_url);\nlong twr_audio_play_file_ex(char* file_url, double volume, double playback_rate, int loop);\n\nstruct AudioMetadata {\n long length;\n long sample_rate;\n long channels;\n};\n\nvoid twr_audio_get_metadata(long node_id, struct AudioMetadata* metadata);\n
"},{"location":"api/api-c-con/","title":"WebAssembly Character Console API","text":"twr-wasm for WebAssembly provides Consoles for interactive user I/O. Character and graphic 2D draw consoles exist. This section covers the streaming and addressable character APIs that can be used with an instance of twrConsoleDebug, twrConsoleTerminal, twrConsoleDiv. This API works with stdin, stdout, stderr and custom named consoles.
Also see the Consoles section in Getting Started
"},{"location":"api/api-c-con/#examples","title":"Examples","text":"Name View Live Link Source Link \"terminal\" in/out with a <canvas>
View mini-term demo Source"},{"location":"api/api-c-con/#getting-a-console","title":"Getting a Console","text":""},{"location":"api/api-c-con/#stdin-stdout-stderr","title":"stdin, stdout, stderr","text":"stdin
, stdout
, stderr
are defined in <stdio.h>
.
This section describes how to configure stdio
In C, consoles are represented by a twr_ioconsole_t
.
stdio.h also defines FILE
like this:
typedef twr_ioconsole_t FILE; \n
from <stdio.h>
:
#define stderr (FILE *)(twr_get_stderr_con())\n#define stdin (FILE *)(twr_get_stdio_con())\n#define stdout (FILE *)(twr_get_stdio_con())\n
"},{"location":"api/api-c-con/#twr_get_console","title":"twr_get_console","text":"This function will retrieve a console by its name. The standard names are stdio
, stderr
, and std2d
. In addition, any named console that was passed to a module using the io
option can be retrieved with this function.
See io doc.
See the multi-io example.
#include \"twr-crt.h\"\n\ntwr_ioconsole_t* twr_get_console(const char* name)\n
"},{"location":"api/api-c-con/#io_nullcon","title":"io_nullcon","text":"Returns an IoConsole that goes to the bit bucket. io_getc32 will return 0.
#include \"twr-io.h\"\n\ntwr_ioconsole_t* io_nullcon(void);\n
"},{"location":"api/api-c-con/#io-console-functions","title":"IO Console Functions","text":""},{"location":"api/api-c-con/#io_cls","title":"io_cls","text":"For addressable display consoles only.
Clears the screen. That is, all character cells in the console are set to a space, their colors are reset to the current default colors (see io_set_colors
).
#include <twr_io.h>\n\nvoid io_cls(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_getc32","title":"io_getc32","text":"Waits for the user to press a key and then returns a unicode code point.
To return characters encoded with the current locale, see io_mbgetc
#include <twr_io.h>\n\nint io_getc32(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_colors","title":"io_get_colors","text":"For addressable display consoles only.
Gets the current default foreground and background colors. These colors are used by an new text updates.
The color format is a 24 bit int as RGB.
#include <twr_io.h>\n\nvoid io_get_colors(twr_ioconsole_t* io, unsigned long *foreground, unsigned long *background);\n
"},{"location":"api/api-c-con/#io_get_cursor","title":"io_get_cursor","text":"Returns an integer of the current cursor position. The cursor is where the next io_putc
is going to go.
For addressable display consoles, the cursor position ranges from [0, width*height-1], inclusive.
#include <twr_io.h>\n\nint io_get_cursor(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_prop","title":"io_get_prop","text":"Given a string key (name) of a property, returns its integer value. The available properties varies by console type.
#include <twr_io.h>\n\nint io_get_prop(twr_ioconsole_t* io, const char* key)\n
All consoles support: \"type\". Addressable consoles also support: \"cursorPos\", \"charWidth\", \"charHeight\", \"foreColorAsRGB\", \"backColorAsRGB\", \"widthInChars\", \"heightInChars\", \"fontSize\", \"canvasWidth\", \"canvasHeight\"
You can do a bitwise &
on type with the following C defines to determine a console capabilities:
IO_TYPE_CHARREAD
IO_TYPE_CHARWRITE
IO_TYPE_ADDRESSABLE_DISPLAY
IO_TYPE_CANVAS2D
For example:
if (io_get_prop(stdin, \"type\")&IO_TYPE_CHARREAD) {\n printf (\"okay to read from stdin);\n}\n
"},{"location":"api/api-c-con/#io_get_width","title":"io_get_width","text":"Returns the width in characters of an addressable console.
#include <twr_io.h>\n\nint io_get_width(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_get_height","title":"io_get_height","text":"Returns the height in characters of an addressable console.
#include <twr_io.h>\n\nint io_get_height(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_set_colors","title":"io_set_colors","text":"For addressable display consoles only.
Sets a 24 bit RGB default color for the foreground and background. The prior default colors are changed (lost). For example, if you set the default colors when you created the console (see twrConsoleTerminal Options), the defaults will no longer be active. Use io_get_colors
to save existing colors for later restoration using io_set_colors
.
A call to io_set_colors
doesn't actually cause any on screen changes. Instead, these new default colors are used in future draw and text calls. A foreground and background color is set for each cell in the console window. The cell's colors are set to these default foreground/background colors when a call to io_setc
, io_setreset
, etc is made.
#include <twr_io.h>\n\nvoid io_set_colors(twr_ioconsole_t* io, unsigned long foreground, unsigned long background);\n
"},{"location":"api/api-c-con/#io_setc","title":"io_setc","text":"For addressable display consoles only.
Sets a console cell to the specified character. Sends a byte to an console and supports the current locale's character encoding. This function will \"stream\" using the current code page. In other words, if you are in the \"C\" locale io_setc
it will set ASCII characters. If the current locale is set to 1252, then you can send windows-1252 encoded characters. If the current locale is UTF-8, then you can stream UTF-8 (that is, call io_setc
once for each byte of the multi-byte UTF-8 character).
#include <twr_io.h>\n\nbool io_setc(twr_ioconsole_t* io, int location, unsigned char c);\n
"},{"location":"api/api-c-con/#io_setc32","title":"io_setc32","text":"For addressable display consoles only.
Sets a console cell to a unicode code point. The colors are set to the defaults (see io_set_colors
).
#include <twr_io.h>\n\nvoid io_setc32(twr_ioconsole_t* io, int location, int c);\n
"},{"location":"api/api-c-con/#io_set_cursor","title":"io_set_cursor","text":"Moves the cursor. See io_get_cursor
.
#include <twr_io.h>\n\nvoid io_set_cursor(twr_ioconsole_t* io, int loc);\n
"},{"location":"api/api-c-con/#io_set_cursorxy","title":"io_set_cursorxy","text":"Set's the cursor's x,y position in an addressable console.
#include <twr_io.h>\n\nvoid io_set_cursorxy(twr_ioconsole_t* io, int x, int y);\n
"},{"location":"api/api-c-con/#io_setfocus","title":"io_setfocus","text":"Sets the input focus to the indicated console.
#include <twr_io.h>\n\nvoid io_setfocus(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_set_range","title":"io_set_range","text":"Sets a range of characters in an addressable display.
#include <twr_io.h>\n\nvoid io_set_range(twr_ioconsole_t* io, int *chars32, int start, int len)\n
"},{"location":"api/api-c-con/#io_setreset","title":"io_setreset","text":"For addressable display consoles only.
Sets or resets (clears) a chunky graphics \"pixel\". Each character cell can also be a 2x3 grid of graphic \"pixels\". In other words, the terminal window has pixel dimensions of width2 x height3.
The color will be set to the defaults if the impacted cell is not a graphics cell. If it is an existing graphics cell, the colors don't change.
See the terminal
example.
#include <twr_io.h>\n\nbool io_setreset(twr_ioconsole_t* io, int x, int y, bool isset);\n
"},{"location":"api/api-c-con/#io_mbgetc","title":"io_mbgetc","text":"io_mbgetc
will get a character from stdin and encode it using the character encoding of the LC_CTYPE category of the current locale. \"C\" will use ASCII. UTF-8 and windows-1252 are also supported.
#include <twr_io.h>\n\nvoid io_mbgetc(twr_ioconsole_t* io, char* strout);\n
"},{"location":"api/api-c-con/#io_mbgets","title":"io_mbgets","text":"Gets a string from a Console. Returns when the user presses \"Enter\". Displays a cursor character and echos the inputted characters, at the current cursor position. Uses character encoding of LC_TYPE of current locale. If the encoding is UTF-8, then the result will be multibyte.
This function is commonly used with stdin
.
This function requires that you use twrWasmModuleAsync
.
#include <twr_io.h>\n\nchar *io_mbgets(twr_ioconsole_t* io, char *buffer );\n
"},{"location":"api/api-c-con/#io_point","title":"io_point","text":"For addressable display consoles only.
Checks if a chunky graphics \"pixel\" is set or clear. See io_setreset
.
#include <twr_io.h>\n\nbool io_point(twr_ioconsole_t* io, int x, int y);\n
"},{"location":"api/api-c-con/#io_putc","title":"io_putc","text":"Sends a byte to an IoConsole and supports the current locale's character encoding. This function will \"stream\" using the current code page. In other words, if you io_putc
ASCII, it will work as \"normal\". If the current locale is set to 1252, then you can send windows-1252 encoded characters. If the current locale is UTF-8, then you can stream UTF-8 (that is, call io_putc
once for each byte of the multi-byte UTF-8 character).
Note that when characters are sent to the browser console using stderr
they will not render to the console until a newline or return is sent.
#include \"twr-io.h\"\n\nvoid io_putc(twr_ioconsole_t* io, unsigned char c);\n
"},{"location":"api/api-c-con/#io_putstr","title":"io_putstr","text":"Calls io_putc
for each byte in the passed string.
#include \"twr-io.h\"\n\nvoid io_putstr(twr_ioconsole_t* io, const char* s);\n
"},{"location":"api/api-c-con/#io_printf","title":"io_printf","text":"Identical to fprintf
, however io_printf will call io_begin_draw
and io_end_draw
around its drawing activities -- resulting in snapper performance.
For example:
#include \"twr-io.h\"\n\nio_printf(twr_debugcon(), \"hello over there in browser debug console land\\n\");\n
or
#include <stdio.h>\n#include <twr_io.h>\n\nio_printf(stdout, \"hello world\\n\");\n
#include <twr_io.h>\n\nvoid io_printf(twr_ioconsole_t *io, const char *format, ...);\n
"},{"location":"api/api-c-con/#io_begin_draw","title":"io_begin_draw","text":"For addressable display consoles only.
This call (and its matching io_end_draw) are not required. But if you bracket any call sequence that draws to the terminal window with an io_begin_draw
and io_end_draw
, the updates will be batched into one update. This will increase performance and usually prevents the user from seeing partial updates.
io_begin_draw
can be nested.
See the terminal example.
#include <twr_io.h>\n\nvoid io_begin_draw(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#io_end_draw","title":"io_end_draw","text":"For addressable display consoles only.
See io_begin_draw
.
#include <twr_io.h>\n\nvoid io_end_draw(twr_ioconsole_t* io);\n
"},{"location":"api/api-c-con/#deprecated-functions","title":"Deprecated Functions","text":""},{"location":"api/api-c-con/#twr_debugcon","title":"twr_debugcon","text":"This function has been removed. Use stderr
or twr_conlog
.
#include \"twr-crt.h\"\n\ntwr_conlog(\"hello 99 in hex: %x\", 99);\n
or
#include <stdio.h>\n\nfprintf(stderr, \"hello over there in browser debug console land\\n\");\n
"},{"location":"api/api-c-con/#twr_divcon","title":"twr_divcon","text":"This function has been removed.
"},{"location":"api/api-c-con/#twr_windowcon","title":"twr_windowcon","text":"This function has been removed.
"},{"location":"api/api-c-d2d/","title":"2D Draw C API for WebAssembly","text":"This section describes twr-wasm's C D2D API, which allows your WebAssembly module to call many of the JavaScript Canvas APIs.
"},{"location":"api/api-c-d2d/#examples","title":"Examples","text":"Name View Live Link Source Link Bouncing Balls (C++) View bouncing balls Source for balls Pong (C++) View Pong Source for Pong Maze (Win32 C Port) View live maze here Source for maze"},{"location":"api/api-c-d2d/#code-example","title":"Code Example","text":"Draw A Rectangle#include \"twr-draw2d.h\"\n\nvoid square() {\n // batch draw commands, with a maximum of 100 commands before render\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n // set color using CSS color string\n d2d_setfillstyle(ds, \"blue\");\n // draw a the rect\n d2d_fillrect(ds, 10, 10, 100, 100);\n // this will cause the JavaScript thread to render\n d2d_end_draw_sequence(ds);\n}\n
"},{"location":"api/api-c-d2d/#overview","title":"Overview","text":"The Draw 2D APIs are C APIs and are part of the twr-wasm library that you access with #include \"twr-draw2d.h\"
. There is also a C++ canvas wrapper class in examples/twr-cpp
used by the balls and pong examples.
To create a canvas surface, that you can draw to using the twr-wasm 2D C drawing APIs, use the twrConsoleCanvas
class in your JavaScript/HTML (see Consoles Section). Or more simply, if you add a canvas tag to your HTML named twr_d2dcanvas
, the needed twrConsoleCanvas
will be created automatically.
<canvas id=\"twr_d2dcanvas\" width=\"600\" height=\"600\"></canvas>\n//Feel free to change the `width=\"600` and/or `height=\"600` attributes.\n
To draw using the C 2D Draw API:
- call
d2d_start_draw_sequence
(or alternately d2d_start_draw_sequence_with_con
) - call one or more (a sequence) of 2D draw commands, like
d2d_fillrect
- call
d2d_end_draw_sequence
- repeat as desired
d2d_start_draw_sequence
will draw to the default twrConsoleCanvas
, as explained at the start of this section. d2d_start_draw_sequence_with_con
is optional, and allows you to specify the twrConsoleCanvas
to draw to. You would typically get this console in C using the twr_get_console
function (which retrieves a named console that you specified in the io
module option.)
Commands are queued until flushed -- which will take the batch of queued draw commands, and execute them. The 2D draw APIs will work with either twrWasmModule
or twrWasmModuleAsync
. With twrWasmModuleAsync
, the batch of commands is sent from the worker thread over to the JavaScript main thread for execution. By batching the calls between calls to d2d_start_draw_sequence
and d2d_end_draw_sequence
, performance is improved.
d2d_flush
waits for the commands to finish execution before returning. d2d_flush
is called automatically by d2d_end_draw_sequence
and so you generally don't need to call it manually.
You pass an argument to d2d_start_draw_sequence
specifying how many instructions will trigger an automatic call to d2d_flush
. You can make this larger for efficiency, or smaller if you want to see the render progress more frequently. There is no limit on the size of the queue, except memory used in the Wasm module. The d2d_flush
function can be called manually, but this is not normally needed, unless you would like to ensure a sequence renders before your d2d_end_draw_sequence
is called, or before the count passed d2d_start_draw_sequence
is met.
If you are using twrWasmModuleAsync
, or if you are re-rendering the entire frame for each animation update, you should ensure that all of your draws for a complete frame are made without an explicit or implicit call to d2d_flush
in the middle of the draw sequence, as this may cause flashing.
"},{"location":"api/api-c-d2d/#possible-pitfalls","title":"Possible Pitfalls","text":"Some commands have extra details that you need to be aware of to avoid performance loss or bugs.
- Getters, like d2d_measuretext, will flush the queue in order to retrieve the requested data. If your program relies on not flushing early (for example, to avoid flashes), then getters should be avoided in your main render loops.
- putImageData references the provided pointer, so the given image data needs to stay valid on the caller's stack or heap until flush is called.
- getLineDash takes in a buffer_length, double * array (the buffer), and returns the amount of the buffer filled. If there are more line segments than can fit in the buffer_length, a warning is printed and the excess is voided. If you want to know the size before hand for allocation, the getLineDashLength function is available.
"},{"location":"api/api-c-d2d/#notes","title":"Notes","text":"The functions listed below are based on the JavaScript Canvas 2D API (found here). However, there are some slight differences since these APIs are made for C rather than JavaScript. For example some items keep resources stored on the JavaScript side (such as d2d_createlineargradient) which are referenced by a numeric ID , rather than an actual object reference.
Additionally, there are alternative functions like d2d_setstrokestylergba, which calls the same underlying function as d2d_setstrokestyle, but takes in a color as a number rather than CSS style string.
As noted above, putImageData requires that the image data be valid until flush is called.
Other functions that take a string, like d2d_filltext, don't have this same issue because they make a copy of the string argument. These string copies will be automatically freed.
getCanvasPropDouble, getCanvasPropString, setCanvasPropDouble, and setCanvasPropString allow you to change canvas properties by name. If the previous values type is either undefined, a string rather than a number, etc. then it will throw an error so ensure that you have your property names correct.
d2d_load_image is not called like other instructions which rely on d2d_start_draw_sequence. This means it always gets called immediately and doesn't queue up in or flush the instruction queue. This can cause some issues such as the example below. Load Image Pitfall
#include \"twr-draw2d.h\"\nbool has_background = false;\nconst long BACKGROUND_ID = 1;\n//draws the background\nvoid draw_background(struct d2d_draw_seq* ds) {\n assert(has_background);\n d2d_drawimage(ds, BACKGROUND_ID, x, y);\n}\n//loads a new background image\nvoid load_background_image(struct d2d_draw_seq* ds, const char * url) {\n if (has_background) {\n //free previous background\n //this isn't called until the buffer in ds get's flushed.\n // For this program, that doesn't happen until d2d_end_draw_sequence is called,\n // so d2d_load_image processes before d2d_releasid throws a warning and then is deleted when d2d_releaseid\n // is eventually called.\n d2d_releaseid(ds, BACKGROUND_ID);\n //d2d_flush(ds) //by adding a flush like so, it ensures releaseid is called before d2d_load_image\n } else {\n has_background = true;\n }\n d2d_load_image(url, BACKGROUND_ID);\n}\nvoid render() {\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n\n //load background\n load_background_image(ds, \"example_image.com\");\n\n draw_background(ds); //draw it\n\n d2d_end_draw_sequence(ds);\n\n\n struct d2d_draw_seq* ds=d2d_start_draw_sequence(100);\n\n //load new background image\n load_background_image(ds, \"example_image2.com\");\n draw_background(ds);\n\n d2d_end_draw_sequence(ds);\n}\n
"},{"location":"api/api-c-d2d/#functions","title":"Functions","text":"These are the Canvas APIs currently available in C:
struct d2d_draw_seq* d2d_start_draw_sequence(int flush_at_ins_count);\nstruct d2d_draw_seq* d2d_start_draw_sequence_with_con(int flush_at_ins_count, twr_ioconsole_t * con);\nvoid d2d_end_draw_sequence(struct d2d_draw_seq* ds);\nvoid d2d_flush(struct d2d_draw_seq* ds);\nint d2d_get_canvas_prop(const char* prop);\n\nvoid d2d_fillrect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_strokerect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_filltext(struct d2d_draw_seq* ds, const char* str, double x, double y);\nvoid d2d_fillcodepoint(struct d2d_draw_seq* ds, unsigned long c, double x, double y);\nvoid d2d_stroketext(struct d2d_draw_seq* ds, const char* text, double x, double y);\n\nvoid d2d_measuretext(struct d2d_draw_seq* ds, const char* str, struct d2d_text_metrics *tm);\nvoid d2d_save(struct d2d_draw_seq* ds);\nvoid d2d_restore(struct d2d_draw_seq* ds);\n\nvoid d2d_setlinewidth(struct d2d_draw_seq* ds, double width);\nvoid d2d_setstrokestylergba(struct d2d_draw_seq* ds, unsigned long color);\nvoid d2d_setfillstylergba(struct d2d_draw_seq* ds, unsigned long color);\nvoid d2d_setstrokestyle(struct d2d_draw_seq* ds, const char* css_color);\nvoid d2d_setfillstyle(struct d2d_draw_seq* ds, const char* css_color);\nvoid d2d_setfont(struct d2d_draw_seq* ds, const char* font);\nvoid d2d_setlinecap(struct d2d_draw_seq* ds, const char* line_cap);\nvoid d2d_setlinejoin(struct d2d_draw_seq* ds, const char* line_join);\nvoid d2d_setlinedash(struct d2d_draw_seq* ds, unsigned long len, const double* segments);\nunsigned long d2d_getlinedash(struct d2d_draw_seq* ds, unsigned long length, double* buffer);\nunsigned long d2d_getlinedashlength(struct d2d_draw_seq* ds);\nvoid d2d_setlinedashoffset(struct d2d_draw_seq* ds, double line_dash_offset);\n\nvoid d2d_createlineargradient(struct d2d_draw_seq* ds, long id, double x0, double y0, double x1, double y1);\nvoid d2d_createradialgradient(struct d2d_draw_seq* ds, long id, double x0, double y0, double radius0, double x1, double y1, double radius1);\nvoid d2d_addcolorstop(struct d2d_draw_seq* ds, long gradID, long position, const char* csscolor);\nvoid d2d_setfillstylegradient(struct d2d_draw_seq* ds, long gradID);\nvoid d2d_releaseid(struct d2d_draw_seq* ds, long id);\n\nvoid d2d_beginpath(struct d2d_draw_seq* ds);\nvoid d2d_fill(struct d2d_draw_seq* ds);\nvoid d2d_stroke(struct d2d_draw_seq* ds);\nvoid d2d_moveto(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_lineto(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_arc(struct d2d_draw_seq* ds, double x, double y, double radius, double start_angle, double end_angle, bool counterclockwise);\nvoid d2d_arcto(struct d2d_draw_seq* ds, double x1, double y1, double x2, double y2, double radius);\nvoid d2d_bezierto(struct d2d_draw_seq* ds, double cp1x, double cp1y, double cp2x, double cp2y, double x, double y);\nvoid d2d_roundrect(struct d2d_draw_seq* ds, double x, double y, double width, double height, double radii);\nvoid d2d_ellipse(struct d2d_draw_seq* ds, double x, double y, double radiusX, double radiusY, double rotation, double startAngle, double endAngle, bool counterclockwise);\nvoid d2d_quadraticcurveto(struct d2d_draw_seq* ds, double cpx, double cpy, double x, double y);\nvoid d2d_rect(struct d2d_draw_seq* ds, double x, double y, double width, double height);\nvoid d2d_closepath(struct d2d_draw_seq* ds);\n\n//deprecated, use d2d_ctoimagedata instead\nvoid d2d_imagedata(struct d2d_draw_seq* ds, long id, void* mem, unsigned long length, unsigned long width, unsigned long height);\n\nvoid d2d_ctoimagedata(struct d2d_draw_seq* ds, long id, void* mem, unsigned long length, unsigned long width, unsigned long height);\nvoid d2d_putimagedata(struct d2d_draw_seq* ds, long id, unsigned long dx, unsigned long dy);\nvoid d2d_putimagedatadirty(struct d2d_draw_seq* ds, long id, unsigned long dx, unsigned long dy, unsigned long dirtyX, unsigned long dirtyY, unsigned long dirtyWidth, unsigned long dirtyHeight);\n\nvoid d2d_reset(struct d2d_draw_seq* ds);\nvoid d2d_clearrect(struct d2d_draw_seq* ds, double x, double y, double w, double h);\nvoid d2d_scale(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_translate(struct d2d_draw_seq* ds, double x, double y);\nvoid d2d_rotate(struct d2d_draw_seq* ds, double angle);\nvoid d2d_gettransform(struct d2d_draw_seq* ds, struct d2d_2d_matrix *transform);\nvoid d2d_settransform(struct d2d_draw_seq* ds, double a, double b, double c, double d, double e, double f);\nvoid d2d_settransformmatrix(struct d2d_draw_seq* ds, const struct d2d_2d_matrix * transform);\nvoid d2d_transform(struct d2d_draw_seq* ds, double a, double b, double c, double d, double e, double f);\nvoid d2d_transformmatrix(struct d2d_draw_seq* ds, const struct d2d_2d_matrix * transform);\nvoid d2d_resettransform(struct d2d_draw_seq* ds);\n\nbool d2d_load_image(const char* url, long id);\nbool d2d_load_image_with_con(const char* url, long id, twr_ioconsole_t * con);\nvoid d2d_drawimage(struct d2d_draw_seq* ds, long id, double dx, double dy);\nvoid d2d_drawimage_ex(struct d2d_draw_seq* ds, long id, double sx, double sy, double sWidth, double sHeight, double dx, double dy, double dWidth, double dHeight);\nvoid d2d_getimagedata(struct d2d_draw_seq* ds, long id, double x, double y, double width, double height);\nunsigned long d2d_getimagedatasize(double width, double height);\nvoid d2d_imagedatatoc(struct d2d_draw_seq* ds, long id, void* buffer, unsigned long buffer_len);\n\ndouble d2d_getcanvaspropdouble(struct d2d_draw_seq* ds, const char* prop_name);\nvoid d2d_getcanvaspropstring(struct d2d_draw_seq* ds, const char* prop_name, char* buffer, unsigned long buffer_len);\nvoid d2d_setcanvaspropdouble(struct d2d_draw_seq* ds, const char* prop_name, double val);\nvoid d2d_setcanvaspropstring(struct d2d_draw_seq* ds, const char* prop_name, const char* val);\n
d2d_measuretext() returns this structure:
struct d2d_text_metrics {\n double actualBoundingBoxAscent;\n double actualBoundingBoxDescent;\n double actualBoundingBoxLeft;\n double actualBoundingBoxRight;\n double fontBoundingBoxAscent;\n double fontBoundingBoxDescent;\n double width;\n};\n
d2d_get_canvas_prop() returns a value of:
export interface ICanvasProps {\n charWidth: number,\n charHeight: number,\n foreColor: number,\n backColor: number,\n widthInChars: number,\n heightInChars: number,\n canvasWidth:number,\n canvasHeight:number\n}\n
d2d_gettransform() returns this structure:
struct d2d_2d_matrix {\n double a, b, c, d, e, f;\n};\n
d2d_getlinedash() returns this structure:
struct d2d_line_segments {\n long len;\n double *segments;\n};\n
"},{"location":"api/api-c-general/","title":"General C API for Wasm","text":""},{"location":"api/api-c-general/#overview","title":"Overview","text":"This sections describes the \"general\" twr-wasm functions available that don't fit neatly into another category (such as standard C library functions, Draw 2D functions, etc.)
These functions often start with \"twr_\" and are generally found in this include file:
\\twr-wasm\\include\\twr-crt.h
"},{"location":"api/api-c-general/#bzero","title":"bzero","text":"Set a block of memory to zeros. Calls memset(to, 0, count)
.
#include <string.h>\n\nvoid bzero (void *to, size_t count);\n
"},{"location":"api/api-c-general/#getc","title":"getc","text":"This is the standard c library function (see the the standard library docs available on the internet).
Of note this function will return extended ASCII (128-255 inclusive). The extend ASCII are always encoded with Windows-1252 encoding.
See twr_getc32
for a list of related functions.
Note that C character input is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
"},{"location":"api/api-c-general/#twr_atod","title":"twr_atod","text":"Similar to stdlib atof
.
#include \"twr-crt.h\"\n\ndouble twr_atod(const char* str);\n
"},{"location":"api/api-c-general/#twr_atou64","title":"twr_atou64","text":"Convert a string to a 64 bit unsigned integer, stopping when the first non-valid character is encountered. If len is provided, it will be set to the number of characters read. Radix should be >=2 and <=36 -- for example, 10 is a normal base 10 number and 16 is hexadecimal.
#include \"twr-crt.h\"\n\nint64_t twr_atou64(const char *str, int* len, int radix);\n
"},{"location":"api/api-c-general/#twr_dtoa","title":"twr_dtoa","text":"The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
#include \"twr-crt.h\"\n\nvoid twr_dtoa(char* buffer, int sizeInBytes, double value, int max_precision);\n
"},{"location":"api/api-c-general/#twr_cache_mallocfree","title":"twr_cache_malloc/free","text":"These functions keep allocated memory in a cache for much faster re-access than the standard malloc/free.
#include \"twr-crt.h\"\n\nvoid *twr_cache_malloc(twr_size_t size);\nvoid twr_cache_free(void* mem);\n
"},{"location":"api/api-c-general/#twr_code_page_to_utf32_streamed","title":"twr_code_page_to_utf32_streamed","text":"Return a unicode code point (aka utf-32 value) when passed a byte stream that represents an encoded character using the current local's LC_CTYPE code page. A zero is returned if the byte stream has not yet completed a decode.
For example:
int cp\n\nsetlocale(LC_ALL, \"\"); // set to default locale, which will be UTF-8 encoding with local language/region\n\n// turn a UTF-8 Euro into a UTF-32 value\ncp==twr_code_page_to_utf32_streamed(0xE2);\nassert (cp==0);\ncp=twr_code_page_to_utf32_streamed(0x82);\nassert (cp==0);\ncp=twr_code_page_to_utf32_streamed(0xAC);\nassert (cp==0x000020AC); // Euro Code points\n
#include <locale.h>\n\nint twr_code_page_to_utf32_streamed(unsigned char byte) \n
"},{"location":"api/api-c-general/#twr_conlog","title":"twr_conlog","text":"twr_conlog
prints debug messages to stderr
(usually your browser console) from your C code.
#include \"twr-crt.h\"\n\nvoid twr_conlog(char* format, ...);\n
This call is identical to fprintf(stderr, ...)
, except that it adds a newline. When stderr
is set to twrConsoleDebug
each call to twr_conlog() will generate a single call to console.log() in JavaScript to ensure that you see debug prints.
The current implementation does not wait for the debug string to output to the console before returning from twr_conlog, when using twrWasmModuleAsync. In this case, it can take a small bit of time for the string to make its way across the Worker Thread boundary. This is normally not a problem and results in faster performance. But if your code crashes soon after the debug print, the print might not appear. If you think this is an issue, you can call twr_sleep(1)
after your twr_conlog call. This will force a blocking wait for the print to print.
"},{"location":"api/api-c-general/#twr_epoch_timems","title":"twr_epoch_timems","text":"Returns the number of milliseconds since the start of the epoch.
#include \"twr-crt.h\"\n\nuint64_t twr_epoch_timems();\n
"},{"location":"api/api-c-general/#twr_getc32","title":"twr_getc32","text":"Gets a 32 bit unicode code point character from stdin. Unlike the standard C library function getchar
, twr_getc32
does not buffer a line (that is, twr_getc32
will return a character before the user presses Enter).
twr_getc32
is implemented as:
int twr_getc32() {\n return io_getc32(twr_get_stdio_con());\n}\n
Note that stdlib getchar
and ungetc
are not currently implemented.
Note that C character input with these functions is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
Also see:
io_mbgets
- get a multibyte string from a console using the current locale character encoding. Console must support IO_TYPE_CHARREAD. twr_mbgets
- the same as io_mbgets
with the console set to stdin
. io_mbgetc
- get a multibyte character from an IoConsole (like stdin
) using the current locale character encoding getc
(sames as fgetc
) - get a single byte from a FILE * (IoConsole) -- returning ASCII or extended ASCII (window-1252 encoding) io_getc32
- gets a 32 bit unicode code point from an IoConsole (which must support IO_TYPE_CHARREAD)
#include \"twr-crt.h\"\n\nint twr_getc32();\n
"},{"location":"api/api-c-general/#twr_get_navlang","title":"twr_get_navlang","text":"Returns the BCP 47 language tag as found in javacript navigator.language
. If len is not null, it will be filled in with the string length of the language tag.
#include \"twr-crt.h\"\n\nconst char* twr_get_navlang(int *len);\n
"},{"location":"api/api-c-general/#twr_get_current_locale","title":"twr_get_current_locale","text":"extern inline locale_t twr_get_current_locale(void);\n
twr_get_current_locale
will return the locale that has been set by setlocale
. It can be used to pass to a function that takes a locale_t.
"},{"location":"api/api-c-general/#twr_localize_numeric_string","title":"twr_localize_numeric_string","text":"Functions like twr_dtoa
do not localize the decimal point. To get a localized decimal point, you can use printf
, or alternately twr_localize_numeric_string
to post process a string. For example:
char b[10];\nstrcpy(b, \"1.23\");\ntwr_localize_numeric_string(b, twr_get_current_locale());\n// if locale was set to french, then b is now 1,23\n
#include <locale.h>\n\nvoid twr_localize_numeric_string(char* str, locale_t locale);\n
"},{"location":"api/api-c-general/#twr_mem_debug_stats","title":"twr_mem_debug_stats","text":"Print memory map and malloc stats to stderr or stdout.
(note FILE * is the same as twr_ioconsole_t*)
#include <stdio.h>\n\nvoid twr_mem_debug_stats(twr_ioconsole_t* outcon);\n
"},{"location":"api/api-c-general/#twr_mbgets","title":"twr_mbgets","text":"Gets a string from stdin. The string will be in the current locale's character encoding -- ASCII for \"C\", and either UTF-8 or windows-1252 for \"\". See Character Encoding Support with twr-wasm.
#include \"twr-crt.h\"\n\nchar* twr_mbgets(char* buffer);\n
Internally this function uses the stdio IoConsole -- see the IoConsole section for more advanced input/output.
This function will encode characters as specified by the LC_CTYPE category of the current locale. ASCII is used for \"C\", and UTF-8 and Windows-1252 are also supported (see localization)
Note that C character input is blocking and you must use twrWasmModuleAsync -- see stdin for details on how to enable blocking character input.
"},{"location":"api/api-c-general/#twr_mbslen_l","title":"twr_mbslen_l","text":"Returns the number of characters in a string using the character encoding of the passed locale (ASCII for \"C\", UTF-8, or windows-1252 for \"\"). You can use twr_get_current_locale
to find the current locale.
#include <string.h>\n\nsize_t twr_mbslen_l(const char *str, locale_t locale);\n
"},{"location":"api/api-c-general/#twr_sleep","title":"twr_sleep","text":"twr_sleep
is a traditional blocking sleep function. This function is blocking, and so is only available if you use twrWasmModuleAsync
.
#include \"twr-crt.h\"\n\nvoid twr_sleep(int ms);\n
"},{"location":"api/api-c-general/#twr_register_callback","title":"twr_register_callback","text":"Returns a new event ID that is paired with the specified C function. This event ID can be passed to functions that accept an event ID. When the event is triggered, the specified callback is called.
The callback function's first argument will be the event ID. Subsequent arguments are event specific. It is legal to register the same callback for multiple event IDs.
#include \"twr-crt.h\"\n\nint twr_register_callback(const char* func_name);\n
For example:
// timer event callback (called once)\n__attribute__((export_name(\"on_timer1\")))\nvoid on_timer1(int event_id) {\n printf(\"timer callback 1 entered (event id=%d) !\\n\", event_id);\n}\n\n// entry point\n__attribute__((export_name(\"twr_main\")))\nvoid twr_main() {\n int timer1=twr_register_callback(\"on_timer1\");\n twr_timer_single_shot(2000, timer1);\n}\n
"},{"location":"api/api-c-general/#twr_timer_single_shot","title":"twr_timer_single_shot","text":"Triggers the specified event (callback) once after milliSeconds
. Returns a timerID
which can be used with twr_timer_cancel
.
int twr_timer_single_shot(int milliSeconds, int eventID);\n
"},{"location":"api/api-c-general/#twr_timer_repeat","title":"twr_timer_repeat","text":"Triggers the specified event (callback) repeatedly after milliSeconds
. Returns a timerID
which can be used with twr_timer_cancel
.
int twr_timer_repeat(int milliSeconds, int eventID);\n
"},{"location":"api/api-c-general/#twr_timer_cancel","title":"twr_timer_cancel","text":"Cancels the specfied timer.
void twr_timer_cancel(int timerID);\n
"},{"location":"api/api-c-general/#twr_tofixed","title":"twr_tofixed","text":"This function is identical to its JavaScript version.
#include \"twr-crt.h\"\n\nvoid twr_tofixed(char* buffer, int buffer_size, double value, int dec_digits);\n
The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
"},{"location":"api/api-c-general/#twr_toexponential","title":"twr_toexponential","text":"This function is identical to its JavaScript version.
#include \"twr-crt.h\"\n\nvoid twr_toexponential(char* buffer, int buffer_size, double value, int dec_digits);\n
The functions to convert double to text are snprintf
, fcvt_s
,twr_dtoa
, twr_toexponential
, and twr_tofixed
"},{"location":"api/api-c-general/#twr_strhorizflip","title":"twr_strhorizflip","text":"Mirror image the passed in string.
#include \"twr-crt.h\"\n\nvoid twr_strhorizflip(char * buffer, int n);\n
"},{"location":"api/api-c-general/#twr_utf8_char_len","title":"twr_utf8_char_len","text":"Returns the number of bytes in a UTF-8 character (passed as a string pointer). UTF-8 characters can be 1 to 4 bytes in length.
#include <string.h>\n\nint twr_utf8_char_len(const char *str);\n
"},{"location":"api/api-c-general/#twr_utf32_to_code_page","title":"twr_utf32_to_code_page","text":"Takes a utf32 value (aka unicode code point value), and fills in the passed character array buffer with the character encoding of the utf32 value, using the current locale's LC_CTYPE code page. The buffer is 0 terminated.
Also see c32rtomb
and c16rtomb
.
For example:
char strbuf[6]; // max size of utf-8 is 4+terminating zero. Max size of ASCII or windows 1252 is 1 + terminating zero\nsetlocale(LC_ALL, \"\"); // set to default locale, which will be UTF-8 encoding with local language/region\ntwr_utf32_to_code_page(strbuf, 0x000020AC); // encode a Euro code point \nprintf(\"%s\", strbuf); \nassert ( strcmp(strbuf,\"\\xE2\\x82\\xAC\")==0 ); // utf-8 encoding of euro\nassert ( strcmp(strbuf,\"\u20ac\")==0 ); // clang string literals default to utf-8 encoding\n
include <locale.h>\n\nvoid twr_utf32_to_code_page(char* out, int utf32)\n
"},{"location":"api/api-c-general/#twr_vprintf","title":"twr_vprintf","text":"Performs a printf by calling the callback with cbdata for each character.
#include \"twr-crt.h\"\n\nvoid twr_vprintf(twr_cbprintf_callback out, void* cbdata, const char *format, va_list* args);\n
"},{"location":"api/api-c-general/#floating-math-helpers","title":"floating math helpers","text":"int twr_isnan(double v);\nint twr_isinf(double v);\ndouble twr_nanval();\ndouble twr_infval();\n
"},{"location":"api/api-c-localization/","title":"Localization Reference for twr-wasm","text":"This section details twr-wasm's WebAssembly localization support.
Also see Introduction to Character Encoding Support with twr-wasm
"},{"location":"api/api-c-localization/#using-c","title":"Using C:","text":"Standard C locale functions are supported by twr-wasm. ASCII, UTF-8 and windows-1252 encoding is supported by the twr-wasm standard C library locale. twr-wasm also includes C functions for UTF-32 support.
"},{"location":"api/api-c-localization/#using-c_1","title":"Using C++:","text":" - libc++ locale and unicode functions are supported by twr-wasm.
- libc++ unicode support includes utf-16 and utf-32 strings.
"},{"location":"api/api-c-localization/#character-encodings","title":"Character Encodings","text":"twr-wasm C locales support ASCII, UTF-8 or windows-1252 encoding. UTF-16/32 are not supported as a std c lib locale setting, but functions are provided to convert utf-32 (unicode code points) to and from ASCII, UTF-8, and windows-1252 \"code pages\" (there are other miscellaneous utf-32 based functions as well.)
"},{"location":"api/api-c-localization/#locales-standard-c-library","title":"Locales (Standard C Library)","text":""},{"location":"api/api-c-localization/#c","title":"\"C\"","text":"\"C\" is the default locale, as usual. When \"C\" is selected, the functions operate as usual. One subtly is that console i/o functions (such as printf
) will generally function as expected with UTF-8, since the div
and window
consoles correctly handle UTF-8 character encoding. This is normal on some OSs, such as linux, but not the default on Windows (which often defaults to windows-1252 for backward compatibility).
isgraph
style functions will only recognize ASCII characters, as is normal. Functions such as strcmp
operate on the byte sequence, which will typically results in UTF-8 codes being compared lexically. strcoll
will use lexical ordering.
"},{"location":"api/api-c-localization/#posix","title":"\"POSIX\"","text":"\"POSIX\" is the same as \"C\"
"},{"location":"api/api-c-localization/#_1","title":"\"\"","text":"\"\" is the locale to specify the users default setting (this selects the setting used by the browser). This will also enable UTF-8 in functions such as strcoll
. For example, if your browser is set to \"en-US\" as its default locale, setlocale(LC_ALL, \"\")
will return en-US.UTF-8
.
isgraph
style functions will still only recognize ASCII characters (since UTF-8 doesn't encode any single bytes greater than 127). strcoll
uses locale specific ordering, and printf
will use locale specific decimal points. strcmp
still compares two strings lexicographically (byte-by-byte) without considering locale-specific rules, per the spec.
"},{"location":"api/api-c-localization/#utf-8","title":"\".UTF-8\"","text":"\".UTF-8\" is the same as \"\" with twr-wasm.
"},{"location":"api/api-c-localization/#1252","title":"\".1252\"","text":"\".1252\" will select the current default locale, but use windows-1252 character encoding (instead of UTF-8). Windows-1252 is a super set of ISO-8859-1 and is the most commonly used encoding for many european languages when unicode is not used. This mode is primarily for legacy software, backwards compatibly, and windows compatibility.
"},{"location":"api/api-c-localization/#others","title":"Others","text":"Setting arbitrary locales, such as \"fr-FR\" when the browser is defaulted to another locale, is not supported.
"},{"location":"api/api-c-localization/#select-the-default-locale","title":"Select the default locale","text":"To select the user's browser's default locale using the C language, and enable consistent utf-8 support, use a call like this:
setlocale(LC_ALL, \"\")\n
"},{"location":"api/api-c-localization/#c-and-libc-functions","title":"C and libc++ functions","text":"If you are using twr-wasm's build of libc++, libc++ locale and unicode functions work as normal.
The usual standard C library locale support is available, along with some POSIX extensions. In addition, some locale useful twr-wasm specific functions are documented in C API, such as twr_get_current_locale
,twr_mbgets
, twr_getc32
, twr_utf8_char_len
, twr_mbslen_l
, twr_utf32_to_code_page
, twr_code_page_to_utf32_streamed
, twr_get_navlang
, twr_localize_numeric_string
.
Note that io_getc32
, getc(stdin)
, fgetc(stdin)
do not look at the current locale. io_getc32
returns a 32 bit unicode code point, and getc
/fgetc
return extended ASCII.
For a locale aware character input, use io_mbgetc()
or twr_mbgets()
. Both use the locale category LC_CTYPE. See C API.
Note that when the locale is not set (or whenever the \"C\" locale is set) functions that get character(s) from stdin that are locale aware, like twr_mbgets()
, behave different than functions that output characters to stdout (like puts
, io_putstr
, io_putc
, putchar
). Characters to stdout in \"C\" locale will handle UTF-8 characters. For stdin, \"C\" locale uses ASCII.
For consistent UTF-8 (or windows-1252) behavior, set the locale as discussed above ( use setlocale
)
The primary standard C library locale functions are:
char* setlocale(int category, const char* locale);\nstruct lconv *localeconv(void);\n
As well as the two standard library functions above, appropriate functions take into account the current locale (printf, strcoll, etc).
Note that setlocale
returns a string using BCP 47 format (like a web browser). Locale strings look like \"en-US.UTF-8\", instead of \"en_US.UTF-8\". A dash, not an underscore, is used as a separator.
POSIX functions These are the extended POSIX style functions provided that are related to locale:
locale_t newlocale(int category_mask, const char *locale, locale_t base);\nlocale_t uselocale(locale_t);\nvoid freelocale(locale_t);\nlocale_t duplocale(locale_t);\n\nint isalnum_l(int c, locale_t loc);\nint isalpha_l(int c, locale_t loc);\nint isblank_l(int c, locale_t loc);\nint iscntrl_l(int c, locale_t loc);\nint isdigit_l(int c, locale_t loc);\nint isgraph_l(int c, locale_t loc);\nint islower_l(int c, locale_t loc);\nint isprint_l(int c, locale_t loc);\nint ispunct_l(int c, locale_t loc);\nint isspace_l(int c, locale_t loc);\nint isupper_l(int c, locale_t loc);\nint isxdigit_l(int c, locale_t loc);\nint tolower_l(int c, locale_t loc);\nint toupper_l(int c, locale_t loc);\n\nlong long strtoll_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long long strtoull_l(const char *str, char **str_end, int base, locale_t loc);\nfloat strtof_l(const char *str, char ** str_end, locale_t locale);\ndouble strtod_l(const char *str, char **str_end, locale_t locale);\nlong double strtold_l(const char *str, char **str_end, locale_t locale);\n\nint strcoll_l(const char* lhs, const char* rhs, locale_t loc);\n\nsize_t strftime_l(char *s, size_t maxsize, const char *format, const struct tm *timeptr, locale_t locale);\n
"},{"location":"api/api-c-stdlib/","title":"Standard C library for WebAssembly","text":"This section describes twr-wasm's support for the Standard C Library. twr-wasm includes its own implementation of the standard C library optimized for WebAssembly and Wasm running in a web browser. This is a core feature of twr-wasm.
For documentation of these functions, see the many standard C library documentation web sites.
The following subset of the standard C library is available. Also see twr-wasm/include
folder for include files.
"},{"location":"api/api-c-stdlib/#stdioh","title":"stdio.h","text":"* fprintf will only work with these -- stderr, stdin, stdout */\n/* these return 'twr_ioconsole_t *' which is same as 'FILE *' */\n#define stderr (FILE *)(twr_get_stderr_con())\n#define stdin (FILE *)(twr_get_stdio_con())\n#define stdout (FILE *)(twr_get_stdio_con())\n\nint snprintf(char *buffer, size_t bufsz, const char *format, ... );\nint sprintf( char *buffer, const char *format, ... );\nint vsnprintf(char *buffer, size_t bufsz, const char *format, va_list vlist);\nint vasprintf(char **strp, const char* format, va_list vlist );\nint printf(const char* format, ...);\nint vprintf(const char* format, va_list vlist );\nint puts(const char *str);\nint putchar(int c);\n\ntypedef twr_ioconsole_t FILE; \nint vfprintf(FILE *stream, const char *format, va_list vlist);\nint fprintf(FILE *stream, const char* format, ...);\nsize_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);\nint ferror(FILE *stream);\nint feof(FILE *stream);\nint fflush(FILE *stream);\nint is_terminal(FILE *stream);\nint fputc(int ch, FILE* stream);\nint putc(int ch, FILE* stream);\nint fgetc(FILE *stream );\nint getc(FILE *stream);\n
"},{"location":"api/api-c-stdlib/#stdlibh","title":"stdlib.h","text":"void *malloc(size_t size);\nvoid free(void *mem);\nsize_t avail(void);\nvoid *realloc( void *ptr, size_t new_size );\nvoid* calloc( size_t num, size_t size );\nvoid *aligned_alloc( size_t alignment, size_t size );\n\nint rand(void);\nvoid srand(int seed);\n\n#define __min(a,b) (((a) < (b)) ? (a) : (b))\n#define __max(a,b) (((a) > (b)) ? (a) : (b))\n\nint abs(int n);\n\nint _fcvt_s(\n char* buffer,\n size_t sizeInBytes,\n double value,\n int fracpart_numdigits,\n int *dec,\n int *sign\n);\ndouble atof(const char* str);\nint atoi(const char *str);\nlong atol( const char *str );\nlong long atoll( const char *str );\nlong strtol(const char *str, char **str_end, int base);\nlong long strtoll(const char *str, char **str_end, int base);\nlong long strtoll_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long long strtoull(const char *str, char **str_end, int base);\nunsigned long long strtoull_l(const char *str, char **str_end, int base, locale_t loc);\nunsigned long strtoul(const char *str, char ** str_end, int base);\nfloat strtof(const char *str, char ** str_end);\nfloat strtof_l(const char *str, char ** str_end, locale_t locale);\ndouble strtod(const char *str, char **str_end);\ndouble strtod_l(const char *str, char **str_end, locale_t locale);\nlong double strtold(const char *str, char **str_end);\nlong double strtold_l(const char *str, char **str_end, locale_t locale);\nint _itoa_s(int64_t value, char * buffer, size_t size, int radix);\n\ndiv_t div( int x, int y );\nldiv_t ldiv( long x, long y );\nlldiv_t lldiv( long long x, long long y );\n\n_Noreturn void abort(void);\nint atexit(void (*func)(void));\n
Note that _fcvt_s as currently enabled has these limitations: - fractional digits <=100 - values must be less than 1e+21 - values negative exponents must be smaller than 1e-99
There is a full featured version of _fcvt_s in the source code, but it is not currently enabled, since the version enabled is smaller and works in most use cases.
"},{"location":"api/api-c-stdlib/#asserth","title":"assert.h","text":"void assert(int expression);\n
"},{"location":"api/api-c-stdlib/#mathh","title":"math.h","text":"double acos(double arg);\ndouble asin(double arg);\ndouble atan(double arg);\ndouble atan2(double y, double x);\ndouble ceil(double arg);\ndouble cos(double arg);\ndouble exp(double arg);\ndouble fabs(double arg);\ndouble floor(double arg);\ndouble fmod(double x, double y);\ndouble log(double arg);\ndouble pow(double base, double exp);\ndouble sin(double arg);\ndouble sqrt(double arg);\ndouble tan(double arg);\ndouble trunc(double arg);\n
"},{"location":"api/api-c-stdlib/#stdargh","title":"stdarg.h","text":"#define va_start(v,l) __builtin_va_start(v,l)\n#define va_end(v) __builtin_va_end(v)\n#define va_arg(v,l) __builtin_va_arg(v,l)\n#define va_copy(d,s) __builtin_va_copy(d,s)\ntypedef __builtin_va_list va_list;\n
"},{"location":"api/api-c-stdlib/#ctypeh","title":"ctype.h","text":"int isascii(int);\nint toascii(int);\nint isalnum(int c);\nint isalpha(int c);\nint isblank(int);\nint iscntrl(int);\nint isdigit(int c);\nint isgraph(int c);\nint islower(int);\nint isprint(int);\nint ispunct(int);\nint isspace(int c);\nint isupper(int);\nint isxdigit(int);\nint tolower(int c);\nint toupper(int c);\n\nint isalnum_l(int c, locale_t loc);\nint isalpha_l(int c, locale_t loc);\nint isblank_l(int c, locale_t loc);\nint iscntrl_l(int c, locale_t loc);\nint isdigit_l(int c, locale_t loc);\nint isgraph_l(int c, locale_t loc);\nint islower_l(int c, locale_t loc);\nint isprint_l(int c, locale_t loc);\nint ispunct_l(int c, locale_t loc);\nint isspace_l(int c, locale_t loc);\nint isupper_l(int c, locale_t loc);\nint isxdigit_l(int c, locale_t loc);\nint tolower_l(int c, locale_t loc);\nint toupper_l(int c, locale_t loc);\n
"},{"location":"api/api-c-stdlib/#stddefh","title":"stddef.h","text":"#define offsetof(TYPE, MEMBER) __builtin_offsetof (TYPE, MEMBER)\ntypedef __PTRDIFF_TYPE__ ptrdiff_t;\ntypedef double max_align_t;\n
"},{"location":"api/api-c-stdlib/#stringh","title":"string.h","text":"size_t strlen(const char * str);\nchar *strdup(const char * source);\nchar *strcpy(char *dest, const char *source);\nint strcat_s(char *dest, size_t destsz, const char *src);\nchar* strcat(char *dest, const char *src);\nchar *strncpy(char *dest, const char *source, size_t count);\nint strcmp(const char* string1, const char* string2);\nint strncmp(const char* lhs, const char* rhs, size_t count);\nint stricmp(const char* string1, const char* string2);\nint strnicmp(const char* string1, const char* string2, size_t count);\nint strcoll(const char* lhs, const char* rhs);\nint strcoll_l(const char* lhs, const char* rhs, locale_t loc);\nchar *strchr(const char *str, int ch);\nvoid *memchr(const void *ptr, int ch, size_t count);\nchar *strstr(const char *haystack, const char *needle);\nchar * strerror(int errnum );\nchar * _strerror(const char *strErrMsg);\nvoid *memmove(void *dest, const void *src, size_t n);\nint memcmp( const void* lhs, const void* rhs, size_t count );\nvoid bzero (void *to, size_t count);\n\n// implemented in memcpy.wat\nvoid *memcpy(void *dest, const void * src, size_t n);\nvoid *memset(void *mem, int c, size_t n);\n
"},{"location":"api/api-c-stdlib/#timeh","title":"time.h","text":"typedef unsigned long time_t;\nunsigned long time(unsigned long *time);\nsize_t strftime(char *s, size_t maxsize, const char *format, const struct tm *timeptr);\nsize_t strftime_l(char *s, size_t maxsize, const char *format, const struct tm *timeptr, locale_t locale);\nstruct tm *localtime(const time_t *timer);\nint gettimeofday(struct timeval *tv, void* notused);\n#define timerisset(tvp) ((tvp)->tv_sec || (tvp)->tv_usec)\n#define timercmp(tvp,uvp,cmp) \\\n ((tvp)->tv_sec cmp (uvp)->tv_sec || \\\n ((tvp)->tv_sec == (uvp)->tv_sec && (tvp)->tv_usec cmp (uvp)->tv_usec))\n#define timerclear(tvp) (tvp)->tv_sec = (tvp)->tv_usec = 0\n
"},{"location":"api/api-c-stdlib/#localeh","title":"locale.h","text":"#define LC_GLOBAL_LOCALE twr_get_current_locale()\nchar* setlocale(int category, const char* locale);\nstruct lconv *localeconv(void);\nlocale_t newlocale(int category_mask, const char *locale, locale_t base);\nlocale_t uselocale(locale_t);\nvoid freelocale(locale_t);\nlocale_t duplocale(locale_t);\nextern inline locale_t twr_get_current_locale(void);\n
"},{"location":"api/api-c-stdlib/#ucharh","title":"uchar.h","text":"typedef uint_least32_t char32_t;\ntypedef uint_least16_t char16_t;\n\nsize_t c32rtomb( char* s, char32_t c32, mbstate_t* ps );\n
"},{"location":"api/api-c-stdlib/#errnoh","title":"errno.h","text":"typedef int errno_t;\n\nextern int * _errno(void);\n#define errno (*_errno())\n\nerrno_t _set_errno(int _Value);\nerrno_t _get_errno(int *_Value);\n
"},{"location":"api/api-c-stdlib/#_stdtypesh","title":"_stdtypes.h","text":"// don't include directly -- included by various .h files
typedef unsigned long size_t;\n#define MAX_SIZE_T 2147483647 \n\n#ifdef __cplusplus\n#define NULL __null\n#else\n#define NULL ((void*)0)\n#endif\n\ntypedef struct __locale_t_struct * locale_t;\n
"},{"location":"api/api-c-stdlib/#other-include-files-available","title":"Other include files available","text":"float.h\nlimits.h\nstdbool.h\nstdint.h\n
"},{"location":"api/api-libcpp/","title":"libc++ for WebAssembly","text":"This section describes twr-wasm's support for using the standard c++ library libc++ with WebAssembly.
twr-wasm includes libc++ built for WebAssembly in the twr-wasm/lib-c
folder.
For C++ the use of libc++ is optional. That is you can build twr-wasm projects in C++ with or without libc++.
See the examples tests-libcx and tests-user for examples of using libc++.
See the balls example for how to create a C++ WebAssembly program without the standard C++ library. The primary advantage to this approach is a bit smaller code size. You don't need to staticly link libc++.
Some of the key options twr-wasm's libc++ for WebAssembly was built with are these:
DLIBCXX_ENABLE_LOCALIZATION=ON \nDLIBCXX_ENABLE_UNICODE=ON \nDLIBCXX_ENABLE_RTTI=ON \nDLIBCXX_ENABLE_STATIC_ABI_LIBRARY=ON \n\nDCMAKE_BUILD_TYPE=Release \nDCMAKE_CXX_STANDARD=20 \n\nDLIBCXX_ENABLE_EXCEPTIONS=OFF \nDLIBCXX_ENABLE_THREADS=OFF \nDLIBCXX_ENABLE_SHARED=OFF \nDLIBCXX_ENABLE_WIDE_CHARACTERS=OFF \nDLIBCXX_ENABLE_FILESYSTEM=OFF \nDLIBCXX_ENABLE_TIME_ZONE_DATABASE=OFF \nDLIBCXX_ENABLE_MONOTONIC_CLOCK=OFF \nDLIBCXX_ENABLE_RANDOM_DEVICE=OFF\n
"},{"location":"api/api-ts-consoles/","title":"Console Classes","text":"This section describes the twr-wasm TypeScript/JavaScript classes that you use to create I/O Consoles for character streaming, a terminal, or 2D Canvas Drawing
The classes twrConsoleDiv
, twrConsoleTerminal
, twrConsoleDebug
, and twrConsoleCanvas
create consoles that enable user i/o. Your C/C++ can direct user interactive i/o to these consoles.
"},{"location":"api/api-ts-consoles/#related-console-documentation","title":"Related Console Documentation","text":" - Console Introduction
- Console C APIs
"},{"location":"api/api-ts-consoles/#class-twrconsoledebug","title":"class twrConsoleDebug","text":"twrConsoleDebug
streamings characters to the browser debug console.
C type: IO_TYPE_CHARWRITE
There are no constructor parameters.
"},{"location":"api/api-ts-consoles/#class-twrconsolediv","title":"class twrConsoleDiv","text":"twrConsoleDiv
streams character input and output to a div tag .
C type: IO_TYPE_CHARREAD
and IO_TYPE_CHARWRITE
The div tag will expand as you add more text (via printf, etc).
You pass a <div>
element to use to render the Console to to the twrConsoleDiv
constructor. For example:
<div id=\"div1\" tabindex=\"0\"></div>\n\n<script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleDiv} from \"twr-wasm\";\n\n const stream1Element=document.getElementById(\"div1\");\n\n // adding keyDown events is needed if the console will accept key input\n // don't forget to set \"tabindex\" in your tag, otherwise it won't get key events\n stream1Element.addEventListener(\"keydown\",(ev)=>{stream1.keyDown(ev)});\n\n const stream1 = new twrConsoleDiv(stream1Element);\n const mod = new twrWasmModuleAsync( {stdio: stream1} );\n // mod.callC would go here...\n</script>\n
There are constructor options to set the color and font size. You can also set these directly in the HTML for your <div>
tag. If you wish to change the default font, set the font in the div
tag with the normal HTML tag options.
twrConsoleDiv constructor optionsconstructor(element:HTMLDivElement, params:IConsoleDivParams)\n\nexport interface IConsoleDivParams {\n foreColor?: string,\n backColor?: string,\n fontSize?: number,\n}\n
You can use the putStr
member function to print a string to the div console in JavaScript.
"},{"location":"api/api-ts-consoles/#class-twrconsoleterminal","title":"class twrConsoleTerminal","text":"twrConsoleTerminal
provides streaming and addressable character input and output. A <canvas>
tag is used to render into.
C types: IO_TYPE_CHARREAD
, IO_TYPE_CHARWRITE
, IO_TYPE_ADDRESSABLE_DISPLAY
twrConsoleTerminal is a simple windowed terminal and supports the same streamed output and input features as a does twrConsoleDiv
, but also supports x,y coordinates, colors, and other features. The window console supports chunky (low res) graphics (each character cell can be used as a 2x3 graphic array).
The canvas width and height, in pixels, will be set based on your selected font size and the width and height (in characters) of the terminal. These are passed as constructor options when you instantiate the twrConsoleTerminal
.
You can use the putStr
member function on twrConsoleTerminal to print a string to the terminal in JavaScript.
As you add more text (via printf, etc), the twrConsoleTerminal
will scroll if it becomes full (unlike twrConsoleDiv
, which expands)
A list of C functions that operate on twrConsoleTerminal
are available.
Here is an example:
<body>\n\n <canvas id=\"canvas1forterm\" tabindex=\"0\"></canvas>\n\n <script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleTerminal} from \"twr-wasm\";\n\n // find the HTML elements that we will use for our console to render into\n const term1Element=document.getElementById(\"canvas1forterm\");\n\n // adding keyDown events is needed if the console will accept key input\n // don't forget to set \"tabindex\" in your tag, otherwise it won't get key events\n term1Element.addEventListener(\"keydown\",(ev)=>{term1.keyDown(ev)});\n\n // create the console\n const term1 = new twrConsoleTerminal(term1Element, {widthInChars: 50, heightInChars: 20});\n\n const amod = new twrWasmModuleAsync( \n {io:{\n stdio: debug, stderr: debug, stream1: stream1, stream2: stream2, term1: term1, term2: term2, draw1: draw1, draw2: draw2\n }} );\n\n // set the input focus so user doesn't have to click\n stream1Element.focus();\n\n // load the wasm code and call the multi C function\n await amod.loadWasm(\"./multi-io.wasm\");\n await amod.callC([\"multi\"]);\n\n // example of using a console in in JavaScript\n stream1.putStr(`Hello stream1 of type ${stream1.getProp(\"type\")} from JavaScript!\\n`);\n\n </script>\n</body>\n
twrConsoleTerminal constructor optionsconstructor (canvasElement:HTMLCanvasElement, params:IConsoleTerminalParams)\n\n// see twrConsoleDiv options elsewhere, which are also supported\nexport interface IConsoleTerminalParams extends IConsoleDivParams {\n widthInChars?: number,\n heightInChars?: number,\n}\n
"},{"location":"api/api-ts-consoles/#class-twrconsolecanvas","title":"class twrConsoleCanvas","text":"twrConsoleCanvas
creates a 2D drawing surface that the Canvas compatible 2d drawing APIs can be used with.
C type: IO_TYPE_CANVAS2D
.
constructor(element:HTMLCanvasElement)\n
twrConsoleCanvas Example<body>\n canvas id=\"canvas1for2d\"></canvas>\n\n <script type=\"module\">\n import {twrWasmModule, twrConsoleCanvas} from \"twr-wasm\";\n\n // find the HTML elements that we will \n // use for our console to render into\n const draw1Element=document.getElementById(\"canvas1for2d\");\n\n // create the console\n const draw1 = new twrConsoleCanvas(draw1Element);\n\n const mod = new twrWasmModule( {io: {std2d: draw1} }} );\n\n // callC here...\n </script>\n
"},{"location":"api/api-ts-library/","title":"twr-wasm Libraries","text":"twr-wasm Libraries are used to expose TypeScript code to C/C++ as C APIs. All of the twr-wasm C APIs are implemented with twr-wasm libraries. You can also use a library to implement your own C APIs using TypeScript.
There are two kinds of Libraries:
- Those that have only once instance (such as the math library)
- Those that can have multiple instances across one or more library types, where each library type implements the same interface. Consoles are an example of this (see interfaceName).
"},{"location":"api/api-ts-library/#basic-steps","title":"Basic Steps","text":"twr-wasm Libraries support both twrWasmModule
and twrWasmModuleAsync
. That is, when you create a twrLibrary, it will function with either type of module. In many cases no extra work is needed for the twrWasmModuleAsync
, but in some cases, extra code is needed.
The class twrLibrary
provides the core functionality:
- Support for functions to be imported into the WebAssembly Module
- Support for
twrWasmModuleAsync
proxy Web Worker thread - An event framework, allowing you to post events to WebAssembly C code.
To implement a twr-wasm library you:
- create a new TypeScript class that extends
class twrLibrary
. - create a C .h file for your new functions (with function signatures)
- add one or more functions to your TypeScript class
- add the functions to the
import
object (that is part of your class) - consider if special handling is needed for
twrWasmModuleAsync
(more on this below)
"},{"location":"api/api-ts-library/#lib-example","title":"Lib Example","text":"See the lib
example here for a more complete example which shows how each of the different use cases can be handled. In addition, you can look in /source/twr-ts
for files that start with twrlib*
or twrcon*
for examples.
"},{"location":"api/api-ts-library/#example-twrlibtimer","title":"Example twrLibTimer","text":"The following code is from the twr-wasm source for twrlibtimer.
twr_timer_single_shot
- sends an event to C after the timer times out. twr_sleep
- blocks C execution for a period of time.
import {IWasmModule,} from \"./twrmod.js\"\nimport {IWasmModuleAsync} from \"./twrmodasync.js\"\nimport {twrLibrary, TLibImports, twrLibraryInstanceRegistry} from \"./twrlibrary.js\"\n\n// Libraries use default export\nexport default class twrLibTimer extends twrLibrary {\n id: number;\n imports:TLibImports = {\n twr_timer_single_shot:{},\n twr_sleep:{isAsyncFunction: true, isModuleAsyncOnly: true},\n };\n\n libSourcePath = new URL(import.meta.url).pathname;\n\n constructor() {\n // all library constructors should start with these two lines\n super();\n this.id=twrLibraryInstanceRegistry.register(this);\n }\n\n twr_timer_single_shot(callingMod:IWasmModule|IWasmModuleAsync, milliSeconds:number, eventID:number) {\n setTimeout(()=>{\n callingMod.postEvent(eventID)\n }, milliSeconds); \n }\n\n async twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number) {\n const p = new Promise<void>( (resolve)=>{\n setTimeout(()=>{ resolve() }, milliSeconds); \n });\n\n return p;\n }\n\n}\n
"},{"location":"api/api-ts-library/#c-header-files","title":"C Header Files","text":"You need to create a .h file that provides signatures to the C users of your new API. For example, for this library your .h file would be this:
__attribute__((import_name(\"twr_timer_single_shot\"))) void twr_timer_single_shot(int ms, int event_id);\n__attribute__((import_name(\"twr_sleep\"))) void twr_sleep(int ms);\n
The purpose of import_name
code is to export your functions from WebAssembly to JavaScript. These are an equivalent alternative to adding the functions to an wasm-ld
-export
option.
"},{"location":"api/api-ts-library/#registering-your-api","title":"Registering your API","text":"To register you class so that the APIs are available to C code, you use code akin to this in your index.html
(or similar):
import twrLibTimerMod from \"./twrlibtimer.js\" // default export\n\nnew twrLibTimerMod(); // will register itself\n
If you are a contributor to twr-wasm and plan to add your library as new built-in APIs, add the registration to twrLibBultins.ts
"},{"location":"api/api-ts-library/#example-function-explained","title":"Example Function Explained","text":"Here is what is happening in this code:
imports:TLibImports = {\n twr_timer_single_shot:{},\n}\n\n// this function will work in both twrWasmModule and twrWasmModuleAsync\ntwr_timer_single_shot(callingMod:IWasmModule|IWasmModuleAsync, milliSeconds:number, eventID:number) {\n setTimeout(()=>{\n callingMod.postEvent(eventID)\n }, milliSeconds); \n}\n
twr_timer_single_shot
is listed in the imports
class member variable which causes the function twr_timer_single_shot
to be added to the WebAssembly.ModuleImports imports.
The argument for callingMod:IWasmModule|IWasmModuleAsync
is filled in by the twrLibrary
code -- the calling C function does not include this as an argument. All Parameters following callingMod
are passed by the C code calling the function.
twr_timer_single_shot
creates a JavaScript timer. When the timer completes, an event is posted, which will trigger a callback in the C code.
"},{"location":"api/api-ts-library/#events","title":"Events","text":"To receive an Event, the C code needs to register an event callback, and then pass the event ID to the function that will generate events. Like this:
__attribute__((export_name(\"on_timer\")))\nvoid on_timer(int event_id) {\n printf(\"timer callback 2 entered (event id=%d)\\n\", event_id);\n}\n\n__attribute__((export_name(\"twr_main\")))\nvoid twr_main() {\n int timer1=twr_register_callback(\"on_timer\");\n twr_timer_single_shot(2000, timer);\n
In this example, on_timer
will be called after 2 seconds.
__attribute__((export_name(\"on_timer\")))
is needed because this C function is exported out of the WebAssembly module and into the JavaScript code. JavaScript needs to call this function.
In this example, the event does not have any arguments. But it may -- integers (which includes pointers) can be accepted as arguments to the event callback. These arguments are event specific.
"},{"location":"api/api-ts-library/#imports","title":"imports","text":"All TypeScript functions that you wish to import into a WebAssembly module as C APIs, should be listed in the imports
object.
Each function in the imports
object has optional options, that are primarily for use with twrWasmModuleAsync
modules.
imports
are added to WebAssembly.ModuleImports imports.
"},{"location":"api/api-ts-library/#callingmod","title":"callingMod","text":"Each function listed in the import
section will be passed a module as the first parameter. In general, a function should be written to handle being called either with IWasmModule
or IWasmModuleAsync
as the calling module interface ( callingMod:IWasmModule|IWasmModuleAsync
). This is generally straight forward.
Examples that might cause some extra work, and that are covered below, are:
- Implementing blocking functions like
twr_sleep
- Allocating memory, for example, to return a string
"},{"location":"api/api-ts-library/#numbers-only","title":"Numbers Only","text":"All of the parameters received by an import
function need to be numbers. These functions interface directly with the WebAssembly module with no conversion. If you are passing or returning strings, or accessing structures, you will need to use the data access functions that are provided in callingMod.memWasm
(more on this below). The general issue and approach is explained in this document..
"},{"location":"api/api-ts-library/#memwasm","title":"memWasm","text":"A callingMod
member function that you may need to use is memWasm
(callingMod.memWasm
). memWasm
is used to access data in the WebAssembly Memory. This will happen when you need to dereference a pointer, access strings, or access structures. See wasmMem
documentation here.
memWasm
is exposed by both IWasmModule
and IWasmModuleAsync
.
IWasmModule
exposes wasmMem: IWasmMemory
IWasmModuleAsync
exposes wasmMem: IWasmMemoryAsync
.
If you wish to write a function that accesses the async
PutXXX
functions, you should use the isAsyncFunction: true
option.
"},{"location":"api/api-ts-library/#twrwasmmodule-and-twrwasmmoduleasync","title":"twrWasmModule
and twrWasmModuleAsync
.","text":"twrLibrary's are designed to work with either twrWasmModule
and twrWasmModuleAsync
. (Recall that twr-wasm has two different module types: twrWasmModule
and twrWasmModuleAsync
).
twrWasmModule
runs in the JavaScript main thread, and thus all functions that it exposes are asynchronous -- meaning that they should return relatively quickly and not block execution. twrWasmModuleAsync
runs in a JavaScript Worker thread , and thus all functions that it exposes are synchronous -- meaning they can block C execution. The \"Async\" in twrWasmModuleAsync
refers to the fact that javaScript can await
on twrWasmModuleAsync
blocking APIs. It takes blocking APIs and makes them \"asynchronous\" to the JavaScript main thread.
Although many functions can be listed in imports
and written without worrying about which type of module is using them, this isn't always true. Some tomes extra code or thought is needed to have optimal APIs for twrWasmModuleAsync
.
In the above example, twr_timer_single_shot
will work correctly with both twrWasmModule
and twrWasmModuleAsync
. It is an asynchronous function -- meaning that it returns quickly.
However, the function twr_sleep
blocks C code, and will only work with twrWasmModuleAsync
.
"},{"location":"api/api-ts-library/#twrwasmmoduleasync-thread-structure","title":"twrWasmModuleAsync
thread structure","text":"To understand more clearly why twrWasmModuleAsync
might need more attention, it is helpful to understand how it is allocates task between its two threads: the JavaScript main thread, and a worker thread.
"},{"location":"api/api-ts-library/#the-twr_sleep-function-is-used-to-illustrate-thread-structure","title":"The twr_sleep
function is used to illustrate thread structure","text":"The function twr_sleep_async
will only function with twrWasmModuleAsync
because it causes the C code to block. For example, twr_sleep
can be used like this:
printf(\"going to sleep...\");\ntwr_sleep(1000);\nprintf(\"awake!\\n\");\n
"},{"location":"api/api-ts-library/#twrwasmmoduleasync-uses-two-threads","title":"twrWasmModuleAsync
uses Two Threads","text":"ThetwrWasmModuleAsync
consists of two threads: The JavaScript main thread, and the Web Worker. By default the code for your library functions is always executed in the JavaScript main thread. In the Web Worker, an internal class called twrWasmModuleAsyncProxy
executes. The default execution (unless isCommonCode
is specified -- which is explained later), happens like this:
- in C: twr_sleep() is called
- in
twrWasmModuleAsyncProxy
, a message is sent to the JavaScript main thread, requesting execution of the twrLibTimer.twr_sleep
function. twrWasmModuleAsyncProxy
is paused (thus twr_sleep
is blocking), waiting for a response to the message sent ins step 2. - The JavaScript main thread receives the message, and
awaits
on twrLibTimer.twr_sleep
- When the Promise that is being awaited on resolves, the JavaScript main threads sends a reply back to
twrWasmModuleAsyncProxy
indicating that execution is complete. If there are any return codes they are also sent (twr_sleep does not have a return code) twr_sleep
returns to the C caller
The above sequence actually happens for all import
functions by default when using twrWasmModuleAsync
, irregardless if or how long they block for. This is because certain JavaScript code can only execute in the JavaScript main thread. Import function options exists to modify this behavior, in the cases where it is not desired.
The above steps also glosses over an important point -- the method that the twrWasmModuleAsyncProxy
uses to wait for a response from the main JavaScript thread. In step 3 above (Worker thread is blocking from twr_sleep
call), the worker thread is blocking on a call to Atomics.wait
. Communication from the JavaScript main thread to the Worker is through shared memory and a circular buffer. This is how twrWasmModuleAsync is able to block execution of the C code. This means that the Worker Thead main event loop can block for long periods of time -- perhaps indefinitely. And this means common JavaScript code can not run reliably in the Worker thread. For example, a setTimeout callback may not happen (because it is dispatched by the main JavaScript event loop). Likewise, Animations won't work since they are often executed inside the JavaScript event loop. This is another important reason that all the import
code generally runs inside the JavaScript main thread.
"},{"location":"api/api-ts-library/#blocking-function-explained","title":"Blocking Function Explained","text":"twr-wasm supports blocking functions like sleep when the API user is using twrWasmModuleAsync
. This section explains the sleep
function which causes C code to block. In other words, in C, code like this will work:
printf(\"going to sleep...\");\ntwr_sleep(1000);\nprintf(\"awake!\\n\");\n
The TypeScript twrLibrary derived class implementation looks like this:
// this function will only work in twrWasmModuleAsync since it blocks the C caller.\nasync twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number) {\n const p = new Promise<void>( (resolve)=>{\n setTimeout(()=>{ resolve() }, milliSeconds); \n });\n\n return p;\n}\n
And has these import options set:
imports:TLibImports = {\n twr_sleep:{isAsyncFunction: true, isModuleAsyncOnly: true},\n};\n
isAsyncFunction: true
is telling twrLibrary to call the twr_sleep_async
function with await
and so allow the function being called to await
. isModuleAsyncOnly: true
is telling twrLibrary that this function only exists when twrWasmModuleAsync
is used.
This code will execute in the JavaScript main thread. The Calling C code (twr_sleep
) will block in a Worker thread while waiting for this code in the JavaScrit main thread to complete. The function twr_sleep_async
creates a JavaScript promise that the calling code will await
on. Once the promise resolves, the calling function will unblock the C twr_sleep
function that is in the Worker Thread.
"},{"location":"api/api-ts-library/#import-options","title":"import
options","text":"The various import
options are used to handle different cases for twrWasmModuleAsync
.
The import options are:
isAsyncFunction?:boolean;\nisModuleAsyncOnly?:boolean;\nisCommonCode?:boolean;\n
"},{"location":"api/api-ts-library/#isasyncfunction","title":"isAsyncFunction
","text":"This option is used when you wish to await
inside the implementation of your function. This option also specifies that the function will be called with await
.
This option will only modify behavior this way when your function is called from twrWasmModuleAsync
.
- If this option is not specified, the same
import
function will be used for both module types, and the function can not use the await
keyword. - if this option is specified, then when
twrWasmModuleAsync
calls the indicated import
function, it will call a version of the function that has _async
append to the function name. This means that you will create two versions of the import
funcA
- funcA
and funcA_async
. - If, however, you also specify the option
isModuleAsyncOnly
, then only the _async
function is expected.
Here is an example of how declare a function with the import
option isAsyncFunction
:
async twr_sleep_async(callingMod:IWasmModuleAsync, milliSeconds:number)\n
Note that:
- the function declaration starts with the async keyword
- the function has the suffix
_async
appended to its import
name - that the function is passed an
IWasmModuleAsync
as the callingMod.
By using this option, you may need to create two versions of the import
function -- one that is async
(for use by twrWasmModuleAsync
), and one that does not use the async
keyword--for use by twrWasmModule
.
In a case like twr_sleep
, there is only one function implemented for sleep - the async function. But this is not always the case. For example, if your import
function uses the wasmMem.PutXX
functions, you will need to create two functions for the import
. Here is an example:
imports:TLibImports = {\n ex_append_two_strings:{isAsyncFunction: true},\n };\n\n ex_append_two_strings(callingMod:IWasmModule, str1Idx:number, str2Idx:number) {\n const newStr=callingMod.wasmMem.getString(str1Idx)+callingMod.wasmMem.getString(str2Idx);\n const rv=callingMod.wasmMem.putString(newStr);\n return rv;\n }\n\n async ex_append_two_strings_async(callingMod:IWasmModuleAsync, str1Idx:number, str2Idx:number) {\n const newStr=callingMod.wasmMem.getString(str1Idx)+callingMod.wasmMem.getString(str2Idx);\n const rv=await callingMod.wasmMem.putString(newStr);\n return rv;\n }\n
"},{"location":"api/api-ts-library/#ismoduleasynconly","title":"isModuleAsyncOnly
","text":"This option specifies that the indicated function is only available to twrWasmModuleAsync
. This option should be used for functions that block C execution. The twr_sleep
is an example.
"},{"location":"api/api-ts-library/#iscommoncode","title":"isCommonCode
","text":"This option is used to specify a function that should be used directly by the twrWasmModuleAsync
Web Worker. Without this option, the behavior is that code running in the Web Worker will send a message to the JavaScript Main thread requesting that the function be executed in the context of the JavaScript main thread. The Web Worker will then wait for a reply to the message before continuing C execution. However, in certain cases, it is possible, and might be more performant, to have the code execute directly in the WorkerThread.
There are limitations on the code that will work correctly with isCommonCode
:
- The functions must be available to a Worker thread
- The function can not be
async
(that is, it can not use await
) - The function can not call
PostEvent
- The functions must not depend on the Worker's main event loop running (this event loop often doesn't execute with the
twrWasmModuleAsync
Worker thread.) - The function can not use a callback (the callback won't get called because callbacks are often dispatched in the JavaScript main event loop)
Here is an Example of using isCommonCode
:
export default class twrLibMath extends twrLibrary {\n imports:TLibImports = {\n twrSin:{isCommonCode: true},\n }\n\n libSourcePath = new URL(import.meta.url).pathname;\n\n twrSin(callingMod:IWasmModule|twrWasmBase, angle:number ) {return Math.sin(angle)}\n}\n
In this case the Math.sin
function is available in both a Web Worker and the JavaScript main thread. It is a simple function, that works fine without the JavaScript event loop operating.
"},{"location":"api/api-ts-library/#noblock","title":"noBlock","text":"noBlock
will cause a function call in an twrWasmModuleAsync
to send the message from the Worker proxy thread to the JS main thread to execute the function, but not to wait for the result. This should only be used for functions with a void
return value. This has the advantage that (a) the C code will not block waiting for a void return value (so it returns faster), and (b) it takes advantage of multiple cores by allowing the JS Main thread and the Worker thread to execute in parallel.
Note that the messages sent from the proxy thread to the JS main thread (for function execution) will cause execution of function calls to serialize, and so if a function that blocks (waits for results from JS main thread) is called after a call with noBlock
, everything should work as expected.
Do not use noBlock
if:
- the function returns a value
- the C code should not continue executing until the function completes execution.
- if the following scenario could arise:
- funcA (with noBlock) called
- funcB called and returns a value or otherwise depends on funcA completing execution, and funcA uses async keyword.
Use noBlock
carefully.
"},{"location":"api/api-ts-library/#libsourcepath","title":"libSourcePath","text":"Always set this as follows:
libSourcePath = new URL(import.meta.url).pathname;\n
libSourcePath
is used to uniquely identify the library class, as well as to dynamically import the library when isCommonCode
is used.
"},{"location":"api/api-ts-library/#interfacename","title":"interfaceName","text":"In a twrLibrary,
- An \"interface\" refers to the set of functions that the library exposes to C. Ie, the functions in the
import
object. - The name of the interface is anonymous, unless
interfaceName
is set. - An undefined interfaceName (anonymous interface) also means that only one instance of that class is allowed (for example
twrLibMath
) - Set
interfaceName
to a unique name when multiple instances that support the same interface are allowed (for example the twr-wasm Consoles). - Multiple classes may have the same interfaceName (a class is identified by its libSourcePath). For example
twrConDiv
, twrConDebug
, twrConTerminal
all have the same interface.
When multiple instances of classes with the same interface are enabled (by setting interfaceName
), the first argument in every C function call is expected to be the twrLibrary id
(a member variable of the twrLibrary derived class). The twrLibrary will use this id
to route the function call to the correct instance of the library. The id
is not passed to the twrLibrary function (even though it is required to be the first C arg).
The twrLibrary instance should be created in the JavaScript main thread, and passed to the module in the io
option. The C code can discover the id
, by using the twr_get_console
.
exampleinterfaceName = \"twrConsole\";\n
"},{"location":"api/api-ts-library/#the-twrwasmmoduleasync-event-loop","title":"The twrWasmModuleAsync
Event Loop","text":"TODO
"},{"location":"api/api-ts-memory/","title":"Accessing Data in WebAssembly Memory","text":"There are situations where you may need to access WebAssembly memory from your TypeScript code. For example, if you need to allocate a new string, de-reference a pointer, or examine or modify a structure. A TwrLibrary
in particular may need to do this. ( For background on the issues involved in using WebAssembly Memory with C and TypeScript see Passing Function Arguments from JavaScript to C/C++ with WebAssembly.).
To access WebAssembly memory you will use the wasmMem
public member variable:
twrWasmModule
has the public member variable wasmMem:IWasmMemory
twrWasmModuleAsync
has the public member variable wasmMem:IWasmMemoryAsync
If you are writing a twrLibrary
, the appropriate wasmMem
is the first parameter of your import functions.
Both versions of wasmMem extend IWasmMemoryBase
which has common functions for retrieving or setting values from WebAssembly memory. With IWasmMemoryAsync
, for functions that call malloc
internally, await
must be used. This situation arises in the IWasmMemoryAsync
versions of the PutXXX
functions -- they return a Promise. PutXX
makes a call to malloc
, and in twrWasmModuleAsync
, malloc
needs to message the Worker thread and await
for a response.
mem8
, mem32
, memF
, and memD
can be used to access the WebAssembly memory directly. They are different views on the same memory.
getU8Arr
and getU32Arr
expect that idx
(a pointer) points to a:
struct {\n int size;\n void* data;\n}\n
Note: In prior versions of twr-wasm, these functions were available directly on the module instance. For example, mod.GetString
. These functions have been deprecated. Now you should use mod.wasmMem.getString
(for example).
// IWasmMemoryBase operate on shared memory, so they will function in any WasmModule \nexport interface IWasmMemoryBase {\n memory:WebAssembly.Memory;\n mem8:Uint8Array;\n mem32:Uint32Array;\n memF:Float32Array;\n memD:Float64Array;\n stringToU8(sin:string, codePage?:number):Uint8Array;\n copyString(buffer:number, buffer_size:number, sin:string, codePage?:number):void;\n getLong(idx:number): number;\n setLong(idx:number, value:number):void;\n getDouble(idx:number): number;\n setDouble(idx:number, value:number):void;\n getShort(idx:number): number;\n getString(strIndex:number, len?:number, codePage?:number): string;\n getU8Arr(idx:number): Uint8Array;\n getU32Arr(idx:number): Uint32Array;\n}\n\n// IWasmMemory does not support await, and so will only work in a thread that has the module loaded\n// That would be twrWasmModule, twrWasmModuleAsyncProxy\nexport interface IWasmMemory extends IWasmMemoryBase {\n malloc:(size:number)=>number;\n free:(size:number)=>void;\n putString(sin:string, codePage?:number):number;\n putU8(u8a:Uint8Array):number;\n putArrayBuffer(ab:ArrayBuffer):number;\n}\n\n// IWasmMemoryAsync must be used from an async function since await is needed\nexport interface IWasmMemoryAsync extends IWasmMemoryBase {\n malloc:(size:number)=>Promise<number>;\n free:(size:number)=>Promise<void>;\n putString(sin:string, codePage?:number):Promise<number>;\n putU8(u8a:Uint8Array):Promise<number>;\n putArrayBuffer(ab:ArrayBuffer):Promise<number>;\n}\n
"},{"location":"api/api-ts-modules/","title":"Wasm Modules","text":"This section describes the twr-wasm TypeScript/JavaScript classes twrWasmModule
and twrWasmModuleAsync
that are used to load .wasm
modules, call their C functions, and access wasm memory. Both classes have similar APIs.
"},{"location":"api/api-ts-modules/#about-twrwasmmodule","title":"About twrWasmModule
","text":"class twrWasmModule
allows you to integrate WebAssembly C/C++ code into your Web Page. You can call C/C++ functions, and read and write WebAssembly memory. Function calls are asynchronous, as is normal for a JavaScript function. That is C/C++ functions should not block - they should return quickly (just as happens in JavaScript).
The constructor accepts an optional object (type IModOpts
), which is explained further down.
import {twrWasmModule} from \"twr-wasm\";\n\nconst mod = new twrWasmModule();\n
"},{"location":"api/api-ts-modules/#about-twrwasmmoduleasync","title":"About twrWasmModuleAsync
","text":"class twrWasmModuleAsync
allows you to integrate WebAssembly C/C++ code into your Web Page that uses a Read-Eval-Print Loop (REPL) pattern, a CLI pattern or code that blocks. For example, with twrWasmModuleAsync
your C/C++ code can call a synchronous function for keyboard input (that blocks until the user has entered the keyboard input). Or your C/C++ code can sleep
or otherwise block. This is the pattern that is used by many standard C library functions - fread
, etc.
class twrWasmModuleAsync
creates a WorkerThread that runs in parallel to the JavaScript main thread. This Worker thread executes your C/C++ code, and proxies functionality that needs to execute in the JavaScript main thread via remote procedure calls. This allows the JavaScript main thread to await
on a blocking callC
in your JavaScript main thread.
The Async
part of the twrWasmModuleAsync
name refers to the property of twrWasmModuleAsync
that makes your synchronous C/C++ code asynchronous.
The APIs in class twrWasmModuleAsync
are identical to class twrWasmModule
, except that certain functions use the async
keyword and thus need to be called with await
. This happens whenever the function needs to cross the JavaScript main thread and the Worker thread boundary. For example: callC
or malloc
.
The constructor accepts an optional object (type IModOpts
), which is explained further down.
import {twrWasmModuleAsync} from \"twr-wasm\";\n\nconst amod = new twrWasmModuleAsync();\n
"},{"location":"api/api-ts-modules/#loadwasm","title":"loadWasm","text":"This function is available on both class twrWasmModule
and class twrWasmModuleAsync
.
Use loadWasm
to load your compiled C/C++ code (the .wasm
file).
await mod.loadWasm(\"./mycode.wasm\")\n
"},{"location":"api/api-ts-modules/#callc","title":"callC","text":"This function is available on both class twrWasmModule
and class twrWasmModuleAsync
. twrWasmModuleAsync
returns a Promise, twrWasmModule
does not.
After your .wasm
module is loaded with loadWasm
, you call functions in your C/C++ from TypeScript/JavaScript like this:
twrWasmModulelet result=mod.callC([\"function_name\", ...params])\n
twrWasmModuleAsynclet result=await mod.callC([\"function_name\", ...params])\n
If you are calling into C++, you need to use extern \"C\"
like this in your C++ function:
extern \"C\" int function_name() {}\n
Each C/C++ function that you wish to call from TypeScript/JavaScript needs to be exported in your wasm-ld
command line with an option like this:
--export=function_name\n
Or like this in your source file: __attribute__((export_name(\"function_name\")))\nvoid function_name() {\n ...\n}\n
Fo more details, see the Compiler Options.
callC
takes an array where:
callC
returns the value returned by the C function. long
, int32_t
, int
, float
or double
and the like are returned as a number
. int64_t
is returned as a bigint
, and pointers are returned as a number
. The contents of the pointer will need to be extracted using the functions listed below. More details can be found in this article: Passing Function Arguments to WebAssembly and in this example. The FFT example demonstrates passing and modifying a Float32Array
view of an ArrayBuffer
.
Although you can always use await
on a callC
, it is only strictly necessary if the module is of class twrWasmModuleAsync
.
CallC
is mapped to wasmCall.callC
. wasmCall
also exposes callCImpl
, which can be used if no argument conversion is needed.That is if the arguments are all numbers).
"},{"location":"api/api-ts-modules/#fetchandputurl","title":"fetchAndPutURL","text":"fetchAndPutURL
is available as a member function of both twrWasmModule
and twrWasmModuleAsync
.
fetchAndPutURL(fnin:URL) : Promise<[number, number]>;\n
The returned array contains the index of where the URL contents have been loaded into wasm memory (in index 0), and the length (in index 1).
In prior versions of twr-wasm, callC
could accept an argument type of URL. This is no longer supported, and instead fetchAndPutURL
should be used.
"},{"location":"api/api-ts-modules/#log","title":"log","text":"log
is available as a member function of both twrWasmModule
and twrWasmModuleAsync
.
log
is similar to the JavaScript console.log
, except that the output is sent to the stdio
console.
log(...params: string[]):void;\n
Also note that most consoles have a putStr
function.
"},{"location":"api/api-ts-modules/#twrwasmmoduleasync-details","title":"twrWasmModuleAsync Details","text":"twrWasmModuleAsync
implements all of the same functions as twrWasmModule
, plus allows blocking inputs, and blocking code generally. This is achieved by proxying all the calls through a Web Worker thread.
For example, with this C function in your Wasm module:
void mysleep() {\n twr_sleep(5000); // sleep 5 seconds\n}\n
can be called from your JavaScript main loop like this:
await amod.callC([\"mysleep\"]);\n
You must use twrWasmModuleAsync
in order to:
- call any blocking C function (meaning it takes \"a long time\") to return
- use blocking input from a div or canvas ( eg.
twr_mbgets
) - use
twr_sleep
"},{"location":"api/api-ts-modules/#linking-requirements","title":"Linking Requirements","text":"When linking your C/C++ code, twrWasmModule
and twrWasmModuleAsync
use slightly different wasm-ld
options since twrWasmModuleAsync
uses shared memory. twrWasmModule
will operate with shared memory, so technically you could just use the same share memory options with either module, but you don't need the overhead of shared memory when using twrWasmModule, and so better to not enable it.
See wasm-ld Linker Options.
"},{"location":"api/api-ts-modules/#javascript-needed-for-char-input","title":"JavaScript Needed for Char Input","text":"When a console will handle key input, you need to add a line to your JavaScript to send key events to the console. There are two options for this: You can send the key events directly to the console, or if the key events are always directed to stdio
, you cam send the key events to the module. This latter case is primarily for when you are using tag shortcuts.
To send key events to the console, you add a line like this:
yourDivOrCanvasElement.addEventListener(\"keydown\",(ev)=>{yourConsoleClassInstance.keyDown(ev)});\n
To send key events to the module's stdio
, you add a line like this:
yourDivOrCanvasElement.addEventListener(\"keydown\",(ev)=>{yourModuleClassInstance.keyDown(ev)});\n
You likely want a line like this to automatically set the focus to the div or canvas element (so the user doesn't have to click on the element to manually set focus. Key events are sent to the element with focus.):
yourDivOrCanvasElement.focus();\n
You will also need to set the tabindex
attribute in your tag like this to enable key events:
<div id=\"twr_iodiv\" tabindex=\"0\"></div>\n<canvas id=\"twr_iocanvas\" tabindex=\"0\"></canvas>\n
See this example on character input.
Note that this section describes blocking input. As an alternative, you can send events (keyboard, mouse, timer, etc) to a non-blocking C function from JavaScript using callC
. See the balls
or pong
examples.
"},{"location":"api/api-ts-modules/#sharedarraybuffers","title":"SharedArrayBuffers","text":"twrWasmModuleAsync
uses SharedArrayBuffers which require certain CORS HTTP headers to be set. Note that twrWasmModule
does not use SharedArrayBuffers. If you limit yourself to twrWasmModule
you will not need to worry about configuring the CORS http headers on your web server.
See this note on enabling CORS HTTP headers for SharedArrayBuffers.
"},{"location":"api/api-ts-modules/#module-options","title":"Module Options","text":"The twrWasmModule
and twrWasmModuleAsync
constructor both take optional options.
For example:
let amod=new twrWasmModuleAsync();\n\nlet amod=new twrWasmModuleAsync({\n stdio: new twrConsoleDebug(); // send stdio to debug console\n });\n
These are the options: twrWasmModule & twrWasmModuleAsync Options
export interface IModOpts {\n stdio?: IConsoleStream&IConsoleBase,\n d2dcanvas?: IConsoleCanvas&IConsoleBase,\n io?: {[key:string]: IConsole},\n}\n
"},{"location":"api/api-ts-modules/#stdio-option","title":"stdio
Option","text":"Set this to a Console class instance. If you leave it undefined, twrConsoleDebug
will be used (or a tag shortcut, if set)
This option is a shortcut to setting stdio
using the io
option.
"},{"location":"api/api-ts-modules/#d2dcanvas-option","title":"d2dcanvas
Option","text":"Set this to a twrConsoleCanvas
instance to configure a 2D drawing surface. If you leave it undefined, a tag shortcut will be used.
This option is a shortcut to setting std2d
using the io
option (note the different names).
"},{"location":"api/api-ts-modules/#io-option-multiple-consoles-with-names","title":"io
Option: Multiple Consoles with Names","text":"This option allows you to assign names to consoles. The C/C++ code can then retrieve a console by name.
When using the io
object to specify named consoles:
- You can use the attribute
stdio
to set stdio. - You can use the attribute
stderr
to set stderr - You can use the attribute
std2d
to set the default 2D Drawing Surfaces -- used by twr-wasm 2D APIs. - all other attribute names are available for your consoles. Use this to access consoles in C/C++ beyond (or instead of) stdio, etc.
Alternately, you can specify stdio
and std2d
directly as module attributes (outside of io
) as a shortcut (see above).
There is a twr-wasm C API to access named consoles: twr_get_console
.
This code snippet shows how to use the io
option to pass in an object containing named console attributes:
const stream1Element=document.getElementById(\"stream1\");\nconst stream2Element=document.getElementById(\"stream2\");\n\nconst debug = new twrConsoleDebug();\nconst stream1 = new twrConsoleDiv(stream1Element);\nconst stream2 = new twrConsoleDiv(stream2Element);\n\nstream1Element.addEventListener(\"keydown\",(ev)=>{stream1.keyDown(ev)});\nstream2Element.addEventListener(\"keydown\",(ev)=>{stream2.keyDown(ev)});\n\n// setting stdio and/or stderr to a debug console isn't necessary since that will be the default if stdio or stderr is not set.\n// but here to show how to set stdio and/or stderr. They can be set to any console.\nconst amod = new twrWasmModuleAsync( {io:{stdio: debug, stderr: debug, stream1: stream1, stream2: stream2}} );\nconst mod = new twrWasmModule( {io:{stdio: debug, stderr: debug, stream1: stream1, stream2: stream2}} );\n
In this case, as well as setting stdio and stderr, consoles named \"stream1\" and \"stream2\" are made available to the C/C++ code.
Using a Named Consoletwr_ioconsole_t * stream1=twr_get_console(\"stream1\");\nfprintf(stream1, \"Hello Stream One!\\n\");\n
A complete example multi-io is provided.
"},{"location":"api/api-ts-modules/#deprecated-options","title":"Deprecated Options","text":"The following options are deprecated. Instead of these, use options available to twrConsoleDiv
and twrConsoleTerminal
constructors.
deprecatedexport interface IModOpts {\n windim?:[number, number],\n forecolor?:string,\n backcolor?:string,\n fontsize?:number,\n}\n
Note:
windim
- if stdio is set to a twrConsoleTerminal
, this will set the width and height, in characters. Instead, use constructor options on twrConsoleTerminal. forecolor
and backcolor
- if stdio is set to twrConsoleDiv
or twrConsoleTerminal
, these can be set to a CSS color (like '#FFFFFF' or 'white') to change the default background and foreground colors. However, these are deprecated, and instead, use the twrConsoleDiv
or twrConsoleTerminal
constructor options. fonsize
- Changes the default fontsize if stdio is set to twrConsoleDiv
or twrConsoleTerminal
. Deprecated, instead use twrConsoleDiv
or twrConsoleTerminal
constructor options. TStdioVals
have been removed (they were a not too useful option in prior versions of twr-wasm) divLog
has been renamed log
. Or use the putStr
member function on most consoles.
"},{"location":"examples/examples-balls/","title":"Bouncing Balls - 2D Draw API Wasm Example","text":"This example uses twr-wasm's 2D Draw API and a C++ Canvas class with WebAssembly and C++ to bounce balls around your HTML page.
- View bouncing balls
- Source for balls
The bouncing balls example demonstrates
- C++
- Using the twr-wasm draw 2D APIs that match Javascript Canvas APIs.
- Using the twr-wasm canvas.cpp wrapper class.
This example does not use libc++, which results in smaller code size. For an example that uses libc++ see tests-libcxx.
"},{"location":"examples/examples-balls/#screen-grab-of-balls-example","title":"Screen Grab of Balls Example","text":""},{"location":"examples/examples-callc/","title":"callC - Calling WebAssembly Functions Example","text":"This example demonstrates how to pass and return values between TypeScript/JavaScript and C/C++ when you are using WebAssembly with twr-wasm.
This article explains the key concepts to pass arguments between JavaScript/TypeScript and Wasm C/C++.
- View callC example running live
- View callC example source
"},{"location":"examples/examples-divcon/","title":"divcon - Printf and Input Using a div
Tag","text":""},{"location":"examples/examples-divcon/#what-it-does","title":"What It Does","text":"This example inputs a number, squares it, and prints the result using standard C library functions.
The divcon example demos:
- A Read-Eval-Print Loop (REPL)
- using twr-wasm
class twrWasmModuleAsync
to await
on blocking C code - getting and print characters to a
div
tag using twr-wasm class twrConsoleDiv
"},{"location":"examples/examples-divcon/#running-examples-and-source","title":"Running Examples and Source:","text":" - view divcon example running live
- View divcon source code
"},{"location":"examples/examples-divcon/#screen-grab-of-square-calculator","title":"Screen Grab of Square Calculator","text":""},{"location":"examples/examples-divcon/#c-code","title":"C Code","text":"divcon.c#include <stdio.h>\n#include <stdlib.h>\n#include \"twr-crt.h\"\n\nvoid stdio_div() {\n char inbuf[64];\n char *r;\n int i;\n\n printf(\"Square Calculator\\n\");\n\n while (1) {\n printf(\"Enter an integer: \");\n r=twr_mbgets(inbuf); // r is NULL if esc entered. Otherwise r == inbuf\n if (r) { \n i=atoi(inbuf);\n printf(\"%d squared is %d\\n\\n\",i,i*i);\n }\n else {\n printf(\"\\n\");\n }\n }\n}\n
"},{"location":"examples/examples-divcon/#html-code","title":"HTML Code","text":"We are using twrWasmModuleAsync
which integrates blocking C code into JavaScript. twrWasmModuleAsync
can also be used to receive key input from a <div>
or <canvas>
tag.
index.html<body>\n <div id=\"stdioDiv\" \n tabindex=\"0\" \n style=\"color: DarkGreen; background-color: LightGray; font-size: 18px;font-family: Arial, sans-serif;\" >\n Loading... <br>\n </div>\n\n <script type=\"module\">\n import {twrWasmModuleAsync, twrConsoleDiv} from \"twr-wasm\";\n\n const con = new twrConsoleDiv(document.getElementById(\"stdioDiv\"));\n const amod = new twrWasmModuleAsync({stdio: con});\n\n // remove 'Loading...'\n document.getElementById(\"stdioDiv\").innerHTML =\"<br>\"; \n // send key events to twrConsoleDiv\n document.getElementById(\"stdioDiv\").addEventListener(\"keydown\",(ev)=>{con.keyDown(ev)});\n\n await amod.loadWasm(\"./divcon.wasm\");\n await amod.callC([\"stdio_div\"]);\n\n </script>\n</body>\n
"},{"location":"examples/examples-fft/","title":"FFT - Example of using C FFT with HTML/JavaScript","text":"This example is a demo of integrating the popular KISS FFT C library with TypeScript/JavaScript/HTML using WebAssembly. The FFT C library is compiled into a Wasm (WebAssembly) module using clang, with the help of twr-wasm. The FFT Wasm module is used by the HTML page to calculate the FFT. The FFT input and output is drawn to the web page using JavaScript canvas functions.
The FFT library exposes APIs to process data, and doesn't use stdio.
The FFT APIs use float32 arrays for complex-number input and output data, and a configuration C struct. In the example I generate the input data by adding a 1K and 5K sine waves, call the kiss FFT API to perform the FFT on the generated sine waves, and then graph the input and output data using a JavaScript Canvas.
- View example running on the web
- View example source code
"},{"location":"examples/examples-fft/#screen-grab-of-output","title":"Screen Grab of Output","text":""},{"location":"examples/examples-fft/#code","title":"Code","text":"Here is part of the code. The rest can be found on github.
index.html
<head>\n <title>Fast Fourier transform (FFT)</title>\n</head>\n<body style=\"background-color:white\">\n\n <br>\n\n <div style=\"font:24px arial\">Input Signal</div>\n <canvas id=\"c-input\" width=\"1024\" height=\"300\" style=\"background-color:lightgray\"></canvas>\n\n <br><br><br>\n\n <div style=\"font:24px arial\">FFT Output</div>\n <canvas id=\"c-output\" width=\"1024\" height=\"300\" style=\"background-color:lightgray\"></canvas>\n\n <script type=\"module\">\n import {fftDemo} from \"./fft-script.js\";\n\n fftDemo();\n\n </script>\n</body>\n
fft-script.jsimport {twrWasmModule} from \"twr-wasm\";\n\nexport async function fftDemo() {\n\n const mod=new twrWasmModule();\n\n // load the kiss_fft C code as is, unmodified\n await mod.loadWasm('kiss_fft.wasm');\n\n // kissFFTData stores and graphs the input and output data\n // in this example the fft has 1024 bins, and I am using a 48K sampling rate\n let fft=new kissFFTData(1024, 48000);\n fft.genSin(1000)\n fft.addSin(5000)\n fft.graphIn(\"c-input\");\n\n // see kiss_fft README, but in summary you: (a) alloc config, (b) compute the FFT, (c) free the config\n // kiss_fft_alloc() returns a malloced structure. Pointers are numbers (index into Wasm module memory) in JS land \n //\n //kiss_fft_cfg cfg = kiss_fft_alloc( nfft ,is_inverse_fft ,0,0 );\n let cfg:number = await mod.callC([\"kiss_fft_alloc\", fft.nfft, 0, 0, 0 ]);\n\n // The FFT input and output data are C arrays of complex numbers.\n // typedef struct {\n // kiss_fft_scalar r;\n // kiss_fft_scalar i;\n // } kiss_fft_cpx;\n //\n // /* default is float */\n // define kiss_fft_scalar float\n\n // So if the FFT data has 1024 bins, then 1024 * 2 floats (r & i) * 4 bytes per float are needed.\n // I use a JS Float32Array view on the ArrayBuffer to access the floats\n\n // When an arrayBuffer is passed in as an argument to mod.callC,\n // callC will malloc memory in the Wasm module of a size that matches the array buffer, then\n // copy the arraybuffer into the malloc'd memory prior to the function call, \n // then copy the malloc'd memory contents back into the arrayBuffer post call.\n // The malloc'd memory is free'd post call. \n\n // void kiss_fft(kiss_fft_cfg cfg,const kiss_fft_cpx *fin,kiss_fft_cpx *fout);\n await mod.callC([\"kiss_fft\", cfg, fft.inArrayBuf, fft.outArrayBuf]);\n\n fft.graphOut(\"c-output\");\n\n await mod.callC([\"free\", cfg]); // not much point to this since all the module memory is about to disappear\n}\n
"},{"location":"examples/examples-helloworld/","title":"Hello World - WebAssembly C Example","text":"This example is a very simple twr-wasm program. It uses WebAssembly and C to print \"hello, world!\" to an HTML <div>
tag.
Also see: Hello World - Step-by-Step C to Wasm.
- View helloworld example running live
- View helloworld source code
"},{"location":"examples/examples-lib/","title":"class twrLibrary Example","text":""},{"location":"examples/examples-lib/#what-it-does","title":"What It Does","text":"This example is a twr-wasm Library that implements functions that can be called by your C/C++ code. A twr-wasm Library is written in TypeScript, and derives from the class twrLibrary.
The lib example demos:
- Creating functions in TypeScript that can be called from C/C++
- Posting Events to C
- Implementing blocking as well as non-blocking functions
"},{"location":"examples/examples-lib/#running-examples-and-source","title":"Running Examples and Source:","text":" - View lib output
- Source for lib
Also see twr-wasm Libraries Documentation
"},{"location":"examples/examples-libcxx/","title":"tests-libcxx - WebAssembly libc++ Smoke Test","text":"This is a simple test of various libc++ functions using WebAssembly with twr-wasm. The C++ program links with libc++. An example makefile is provided.
- view tests-libcxx example running live
- View tests-libcxx source code
Also see this WebAssembly program that uses libc++ with twr-wasm to implement a CLI console.
- tests-user Live
- tests-user Source
"},{"location":"examples/examples-maze/","title":"Maze Generator/Solver","text":"This example is a port to Wasm of a 20 year old Win32 C Maze creator, with the help of twr-wasm 2D Draw APIs.
- View live maze here
- Source for maze
"},{"location":"examples/examples-maze/#screen-grab-of-output","title":"Screen Grab of Output","text":""},{"location":"examples/examples-maze/#overview","title":"Overview","text":"This Maze generator uses the twr-wasm \"d2d\" (Draw 2D) C APIs. These allow drawing onto an HTML canvas from C/C++. (Also see the balls C++ example).
This C code is interesting in that it is a combination of blocking and non blocking functions. The CalcMaze
function is blocking when the \"slow draw\" flag is set. It uses Sleep
in this case. For this reason, I use twrWasmModuleAsync. The solve section uses repeated calls to SolveStep
, which works well with a JavaScript main loop. I used a javascript interval timer to make repeated calls to the C SolveStep
function. If all the C code was structured this way, twrWasmModule
could have been used (instead of the Async version)
To port this code to twr-wasm I wrote a (very tiny) Win32 compatible API (in winemu.c/winemu.h). It only implements the features needed to port Maze, but it might be useful to use as a starting point for porting your Win32 code to the web.
index.html<head>\n <title>Maze</title>\n</head>\n<body style=\"background-color:powderblue\">\n <canvas id=\"twr_d2dcanvas\" width=\"600\" height=\"600\"></canvas>\n\n <script type=\"module\">\n import {mazeRunner} from \"./maze-script.js\";\n\n mazeRunner();\n </script>\n</body>\n
maze-script.jsimport {twrWasmModuleAsync} from \"twr-wasm\";\n\nexport async function mazeRunner() {\n\n const amod=new twrWasmModuleAsync();\n\n await amod.loadWasm('maze.wasm');\n\n //void CalcMaze(HWND hWnd, LONG cell_size, LONG is_black_bg, LONG isd - slow draw)\n await amod.callC([\"CalcMaze\", 0, 7, 0, 1]);\n await amod.callC([\"SolveBegin\"]);\n\n let timer = setInterval(async ()=>{\n let isdone=await amod.callC([\"SolveStep\", 0]); //SolveStep(hwnd))\n if (isdone) clearInterval(timer);\n }, 50);\n}\n
"},{"location":"examples/examples-multi-io/","title":"Multi-io Multiple Console Example","text":""},{"location":"examples/examples-multi-io/#what-it-does","title":"What It Does","text":"This example demos six simultaneous consoles:
- Two character streaming consoles directed to a
<div>
tag - Two character terminal consoles directed to a
<canvas>
tag - Two 2D draw surface consoles directed to a
<canvas>
tag
The multi-io example also demos:
- Creating consoles in JavaScript/TypeScript
- Using consoles in C/C++ and JavaScript/TypeScript
- Mixing multiple Consoles
- Using multiple .wasm modules using the same console
- Inputting, printing, and drawing to consoles
"},{"location":"examples/examples-multi-io/#running-examples-and-source","title":"Running Examples and Source:","text":" - View multi-io
- Source for multi-io
Also see Console Introduction
"},{"location":"examples/examples-overview/","title":"WebAssembly C/C++ Examples","text":""},{"location":"examples/examples-overview/#overview","title":"Overview","text":"These C and C++ examples demonstrate how to create different types of WebAssembly (wasm) programs with the twr-wasm library.
These are good examples to use as starting points for your own Wasm projects.
These examples are a good place to learn how to configure clang and wasm-ld to compile and link C/C++ code for use with WebAssembly (wasm).
"},{"location":"examples/examples-overview/#example-quick-links","title":"Example Quick Links","text":" - Click here to view C/C++ WebAssembly twr-wasm examples running live
- Click here to view source code and make files
"},{"location":"examples/examples-overview/#hello-world","title":"Hello World","text":"Name Description Link helloworld A very simple C Wasm example to get you started helloworld"},{"location":"examples/examples-overview/#console-examples","title":"Console Examples","text":"Name Description Link divcon A simple C program demos inputting and printing characters to a div
tag divcon terminal A simple C program demos writing and inputting from a <canvas>
tagthat twr-wasm configures as a windowed \"terminal\" terminal multi-io Demo 6 simultaneous consoles: stream i/o, terminal, and 2D Drawing. multi-io"},{"location":"examples/examples-overview/#draw-2d-and-audio-examples","title":"Draw 2D and Audio Examples","text":"Name Description Link balls These fun Bouncing Balls are written in C++ and demo the 2D drawingAPIs with a C++ Canvas wrapper class balls pong A simple game of Pong written in C++ to demo 2D drawing and Audio APIs witha C++ canvas wrapper class and taking user input from JS pong maze This is an old Win32 program ported to wasm and demos 2D Draw APIs maze"},{"location":"examples/examples-overview/#call-argument-examples","title":"Call Argument Examples","text":"Name Description Link callC A demo of passing and returning values between JavaScript and Wasm module callc fft A demo of calling a C library to perform an FFT that is graphed in TypeScript fft"},{"location":"examples/examples-overview/#twrlibrary-examples","title":"twrLibrary Examples","text":"Name Description Link lib A demo of createing a twrLibrary (use TypeScript to create C/C++ APIs) library"},{"location":"examples/examples-overview/#unit-tests","title":"Unit Tests","text":"Name Description Link tests twr-wasm unit tests tests tests-user \"cli\" for tests using libc++ and <canvas>
tests-user tests-libcxx Smoke test for libc++. Shows how to use libc++. tests-libcxx tests-d2d Unit tests for Draw 2D canvas console tests-d2d tests-audio Unit tests for the Audio Library tests-audio"},{"location":"examples/examples-overview/#running-or-building-the-examples-locally","title":"Running or Building the examples locally","text":"Online versions of the examples can be viewed here.
You can also run the examples locally, or build them..
"},{"location":"examples/examples-overview/#copying-examples-to-start-your-own-project","title":"Copying Examples to Start your own Project","text":"All of the examples have makefiles that use a relative path for twr.a
and includes
. These paths will work fine if your code is in an examples sub-folder as a peer to the other examples. But assuming your code is in your own project folder elsewhere, you will need to determine the correct path to twr.a
and includes
for your project's makefile. Details on how to do this can be found in the following sections: Hello World walk through and the Compiler and Linker Options section.
Also see the section on Import Resolution if you installed with git clone.
"},{"location":"examples/examples-pong/","title":"Pong - 2D Game Example","text":"Similar to the balls example, this example uses twr-wasm's 2D Draw API and a C++ canvas class to implement a simple game of 2 player and single player Pong with WebAssembly.
- View Pong
- Source for Pong
The Pong example demonstrates
- C++
- Using twr-wasm draw 2D APIs that match Javascript Canvas APIs.
- Using the twr-wasm canvas.cpp class.
- A custom typescript library
- User mouse and keyboard input via events
- Using the Audio Library to play simple sounds
This example does not use libc++, which results in smaller code size. For an example that uses libc++ see tests-libcxx.
"},{"location":"examples/examples-pong/#screen-grab-of-pong-example","title":"Screen Grab of Pong Example","text":""},{"location":"examples/examples-terminal/","title":"Terminal Console Demo","text":"A simple WebAssembly C \"terminal\" is demoed with input and output directed to an HTML <canvas>
tag.
"},{"location":"examples/examples-terminal/#what-it-does","title":"What it Does","text":" - uses class twrConsoleTerminal, a Console
- moves a string up or down in the terminal window when you press the u or d or arrow keys.
- shows basic color usage
- draws a graphic box around the terminal window.
"},{"location":"examples/examples-terminal/#run-and-view-the-code","title":"Run and View the Code","text":" - View terminal example running live
- View terminal source code
- For another 'terminal' demo View tests-user
"},{"location":"examples/examples-terminal/#screen-grab-of-terminal","title":"Screen Grab of Terminal","text":""},{"location":"examples/examples-tests-audio/","title":"tests-audio - Unit tests for Audio Library","text":"This is a simple set of Unit Tests for testing the Audio API functions. It includes a test for each Audio function. WARNING: Some of the audio played can be loud, so turn down your volume before playing.
- view tests-audio example running live
- View tests-audio source code
Also see these WebAssembly programs that use this API
"},{"location":"examples/examples-tests-d2d/","title":"tests-d2d - Unit tests for Draw 2D canvas console","text":"This is a simple set of Unit Tests for testing the D2D API functions. It includes a test for each D2D function as well as a test for memory leaks within the API itself.
- view tests-d2d example running live
- View tests-d2d source code
Also see these WebAssembly programs that use this API
"},{"location":"gettingstarted/basicsteps/","title":"Basic Steps To Create Your Wasm Project","text":"This section describes the basic steps to integrate your TypeScript/JavaScript with C/C++ WebAssembly code.
"},{"location":"gettingstarted/basicsteps/#overview-of-webassembly-project","title":"Overview of WebAssembly Project","text":"Your C/C++ WebAssembly project consists of HTML (and related JavaScript or TypeScript) and C/C++ source. The C/C++ is compiled using clang
into a .wasm
binary module. The .wasm
module is loaded as a WebAssembly module by your JavaScript using twr-wasm
.
"},{"location":"gettingstarted/basicsteps/#javascripttypescript-part-of-wasm-project","title":"JavaScript/TypeScript Part of Wasm Project","text":"On the JavaScript side of your WebAssembly project you will use the twr-wasm JavaScript/TypeScript class twrWasmModule
or twrWasmModuleAsync
to load the .wasm
module, and then call C functions in it using callC
(more details are in the TypeScript/Javascript API section).
"},{"location":"gettingstarted/basicsteps/#cc-part-of-wasm-project","title":"C/C++ Part of Wasm Project","text":"You will call C functions (or C++ with ' extern \"C\" ' linkage) in the .wasm
module from your JavaScript. You can also call JavaScript functions from your C/C++ code, but this is less common.
There is no direct equivalent to a C \"main\". Instead, a Wasm module provides exported C functions that you can call from JavaScript/TypeScript. A Wasm module is more like a runtime loaded dynamic library.
twr-wasm
supports C/C++ code that is either asynchronous (non-blocking) or syncronous (blocking). A CLI app is an example of typical blocking code. A CLI app typically blocks waiting for keyboard input (blocking means that it \"takes a long time\" to return). If your C code is a big loop that never returns, that would be very blocking. Alternately,if you send mouse events to C code, have the code process them then return, this would be non-blocking. You can use the twr-wasm class twrWasmModuleAsync
to execute blocking code from JavaScript or twrWasmModule
to integrate asynchronous C/C++ code. The example maze demonstrates both non-blocking and blocking C calls.
See the examples of different types of C/C++ apps.
"},{"location":"gettingstarted/basicsteps/#steps-to-integrate-c-code-with-javascript-code","title":"Steps to integrate C code with JavaScript code","text":"Here are the general steps to integrate your C with your JavaScript:
- Compile your C code with
clang
and link with wasm-ld
to create the .wasm
file. - On the JavaScript side you:
- Access
twr-wasm
\"ES\" modules in the normal way with import
. - Add a
<div id=twr_iodiv>
or <canvas id=twr_iocanvas>
to your HTML (see stdio) - Use
new twrWasmModule
, followed by a call to loadWasm
, then one or more callC
. - Alternately, use
twrWasmModuleAsync
-- which is interchangeable with twrWasmModule
, but proxies through a worker thread, which allows you to call blocking functions from the asynchronous JavaScript main thread. - For more details, see the remainder of this documentation, or see the hello world or other exampes.
"},{"location":"gettingstarted/charencoding/","title":"Character Encoding Support with twr-wasm","text":"This section explains twr-wasm's WebAssembly support for ASCII, UTF-8, windows-1252, and UTF-32 character encoding.
"},{"location":"gettingstarted/charencoding/#getting-started","title":"Getting Started","text":"When using C with twr-wasm, you will likely want to add this line to the start of your code:
setlocale(LC_ALL, \"\")\n
This will change the C locale language to the one selected in the browser, and will enable consistent UTF-8 character encoding support.
Without this line, the standard C runtime will default character encoding to ASCII, as per the std c lib standard. However, just as with gcc, twr-wasm consoles support outputting UTF-8, even in the default setting.
"},{"location":"gettingstarted/charencoding/#character-encodings","title":"Character Encodings","text":"twr-wasm supports ASCII, UNICODE, and extended-ASCII (in the form of Windows-1252).
"},{"location":"gettingstarted/charencoding/#utf-8","title":"UTF-8","text":"These days UNICODE with UTF-8 encoding is the most popular method of displaying and encoding text. UTF-8 is popular because it has the deep character glyph definitions of UNICODE with an encoding that provides (a) the best backwards compatibility with ASCII, and (b) a compact memory footprint. It does this at the expense of some multibyte complexities.
UTF-8 is variable length, and uses between one to four bytes to represent any unicode code point, with ASCII compatibility in the first 128 characters. It is also the standard for the web, and the default for clang. But because UTF-8 uses a variable number of bytes per character it can make string manipulation in C a bit harder than ASCII, Windows-1252 or UTF-32.
"},{"location":"gettingstarted/charencoding/#locale","title":"Locale","text":"In this document you will see the term \"locale\". This term originated (at least as its commonly used in programming) in the standard C library, and is also used in the standard C++ library (libc++ in twr-wasm). A locale refers to a region of the world, along with a specific character encoding. The twr-wasm standard c runtime uses a label akin to this to define a locale: en-US.UTF-8
. Of note is that libc++ and the standard C runtime have different domains for their locales (ie, they don't directly impact each other). You can learn more about locales by searching the internet.
twr-wasm C locales support ASCII, UTF-8 or windows-1252 character encoding.
"},{"location":"gettingstarted/charencoding/#utf-32","title":"UTF-32","text":"UTF-16/32 are not supported as a std c lib locale setting, but functions are provided to convert utf-32 (unicode code points) to and from ASCII, UTF-8, and windows-1252 \"code pages\" (there are other miscellaneous utf-32 based functions as well.)
You can also use libc++, which has classes that directly support utf-16 and utf-32.
"},{"location":"gettingstarted/charencoding/#windows-compatibility-with-windows-1252","title":"Windows Compatibility with Windows-1252","text":"Windows-1252 is the default character encoding on Windows computers in many countries - particularly the Americas and western Europe -- and particularly when using MSVC. Linux, clang, gcc, and the web commonly default in some way to UTF-8 character encoding. Windows-1252 is an extension of ASCII that uses a single byte per character. This makes it easier than UTF-8 from a programmers perspective, but it doesn't represent as many characters. It is supported by twr-wasm to make it easier to port legacy C code, windows code, as well as a simpler alternative to UTF-8.
twr-wasm supports Windows-1252, and you can enable it like this:
setlocale(LC_ALL, \".1252\")\n
This will set the locale to the default browser language, and character encoding to Windows-1252.
1252 String Literals These days text editors generally default to UTF-8. In order to use windows-1252 source code and/or string literals, such as const char * str=\"\u20ac100\"
you may need to:
- Configure your text editor to save in Windows-1252/ISO-8859-1 format (instead of UTF-8)
- use compiler flags like
--finput-charset
and -fexec-charset
By default, the Microsoft Visual Studio C compiler (MSVC) does not treat string literals as UTF-8. Instead, it treats them as being encoded in the current code page of the system, which is typically Windows-1252 on western european language Windows systems. twr-wasm is designed to work with clang, which does default to utf-8, so if you are compiling code written for MSVC, and you use extend character sets (non ASCII), you may need to adjust your compiler settings with the flags mentioned above.
"},{"location":"gettingstarted/charencoding/#more","title":"More","text":"For more details see Localization Reference for twr-wasm
"},{"location":"gettingstarted/compiler-opts/","title":"Compiling, Linking, and Memory Options","text":"This section describes how to use clang
to compile C/C++ code for WebAssembly, and how to use wasm-ld
to link your files into a .wasm module, when using twr-wasm.
twr-wasm lets you use clang directly, without a wrapper. This section describes the needed clang compile options and the wasm-ld link options. You can also take a look at the example makefiles.
"},{"location":"gettingstarted/compiler-opts/#compiler-notes","title":"Compiler Notes","text":"twr-wasm has been tested with clang 17.0.6 and wasm-ld 17.0.6.
If you are using nix, the default clang packages are wrapped with flags that break compilation. The following packages don't have this issue:
- llvmPackages_18.clang-unwrapped (clang 18.1.7)
- llvmPackages_17.clang-unwrapped (clang 17.0.6)
"},{"location":"gettingstarted/compiler-opts/#c-clang-compiler-options","title":"C clang Compiler Options","text":"When compiling C code with clang for use with Wasm and twr-wasm, use these clang options:
--target=wasm32 -nostdinc -nostdlib -isystem ../../include\n
Here is an example of a compile command:
clang --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include -c helloworld.c -o helloworld.o\n
-isystem
should be adjusted to point to where the folder twr-wasm/include
is installed. For example:
../../include
is a relative link to include
that works if your project is a sub folder in the examples
folder. ./node_modules/twr-wasm/include
assumes you installed with npm
into your project folder. (see the Hello World Walk Through).
"},{"location":"gettingstarted/compiler-opts/#c-clang-compiler-options_1","title":"C++ clang Compiler Options","text":"When compiling C++ code with clang for use with Wasm and twr-wasm, use these clang options:
--target=wasm32 -fno-exceptions -nostdlibinc -nostdinc -nostdlib -isystem ../../include\n
"},{"location":"gettingstarted/compiler-opts/#wasm-ld-linker-options","title":"wasm-ld Linker Options","text":"Use the wasm-ld linker directly with twr-wasm.
For example:
wasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm --no-entry --initial-memory=131072 --max-memory=131072 --export=hello \n
For C and C++ link to twr.a
to link to the twr-wasm library.
For C++ link to libc++.a
if you are using libc++. (see the tests-libcxx example makefile).
Be sure to adjust the path to twr.a
and libc++.a
as needed to the location where twr-wasm/lib-c/
is installed.
All of the twr-wasm functions are staticly linked from the library lib-c/twr.a
. There is also a version ( lib-c/twrd.a
) of twr-wasm library available with debug symbols. One of these two static libraries should be added to the list of files to link (normally this is twr.a
). Both versions are built with asserts enabled. twr.a
is built with -O3
. twrd.a
is built with -g -O0
.
C functions that you wish to call from JavaScript should either have an -export
option passed to wasm-ld
, or you can use the __attribute__((export_name(\"function_name\")))
option in your C function definition.
All exported functions to JavaScript should be C linkage (extern \"C\"
if using C++).
wasm-ld should be passed the following options:
If Using twrWasmModule:
--no-entry --initial-memory=<size> --max-memory=<size>\n
If Using twrWasmModuleAsync:
--no-entry --shared-memory --no-check-features --initial-memory=<size> --max-memory=<size>\n
"},{"location":"gettingstarted/compiler-opts/#memory-options-memory-size-stack-size-etc","title":"Memory Options (Memory Size, Stack Size, etc)","text":"WebAssembly.Memory
contains all the data used by your code (including the data needs of staticly linked libraries such as twr-wasm or libc++), but it does not store your actual code. It provides a contiguous, mutable array of raw bytes. Code execution and storage in WebAssembly are handled separately using the WebAssembly.Module
and WebAssembly.Instance
objects. The code (compiled WebAssembly instructions) is stored in the WebAssembly.Module
, while WebAssembly.Memory
is used to manage the linear memory accessible to the WebAssembly instance for storing data. Examples of data include your static data (.bss section or the .data section), the heap (used by malloc
and free
), and the stack (used for function calls and local variables).
The memory size should be a multiple of 64*1024 (64K) chunks. \"initial-memory\" and \"max-memory\" should be set to the same number since there is no support for automatically growing memory in twr-wasm. The memory is an export out of the .wasm
into the JavaScript code -- you should not create or set the size of WebAssembly.Memory
in JavaScript when using twr-wasm.
You set the memory size for your module (WebAssembly.Memory
) using wasm-ld
options as follows (this examples sets your Wasm memory to 1MB).
"},{"location":"gettingstarted/compiler-opts/#twrwasmmodule","title":"twrWasmModule","text":"if using twrWasmModule
:
--initial-memory=1048576 --max-memory=1048576\n
"},{"location":"gettingstarted/compiler-opts/#twrwasmmoduleasync","title":"twrWasmModuleAsync","text":"If you are using twrWasmModuleAsync
, shared memory must also be enabled. Like this:
--shared-memory --no-check-features --initial-memory=1048576 --max-memory=1048576\n
See this note on CORS headers with shared memory.
"},{"location":"gettingstarted/compiler-opts/#stack-size","title":"Stack Size","text":"You can change your C/C++ stack size from the default 64K with the following wasm-ld
option. This example sets the stack at 128K
-z stack-size=131072\n
"},{"location":"gettingstarted/compiler-opts/#print-memory-map","title":"Print Memory Map","text":"You can print your module memory map, heap stats, and stack size using the function from C:
void twr_mem_debug_stats(twr_ioconsole_t* outcon);\n
You can call it from Javascript with the output sent to the debug console (stderr) like this: twrWasmModule/Async.callC([\"twr_wasm_print_mem_debug_stats\"])\n
"},{"location":"gettingstarted/compiler-opts/#typescriptjavascript-malloc-and-memory-access","title":"TypeScript/JavaScript malloc and Memory Access","text":"twrWasmModule
and twrWasmModuleAsync
expose malloc
as an async function, as well as the WebAssembly Module memory as:
async malloc(size:number);\n\nmemory?:WebAssembly.Memory;\nmem8:Uint8Array;\nmem32:Uint32Array;\nmemD:Float64Array;\n
to call free
from JavaScript (you probably won't need to), you can use: twrWasmModule/Async.callC([\"twr_free\", index]); // index to memory to free, as returned by malloc\n
more information on these functions and module public variables can be found in the examples in this section: Passing Function Arguments to WebAssembly.
"},{"location":"gettingstarted/debugging/","title":"Debugging WebAssembly","text":"This section describes tips for debugging your WebAssembly (Wasm) program. Some of these techniques are WebAssembly generic, some are specific to using twr-wasm.
"},{"location":"gettingstarted/debugging/#debug-and-release-libraries","title":"Debug and Release libraries","text":"There are release (twr.a) and debug (twrd.a) versions of the twr-wasm C library. The \"debug\" version has debug symbols enabled with -g
and is built with optimizations disabled via -O0
. The \"release\" version has no debug symbols and optimization is set to -O3
. Both have asserts enabled. In general, you should use the \"release\" version unless you wish to step through the twr-wasm source -- in which case use the \"debug\" version.
libc++.a is not built with debug symbols.
"},{"location":"gettingstarted/debugging/#source-level-debugging-webassembly-cc","title":"Source Level Debugging WebAssembly C/C++","text":"In order to enable C/C++ source debugging with Wasm and clang, do the following:
- Use Chrome
- Install the Chrome extension: C/C++ DevTools Support (DWARF)
- Use the clang compile flag -g to add debug annotation to your object files
- You may want to turn off optimization to allow the debugger to have a bit more logical behavior (remove the
-O
flag or set to -O0
) - You may want to use the version of the twr-wasm C library that has debug symbols enabled (twrd.a). Only if you want to step into the twrd.a source.
- You need to serve your files with a (likely local) web server.
- For example, 'python server.py' is provided. 'server.py' can be found in the examples root folder. Note that your local server needs to enable SharedArrayBuffers if you are using
twrWasmModuleAsync
-- see these CORS notes. - your code can be bundled or unbundled, but
- you need to ensure that the web server/browser can find the source code
- also see Example Readme
"},{"location":"gettingstarted/debugging/#resolving-imports","title":"Resolving Imports","text":"If you are having issues with import resolution, see this section.
"},{"location":"gettingstarted/debugging/#useful-twr-wasm-debug-functions","title":"Useful twr-wasm Debug Functions","text":"Use twr_conlog
to print to the JavaScript console from C (see API ref section).
#include \"twr-crt.h\"\n\ntwr_conlog(\"hello 99 in hex: %x\",99);\n
Inside JavaScript, you can print to a console using the putStr
console member function that is available on all consoles.
For example:
const stream1 = new twrConsoleDiv(stream1Element);\nstream1.putStr(`Hello stream1 of type ${stream1.getProp(\"type\")} from JavaScript!\\n`);\n
"},{"location":"gettingstarted/debugging/#testing-webassembly-without-a-web-server","title":"Testing WebAssembly Without a Web Server","text":"Note: If you use this technique, you will not be able to get the C/C++ DevTool chrome extension to run, and so source level debugging won't work. (If you know how to fix this, please contact me on github.)
You can execute and debug JavaScript with Wasm from local files without an HTTP server. It might be helpful to download the twr-wasm source code from github when you do this (so you can step through the twr-wasm typescript code as needed).
See the examples and Example Readme for more detail on how this works.
In general, you will need to add a clip of code similar to this to your HTML:
<script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"./../../lib-js/index.js\"\n }\n }\n</script>\n
Make sure the paths to twr-wasm/lib-js/index.js
are correct for where your source is located. The above is correct for the provided examples.
You will need to set the following flags when running chrome from the shell or VS Code (the first is only strictly required if using twrWasmModuleAsync).
--enable-features=SharedArrayBuffer\n--allow-file-access-from-files\n
If you are using VS Code, You can create a launch.json entry similar to this:
launch.json{\n \"configurations\": [\n {\n \"name\": \"Launch Chrome\",\n \"request\": \"launch\",\n \"type\": \"chrome\",\n \"runtimeArgs\": [\n \"--allow-file-access-from-files\",\n \"--autoplay-policy=no-user-gesture-required\",\n \"--enable-features=SharedArrayBuffer\"\n ],\n \"file\": \"${workspaceFolder}/index.html\",\n \"cwd\": \"${workspaceFolder}/\",\n }\n ]\n}\n
"},{"location":"gettingstarted/events/","title":"Overview of Events","text":"This section describes how to use twr-wasm to:
- register event callbacks in C/C++
- use events in C/C++
"},{"location":"gettingstarted/events/#quick-example","title":"Quick Example","text":"timer events#include <stdio.h>\n#include \"twr-crt.h\"\n\nint t2_count=0;\nint t2_id;\n\n// timer2 event callback (called multiple times)\n__attribute__((export_name(\"on_timer2\")))\nvoid on_timer2(int event_id) {\n t2_count++;\n printf(\"timer callback 2 entered (event id=%d, count=%d)\\n\", event_id, t2_count);\n\n if (t2_count==5) {\n twr_timer_cancel(t2_id);\n printf(\"timer example complete\\n\")\n }\n}\n\n// C entry point to call from JavaScript\nint timer_main() {\n printf(\"the timer will trigger 5 times...\\n\");\n\n int t2_eventid=twr_register_callback(\"on_timer2\");\n t2_id=twr_timer_repeat(500, t2_eventid);\n}\n
"},{"location":"gettingstarted/events/#examples","title":"Examples","text":"Name View Live Link Source Link timer example View timer test Source library example View library example Source"},{"location":"gettingstarted/events/#events","title":"Events","text":"In twr-wasm, certain APIs can trigger events. For example a timer can trigger a \"timer complete\" event, or an audio api my trigger a \"file has finished playing\" event. An event had an id
and an associated callback. The id
is an integer that identifies the event (int event_id
). In order to receive an event call back:
- Write your callback in C/C++. It must be C linkage, and you should be exported from your C code to JavaScript/TypeScript using the
export_name
clang __attribute__
like this: __attribute__((export_name(\"on_timer2\")))
. Replace on_timer2
with your callback function name. - Register your callback. This will also allocate the event ID paired to this callback. For example:
int t2_event_id=twr_register_callback(\"on_timer2\");
- Call an API that takes an event
id
. For example: twr_timer_repeat(500, t2_eventid);
. This will call the t2_event
callback every 500ms.
You can use the same event/callback with multiple APIs if you wish. When the event callback is called, the first argument will be the event id
triggering the callback. There may then be optional parameters. These are event specific.
As in JavaScript, twr-wasm event callbacks only occur when your C/C++ code is not running.
"},{"location":"gettingstarted/events/#when-using-twrwasmmoduleasync","title":"When using twrWasmModuleAsync","text":"With a twrWasmModuleAsync
module, various blocking APIs are available. For example: twr_sleep
. When these functions are blocking (waiting), event callbacks are queued and not processed until your C functions return back to JavaScript.
With a twrWasmModuleAsync
module, events are sent from the JavaScript main thread to the worker thread that is running the C/C++ code.
"},{"location":"gettingstarted/events/#twr_register_callback","title":"twr_register_callback","text":"See twr_register_callback
"},{"location":"gettingstarted/helloworld/","title":"Create and Run WebAssembly Hello World","text":"This section shows you, step by step, how to to create a C \"hello world\" program for WebAssembly (Wasm) with twr-wasm, C, HTML, and JavaScript.
You will learn how to:
- Create the helloworld.c file
- Create the index.html file
- Compile the helloworld.c code with
clang
- Link the helloworld.o and twr.a files with
wasm-ld
to create a helloworld.wasm file - Set the needed library and include paths to allow the twr-wasm libraries to be discovered
- Create an optional Makefile
- Execute the \"hello world\" program using a local web server or directly with VS Code and Chrome
You can find code for a hello world example in the folder examples\\helloworld. It is similar, but not identical to this walk through. The primary differences are the paths for lib-c, lib-js, and include.
"},{"location":"gettingstarted/helloworld/#step-0-installation","title":"Step 0: Installation","text":""},{"location":"gettingstarted/helloworld/#step-1-create-the-c-code","title":"Step 1: Create the C code","text":"Create a file helloworld.c
in hello-proj
helloworld.c
#include <stdio.h>\n\nvoid hello() {\n printf(\"hello world\\n\");\n}\n
"},{"location":"gettingstarted/helloworld/#step-2-create-the-html","title":"Step 2: Create the HTML","text":"Create a file index.html
in hello-proj
index.html
<!doctype html>\n<html>\n<head>\n <title>Hello World</title>\n\n <script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"./node_modules/twr-wasm/lib-js/index.js\"\n }\n }\n </script>\n\n</head>\n<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n </script>\n</body>\n</html>\n
This example uses Import Maps, which are used when not using a bundler like WebPack or Parcel. For smaller projects, this can be simpler with a more clear debugging and development environment. This is the approach we will use for this example (no bundler).
The path in the importmap
section of index.html
should point to the location where you installed twr-wasm/lib-js
. The path above is correct for this project example with the indicated folder structure.
For more detail on import resolution see this section.
"},{"location":"gettingstarted/helloworld/#step-3-compile-your-c-code-to-create-your-wasm-file","title":"Step 3: Compile your C code to create your .wasm file","text":"cd hello-proj\nclang --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include -c helloworld.c -o helloworld.o\nwasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm --no-entry --initial-memory=131072 --max-memory=131072 --export=hello \n
The path to twr.a
and to include
should match your installation. The above path is correct for this example.
As an alternate to executing clang and wasm-ld from the shell, here is a Makefile that will work for this example:
MakefileCC := clang\nTWRCFLAGS := --target=wasm32 -nostdinc -nostdlib -isystem ./node_modules/twr-wasm/include\nCFLAGS := -c -Wall -O3 $(TWRCFLAGS)\nCFLAGS_DEBUG := -c -Wall -g -O0 $(TWRCFLAGS)\n\n.PHONY: default\n\ndefault: helloworld.wasm\n\nhelloworld.o: helloworld.c\n $(CC) $(CFLAGS) $< -o $@\n\nhelloworld.wasm: helloworld.o \n wasm-ld helloworld.o ./node_modules/twr-wasm/lib-c/twr.a -o helloworld.wasm \\\n --no-entry --initial-memory=131072 --max-memory=131072 \\\n --export=hello \n
Copy the above into a file named Makefile
and execute with make
(or mingw32-make
in windows).
"},{"location":"gettingstarted/helloworld/#step-4-load-and-execute-your-web-page","title":"Step 4: Load and execute your web page","text":"The two easiest ways to load and execute your index.html
web page locally are:
"},{"location":"gettingstarted/helloworld/#option-a-run-a-local-web-server","title":"Option A: Run a local web Server","text":"You can run a local server to view your helloworld program.
- Copy the file server.py from the examples folder to your
hello-proj
folder (where your index.html
resides). - Execute with the shell command
python server.py
. - Open your web browser and browse to
http://localhost:8000/index.html
- You should see 'Hello World' in the browser window!
At this pont your folder structure should look like this:
hello-proj\\\n\u2514\u2500\u2500node_modules\\\n\u2514\u2500\u2500helloworld.c\n\u2514\u2500\u2500helloworld.o\n\u2514\u2500\u2500helloworld.wasm\n\u2514\u2500\u2500index.html\n\u2514\u2500\u2500Makefile\n\u2514\u2500\u2500package.json\n\u2514\u2500\u2500server.py\n
"},{"location":"gettingstarted/helloworld/#option-b-vs-code-launchjson","title":"Option B: VS Code launch.json","text":"Alternately, you can launch chrome without a local web server. Add an entry similar to the following to hello-proj\\.vscode\\launch.json
. This assumes your workspaceFolder is hello-proj
.
launch.json{\n \"configurations\": [\n {\n \"name\": \"Launch Chrome Hello, World!\",\n \"request\": \"launch\",\n \"type\": \"chrome\",\n \"runtimeArgs\": [\n \"--allow-file-access-from-files\",\n \"--autoplay-policy=no-user-gesture-required\",\n \"--enable-features=SharedArrayBuffer\"\n ],\n \"file\": \"${workspaceFolder}/index.html\",\n \"cwd\": \"${workspaceFolder}/\",\n }\n ]\n}\n
Once you have created this file, you:
- select the Run and Debug icon on left
- Select the green play icon at the top, with \"Launch Chrome Hello, World!\" selected
- Chrome should launch, and you should see 'Hello World' in the browser window!
--autoplay-policy=no-user-gesture-required
and --enable-features=SharedArrayBuffer
are not required for this simple \"hello world\" example, but will be needed if you request user input or you are using twrWasModuleAsync
.
"},{"location":"gettingstarted/helloworld/#see-live-version","title":"See live version","text":"You can find a live link to hello world on this page.
"},{"location":"gettingstarted/helloworld/#next-steps-after-hello-world","title":"Next steps after hello world","text":"A good way to get your own code up and running is to copy one of the examples, get it to build and run, then start modifying it. Note you will need to modify the paths for include
, lib-js
, lib-c
, etc. based on your project structure. The examples are all setup with relative paths assuming the folder structure twr-wasm\\examples\\<example>
The examples include Makefiles.
\"Hello World\" uses the twr-wasm class twrWasmModule
. If you wish to use C blocking functions, such as twr_getc32
or twr_sleep
, you should use twrWasmModuleAsync
. This square calculator example shows how to do this.
If you wish to build an app that makes non-block calls into C, the balls example shows how to do this. The maze example uses a combination of blocking and non-blocking C functions.
"},{"location":"gettingstarted/helloworld/#debugging","title":"Debugging","text":"See the debugging section for debugging tips, including setting up Wasm source level debugging.
"},{"location":"gettingstarted/installation/","title":"Installing twr-wasm","text":"A simple way to install twr-wasm is:
npm install twr-wasm\n
See the \"Hello World walk through\" in the following section for more specifics.
There are actually two methods of installation with different pros and cons:
npm install
will install everything necessary to build your software: built libraries (lib-js, lib-c) and includes. In addition the examples are installed. git clone
will copy the above as well as the source and VS Code settings.
When using twr-wasm
your applications needs to access both JavaScript and C twr-wasm libraries. This is explained in the installation sections below, as well as in the Hello World walk through.
"},{"location":"gettingstarted/installation/#npm-install","title":"npm install","text":"npm install twr-wasm\n
After installation from npm, you will have a folder structure like this: node_modules\\\n twr-wasm\\\n examples\\\n include\\\n lib-c\\\n lib-js\\\n LICENSE\n package.json\n readme.md\n
The JavaScript and TypeScript exports are in lib-js
and should be found by VS Code, TypeScript or your bundler as usual when using a statement like import {twrWasmModule} from \"twr-wasm\"
. The C library (twr.a
) that you will need to link your C/C++ program to is found in the libs-c
folder, and the C/C++ include files that you will need to use in your C/C++ program are found in the include
folder. You will need to use paths to to these folders in your makefile. See the Hello World walk through for details.
There is no real downside to this installation method, except possibly: (1) it does not include source code (use git clone for that), and (b) the C libraries are buried inside your node_modules.
"},{"location":"gettingstarted/installation/#git-install","title":"git install","text":" git clone https://github.com/twiddlingbits/twr-wasm\n
This method of installation installs the complete code base, including source and built binaries.
After twr-wasm is cloned, use VS Code File | Open Folder
.
See here for information on running the examples or building the examples.
See here for information on building the source.
The primary downside to this method is that the JavaScript side of twr-wasm will not be placed in a node_modules folder. This will create a little extra work to configure a bundler, TypeScript or VS Code to find the location of the twr-wasm module imports.
There are a few solutions to this. For example, in the provided Hello World example, a package.json
file with an alias
entry is used. This syntax is supported by the Parcel bundler:
{\n \"@parcel/resolver-default\": {\n \"packageExports\": true\n },\n \"alias\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n },\n \"dependencies\": {\n \"twr-wasm\": \"^2.0.0\"\n }\n}\n
The FFT example uses the paths
entry in the tsconfig.json
file. This is found by TypeScript, VS Code and the Parcel bundler. This is probably the best solution if you are using TypeScript.
\"paths\": {\n \"twr-wasm\": [\"./../../lib-js/index\"]\n}\n
The paths for alias
and paths
shown above are correct for the included examples, but will likely need to be adjust for your project.
For more details see this section on Import Resolution
"},{"location":"gettingstarted/installation/#clang-and-wasm-ld","title":"clang and wasm-ld","text":"To build C/C++ code for use in your Wasm project, you will need to install clang and the wasm-ld linker. If you are using Windows, more details can be found at the end of the Building Source section.
"},{"location":"gettingstarted/installation/#python-and-more","title":"python and more","text":"To use the included examples\\server.py
to execute your project you will need to install python. server.py
is a simple HTTP server for local testing that sets the correct CORS headers for twrWasmModuleAsync
. As explained in the Hello World walk through, you can alternately execute HTML files directly using VS Code and Chrome.
You will likely want these tools installed to use twr-wasm:
- gnu make (all the examples use make)
- VS Code (not required, but the repo includes VS Code launch.json, etc)
- TypeScript (not required, but the twr-wasm source code is TypeScript)
"},{"location":"gettingstarted/installation/#note-on-examples","title":"Note on Examples","text":"You can run the examples either locally or with the online versions.
"},{"location":"gettingstarted/parameters/","title":"Passing Function Arguments to WebAssembly","text":"This article describes techniques to transfer data between JavaScript/TypeScript and C/C++ when using WebAssembly. It delves a bit \u201cunder the covers\u201d to explain how this works when you use a library like twr-wasm or Emscripten. In this article, I am using twr-wasm for the examples. Emscripten does something similar.
For an example that illustrates the concepts discussed here, see: the callC example.
"},{"location":"gettingstarted/parameters/#webassembly-virtual-machine-intrinsic-capabilities","title":"WebAssembly Virtual Machine Intrinsic Capabilities","text":"The WebAssembly VM (often referred to as a Wasm \u201cRuntime\u201d) is limited to passing numbers between C functions and the Wasm host (I\u2019ll assume that\u2019s JavaScript for this document). In other words, if you are using the most basic WebAssembly capabilities provided by JavaScript, such as WebAssembly.Module
, WebAssembly.Instance
, and instance.exports
, your function calls and return types can only be:
- Integer 32 or 64 bit
- Floating point 32 or 64 bit
These correspond to the WebAssembly spec support for: i32, i64, f32, and f64.
Note that a JavaScript number
is of type Float 64 (known as a double
in C/C++.). If you are storing an integer into a JavaScript number
, it is converted to a Float 64, and its maximum \"integer\" precision is significantly less than 64 bits (its about 52 bits, but this is a simplification). As a result, to use a 64-bit integers with JavaScript the bigint
type is used.
When using 32-bit WebAssembly (by far the most common default), and you call a C function from JavaScript without using any \u201chelper\u201d libraries (like twr-wasm), the following argument types can be passed:
- Integer 32: JavaScript
number
type is converted to an Integer 32 and passed to C when the C function prototype specifies a signed or unsigned int
, long
, int32_t
, or a pointer type. All of these are 32 bits in length in wasm32. - Integer 64: JavaScript
bigint
type is converted to an Integer 64 and passed to C when the C function prototype specifies signed or unsigned int64_t
(or equivalent). Attempting to pass a JavaScript number
to a C int64_t
will fail with a JavaScript runtime error. - Float 32: JavaScript
number
type is converted to a Float 32 when the C function prototype specifies a float
. - Float 64: JavaScript
number
type is passed as a Float 64 when the C function prototype specifies a double
.
The same rules apply to the return types.
"},{"location":"gettingstarted/parameters/#c-structs-javascript-c","title":"C Structs: JavaScript <--> C","text":"This section shows how to create a C struct
in JavaScript, then pass it to a C function, and then read the modified C struct
in JavaScript.
Although the techniques described here are explained with a struct
example, the basic techniques are used with other data types as well (such as strings). For common data types, like a string, libraries like twr-wasm will handle these details for you automatically.
To create and pass a C struct
from JavaScript to C, the technique is to call the WebAssembly C malloc
from JavaScript to allocate WebAssembly memory and then manipulating the memory in JavaScript. One complexity is that each struct entry\u2019s memory address needs to be calculated. And when calculating the WebAssembly Memory indices for the struct entries, C structure padding must be accounted for.
"},{"location":"gettingstarted/parameters/#struct-entry-padding","title":"struct Entry Padding","text":"Before we delve into the actual code, lets review C struct entry padding.
In clang, if you declare this structure in your C code:
struct test_struct {\n int a;\n char b;\n int *c;\n};\n
- The first entry,
int a
, will be at offset 0 in memory (from the start of the struct
in memory). - The second entry,
char b
, will be at offset 4 in memory. This is expected since the length of an int is 4 bytes. - The third entry,
int *c
, will be at offset 8 in memory, not at offset 5 as you might expect. The compiler adds three bytes of padding to align the pointer to a 4-byte boundary.
This behavior is dependent on your compiler, cpu, and whether you are using 32 or 64-bit architecture. For wasm32 with clang:
- char is 1 byte aligned
- short is 2 byte aligned
- pointers are 4 byte aligned
- int, long, int32_t are 4 byte aligned
- double (Float 64) is 8-byte aligned
If you are not familiar with structure padding, there are many articles on the web.
Alignment requirements are why twr-wasm malloc
(and GCC malloc
for that matter) aligns new memory allocations on an 8-byte boundary.
"},{"location":"gettingstarted/parameters/#creating-a-struct-in-javascript","title":"Creating a struct in JavaScript","text":"We can create and initialize the above struct test_struct
like this in JavaScript:
//...\nconst mod = new twrWasmModule();\n//...\nconst structSize=12;\nconst structIndexA=0;\nconst structIndexB=4;\nconst structIndexC=8; // compiler allocates pointer on 4 byte boundaries\nlet structMem=mod.wasmMem.malloc(structSize);\nlet intMem=mod.wasmMem.malloc(4);\nmod.wasmMem.setLong(structMem+structIndexA, 1);\nmod.wasmMem.mem8[structMem+structIndexB]=2; // you can access the memory directly with the mem8, mem32, and memD (float64 aka double) byte arrays.\nmod.wasmMem.setLong(structMem+structIndexC, intMem);\nmod.wasmMem.setLong(intMem, 200000);\n
note that:
mod.wasmMem.malloc(structSize)
is a shortcut for: mod.callC([\"malloc\", structSize])
mod.wasmMem.malloc
returns a C pointer as a number
. This pointer is also an index into WebAssembly.Memory
-- which is exposed as the byte array (Uint8Array
) via mod.wasmMem.mem8
by twr-wasm. - When accessing a C
struct
in JavaScript/TypeScript, you have to do a bit of arithmetic to find the correct structure entry. - The entry
int *c
is a pointer to an int
. So a separate malloc
to hold the int
is needed. - In twr-wasm there is no function like
setLong
to set a byte. Instead you access the byte array view of the WebAssembly memory with mod.wasmMem.mem8
. Functions like mod.wasmMem.setLong
manipulate this byte array for you. - As well as
mod.wasmMem.mem8
(Uint8Array), you can also access WebAssembly.Memory directly via mod.wasmMem.mem32
(Uint32Array), and mod.wasmMem.memD
(Float64Array). - The list of functions available to access WebAssembly.Memory can be found at the end of this page.
"},{"location":"gettingstarted/parameters/#passing-struct-to-c-from-javascript","title":"Passing struct to C from JavaScript","text":"Assume we have C code that adds 2 to each entry of the test_struct
:
__attribute__((export_name(\"do_struct\")))\nvoid do_struct(struct test_struct *p) {\n p->a=p->a+2;\n p->b=p->b+2;\n (*p->c)++;\n (*p->c)++;\n}\n
Once the struct
has been created in JavaScript, you can call the C function do_struct
that adds 2 to each entry like this in twr-wasm:
await mod.callC([\"do_struct\", structMem]); // will add two to each value\n
"},{"location":"gettingstarted/parameters/#reading-c-struct-in-javascript","title":"Reading C struct in JavaScript","text":"You read the modified elements like this using JavaScript:
success=mod.wasmMem.getLong(structMem+structIndexA)==3;\nsuccess=success && mod.wasmMem.mem8[structMem+structIndexB]==4;\nconst intValPtr=mod.wasmMem.getLong(structMem+structIndexC);\nsuccess=success && intValPtr==intMem;\nsuccess=success && mod.wasmMem.getLong(intValPtr)==200002;\n
You can see the additional complexity of de-referencing the int *
.
"},{"location":"gettingstarted/parameters/#cleanup","title":"Cleanup","text":"You can free the malloced memory like this:
mod.wasmMem.free(intMem);\nmod.wasmMem.free(structMem);\n
The complete code for this example is here.
"},{"location":"gettingstarted/parameters/#passing-strings-from-javascript-to-cc-webassembly","title":"Passing Strings from JavaScript to C/C++ WebAssembly","text":"Although you can use the technique I am about to describe here directly (by writing your own code), it is generally accomplished by using a third-party library such as twr-wasm or Emscripten. These libraries handle the nitty-gritty for you.
To pass a string from JavaScript/TypeScript to a WebAssembly module, the general approach is to:
- Allocate memory for the string inside the WebAssembly memory. This is typically done by calling the C
malloc
from JavaScript. malloc
returns a pointer, which is an index into the WebAssembly Memory. - Copy the JavaScript string to this malloc'd Wasm memory. In the case of twr-wasm, this copying also converts the character encoding as necessary, for example, to UTF-8.
- Pass the malloc'd memory index to your function as an integer (which is accepted as a pointer by C code).
In the case of twr-wasm, the above steps are handled automatically for you by the callC
function:
mod.callC([\"my_function\", \"this is my string\"]); // mod is instance of twrWasmModule\n
Under the covers, to pass \"this is my string\" from JavaScript to the C Web Assembly function, callC
will execute code like this:
// twrWasmMemory member function\nputString(sin:string, codePage = codePageUTF8) {\n const ru8 = this.stringToU8(sin, codePage); // convert a string to UTF8 encoded characters stored in a Uint8Array\n const strIndex = this.malloc(ru8.length + 1); // shortcut for: await this.callC([\"malloc\", ru8.length + 1]);\n this.mem8.set(ru8, strIndex); // mem8 is of type Uint8Array and is the Wasm Module\u2019s Memory\n this.mem8[strIndex + ru8.length] = 0;\n return strIndex;\n}\n
this.malloc
is the standard C runtime malloc
function, provided by twr-wasm, and linked into your .wasm
code that is loaded into the WebAssembly Module. Likewise, twr-wasm will call free
after the function call is executed."},{"location":"gettingstarted/parameters/#returning-a-string-from-cc-webassembly-to-javascript","title":"Returning a String from C/C++ WebAssembly to JavaScript","text":"Returning a string from C to JavaScript is the reverse of passing in a string from JavaScript to C. When the \u201craw\u201d WebAssembly capabilities are used (WebAssembly.Module
, etc.) and your C code looks like this:
return(\"my string\");\n
The WebAssembly VM and JavaScript host will cause your JavaScript to receive an unsigned 32-bit integer. This is the pointer to the string, cast to an unsigned 32-bit integer. This integer is an index into the WebAssembly Memory.
twr-wasm provides a function to pull the string out of WebAssembly Memory and convert the character encoding to a JavaScript string. JavaScript strings are Unicode 16, but twr-wasm supports ASCII, UTF-8, and windows-1252 string encoding. When extracted and converted, a copy of the string is made.
const retStringPtr = await mod.callC([\"ret_string_function\"]);\nconsole.log(mod.wasmMem.getString(retStringPtr));\n
The retStringPtr
is an integer 32 (but converted to a JavaScript number
, which is Float 64). This integer is an index into the WebAssembly Memory.
"},{"location":"gettingstarted/parameters/#passing-arraybuffers-from-javascript-to-cc-webassembly","title":"Passing ArrayBuffers from JavaScript to C/C++ WebAssembly","text":"When callC
in twr-wasm is used to pass an ArrayBuffer to and from C/C++, some details are handled for you. The technique is similar to that used for a string
or as performed manually for a struct
above, with the following differences:
ArrayBuffers
have entries of all the same length, so the index math is straight forward and no struct
padding is needed. - When an
ArrayBuffer
is passed to a function, the function receives a pointer to the malloc
memory. If the length is not known by the function, the length needs to be passed as a separate argument. - Before
callC
returns, any modifications made to the memory by the C code are reflected back into the ArrayBuffer
. - the malloced copy of the ArrayBuffer is freed.
Here is an example:
let ba = new Uint8Array(4);\nba[0] = 99; ba[1] = 98; ba[2] = 97; ba[3] = 6;\nconst ret_sum = await mod.callC([\"param_bytearray\", ba.buffer, ba.length]);\n
See this example for the complete example.
"},{"location":"gettingstarted/parameters/#passing-a-javascript-object-to-webassembly","title":"Passing a JavaScript Object to WebAssembly","text":""},{"location":"gettingstarted/parameters/#simple-case-use-c-struct","title":"Simple Case - use C struct","text":"For a simple object like this:
const a = 'foo';\nconst b = 42;\n\nconst obj = {\n a: a,\n b: b\n};\n
It is straightforward to convert to a C struct like this:
struct obj {\n const char* a;\n int b;\n};\n
To pass this JavaScript object to WebAssembly, a C struct is created (using the struct
techniques described above). Each object entry is then copied into the corresponding C struct
entry (using the struct
and string techniques described above)."},{"location":"gettingstarted/parameters/#more-complicated-object","title":"More Complicated Object","text":"A JavaScript object can contain entries that are of more complexity than simple C data types. For example:
const a = 'foo';\nconst b = 42;\nconst map = new Map();\nmap1.set('a', 1);\nmap1.set('b', 2);\nmap1.set('c', 3);\nconst object2 = { a: a, b: b, c: map };\n
In this case, you are going to have to do more work. An approach is to use the libc++ map
class, which is similar to the JavaScript Map
. You could also perhaps use the libc++ vector
.
To handle this more complicated JavaScript object with a Map
entry, an approach is to export functions from WebAssembly to create and add entries to the libc++ map
(you need to use extern 'C'
to export these C++ access functions as C functions). In otherworld, you might export from your Wasm Module C functions like this:
void* createMap(); // return an unsigned long Map ID\nvoid addIntToMap(void* mapID, int newInt);\n
You would then use these functions in JavaScript to build your C++ map
. JavaScript would access this map
using the unsigned long
identifier (the void *
returned by createMap
). After creating and adding entries to the map
, you would set this MapID to object2.c
.
There are alternative approaches. For example, you could convert the JavaScript Map
to a C struct, by enumerating every entry in the Map
. Your C struct might look like: `
struct entry {\n char* name;\n int value;\n};\n\nstruct mapUnroll {\n int MapLen;\n struct entry* entries[];\n};\n
This approach is probably even more work, less general purpose, and less efficient.
"},{"location":"gettingstarted/parameters/#summary","title":"Summary","text":"I hope this has demystified how JavaScript values are passed to and from WebAssembly. In many cases, functions like twr-wasm's mod.callC
will handle the work for you. But in more bespoke cases, you will have to handle some of the work yourself.
"},{"location":"gettingstarted/stdio/","title":"Overview of Consoles","text":"This section describes how to use twr-wasm to:
- create input/output consoles for use by C/C++ with WebAssembly
- direct stdin, stdout and stderr to a console
- use addressable display and canvas 2D consoles
- use multiple consoles at once
"},{"location":"gettingstarted/stdio/#quick-example","title":"Quick Example","text":"Hello World#include <stdio.h>\n\nvoid hello() {\n printf(\"hello world\\n\");\n}\n
Using twrConsoleDiv<body>\n <div id=\"console-tag\"></div>\n\n <script type=\"module\">\n import {twrConsoleDiv, twrWasmModule} from \"twr-wasm\";\n\n const tag=document.getElementById(\"console-tag\");\n const streamConsole=new twrConsoleDiv(tag); \n const mod = new twrWasmModule({stdio: streamConsole});\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n\n </script>\n</body>\n
Using twr_iodiv Shortcut<body>\n <div id=\"twr_iodiv\"></div>\n\n <script type=\"module\">\n import {twrWasmModule} from \"twr-wasm\";\n\n const mod = new twrWasmModule();\n await mod.loadWasm(\"./helloworld.wasm\");\n await mod.callC([\"hello\"]);\n\n </script>\n</body>\n
"},{"location":"gettingstarted/stdio/#running-examples","title":"Running Examples","text":"Name View Live Link Source Link stdin and stdout to <div>
View square demo Source simple \"terminal\" via <canvas>
View hello world demo Source \"cli\" with a <canvas>
stdio View CLI demo using libc++ Source Multiple Consoles, including Canvas2D View multi-io demo Source"},{"location":"gettingstarted/stdio/#capabilities","title":"Capabilities","text":"With a Console you can:
- read character streams (Use C statements like
getc
or io_mbgets
) - write character streams (use C statements like
printf
or cout
) - position characters, graphics, colors with an addressable display (Use C statements like
io_setc32
or io_set_cursor
). - draw to a Canvas compatible 2D surface (Use C statements like
d2d_fillrect
).
Consoles are primarily designed for use by twr-wasm C/C++ modules, but they can also be used by JavaScript/TypeScript.
Although it is common to have a single console, an arbitrary number of consoles can be created, and they can be used by an arbitrary number of twr-wasm C/C++ modules.
Unicode characters are supported by consoles (see Character Encoding Support with twr-wasm).
"},{"location":"gettingstarted/stdio/#tag-shortcuts","title":"Tag Shortcuts","text":"If you add a <div id=\"twr_iodiv\">
, a <canvas id=\"twr_iocanvas\">
, or a <canvas id=\"twr_d2dcanvas\">
tag to your HTML, twr-wasm will create the appropriate class for you when you instantiate the class twrWasmModule
or twrWasmModuleAsync
. Use these tag shortcuts as an aternative to instantiating the console classes in your JavaScript/TypeScript.
<div id=\"twr_iodiv\">
will be used to create a twrConsoleDiv
as stdio
<canvas id=\"twr_iocanvas\">
will be used to create a twrConsoleTerminal
as stdio
. <canvas id=\"twr_d2dcanvas\">
will be used to create a twrConsoleCanvas
as std2d
-- the default 2D drawing surface. See 2D drawing APIs.
If neither of the above <div>
or <canvas>
is defined in your HTML, and if you have not set stdio
via the io
or stdio
module options, then stdout
is sent to the debug console in your browser. And stdin
is not available.
"},{"location":"gettingstarted/stdio/#console-classes","title":"Console Classes","text":"Consoles are implemented in TypeScript and run in the JavaScript main thread. This allows consoles to be shared by multiple wasm modules.
For simple cases, when you use the tag shortcuts, you won't need to use these console classes directly. For more bespoke cases, they will come in handy. For details on console classes, see the TypeScript/JavaScript API reference
These conosle classes are available in twr-wasm:
twrConsoleDiv
streams character input and output to a div tag twrConsoleTerminal
provides streaming or addressable character input and output using a canvas tag. twrConsoleDebug
streamings characters to the browser debug console. twrConsoleCanvas
creates a 2D drawing surface that the Canvas compatible 2d drawing APIs can be used with.
"},{"location":"gettingstarted/stdio/#multiple-consoles-with-names","title":"Multiple Consoles with Names","text":"When you instantiate a class twrWasmModule
or twrWasmModuleAsync
, you can pass it the module option io
-- a javascript object containing name-console attributes. Your C/C++ code can then retrieve a console by name. This is described in more detail the TypeScript/JavaScript API Reference.
Also see the multi-io example.
"},{"location":"gettingstarted/stdio/#setting-stdio-and-stderr","title":"Setting stdio and stderr","text":"stdio
can be defined automatically if you use a Tag Shortcut. stderr
streams to the browser's debug console by default. Both can be set to a specific console with the module io
option.
For example, given:
const tag=document.getElementById(\"console-tag\");\nconst streamConsole=new twrConsoleDiv(tag);\n
Either of these will set stdio
to a streaming div
console:
const mod = new twrWasmModule({stdio: streamConsole});\n
const mod = new twrWasmModule({ io: {stdio: streamConsole} });\n
This option would send stderr and stdio to the same console:
const mod = new twrWasmModule({ io: \n {stdio: streamConsole, stderr: streamConsole} \n});\n
"},{"location":"gettingstarted/stdio/#utf-8-or-windows-1252","title":"UTF-8 or Windows-1252","text":"Consoles can support UTF-8 or Windows-1252 character encodings (see Character Encoding Support with twr-wasm).
"},{"location":"gettingstarted/stdio/#c-access-to-consoles","title":"C Access To Consoles","text":""},{"location":"gettingstarted/stdio/#io_functions","title":"io_functions","text":"io_functions
are available to operate on all character based Consoles.
"},{"location":"gettingstarted/stdio/#d2d_functions","title":"d2d_functions","text":"d2d_functions
are available to operate on Canvas 2D Consoles.
"},{"location":"gettingstarted/stdio/#reading-from-a-console","title":"Reading from a Console","text":"Reading from a console is blocking, and so twrWasmModuleAsync
must be used to receive keys. There are some specific requirements to note in the twrWasmModuleAsync
API docs.
You can get characters with any of these functions:
io_mbgets
- get a multibyte string from a console using the current locale character encoding. Console must support IO_TYPE_CHARREAD. twr_mbgets
- the same as io_mbgets
with the console set to stdin
. io_mbgetc
- get a multibyte character from an twr_ioconsole_t *
(aka FILE *
) like stdin
using the current locale character encoding getc
(same as fgetc
) - get a single byte from a FILE *
(aka twr_ioconsole_t *
) -- returning ASCII or extended ASCII (window-1252 encoding) io_getc32
- gets a 32 bit unicode code point from an twr_ioconsole_t *
(which must support IO_TYPE_CHARREAD)
"},{"location":"gettingstarted/stdio/#standard-c-library-functions","title":"Standard C Library Functions","text":"Many of the common standard C library functions, plus twr-wasm specific functions, are available to stream characters to and from the standard input and output console that supports character streaming (most do).
In C, a console is represented by twr_ioconsole_t
. In addition, FILE
is the same as a twr_ioconsole_t
(typedef twr_ioconsole_t FILE
). stdout
, stdin
, stderr
are all consoles.
#include <stdio.h>
to access stdout
, stdin
, stderr
, and FILE
.
FILE
is supported for user input and output, and for stderr. FILE
as filesystem I/O is not currently supported.
"},{"location":"gettingstarted/stdio/#stdout-and-stderr-functions","title":"stdout and stderr functions","text":"You can use these functions to output to the standard library defines stderr
or stdout
:
fputc, putc, vfprintf, fprintf, fwrite\n
These functions go to stdout
:
printf, vprintf, puts, putchar\n
Note that when characters are sent to the browser console using stderr
they will not render to the console until a newline, return, or ASCII 03 (End-of-Text) is sent.
For example:
#include <stdio.h>\n\nfprintf(stderr, \"hello over there in browser debug console land\\n\");\n
A more common method to send output to the debug console is to use twr_conlog
.
"},{"location":"more/building/","title":"Building the twr-wasm Source","text":""},{"location":"more/building/#source-for-twr-wasm","title":"Source for twr-wasm","text":"The source can be found at:
https://github.com/twiddlingbits/twr-wasm\n
The main
branch contains the latest release. The dev
branch is work in progress.
"},{"location":"more/building/#tools-needed-to-build-twr-wasm-source","title":"Tools Needed to Build twr-wasm Source","text":"You will need these core tools, versions used in release are in ():
- TypeScript (5.4.5)
- clang tool chain (17.0.6) - for C/C++ code
- wasm-ld (17.0.6) - to link the .wasm files
- wat2wasm (1.0.34) - to compile WebAssembly (.wat) files of which I have a few
- GNU make (4.4.1)
- git - to clone twr-wasm source, or to clone llvm, if you want to build libc++
In addition, you might need:
- VS Code - to use the debug launcher and build tasks
- NPM - package manager
- Parcel v2 - to bundle the examples
- mkdocs, material theme, meta-descriptions plugin - to build the documentation static web site
- python - mkdocs is built with python, and you need python to run server.py in examples
- CMake and ninja - to build llvm libc++
There is a deprecated gcc build that I used to use for testing, but now the tests are executed in wasm.
"},{"location":"more/building/#to-build-the-libraries-lib-c-lib-js","title":"To Build the Libraries (lib-c, lib-js)","text":"cd source\nmake\n
or on windows cd source\nmingw32-make\n
"},{"location":"more/building/#to-build-the-examples","title":"To Build the Examples","text":"See examples/readme.md for more information.
To build the examples, but not bundle them.
cd examples\nsh buildall.sh\n
To build bundles:
sh buildbundles.sh\n
"},{"location":"more/building/#to-build-the-docs","title":"To Build the docs","text":"The docs are created using the material theme for mkdocs.
In twr-wasm root folder:
mkdocs build\n
The destination of the build is found in the mkdocs.yml
file (site_dir: azure/docsite/
).
Usually the docs are built as part of building the static web site that hosts the docs and examples. This is accomplished using this shell script (found in examples folder):
buildazure.sh\n
"},{"location":"more/building/#to-build-libc-for-wasm-and-twr-wasm","title":"To Build libc++ for Wasm and twr-wasm","text":"See the instructions in the comments in the shell script source\\libcxx\\buildlibcxx.sh
"},{"location":"more/building/#installing-clang-and-wasm-ld-on-windows","title":"Installing clang and wasm-ld on Windows","text":"Here is how I installed the tools for windows:
install MSYS2 \n 1. https://www.msys2.org/\n 2. After the install completes, run UCRT64 terminal by clicking on the MSYS2 UCRT64 in the Start menu\n 3. pacman -Syuu\n\n install gcc using MSYS2 UCRT64\n 1. Use MSYS2 UCRT64 terminal (per above)\n 1. pacman -S mingw-w64-ucrt-x86_64-toolchain\n\n install clang and wasm-ld using MSYS2 UCRT64\n 2. Use MSYS2 UCRT64 (per above)\n 1. pacman -S mingw-w64-ucrt-x86_64-clang\n 2. pacman -S mingw-w64-x86_64-lld\n\nupdate PATH env variable using the windows control panel (search for path)\n 2. added C:\\msys64\\ucrt64\\bin \n 3. added C:\\msys64\\mingw64\\bin \n 4. added C:\\msys64\\usr\\bin (for sh.exe used by mingw32-make)\n
wabt tools: can be found here https://github.com/WebAssembly/wabt/releases
"},{"location":"more/imports/","title":"twr-wasm Import Resolution","text":"This section covers path resolution for statements like this:
import {twrWasmModule} from \"twr-wasm\";\n
It can be confusing to determine how tools like VS Code, a bundler, or a Web Browser resolve imports like \"twr-wasm\". This section explains how it works, using the included examples as examples.
Two situations are addressed in this document: (1) browser import resolution when not using a bundler, and (2) if you installed using git clone
.
"},{"location":"more/imports/#if-you-installed-using-git-clone","title":"If you installed using git clone
","text":"If you have installed twr-wasm
using npm
, most tools will automatically resolve imports by finding the node_modules
folder.
However, if you installed using git clone
, then other steps will need to be taken to help a bundler, VS Code, and tsc resolve imports. This situation can apply to the examples, which are in the git repo tree, and as a result there may be no node_modules
for the twr-wasm libraries. This will also be the case for your project, if you installed with git clone
.
"},{"location":"more/imports/#import-path-resolution-by-the-bundler","title":"Import path resolution by the bundler","text":"A bundler will find the twr-wasm library using one of these methods:
- If twr-wasm has been installed with npm install, the bundler will find the
node_modules
folder - Alternately, If all your scripts are in TypeScript, use
paths
entry in tsconfig.json
(see maze example) - Alternately, use alias option in package.json as in the helloworld example
\"alias\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n },\n
In the examples, the alias entry in the package.json
exists so that the parcel bundler can find twr-wasm.
If you are using a bundler, you don't need to add a <script type=\"importmap\">
tag.
"},{"location":"more/imports/#import-resolution-by-vs-code-and-tsc","title":"Import resolution by VS Code and tsc","text":"VS Code Intellisense and the typescript compiler need to find modules. If twr-wasm is installed using npm
into a node_modules
folder, this is probably automatic. But if this is not the case, you can add a line to the tsconfig.json
as follows (this example assumes the tsconfig.json
is in a examples/example folder). See the maze example.
\"paths\": {\n \"twr-wasm\": [\"./../../lib-js/index\"]\n}\n
"},{"location":"more/imports/#import-path-resolution-by-the-browser","title":"Import path resolution by the browser","text":"This section apples to executing your javascript without first \"bundling\" it. Whether execution is from the filesystem directly in a browser or using a web server.
In order for the browser to locate the twr-wasm path when import is used, you can add code like this to your HTML prior to the import. You should make sure the path for twr-wasm is correct for your project (this is correct for the examples).
<script type=\"importmap\">\n {\n \"imports\": {\n \"twr-wasm\": \"../../lib-js/index.js\"\n }\n }\n</script>\n
"},{"location":"more/production/","title":"HTTP CORS headers needed to use twrWasmModuleAsync","text":"If you are using twr-wasm with web pages served by an http server, you may need to enable certain CORS headers. This applies whether using a remote server or using your local machine for development.
twr-wasm class twrWasmModuleAsync
uses SharedArrayBuffers
, and there are special CORS headers that need to be configured to use SharedArrayBuffers
, that are not widely enabled by default on web servers. It may be helpful to also see the SharedArrayBuffer
documentation online.
Github pages doesn't support the needed CORS headers for SharedArrayBuffers. But other web serving sites do have options to enable the needed CORS headers.
Here are two provided examples of how to enable the necessary headers:
- server.py
- staticwebapp.config.json
The azure static web site config file staticwebapp.config.json
looks like this:
{\n \"globalHeaders\": {\n \"Access-Control-Allow-Origin\": \"*\",\n \"Cross-Origin-Embedder-Policy\": \"require-corp\",\n \"Cross-Origin-Opener-Policy\": \"same-origin\"\n }\n}\n
server.py in the examples folder will launch a local server with the correct headers. To use Chrome without a web server, see the Hello World walk through.
"},{"location":"more/production/#using-twrwasmmoduleasync-with-file","title":"Using twrWasmModuleAsync with file:","text":"If you are loading html with chrome from files (not using an https server), you will need to set these command line args:
--enable-features=SharedArrayBuffer\n--allow-file-access-from-files\n
More detail is found in the debugging section.
"},{"location":"more/wasm-problem/","title":"Wasm Runtime Limitations","text":"HTML browsers can load a WebAssembly module, and execute it's bytecode in a virtual machine. To create this bytecode (.wasm file) from C/C++, you compile your code using clang with the target code format being WebAssembly (aka Wasm) byte code. If you are not using a support library like twr-wasm or emscripten, there are a few issues that one immediately encounters trying to execute code that is more complicated than squaring a number.
The Wasm virtual machine simply executes the instructions that are generated by the clang compiler and linked by the linker into the .wasm file. The first issue encountered is that some code that is generated by a compiler assumes a compiler support library will be linked to your code. That is, clang code generation will produce calls to compiler support routines for floating point, memcpy
, and the like. In clang, these support routines are in the \"compile-rt\" support library. Typically clang handles this behind to scenes for you. But support for a WebAssembly version of this compiler support library is not (as of this writing) included in a clang distribution.
The next level up the library stack is the standard C runtime library. This library provides functions like malloc
and printf
. And then built on the standard C runtime library is the standard c++ library - like libc++. WebAssembly versions of these libraries are also not part of a clang distribution.
To get access to WebAssembly versions of these libraries you need to use emscripten or twr-wasm.
The second problem is that all the function calls between your Wasm module and your javascript are limited to parameters and return values that are numbers (integer and float). No strings, arrays, struct pointers, etc. (for more on this see this doc).
The third problem is that legacy C code or games often block, and when written this way they don't naturally integrate with the JavaScript asynchronous programming model.
twr-wasm is a static C library (twr.a) that you can link to your clang C/C++ Wasm code, as well as a set of JavaScript/TypeScript modules that solve these issues.
In addition, twr-wasm provides APIs that you can use in your WebAssembly code - such as Canvas compatible 2D drawing APIs, a simple terminal emulator, character encoding support, and more.
"}]}
\ No newline at end of file
diff --git a/azure/docsite/sitemap.xml b/azure/docsite/sitemap.xml
index 5bf565cd..16b42433 100644
--- a/azure/docsite/sitemap.xml
+++ b/azure/docsite/sitemap.xml
@@ -2,197 +2,197 @@
https://twiddlingbits.dev/docsite/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-audio/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-con/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-d2d/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-general/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-localization/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-c-stdlib/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-libcpp/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-ts-consoles/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-ts-library/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-ts-memory/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/api/api-ts-modules/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-balls/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-callc/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-divcon/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-fft/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-helloworld/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-lib/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-libcxx/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-maze/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-multi-io/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-overview/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-pong/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-terminal/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-tests-audio/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/examples/examples-tests-d2d/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/basicsteps/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/charencoding/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/compiler-opts/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/debugging/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/events/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/helloworld/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/installation/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/parameters/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/gettingstarted/stdio/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/more/building/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/more/imports/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/more/production/
- 2024-09-28
+ 2024-09-29
daily
https://twiddlingbits.dev/docsite/more/wasm-problem/
- 2024-09-28
+ 2024-09-29
daily
\ No newline at end of file
diff --git a/azure/docsite/sitemap.xml.gz b/azure/docsite/sitemap.xml.gz
index 554151c08a3c816b9ba45b6cbf43d42fcf3e0aad..4aa03316b4f2b257e6d94ccb604da4b1b72fe4c6 100644
GIT binary patch
literal 504
zcmV&M1$7F
z>Cikr{AzzR_rqO3XHUSJ5*+E!oS4&3rIuys$T0++sKY!dGNaA04yL~g-$SvV_;W=)}bwZ`3kO%n&S!*09#-fs8pZeK#)`fkF^
zNyiC}8J-5Koqz7tP0y=G{Q>*3Hfi3acyBEz)
z$~+)05u*faVVaaANSW4kfY1X8;g+$@IzJ211_&qzi*~MjBEUt5v>M+V?WnyMMlK7l
zH?qjqohv^td^R7CEDL-B?{UHGtt@xtyYiP{bGgVe2mJ!V#ItAD)?E&LR%5Au&khz@
z>V$EUC1%LM#k#AduhdiyO4V2Se}h+EN>9LC`iXUBuxixg3^61Ugf98>`%5!PYz)bH
z$So$8G~5=(d75O6HJl-e8e|y<(+_ADOUwJ}UvPb3C!KYGvO7weAjFk#hygRGady7;
uW%&wqAz`k2g}6FXMmw9@|6o_@qA9e+J9-HJAA9{E;Ndq^G<`Nx8vp=JdGzN1
literal 504
zcmVW*&mu6M1$7F
z@znhM@x8ro9{am)%ASCCB{C&v(UqCMtOk-3xyQm>uruzzU!yIr}~
z1TU@a>O7i$%|ay&FV0$DesFM1dkeF&Xx0>}Q)}Gq*EDfJ+aC|dukGQXJ>Hj)*S?!D
zbJ1~xbB15N)y_Y4>Za$#qyB*Yji!@Z@7A<>vJPcX4MrJId$cm3BSgsB%fOa+?cK9x
zCuJTGmxw`vwJ=Re5~NJ)IzZ@wgmBB)W}TmeXafY4gGD=4J`v!mLt2e*jdsx93nQ0>
z*Be=6>*mVO3m?trGs^;BztckmGYKX&^d;QlvFcZu>-8vp>^cmSjT
diff --git a/azure/examples/dist/divcon/index.html b/azure/examples/dist/divcon/index.html
index 8e74ddfe..d2b2328a 100644
--- a/azure/examples/dist/divcon/index.html
+++ b/azure/examples/dist/divcon/index.html
@@ -1,7 +1,7 @@
-Div Console WebAssembly Example twrConsoleDiv Demo
This example shows how to map a simple streaming character twr-wasm Console onto a div tag as stdio.
docs and more examples here
Loading...
\ No newline at end of file
+ twrConsoleDiv Demo
This example shows how to map a simple streaming character twr-wasm Console onto a div tag as stdio. A Read-Eval-Print Loop (REPL) is also implemented.
docs and more examples here
Loading...