1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
use serde_json;
use base_x;
use wasm_context::{
FunctionKind,
Context
};
pub struct JsExport {
pub raw_name: String,
pub metadata: ExportMetadata
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub enum TypeMetadata {
I32,
F64,
Custom {
name: Option< String >,
conversion_fn: String
}
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub struct ArgMetadata {
pub name: String,
pub ty: TypeMetadata
}
#[derive(Clone, Serialize, Deserialize, Debug)]
pub struct ExportMetadata {
pub name: String,
pub args: Vec< ArgMetadata >,
pub result: Option< TypeMetadata >
}
const ENCODING_BASE: &'static [u8] = b"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
const PREFIX: &'static str = "__JS_EXPORT_";
pub fn process( ctx: &mut Context ) -> Vec< JsExport > {
let mut output = Vec::new();
for function in ctx.functions.values_mut() {
if let &mut FunctionKind::Definition { ref mut export, ref mut name, .. } = function {
if export.names.len() == 1 && export.names[ 0 ].starts_with( PREFIX ) {
let json_metadata = {
let encoded_metadata = &export.names[ 0 ][ PREFIX.len().. ];
base_x::decode( ENCODING_BASE, encoded_metadata )
.expect( "cannot decode `js_export!` symbol" )
};
let json_metadata = String::from_utf8( json_metadata )
.expect( "one of the `js_export!` symbols has metadata which is not valid UTF-8" );
let metadata: ExportMetadata =
serde_json::from_str( &json_metadata )
.expect( "cannot parse `js_export!` symbol metadata" );
export.names[ 0 ] = metadata.name.clone();
if let &mut Some( ref mut name ) = name {
*name = metadata.name.clone();
}
output.push( JsExport {
metadata,
raw_name: export.names[ 0 ].clone()
});
}
}
}
output
}