ExpressionIntepreter.cs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789
  1. using System;
  2. using System.Linq;
  3. using Jint.Native;
  4. using Jint.Native.Function;
  5. using Jint.Native.Object;
  6. using Jint.Parser.Ast;
  7. using Jint.Runtime.Descriptors;
  8. using Jint.Runtime.Environments;
  9. using Jint.Runtime.References;
  10. namespace Jint.Runtime
  11. {
  12. public class ExpressionInterpreter
  13. {
  14. private readonly Engine _engine;
  15. public ExpressionInterpreter(Engine engine)
  16. {
  17. _engine = engine;
  18. }
  19. private object EvaluateExpression(Expression expression)
  20. {
  21. return _engine.EvaluateExpression(expression);
  22. }
  23. public object EvaluateConditionalExpression(ConditionalExpression conditionalExpression)
  24. {
  25. var test = _engine.EvaluateExpression(conditionalExpression.Test);
  26. var evaluate = TypeConverter.ToBoolean(test) ? conditionalExpression.Consequent : conditionalExpression.Alternate;
  27. return _engine.EvaluateExpression(evaluate);
  28. }
  29. public object EvaluateAssignmentExpression(AssignmentExpression assignmentExpression)
  30. {
  31. object rval = _engine.GetValue(EvaluateExpression(assignmentExpression.Right));
  32. var lref = EvaluateExpression(assignmentExpression.Left) as Reference;
  33. if (assignmentExpression.Operator == "=")
  34. {
  35. if(lref != null && lref.IsStrict() && lref.GetBase() is EnvironmentRecord && (lref.GetReferencedName() == "eval" || lref.GetReferencedName() == "arguments"))
  36. {
  37. throw new JavaScriptException(_engine.SyntaxError);
  38. }
  39. _engine.PutValue(lref, rval);
  40. return rval;
  41. }
  42. object lval = _engine.GetValue(lref);
  43. switch (assignmentExpression.Operator)
  44. {
  45. case "+=":
  46. var lprim = TypeConverter.ToPrimitive(lval);
  47. var rprim = TypeConverter.ToPrimitive(rval);
  48. if (TypeConverter.GetType(lprim) == TypeCode.String ||
  49. TypeConverter.GetType(rprim) == TypeCode.String)
  50. {
  51. lval = TypeConverter.ToString(lprim) + TypeConverter.ToString(rprim);
  52. }
  53. else
  54. {
  55. lval = TypeConverter.ToNumber(lprim) + TypeConverter.ToNumber(rprim);
  56. }
  57. break;
  58. case "-=":
  59. lval = TypeConverter.ToNumber(lval) - TypeConverter.ToNumber(rval);
  60. break;
  61. case "*=":
  62. if (lval == Undefined.Instance || rval == Undefined.Instance)
  63. {
  64. lval = Undefined.Instance;
  65. }
  66. else
  67. {
  68. lval = TypeConverter.ToNumber(lval) * TypeConverter.ToNumber(rval);
  69. }
  70. break;
  71. case "/=":
  72. if (lval == Undefined.Instance || rval == Undefined.Instance)
  73. {
  74. lval = Undefined.Instance;
  75. }
  76. else
  77. {
  78. lval = TypeConverter.ToNumber(lval) / TypeConverter.ToNumber(rval);
  79. }
  80. break;
  81. case "%=":
  82. if (lval == Undefined.Instance || rval == Undefined.Instance)
  83. {
  84. lval = Undefined.Instance;
  85. }
  86. else
  87. {
  88. lval = TypeConverter.ToNumber(lval) % TypeConverter.ToNumber(rval);
  89. }
  90. break;
  91. case "&=":
  92. lval = TypeConverter.ToInt32(lval) & TypeConverter.ToInt32(rval);
  93. break;
  94. case "|=":
  95. lval = TypeConverter.ToInt32(lval) | TypeConverter.ToInt32(rval);
  96. break;
  97. case "^=":
  98. lval = TypeConverter.ToInt32(lval) ^ TypeConverter.ToInt32(rval);
  99. break;
  100. case "<<=":
  101. lval = TypeConverter.ToInt32(lval) << (int)(TypeConverter.ToUint32(rval) & 0x1F);
  102. break;
  103. case ">>=":
  104. lval = TypeConverter.ToInt32(lval) >> (int)(TypeConverter.ToUint32(rval) & 0x1F);
  105. break;
  106. case ">>>=":
  107. lval = (uint)TypeConverter.ToInt32(lval) >> (int)(TypeConverter.ToUint32(rval) & 0x1F);
  108. break;
  109. default:
  110. throw new NotImplementedException();
  111. }
  112. _engine.PutValue(lref, lval);
  113. return lval;
  114. }
  115. public object EvaluateBinaryExpression(BinaryExpression expression)
  116. {
  117. object left = _engine.GetValue(EvaluateExpression(expression.Left));
  118. object right = _engine.GetValue(EvaluateExpression(expression.Right));
  119. object value;
  120. switch (expression.Operator)
  121. {
  122. case "+":
  123. var lprim = TypeConverter.ToPrimitive(left);
  124. var rprim = TypeConverter.ToPrimitive(right);
  125. if (TypeConverter.GetType(lprim) == TypeCode.String || TypeConverter.GetType(rprim) == TypeCode.String)
  126. {
  127. value = TypeConverter.ToString(lprim) + TypeConverter.ToString(rprim);
  128. }
  129. else
  130. {
  131. value = TypeConverter.ToNumber(lprim) + TypeConverter.ToNumber(rprim);
  132. }
  133. break;
  134. case "-":
  135. value = TypeConverter.ToNumber(left) - TypeConverter.ToNumber(right);
  136. break;
  137. case "*":
  138. if (left == Undefined.Instance || right == Undefined.Instance)
  139. {
  140. value = Undefined.Instance;
  141. }
  142. else
  143. {
  144. value = TypeConverter.ToNumber(left) * TypeConverter.ToNumber(right);
  145. }
  146. break;
  147. case "/":
  148. if (left == Undefined.Instance || right == Undefined.Instance)
  149. {
  150. value = Undefined.Instance;
  151. }
  152. else
  153. {
  154. value = TypeConverter.ToNumber(left) / TypeConverter.ToNumber(right);
  155. }
  156. break;
  157. case "%":
  158. if (left == Undefined.Instance || right == Undefined.Instance)
  159. {
  160. value = Undefined.Instance;
  161. }
  162. else
  163. {
  164. value = TypeConverter.ToNumber(left) % TypeConverter.ToNumber(right);
  165. }
  166. break;
  167. case "==":
  168. value = Equal(left, right);
  169. break;
  170. case "!=":
  171. value = !Equal(left, right);
  172. break;
  173. case ">":
  174. value = TypeConverter.ToNumber(left) > TypeConverter.ToNumber(right);
  175. break;
  176. case ">=":
  177. value = TypeConverter.ToNumber(left) >= TypeConverter.ToNumber(right);
  178. break;
  179. case "<":
  180. value = TypeConverter.ToNumber(left) < TypeConverter.ToNumber(right);
  181. break;
  182. case "<=":
  183. value = TypeConverter.ToNumber(left) <= TypeConverter.ToNumber(right);
  184. break;
  185. case "===":
  186. return StriclyEqual(left, right);
  187. case "!==":
  188. return !StriclyEqual(left, right);
  189. case "&":
  190. return TypeConverter.ToInt32(left) & TypeConverter.ToInt32(right);
  191. case "|":
  192. return TypeConverter.ToInt32(left) | TypeConverter.ToInt32(right);
  193. case "^":
  194. return TypeConverter.ToInt32(left) ^ TypeConverter.ToInt32(right);
  195. case "<<":
  196. return TypeConverter.ToInt32(left) << (int)(TypeConverter.ToUint32(right) & 0x1F);
  197. case ">>":
  198. return TypeConverter.ToInt32(left) >> (int)(TypeConverter.ToUint32(right) & 0x1F);
  199. case ">>>":
  200. return (uint)TypeConverter.ToInt32(left) >> (int)(TypeConverter.ToUint32(right) & 0x1F);
  201. case "instanceof":
  202. var f = right as FunctionInstance;
  203. if (f == null)
  204. {
  205. throw new JavaScriptException(_engine.TypeError, "instanceof can only be used with a function object");
  206. }
  207. value = f.HasInstance(left);
  208. break;
  209. case "in":
  210. var o = right as ObjectInstance;
  211. if (o == null)
  212. {
  213. throw new JavaScriptException(_engine.TypeError, "in can only be used with an object");
  214. }
  215. value = o.HasProperty(TypeConverter.ToString(left));
  216. break;
  217. default:
  218. throw new NotImplementedException();
  219. }
  220. return value;
  221. }
  222. public object EvaluateLogicalExpression(LogicalExpression logicalExpression)
  223. {
  224. var left = _engine.GetValue(EvaluateExpression(logicalExpression.Left));
  225. switch (logicalExpression.Operator)
  226. {
  227. case "&&":
  228. if (!TypeConverter.ToBoolean(left))
  229. {
  230. return left;
  231. }
  232. return _engine.GetValue(EvaluateExpression(logicalExpression.Right));
  233. case "||":
  234. if (TypeConverter.ToBoolean(left))
  235. {
  236. return left;
  237. }
  238. return _engine.GetValue(EvaluateExpression(logicalExpression.Right));
  239. default:
  240. throw new NotImplementedException();
  241. }
  242. }
  243. public static bool Equal(object x, object y)
  244. {
  245. var typex = TypeConverter.GetType(x);
  246. var typey = TypeConverter.GetType(y);
  247. if (typex == typey)
  248. {
  249. if (typex == TypeCode.Empty)
  250. {
  251. return true;
  252. }
  253. if (typex == TypeCode.Double)
  254. {
  255. var nx = TypeConverter.ToNumber(x);
  256. var ny = TypeConverter.ToNumber(y);
  257. if (double.IsNaN(nx) || double.IsNaN(ny))
  258. {
  259. return false;
  260. }
  261. if (nx == ny)
  262. {
  263. return true;
  264. }
  265. return false;
  266. }
  267. if (typex == TypeCode.String)
  268. {
  269. return TypeConverter.ToString(x) == TypeConverter.ToString(y);
  270. }
  271. if (typex == TypeCode.Boolean)
  272. {
  273. return (bool) x == (bool) y;
  274. }
  275. return x == y;
  276. }
  277. if (x == Null.Instance && y == Undefined.Instance)
  278. {
  279. return true;
  280. }
  281. if (x == Undefined.Instance && y == Null.Instance)
  282. {
  283. return true;
  284. }
  285. if (typex == TypeCode.Double && typey == TypeCode.String)
  286. {
  287. return Equal(x, TypeConverter.ToNumber(y));
  288. }
  289. if (typex == TypeCode.String && typey == TypeCode.Double)
  290. {
  291. return Equal(TypeConverter.ToNumber(x), y);
  292. }
  293. if (typex == TypeCode.Boolean)
  294. {
  295. return Equal(TypeConverter.ToNumber(x), y);
  296. }
  297. if (typey == TypeCode.Boolean)
  298. {
  299. return Equal(x, TypeConverter.ToNumber(y));
  300. }
  301. if (typey == TypeCode.Object && (typex == TypeCode.String || typex == TypeCode.Double))
  302. {
  303. return Equal(x, TypeConverter.ToPrimitive(y));
  304. }
  305. if (typex == TypeCode.Object && (typey == TypeCode.String || typey == TypeCode.Double))
  306. {
  307. return Equal(TypeConverter.ToPrimitive(x), y);
  308. }
  309. return false;
  310. }
  311. public static bool StriclyEqual(object x, object y)
  312. {
  313. var typea = TypeConverter.GetType(x);
  314. var typeb = TypeConverter.GetType(y);
  315. if (typea != typeb)
  316. {
  317. return false;
  318. }
  319. if (typea == TypeCode.Empty)
  320. {
  321. return true;
  322. }
  323. if (typea == TypeCode.Double)
  324. {
  325. var nx = TypeConverter.ToNumber(x);
  326. var ny = TypeConverter.ToNumber(y);
  327. if (double.IsNaN(nx) || double.IsNaN(ny))
  328. {
  329. return false;
  330. }
  331. if (nx == ny)
  332. {
  333. return true;
  334. }
  335. return false;
  336. }
  337. if (typea == TypeCode.String)
  338. {
  339. return TypeConverter.ToString(x) == TypeConverter.ToString(y);
  340. }
  341. if (typea == TypeCode.Boolean)
  342. {
  343. return TypeConverter.ToBoolean(x) == TypeConverter.ToBoolean(y);
  344. }
  345. return x == y;
  346. }
  347. public object EvaluateIdentifier(Identifier identifier)
  348. {
  349. return _engine.ExecutionContext.LexicalEnvironment.GetIdentifierReference(identifier.Name, _engine.Options.IsStrict());
  350. }
  351. public object EvaluateLiteral(Literal literal)
  352. {
  353. return literal.Value ?? Null.Instance;
  354. }
  355. public object EvaluateObjectExpression(ObjectExpression objectExpression)
  356. {
  357. // http://www.ecma-international.org/ecma-262/5.1/#sec-11.1.5
  358. var obj = _engine.Object.Construct(Arguments.Empty);
  359. foreach (var property in objectExpression.Properties)
  360. {
  361. var propName = property.Key.GetKey();
  362. var previous = obj.GetOwnProperty(propName);
  363. PropertyDescriptor propDesc;
  364. switch (property.Kind)
  365. {
  366. case PropertyKind.Data:
  367. var exprValue = _engine.EvaluateExpression(property.Value);
  368. var propValue = _engine.GetValue(exprValue);
  369. propDesc = new DataDescriptor(propValue) {Writable=true, Enumerable=true,Configurable = true};
  370. break;
  371. case PropertyKind.Get:
  372. var getter = property.Value as FunctionExpression;
  373. if (getter == null)
  374. {
  375. throw new JavaScriptException(_engine.SyntaxError);
  376. }
  377. var get = new ScriptFunctionInstance(
  378. _engine,
  379. getter,
  380. _engine.ExecutionContext.LexicalEnvironment,
  381. getter.Strict || _engine.Options.IsStrict()
  382. );
  383. propDesc = new AccessorDescriptor(get) { Enumerable = true, Configurable = true};
  384. break;
  385. case PropertyKind.Set:
  386. var setter = property.Value as FunctionExpression;
  387. if (setter == null)
  388. {
  389. throw new JavaScriptException(_engine.SyntaxError);
  390. }
  391. var set = new ScriptFunctionInstance(
  392. _engine,
  393. setter,
  394. _engine.ExecutionContext.LexicalEnvironment,
  395. setter.Strict || _engine.Options.IsStrict()
  396. );
  397. propDesc = new AccessorDescriptor(null, set) { Enumerable = true, Configurable = true};
  398. break;
  399. default:
  400. throw new ArgumentOutOfRangeException();
  401. }
  402. if (previous != Undefined.Instance)
  403. {
  404. var previousIsData = previous.IsDataDescriptor();
  405. var previousIsAccessor = previous.IsAccessorDescriptor();
  406. var propIsData = propDesc.IsDataDescriptor();
  407. var propIsAccessor = propDesc.IsAccessorDescriptor();
  408. if (_engine.Options.IsStrict() && previousIsData && propIsData)
  409. {
  410. throw new JavaScriptException(_engine.SyntaxError);
  411. }
  412. if (previousIsData && propIsAccessor)
  413. {
  414. throw new JavaScriptException(_engine.SyntaxError);
  415. }
  416. if (previousIsAccessor && propIsData)
  417. {
  418. throw new JavaScriptException(_engine.SyntaxError);
  419. }
  420. if (previousIsAccessor && propIsAccessor && ((previous.As<AccessorDescriptor>().Get != null && propDesc.As<AccessorDescriptor>().Get != null) || (previous.As<AccessorDescriptor>().Set != null && propDesc.As<AccessorDescriptor>().Set != null)))
  421. {
  422. throw new JavaScriptException(_engine.SyntaxError);
  423. }
  424. }
  425. obj.DefineOwnProperty(propName, propDesc, false);
  426. }
  427. return obj;
  428. }
  429. /// <summary>
  430. /// http://www.ecma-international.org/ecma-262/5.1/#sec-11.2.1
  431. /// </summary>
  432. /// <param name="memberExpression"></param>
  433. /// <returns></returns>
  434. public object EvaluateMemberExpression(MemberExpression memberExpression)
  435. {
  436. var baseReference = EvaluateExpression(memberExpression.Object);
  437. var baseValue = _engine.GetValue(baseReference);
  438. string propertyNameString;
  439. if (!memberExpression.Computed) // index accessor ?
  440. {
  441. propertyNameString = memberExpression.Property.As<Identifier>().Name;
  442. }
  443. else
  444. {
  445. var propertyNameReference = EvaluateExpression(memberExpression.Property);
  446. var propertyNameValue = _engine.GetValue(propertyNameReference);
  447. TypeConverter.CheckObjectCoercible(_engine, baseValue);
  448. propertyNameString = TypeConverter.ToString(propertyNameValue);
  449. }
  450. return new Reference(baseValue, propertyNameString, _engine.Options.IsStrict());
  451. }
  452. public object EvaluateFunctionExpression(FunctionExpression functionExpression)
  453. {
  454. return new ScriptFunctionInstance(
  455. _engine,
  456. functionExpression,
  457. LexicalEnvironment.NewDeclarativeEnvironment(_engine, _engine.ExecutionContext.LexicalEnvironment),
  458. functionExpression.Strict
  459. );
  460. }
  461. public object EvaluateCallExpression(CallExpression callExpression)
  462. {
  463. var callee = EvaluateExpression(callExpression.Callee);
  464. var func = _engine.GetValue(callee);
  465. if (func == Undefined.Instance)
  466. {
  467. throw new JavaScriptException(_engine.TypeError);
  468. }
  469. object thisObject;
  470. // todo: implement as in http://www.ecma-international.org/ecma-262/5.1/#sec-11.2.4
  471. var arguments = callExpression.Arguments.Select(EvaluateExpression).Select(_engine.GetValue).ToArray();
  472. if (TypeConverter.GetType(func) != TypeCode.Object)
  473. {
  474. throw new JavaScriptException(_engine.TypeError);
  475. }
  476. var callable = func as ICallable;
  477. if (callable == null)
  478. {
  479. throw new JavaScriptException(_engine.TypeError);
  480. }
  481. var r = callee as Reference;
  482. if (r != null)
  483. {
  484. if (r.IsPropertyReference())
  485. {
  486. thisObject = r.GetBase();
  487. }
  488. else
  489. {
  490. var env = r.GetBase() as EnvironmentRecord;
  491. thisObject = env.ImplicitThisValue();
  492. }
  493. }
  494. else
  495. {
  496. thisObject = Undefined.Instance;
  497. }
  498. return callable.Call(thisObject, arguments);
  499. }
  500. public object EvaluateSequenceExpression(SequenceExpression sequenceExpression)
  501. {
  502. foreach (var expression in sequenceExpression.Expressions)
  503. {
  504. _engine.EvaluateExpression(expression);
  505. }
  506. return Undefined.Instance;
  507. }
  508. public object EvaluateUpdateExpression(UpdateExpression updateExpression)
  509. {
  510. var value = _engine.EvaluateExpression(updateExpression.Argument);
  511. Reference r;
  512. switch (updateExpression.Operator)
  513. {
  514. case "++":
  515. r = value as Reference;
  516. if (r != null
  517. && r.IsStrict()
  518. && (r.GetBase() is EnvironmentRecord)
  519. && (Array.IndexOf(new[] { "eval", "arguments" }, r.GetReferencedName()) != -1))
  520. {
  521. throw new JavaScriptException(_engine.SyntaxError);
  522. }
  523. var oldValue = _engine.GetValue(value);
  524. var newValue = TypeConverter.ToNumber(oldValue) + 1;
  525. _engine.PutValue(r, newValue);
  526. return updateExpression.Prefix ? newValue : oldValue;
  527. case "--":
  528. r = value as Reference;
  529. if (r != null
  530. && r.IsStrict()
  531. && (r.GetBase() is EnvironmentRecord)
  532. && (Array.IndexOf(new[] { "eval", "arguments" }, r.GetReferencedName()) != -1))
  533. {
  534. throw new JavaScriptException(_engine.SyntaxError);
  535. }
  536. oldValue = _engine.GetValue(value);
  537. newValue = TypeConverter.ToNumber(oldValue) - 1;
  538. _engine.PutValue(r, newValue);
  539. return updateExpression.Prefix ? newValue : oldValue;
  540. default:
  541. throw new ArgumentException();
  542. }
  543. }
  544. public object EvaluateThisExpression(ThisExpression thisExpression)
  545. {
  546. return _engine.ExecutionContext.ThisBinding;
  547. }
  548. public object EvaluateNewExpression(NewExpression newExpression)
  549. {
  550. var arguments = newExpression.Arguments.Select(EvaluateExpression).Select(_engine.GetValue).ToArray();
  551. // todo: optimize by defining a common abstract class or interface
  552. var callee = _engine.GetValue(EvaluateExpression(newExpression.Callee)) as IConstructor;
  553. if (callee == null)
  554. {
  555. throw new JavaScriptException(_engine.TypeError, "The object can't be used as constructor.");
  556. }
  557. // construct the new instance using the Function's constructor method
  558. var instance = callee.Construct(arguments);
  559. return instance;
  560. }
  561. public object EvaluateArrayExpression(ArrayExpression arrayExpression)
  562. {
  563. var arguments = arrayExpression.Elements.Select(EvaluateExpression).Select(_engine.GetValue).ToArray();
  564. // construct the new instance using the Function's constructor method
  565. var instance = _engine.Array.Construct(arguments);
  566. return instance;
  567. }
  568. public object EvaluateUnaryExpression(UnaryExpression unaryExpression)
  569. {
  570. var value = _engine.EvaluateExpression(unaryExpression.Argument);
  571. Reference r;
  572. switch (unaryExpression.Operator)
  573. {
  574. case "+":
  575. return TypeConverter.ToNumber(_engine.GetValue(value));
  576. case "-":
  577. var n = TypeConverter.ToNumber(_engine.GetValue(value));
  578. return double.IsNaN(n) ? double.NaN : n*-1;
  579. case "~":
  580. return ~TypeConverter.ToInt32(_engine.GetValue(value));
  581. case "!":
  582. return !TypeConverter.ToBoolean(_engine.GetValue(value));
  583. case "delete":
  584. r = value as Reference;
  585. if (r == null)
  586. {
  587. return true;
  588. }
  589. if (r.IsUnresolvableReference())
  590. {
  591. if (r.IsStrict())
  592. {
  593. throw new JavaScriptException(_engine.SyntaxError);
  594. }
  595. return true;
  596. }
  597. if (r.IsPropertyReference())
  598. {
  599. var o = TypeConverter.ToObject(_engine, r.GetBase());
  600. return o.Delete(r.GetReferencedName(), r.IsStrict());
  601. }
  602. if (r.IsStrict())
  603. {
  604. throw new JavaScriptException(_engine.SyntaxError);
  605. }
  606. var bindings = r.GetBase() as EnvironmentRecord;
  607. return bindings.DeleteBinding(r.GetReferencedName());
  608. case "void":
  609. _engine.GetValue(value);
  610. return Undefined.Instance;
  611. case "typeof":
  612. r = value as Reference;
  613. if (r != null)
  614. {
  615. if (r.IsUnresolvableReference())
  616. {
  617. return "undefined";
  618. }
  619. }
  620. var v = _engine.GetValue(value);
  621. if (v == Undefined.Instance)
  622. {
  623. return "undefined";
  624. }
  625. if (v == Null.Instance)
  626. {
  627. return "object";
  628. }
  629. switch (TypeConverter.GetType(v))
  630. {
  631. case TypeCode.Boolean: return "boolean";
  632. case TypeCode.Double: return "number";
  633. case TypeCode.String: return "string";
  634. }
  635. if (v is ICallable)
  636. {
  637. return "function";
  638. }
  639. return "object";
  640. default:
  641. throw new ArgumentException();
  642. }
  643. }
  644. }
  645. }