Skip to content

Commit

Permalink
src: wasm build support
Browse files Browse the repository at this point in the history
PR-URL: #93
Reviewed-By: Fedor Indutny <[email protected]>
  • Loading branch information
dnlup authored Apr 4, 2021
1 parent cc5adf3 commit a620012
Show file tree
Hide file tree
Showing 8 changed files with 482 additions and 0 deletions.
6 changes: 6 additions & 0 deletions .dockerignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
*
!package.json
!package-lock.json
!tsconfig.json
!bin
!src
22 changes: 22 additions & 0 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
FROM node:14.16.0-buster

ARG UID=1000
ARG GID=1000
ARG WASI_SDK_VERSION_MAJOR=12
ARG WASI_SDK_VERSION_MINOR=0

ENV WASI_ROOT=/home/node/wasi-sdk-12.0

RUN wget https://github.com/WebAssembly/wasi-sdk/releases/download/wasi-sdk-${WASI_SDK_VERSION_MAJOR}/wasi-sdk-${WASI_SDK_VERSION_MAJOR}.${WASI_SDK_VERSION_MINOR}-linux.tar.gz -P /tmp

RUN tar xvf /tmp/wasi-sdk-${WASI_SDK_VERSION_MAJOR}.${WASI_SDK_VERSION_MINOR}-linux.tar.gz --directory /home/node

RUN mkdir /home/node/llhttp

WORKDIR /home/node/llhttp

COPY . .

RUN npm ci

USER node
80 changes: 80 additions & 0 deletions bin/build_wasm.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
import { execSync } from 'child_process';
import { copyFileSync, mkdirSync, writeFileSync } from 'fs';
import { stringify } from 'javascript-stringify';
import { join, resolve } from 'path';
import { constants } from '..';

const { WASI_ROOT } = process.env;
const WASM_OUT = resolve(__dirname, '../build/wasm');
const WASM_SRC = resolve(__dirname, '../');

if (process.argv[2] === '--setup') {
try {
mkdirSync(join(WASM_SRC, 'build'));
process.exit(0);
} catch (error) {
if (error.code !== 'EEXIST') {
throw error;
}
process.exit(0);
}
}

if (process.argv[2] === '--docker') {
let cmd = 'docker run --rm -it';
// Try to avoid root permission problems on compiled assets
// when running on linux.
// It will work flawessly if uid === gid === 1000
// there will be some warnings otherwise.
if (process.platform === 'linux') {
cmd += ` --user ${process.getuid()}:${process.getegid()}`;
}
cmd += ` --mount type=bind,source=${WASM_SRC}/build,target=/home/node/llhttp/build llhttp_wasm_builder npm run wasm`;
execSync(cmd, { cwd: WASM_SRC, stdio: 'inherit' });
process.exit(0);
}

if (!WASI_ROOT) {
throw new Error('Please setup the WASI_ROOT env variable.');
}

try {
mkdirSync(WASM_OUT);
} catch (error) {
if (error.code !== 'EEXIST') {
throw error;
}
}

// Build ts
execSync('npm run build', { cwd: WASM_SRC, stdio: 'inherit' });

// Build wasm binary
execSync(`${WASI_ROOT}/bin/clang \
--sysroot=${WASI_ROOT}/share/wasi-sysroot \
-target wasm32-unknown-wasi \
-Ofast \
-fno-exceptions \
-fvisibility=hidden \
-mexec-model=reactor \
-Wl,-error-limit=0 \
-Wl,-O3 \
-Wl,--lto-O3 \
-Wl,--strip-all \
-Wl,--allow-undefined \
-Wl,--export-dynamic \
-Wl,--export-table \
-Wl,--export=malloc \
-Wl,--export=free \
${join(WASM_SRC, 'build', 'c')}/*.c \
${join(WASM_SRC, 'src', 'native')}/*.c \
-I${join(WASM_SRC, 'build')} \
-o ${join(WASM_OUT, 'llhttp.wasm')}`, { stdio: 'inherit' });

// Copy constants for `.js` and `.ts` users.
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'constants.js'), join(WASM_OUT, 'constants.js'));
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'constants.js.map'), join(WASM_OUT, 'constants.js.map'));
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'constants.d.ts'), join(WASM_OUT, 'constants.d.ts'));
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'utils.js'), join(WASM_OUT, 'utils.js'));
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'utils.js.map'), join(WASM_OUT, 'utils.js.map'));
copyFileSync(join(WASM_SRC, 'lib', 'llhttp', 'utils.d.ts'), join(WASM_OUT, 'utils.d.ts'));
248 changes: 248 additions & 0 deletions examples/wasm.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,248 @@
/**
* A minimal Parser that mimicks a small fraction of the Node.js parser
* API.
* To run:
* - `npm run build-wasm`
* - `npx ts-node examples/wasm.ts`
*/
import { readFileSync } from 'fs';
import { resolve } from 'path';
import * as constants from '../build/wasm/constants';

const bin = readFileSync(resolve(__dirname, '../build/wasm/llhttp.wasm'));
const mod = new WebAssembly.Module(bin);

const REQUEST = constants.TYPE.REQUEST;
const RESPONSE = constants.TYPE.RESPONSE;
const kOnMessageBegin = 0;
const kOnHeaders = 1;
const kOnHeadersComplete = 2;
const kOnBody = 3;
const kOnMessageComplete = 4;
const kOnExecute = 5;

const kPtr = Symbol('kPtr');
const kUrl = Symbol('kUrl');
const kStatusMessage = Symbol('kStatusMessage');
const kHeadersFields = Symbol('kHeadersFields');
const kHeadersValues = Symbol('kHeadersValues');
const kBody = Symbol('kBody');
const kReset = Symbol('kReset');
const kCheckErr = Symbol('kCheckErr');

const cstr = (ptr: number, len: number): string =>
Buffer.from(memory.buffer, ptr, len).toString();

const wasm_on_message_begin = (p: number) => {
const i = instMap.get(p);
i[kReset]();
return i[kOnMessageBegin]();
};

const wasm_on_url = (p: number, at: number, length: number) => {
instMap.get(p)[kUrl] = cstr(at, length);
return 0;
};

const wasm_on_status = (p: number, at: number, length: number) => {
instMap.get(p)[kStatusMessage] = cstr(at, length);
return 0;
};

const wasm_on_header_field = (p: number, at: number, length: number) => {
const i= instMap.get(p)
i[kHeadersFields].push(cstr(at, length));
return 0;
};

const wasm_on_header_value = (p: number, at: number, length: number) => {
const i = instMap.get(p);
i[kHeadersValues].push(cstr(at, length));
return 0;
};

const wasm_on_headers_complete = (p: number) => {
const i = instMap.get(p);
const type = get_type(p);
const versionMajor = get_version_major(p);
const versionMinor = get_version_minor(p);
const rawHeaders = [];
let method;
let url;
let statusCode;
let statusMessage;
const upgrade = get_upgrade(p);
const shouldKeepAlive = should_keep_alive(p);

for (let c = 0; c < i[kHeadersFields].length; c++) {
rawHeaders.push(i[kHeadersFields][c], i[kHeadersValues][c])
}

if (type === HTTPParser.REQUEST) {
method = constants.METHODS[get_method(p)];
url = i[kUrl];
} else if (type === HTTPParser.RESPONSE) {
statusCode = get_status_code(p);
statusMessage = i[kStatusMessage];
}
return i[kOnHeadersComplete](versionMajor, versionMinor, rawHeaders, method,
url, statusCode, statusMessage, upgrade, shouldKeepAlive);
};

const wasm_on_body = (p: number, at: number, length: number) => {
const i = instMap.get(p);
const body = Buffer.from(memory.buffer, at, length);
return i[kOnBody](body);
};

const wasm_on_message_complete = (p: number) => {
return instMap.get(p)[kOnMessageComplete]();
};

const instMap = new Map();

const inst = new WebAssembly.Instance(mod, {
env: {
wasm_on_message_begin,
wasm_on_url,
wasm_on_status,
wasm_on_header_field,
wasm_on_header_value,
wasm_on_headers_complete,
wasm_on_body,
wasm_on_message_complete,
},
});

const memory = inst.exports.memory as any;
const alloc = inst.exports.llhttp_alloc as CallableFunction;
const malloc = inst.exports.malloc as CallableFunction;
const execute = inst.exports.llhttp_execute as CallableFunction;
const get_type = inst.exports.llhttp_get_type as CallableFunction;
const get_upgrade = inst.exports.llhttp_get_upgrade as CallableFunction;
const should_keep_alive = inst.exports.llhttp_should_keep_alive as CallableFunction;
const get_method = inst.exports.llhttp_get_method as CallableFunction;
const get_status_code = inst.exports.llhttp_get_status_code as CallableFunction;
const get_version_minor = inst.exports.llhttp_get_http_minor as CallableFunction;
const get_version_major = inst.exports.llhttp_get_http_major as CallableFunction;
const get_error_reason = inst.exports.llhttp_get_error_reason as CallableFunction;
const free = inst.exports.free as CallableFunction;
const initialize = inst.exports._initialize as CallableFunction;

initialize(); // wasi reactor

class HTTPParser {
static REQUEST = REQUEST;
static RESPONSE = RESPONSE;
static kOnMessageBegin = kOnMessageBegin;
static kOnHeaders = kOnHeaders;
static kOnHeadersComplete = kOnHeadersComplete;
static kOnBody = kOnBody;
static kOnMessageComplete = kOnMessageComplete;
static kOnExecute = kOnExecute;

[kPtr]: number;
[kUrl]: string;
[kStatusMessage]: null|string;
[kHeadersFields]: []|[string];
[kHeadersValues]: []|[string];
[kBody]: null|Buffer;

constructor(type: constants.TYPE) {
this[kPtr] = alloc(constants.TYPE[type]);
instMap.set(this[kPtr], this);

this[kUrl] = '';
this[kStatusMessage] = null;
this[kHeadersFields] = [];
this[kHeadersValues] = [];
this[kBody] = null;
}

[kReset]() {
this[kUrl] = '';
this[kStatusMessage] = null;
this[kHeadersFields] = [];
this[kHeadersValues] = [];
this[kBody] = null;
}

[kOnMessageBegin]() {
return 0;
}

[kOnHeaders](rawHeaders: [string]) {}

[kOnHeadersComplete](versionMajor: number, versionMinor: number, rawHeaders: [string], method: string,
url: string, statusCode: number, statusMessage: string, upgrade: boolean, shouldKeepAlive: boolean) {
return 0;
}

[kOnBody](body: Buffer) {
this[kBody] = body;
return 0;
}

[kOnMessageComplete]() {
return 0;
}

destroy() {
instMap.delete(this[kPtr]);
free(this[kPtr]);
}

execute(data: Buffer) {
const ptr = malloc(data.byteLength);
const u8 = new Uint8Array(memory.buffer);
u8.set(data, ptr);
const ret = execute(this[kPtr], ptr, data.length);
free(ptr);
this[kCheckErr](ret);
return ret;
}

[kCheckErr](n: number) {
if (n === constants.ERROR.OK) {
return;
}
const ptr = get_error_reason(this[kPtr]);
const u8 = new Uint8Array(memory.buffer);
const len = u8.indexOf(0, ptr) - ptr;
throw new Error(cstr(ptr, len));
}
}


{
const p = new HTTPParser(HTTPParser.REQUEST);

p.execute(Buffer.from([
'POST /owo HTTP/1.1',
'X: Y',
'Content-Length: 9',
'',
'uh, meow?',
'',
].join('\r\n')));

console.log(p);

p.destroy();
}

{
const p = new HTTPParser(HTTPParser.RESPONSE);

p.execute(Buffer.from([
'HTTP/1.1 200 OK',
'X: Y',
'Content-Length: 9',
'',
'uh, meow?'
].join('\r\n')));

console.log(p);

p.destroy();
}
6 changes: 6 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,9 @@
"bench": "ts-node bench/",
"build": "ts-node bin/generate.ts",
"build-ts": "tsc",
"prebuild-wasm": "docker build -t llhttp_wasm_builder . && npm run wasm -- --setup",
"build-wasm": "npm run wasm -- --docker",
"wasm": "ts-node bin/build_wasm.ts",
"clean": "rm -rf lib && rm -rf test/tmp",
"prepare": "npm run clean && npm run build-ts",
"lint": "tslint -c tslint.json bin/*.ts src/*.ts src/**/*.ts test/*.ts test/**/*.ts",
Expand All @@ -38,6 +41,7 @@
"devDependencies": {
"@types/mocha": "^5.2.7",
"@types/node": "^10.17.52",
"javascript-stringify": "^2.0.1",
"llparse-dot": "^1.0.1",
"llparse-test-fixture": "^5.0.1",
"mdgator": "^1.1.2",
Expand Down
Loading

0 comments on commit a620012

Please sign in to comment.