| 1 | #include "jsi.h" | 
|---|
| 2 | #include "jslex.h" | 
|---|
| 3 | #include "jscompile.h" | 
|---|
| 4 | #include "jsvalue.h" | 
|---|
| 5 | #include "jsbuiltin.h" | 
|---|
| 6 |  | 
|---|
| 7 | static void jsB_globalf(js_State *J, const char *name, js_CFunction cfun, int n) | 
|---|
| 8 | { | 
|---|
| 9 | js_newcfunction(J, cfun, name, n); | 
|---|
| 10 | js_defglobal(J, name, JS_DONTENUM); | 
|---|
| 11 | } | 
|---|
| 12 |  | 
|---|
| 13 | void jsB_propf(js_State *J, const char *name, js_CFunction cfun, int n) | 
|---|
| 14 | { | 
|---|
| 15 | const char *pname = strrchr(name, '.'); | 
|---|
| 16 | pname = pname ? pname + 1 : name; | 
|---|
| 17 | js_newcfunction(J, cfun, name, n); | 
|---|
| 18 | js_defproperty(J, -2, pname, JS_DONTENUM); | 
|---|
| 19 | } | 
|---|
| 20 |  | 
|---|
| 21 | void jsB_propn(js_State *J, const char *name, double number) | 
|---|
| 22 | { | 
|---|
| 23 | js_pushnumber(J, number); | 
|---|
| 24 | js_defproperty(J, -2, name, JS_READONLY | JS_DONTENUM | JS_DONTCONF); | 
|---|
| 25 | } | 
|---|
| 26 |  | 
|---|
| 27 | void jsB_props(js_State *J, const char *name, const char *string) | 
|---|
| 28 | { | 
|---|
| 29 | js_pushliteral(J, string); | 
|---|
| 30 | js_defproperty(J, -2, name, JS_DONTENUM); | 
|---|
| 31 | } | 
|---|
| 32 |  | 
|---|
| 33 | static void jsB_parseInt(js_State *J) | 
|---|
| 34 | { | 
|---|
| 35 | const char *s = js_tostring(J, 1); | 
|---|
| 36 | int radix = js_isdefined(J, 2) ? js_tointeger(J, 2) : 10; | 
|---|
| 37 | double sign = 1; | 
|---|
| 38 | double n; | 
|---|
| 39 | char *e; | 
|---|
| 40 |  | 
|---|
| 41 | while (jsY_iswhite(*s) || jsY_isnewline(*s)) | 
|---|
| 42 | ++s; | 
|---|
| 43 | if (*s == '-') { | 
|---|
| 44 | ++s; | 
|---|
| 45 | sign = -1; | 
|---|
| 46 | } else if (*s == '+') { | 
|---|
| 47 | ++s; | 
|---|
| 48 | } | 
|---|
| 49 | if (radix == 0) { | 
|---|
| 50 | radix = 10; | 
|---|
| 51 | if (s[0] == '0' && (s[1] == 'x' || s[1] == 'X')) { | 
|---|
| 52 | s += 2; | 
|---|
| 53 | radix = 16; | 
|---|
| 54 | } | 
|---|
| 55 | } else if (radix < 2 || radix > 36) { | 
|---|
| 56 | js_pushnumber(J, NAN); | 
|---|
| 57 | return; | 
|---|
| 58 | } | 
|---|
| 59 | n = strtol(s, &e, radix); | 
|---|
| 60 | if (s == e) | 
|---|
| 61 | js_pushnumber(J, NAN); | 
|---|
| 62 | else | 
|---|
| 63 | js_pushnumber(J, n * sign); | 
|---|
| 64 | } | 
|---|
| 65 |  | 
|---|
| 66 | static void jsB_parseFloat(js_State *J) | 
|---|
| 67 | { | 
|---|
| 68 | const char *s = js_tostring(J, 1); | 
|---|
| 69 | char *e; | 
|---|
| 70 | double n; | 
|---|
| 71 |  | 
|---|
| 72 | while (jsY_iswhite(*s) || jsY_isnewline(*s)) ++s; | 
|---|
| 73 | if (!strncmp(s, "Infinity", 8)) | 
|---|
| 74 | js_pushnumber(J, INFINITY); | 
|---|
| 75 | else if (!strncmp(s, "+Infinity", 9)) | 
|---|
| 76 | js_pushnumber(J, INFINITY); | 
|---|
| 77 | else if (!strncmp(s, "-Infinity", 9)) | 
|---|
| 78 | js_pushnumber(J, -INFINITY); | 
|---|
| 79 | else { | 
|---|
| 80 | n = js_stringtofloat(s, &e); | 
|---|
| 81 | if (e == s) | 
|---|
| 82 | js_pushnumber(J, NAN); | 
|---|
| 83 | else | 
|---|
| 84 | js_pushnumber(J, n); | 
|---|
| 85 | } | 
|---|
| 86 | } | 
|---|
| 87 |  | 
|---|
| 88 | static void jsB_isNaN(js_State *J) | 
|---|
| 89 | { | 
|---|
| 90 | double n = js_tonumber(J, 1); | 
|---|
| 91 | js_pushboolean(J, isnan(n)); | 
|---|
| 92 | } | 
|---|
| 93 |  | 
|---|
| 94 | static void jsB_isFinite(js_State *J) | 
|---|
| 95 | { | 
|---|
| 96 | double n = js_tonumber(J, 1); | 
|---|
| 97 | js_pushboolean(J, isfinite(n)); | 
|---|
| 98 | } | 
|---|
| 99 |  | 
|---|
| 100 | static void Encode(js_State *J, const char *str, const char *unescaped) | 
|---|
| 101 | { | 
|---|
| 102 | js_Buffer *sb = NULL; | 
|---|
| 103 |  | 
|---|
| 104 | static const char *HEX = "0123456789ABCDEF"; | 
|---|
| 105 |  | 
|---|
| 106 | if (js_try(J)) { | 
|---|
| 107 | js_free(J, sb); | 
|---|
| 108 | js_throw(J); | 
|---|
| 109 | } | 
|---|
| 110 |  | 
|---|
| 111 | while (*str) { | 
|---|
| 112 | int c = (unsigned char) *str++; | 
|---|
| 113 | if (strchr(unescaped, c)) | 
|---|
| 114 | js_putc(J, &sb, c); | 
|---|
| 115 | else { | 
|---|
| 116 | js_putc(J, &sb, '%'); | 
|---|
| 117 | js_putc(J, &sb, HEX[(c >> 4) & 0xf]); | 
|---|
| 118 | js_putc(J, &sb, HEX[c & 0xf]); | 
|---|
| 119 | } | 
|---|
| 120 | } | 
|---|
| 121 | js_putc(J, &sb, 0); | 
|---|
| 122 |  | 
|---|
| 123 | js_pushstring(J, sb ? sb->s : ""); | 
|---|
| 124 | js_endtry(J); | 
|---|
| 125 | js_free(J, sb); | 
|---|
| 126 | } | 
|---|
| 127 |  | 
|---|
| 128 | static void Decode(js_State *J, const char *str, const char *reserved) | 
|---|
| 129 | { | 
|---|
| 130 | js_Buffer *sb = NULL; | 
|---|
| 131 | int a, b; | 
|---|
| 132 |  | 
|---|
| 133 | if (js_try(J)) { | 
|---|
| 134 | js_free(J, sb); | 
|---|
| 135 | js_throw(J); | 
|---|
| 136 | } | 
|---|
| 137 |  | 
|---|
| 138 | while (*str) { | 
|---|
| 139 | int c = (unsigned char) *str++; | 
|---|
| 140 | if (c != '%') | 
|---|
| 141 | js_putc(J, &sb, c); | 
|---|
| 142 | else { | 
|---|
| 143 | if (!str[0] || !str[1]) | 
|---|
| 144 | js_urierror(J, "truncated escape sequence"); | 
|---|
| 145 | a = *str++; | 
|---|
| 146 | b = *str++; | 
|---|
| 147 | if (!jsY_ishex(a) || !jsY_ishex(b)) | 
|---|
| 148 | js_urierror(J, "invalid escape sequence"); | 
|---|
| 149 | c = jsY_tohex(a) << 4 | jsY_tohex(b); | 
|---|
| 150 | if (!strchr(reserved, c)) | 
|---|
| 151 | js_putc(J, &sb, c); | 
|---|
| 152 | else { | 
|---|
| 153 | js_putc(J, &sb, '%'); | 
|---|
| 154 | js_putc(J, &sb, a); | 
|---|
| 155 | js_putc(J, &sb, b); | 
|---|
| 156 | } | 
|---|
| 157 | } | 
|---|
| 158 | } | 
|---|
| 159 | js_putc(J, &sb, 0); | 
|---|
| 160 |  | 
|---|
| 161 | js_pushstring(J, sb ? sb->s : ""); | 
|---|
| 162 | js_endtry(J); | 
|---|
| 163 | js_free(J, sb); | 
|---|
| 164 | } | 
|---|
| 165 |  | 
|---|
| 166 | #define URIRESERVED ";/?:@&=+$," | 
|---|
| 167 | #define URIALPHA "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" | 
|---|
| 168 | #define URIDIGIT "0123456789" | 
|---|
| 169 | #define URIMARK "-_.!~*`()" | 
|---|
| 170 | #define URIUNESCAPED URIALPHA URIDIGIT URIMARK | 
|---|
| 171 |  | 
|---|
| 172 | static void jsB_decodeURI(js_State *J) | 
|---|
| 173 | { | 
|---|
| 174 | Decode(J, js_tostring(J, 1), URIRESERVED "#"); | 
|---|
| 175 | } | 
|---|
| 176 |  | 
|---|
| 177 | static void jsB_decodeURIComponent(js_State *J) | 
|---|
| 178 | { | 
|---|
| 179 | Decode(J, js_tostring(J, 1), ""); | 
|---|
| 180 | } | 
|---|
| 181 |  | 
|---|
| 182 | static void jsB_encodeURI(js_State *J) | 
|---|
| 183 | { | 
|---|
| 184 | Encode(J, js_tostring(J, 1), URIUNESCAPED URIRESERVED "#"); | 
|---|
| 185 | } | 
|---|
| 186 |  | 
|---|
| 187 | static void jsB_encodeURIComponent(js_State *J) | 
|---|
| 188 | { | 
|---|
| 189 | Encode(J, js_tostring(J, 1), URIUNESCAPED); | 
|---|
| 190 | } | 
|---|
| 191 |  | 
|---|
| 192 | void jsB_init(js_State *J) | 
|---|
| 193 | { | 
|---|
| 194 | /* Create the prototype objects here, before the constructors */ | 
|---|
| 195 | J->Object_prototype = jsV_newobject(J, JS_COBJECT, NULL); | 
|---|
| 196 | J->Array_prototype = jsV_newobject(J, JS_CARRAY, J->Object_prototype); | 
|---|
| 197 | J->Function_prototype = jsV_newobject(J, JS_CCFUNCTION, J->Object_prototype); | 
|---|
| 198 | J->Boolean_prototype = jsV_newobject(J, JS_CBOOLEAN, J->Object_prototype); | 
|---|
| 199 | J->Number_prototype = jsV_newobject(J, JS_CNUMBER, J->Object_prototype); | 
|---|
| 200 | J->String_prototype = jsV_newobject(J, JS_CSTRING, J->Object_prototype); | 
|---|
| 201 | J->RegExp_prototype = jsV_newobject(J, JS_COBJECT, J->Object_prototype); | 
|---|
| 202 | J->Date_prototype = jsV_newobject(J, JS_CDATE, J->Object_prototype); | 
|---|
| 203 |  | 
|---|
| 204 | /* All the native error types */ | 
|---|
| 205 | J->Error_prototype = jsV_newobject(J, JS_CERROR, J->Object_prototype); | 
|---|
| 206 | J->EvalError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 207 | J->RangeError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 208 | J->ReferenceError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 209 | J->SyntaxError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 210 | J->TypeError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 211 | J->URIError_prototype = jsV_newobject(J, JS_CERROR, J->Error_prototype); | 
|---|
| 212 |  | 
|---|
| 213 | /* Create the constructors and fill out the prototype objects */ | 
|---|
| 214 | jsB_initobject(J); | 
|---|
| 215 | jsB_initarray(J); | 
|---|
| 216 | jsB_initfunction(J); | 
|---|
| 217 | jsB_initboolean(J); | 
|---|
| 218 | jsB_initnumber(J); | 
|---|
| 219 | jsB_initstring(J); | 
|---|
| 220 | jsB_initregexp(J); | 
|---|
| 221 | jsB_initdate(J); | 
|---|
| 222 | jsB_initerror(J); | 
|---|
| 223 | jsB_initmath(J); | 
|---|
| 224 | jsB_initjson(J); | 
|---|
| 225 |  | 
|---|
| 226 | /* Initialize the global object */ | 
|---|
| 227 | js_pushnumber(J, NAN); | 
|---|
| 228 | js_defglobal(J, "NaN", JS_READONLY | JS_DONTENUM | JS_DONTCONF); | 
|---|
| 229 |  | 
|---|
| 230 | js_pushnumber(J, INFINITY); | 
|---|
| 231 | js_defglobal(J, "Infinity", JS_READONLY | JS_DONTENUM | JS_DONTCONF); | 
|---|
| 232 |  | 
|---|
| 233 | js_pushundefined(J); | 
|---|
| 234 | js_defglobal(J, "undefined", JS_READONLY | JS_DONTENUM | JS_DONTCONF); | 
|---|
| 235 |  | 
|---|
| 236 | jsB_globalf(J, "parseInt", jsB_parseInt, 1); | 
|---|
| 237 | jsB_globalf(J, "parseFloat", jsB_parseFloat, 1); | 
|---|
| 238 | jsB_globalf(J, "isNaN", jsB_isNaN, 1); | 
|---|
| 239 | jsB_globalf(J, "isFinite", jsB_isFinite, 1); | 
|---|
| 240 |  | 
|---|
| 241 | jsB_globalf(J, "decodeURI", jsB_decodeURI, 1); | 
|---|
| 242 | jsB_globalf(J, "decodeURIComponent", jsB_decodeURIComponent, 1); | 
|---|
| 243 | jsB_globalf(J, "encodeURI", jsB_encodeURI, 1); | 
|---|
| 244 | jsB_globalf(J, "encodeURIComponent", jsB_encodeURIComponent, 1); | 
|---|
| 245 | } | 
|---|
| 246 |  | 
|---|