-
Notifications
You must be signed in to change notification settings - Fork 21
/
rspwd_wrap.js
135 lines (126 loc) · 5.75 KB
/
rspwd_wrap.js
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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
//
// Ezpwd Reed-Solomon -- Reed-Solomon encoder / decoder library
//
// Copyright (c) 2014, Hard Consulting Corporation.
//
// Ezpwd Reed-Solomon is free software: you can redistribute it and/or modify it under the terms of
// the GNU General Public License as published by the Free Software Foundation, either version 3 of
// the License, or (at your option) any later version. See the LICENSE file at the top of the
// source tree. Ezpwd Reed-Solomon is also available under Commercial license. c++/ezpwd/rs_base
// is redistributed under the terms of the LGPL, regardless of the overall licensing terms.
//
// Ezpwd Reed-Solomon is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
// without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
// the GNU General Public License for more details.
//
//
// string_to_heapi8 -- allocate buffer of (at least) len of type (eg. i8), initializing from str
// heapi8_to_string -- decode a JS string from the buffer, til NUL
// array_to_heap -- allocate an array of max( arr.length, len ) elements of type typ
// index_in_heap -- return pointer to index ptr[idx] in an array of type typ
//
// The returned buffer may be longer than the C representation of UTF-8 str, but won't be
// shorter, and will be NUL terminated. Account for size of native type in all heap allocations and
// indexing.
//
function string_to_heapi8( str, len ) {
var arr = intArrayFromString( str ); // will NUL terminate
if ( len && arr.length < len )
arr.length = len;
return allocate( arr, 'i8', ALLOC_NORMAL );
}
function heapi8_to_string( ptr, len ) {
return Pointer_stringify( ptr );
};
function array_to_heap( typ, arr, len ) {
if ( len && arr.length / Runtime.getNativeTypeSize( typ ) < len )
arr.length = len * Runtime.getNativeTypeSize( typ );
return allocate( arr, typ, ALLOC_NORMAL );
}
function index_in_heap( typ, ptr, idx ) {
return ptr + idx * Runtime.getNativeTypeSize( typ );
}
//
// rspwd_encode_<N> -- Adds <N> parity to the supplied string (w/ max. size), returning length
// rspwd_decode_<N> -- Decodes the supplied string w/ up to <N> parity, returning confidence
//
// Since arrays are used to communicate, they must be allocated locally and release after the
// wrapped call. The underlying "C" decode returns a -'ve confidence and a (NUL terminated) string
// describing the error on failure, or a zero or +'ve confidence and the decoded lat/lon and
// accuracy on success.
//
// Decoding an EZCOD 3:10/11/12 encoded string returns an array:
//
// [<confidence>, <latitude>, <longitude>, <accuracy>]
//
// <confidence> -- The proportion [0,1] of parity symbols in excess, after error/erasure correction
// <latitude> -- Latitude in degress [-90,90]
// <longitude> -- Longitude in degrees [-180,180]
// <accuracy> -- Estimated accuracty in meters
//
rspwd_encode_N_wrap = function( func_name ) {
var func = Module.cwrap( func_name, 'number',
['number' // array (buf allocated here)
,'number'] ); // array size
return function( lat, lon ) {
var len = 1024; // room for error message, spaces, etc.
var res = -1;
var str = func_name + " invocation failed.";
var buf = string_to_heapi8( "", len );
try { // must de-allocate buf after this point
res = func( lat, lon, buf, len );
str = heapi8_to_string( buf );
} finally {
if ( buf ) _free( buf );
}
if ( res < 0 ) { // call failed, or call attempt failed w/o exception
console.log( str );
throw str;
}
return str;
}
}
ezcod_3_10_encode = ezcod_3_N_encode_wrap( 'ezcod_3_10_encode' );
ezcod_3_11_encode = ezcod_3_N_encode_wrap( 'ezcod_3_11_encode' );
ezcod_3_12_encode = ezcod_3_N_encode_wrap( 'ezcod_3_12_encode' );
ezcod_3_N_decode_wrap = function( func_name ) {
var func = Module.cwrap( func_name, 'number',
['number' // array (buf allocated here)
,'number' // array size
,'number' // lat * (pos allocated here)
,'number' // lon * ''
,'number'] ); // acc * ''
return function( cod ) {
var cnf = -1;
var str = func_name + " invocation failed.";
var ret;
var pos;
var len = 1024; // room for error message, spaces, etc.
var buf = string_to_heapi8( cod, len );
try { // must de-allocate buf and lat/lon/acc after this point
pos = array_to_heap( 'double', [], 3 );
var lat = index_in_heap( 'double', pos, 0 );
var lon = index_in_heap( 'double', pos, 1 );
var acc = index_in_heap( 'double', pos, 2 );
cnf = func( buf, len, lat, lon, acc );
if ( cnf < 0 ) {
str = heapi8_to_string( buf );
} else {
ret = [
cnf / 100.0, getValue( lat, 'double', 1 ), getValue( lon, 'double', 1 ), getValue( acc, 'double', 1 )
];
}
} finally {
if ( buf ) _free( buf );
if ( pos ) _free( pos );
}
if ( cnf < 0 ) { // call failed, or call attempt failed w/o exception
console.log( str );
throw str;
}
return ret;
}
}
ezcod_3_10_decode = ezcod_3_N_decode_wrap( 'ezcod_3_10_decode' );
ezcod_3_11_decode = ezcod_3_N_decode_wrap( 'ezcod_3_11_decode' );
ezcod_3_12_decode = ezcod_3_N_decode_wrap( 'ezcod_3_12_decode' );