Exception: MiniRacer::EvalError
- Defined in:
- lib/mini_racer.rb,
ext/mini_racer_extension/mini_racer_extension.c
Direct Known Subclasses
InternalError, ParseError, RuntimeError, ScriptError, ScriptTerminatedError, V8OutOfMemoryError
Class Method Summary collapse
Instance Method Summary collapse
- #attach(name, proc) ⇒ Object
- #call(*args) ⇒ Object
- #dispose ⇒ Object
- #dump ⇒ Object
- #eval(*args) ⇒ Object
- #heap_snapshot ⇒ Object
- #heap_stats ⇒ Object
- #initialize(*args) ⇒ Object constructor
-
#low_memory_notification ⇒ Object
takes ownership of |b|.
- #pump_message_loop ⇒ Object
- #size ⇒ Object
- #stop ⇒ Object
- #warmup!(arg) ⇒ Object
Constructor Details
#initialize(*args) ⇒ Object
1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1594
static VALUE context_initialize(int argc, VALUE *argv, VALUE self)
{
VALUE kwargs, a, k, v;
pthread_attr_t attr;
const char *cause;
pthread_t thr;
Snapshot *ss;
Context *c;
char *s;
int r;
TypedData_Get_Struct(self, Context, &context_type, c);
rb_scan_args(argc, argv, ":", &kwargs);
if (NIL_P(kwargs))
goto init;
a = rb_ary_new();
rb_hash_foreach(kwargs, collect, a);
while (RARRAY_LENINT(a)) {
v = rb_ary_pop(a);
k = rb_ary_pop(a);
k = rb_sym2str(k);
s = RSTRING_PTR(k);
if (!strcmp(s, "ensure_gc_after_idle")) {
Check_Type(v, T_FIXNUM);
c->idle_gc = FIX2LONG(v);
if (c->idle_gc < 0 || c->idle_gc > INT32_MAX)
rb_raise(rb_eArgError, "bad ensure_gc_after_idle");
} else if (!strcmp(s, "max_memory")) {
Check_Type(v, T_FIXNUM);
c->max_memory = FIX2LONG(v);
if (c->max_memory < 0 || c->max_memory >= UINT32_MAX)
rb_raise(rb_eArgError, "bad max_memory");
} else if (!strcmp(s, "marshal_stack_depth")) { // backcompat, ignored
Check_Type(v, T_FIXNUM);
} else if (!strcmp(s, "timeout")) {
Check_Type(v, T_FIXNUM);
c->timeout = FIX2LONG(v);
if (c->timeout < 0 || c->timeout > INT32_MAX)
rb_raise(rb_eArgError, "bad timeout");
} else if (!strcmp(s, "snapshot")) {
if (NIL_P(v))
continue;
TypedData_Get_Struct(v, Snapshot, &snapshot_type, ss);
if (buf_put(&c->snapshot, RSTRING_PTR(ss->blob), RSTRING_LENINT(ss->blob)))
rb_raise(runtime_error, "out of memory");
} else if (!strcmp(s, "verbose_exceptions")) {
c->verbose_exceptions = !(v == Qfalse || v == Qnil);
} else {
rb_raise(runtime_error, "bad keyword: %s", s);
}
}
init:
if (single_threaded) {
v8_once_init();
c->pst = v8_thread_init(c, c->snapshot.buf, c->snapshot.len, c->max_memory, c->verbose_exceptions);
} else {
cause = "pthread_attr_init";
if ((r = pthread_attr_init(&attr)))
goto fail;
pthread_attr_setstacksize(&attr, 2<<20); // 2 MiB
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
// v8 thread takes ownership of |c|
cause = "pthread_create";
r = pthread_create(&thr, &attr, v8_thread_start, c);
pthread_attr_destroy(&attr);
if (r)
goto fail;
barrier_wait(&c->early_init);
barrier_wait(&c->late_init);
}
return Qnil;
fail:
rb_raise(runtime_error, "Context.initialize: %s: %s", cause, strerror(r));
return Qnil; // pacify compiler
}
|
Class Method Details
.load(blob) ⇒ Object
1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1770
static VALUE snapshot_load(VALUE klass, VALUE blob)
{
Snapshot *ss;
VALUE self;
Check_Type(blob, T_STRING);
self = snapshot_alloc(klass);
TypedData_Get_Struct(self, Snapshot, &snapshot_type, ss);
ss->blob = rb_str_dup(blob);
rb_enc_associate(ss->blob, rb_ascii8bit_encoding());
ENC_CODERANGE_SET(ss->blob, ENC_CODERANGE_VALID);
return self;
}
|
Instance Method Details
#attach(name, proc) ⇒ Object
1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1315
static VALUE context_attach(VALUE self, VALUE name, VALUE proc)
{
Context *c;
VALUE e;
Ser s;
TypedData_Get_Struct(self, Context, &context_type, c);
// request is (A)ttach, [name, id] array
ser_init1(&s, 'A');
ser_array_begin(&s, 2);
add_string(&s, name);
ser_int(&s, RARRAY_LENINT(c->procs));
ser_array_end(&s, 2);
rb_ary_push(c->procs, proc);
// response is an exception or undefined
e = rendezvous(c, &s.b);
handle_exception(e);
return Qnil;
}
|
#call(*args) ⇒ Object
1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1386
static VALUE context_call(int argc, VALUE *argv, VALUE self)
{
VALUE name, args;
VALUE a, e;
Context *c;
Ser s;
TypedData_Get_Struct(self, Context, &context_type, c);
rb_scan_args(argc, argv, "1*", &name, &args);
Check_Type(name, T_STRING);
rb_ary_unshift(args, name);
// request is (C)all, [name, args...] array
ser_init1(&s, 'C');
if (serialize(&s, args)) {
ser_reset(&s);
rb_raise(runtime_error, "Context.call: %s", s.err);
}
// response is [result, err] array
a = rendezvous(c, &s.b); // takes ownership of |s.b|
e = rb_ary_pop(a);
handle_exception(e);
return rb_ary_pop(a);
}
|
#dispose ⇒ Object
1364 1365 1366 1367 1368 1369 1370 1371 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1364
static VALUE context_dispose(VALUE self)
{
Context *c;
TypedData_Get_Struct(self, Context, &context_type, c);
rb_thread_call_without_gvl(context_dispose_do, c, NULL, NULL);
return Qnil;
}
|
#dump ⇒ Object
1762 1763 1764 1765 1766 1767 1768 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1762
static VALUE snapshot_dump(VALUE self)
{
Snapshot *ss;
TypedData_Get_Struct(self, Snapshot, &snapshot_type, ss);
return ss->blob;
}
|
#eval(*args) ⇒ Object
1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1410
static VALUE context_eval(int argc, VALUE *argv, VALUE self)
{
VALUE a, e, source, filename, kwargs;
Context *c;
Ser s;
TypedData_Get_Struct(self, Context, &context_type, c);
filename = Qnil;
rb_scan_args(argc, argv, "1:", &source, &kwargs);
Check_Type(source, T_STRING);
if (!NIL_P(kwargs))
filename = rb_hash_aref(kwargs, rb_id2sym(rb_intern("filename")));
if (NIL_P(filename))
filename = rb_str_new_cstr("<eval>");
Check_Type(filename, T_STRING);
// request is (E)val, [filename, source] array
ser_init1(&s, 'E');
ser_array_begin(&s, 2);
add_string(&s, filename);
add_string(&s, source);
ser_array_end(&s, 2);
// response is [result, errname] array
a = rendezvous(c, &s.b); // takes ownership of |s.b|
e = rb_ary_pop(a);
handle_exception(e);
return rb_ary_pop(a);
}
|
#heap_snapshot ⇒ Object
1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1460
static VALUE context_heap_snapshot(VALUE self)
{
Buf req, res;
Context *c;
TypedData_Get_Struct(self, Context, &context_type, c);
buf_init(&req);
buf_putc(&req, 'H'); // (H)eap snapshot, returns plain bytes
rendezvous_no_des(c, &req, &res); // takes ownership of |req|
return rb_utf8_str_new((char *)res.buf, res.len);
}
|
#heap_stats ⇒ Object
1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1438
static VALUE context_heap_stats(VALUE self)
{
VALUE a, h, k, v;
Context *c;
int i, n;
Buf b;
TypedData_Get_Struct(self, Context, &context_type, c);
buf_init(&b);
buf_putc(&b, 'S'); // (S)tats, returns object
h = rendezvous(c, &b); // takes ownership of |b|
a = rb_ary_new();
rb_hash_foreach(h, collect, a);
for (i = 0, n = RARRAY_LENINT(a); i < n; i += 2) {
k = rb_ary_entry(a, i+0);
v = rb_ary_entry(a, i+1);
rb_hash_delete(h, k);
rb_hash_aset(h, rb_str_intern(k), v); // turn "key" into :key
}
return h;
}
|
#low_memory_notification ⇒ Object
takes ownership of |b|
1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1483
static VALUE context_low_memory_notification(VALUE self)
{
Buf req, res;
Context *c;
TypedData_Get_Struct(self, Context, &context_type, c);
buf_init(&req);
buf_putc(&req, 'L'); // (L)ow memory notification, returns nothing
rendezvous_no_des(c, &req, &res); // takes ownership of |req|
return Qnil;
}
|
#pump_message_loop ⇒ Object
1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1472
static VALUE context_pump_message_loop(VALUE self)
{
Context *c;
Buf b;
TypedData_Get_Struct(self, Context, &context_type, c);
buf_init(&b);
buf_putc(&b, 'P'); // (P)ump, returns bool
return rendezvous(c, &b); // takes ownership of |b|
}
|
#size ⇒ Object
1784 1785 1786 1787 1788 1789 1790 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1784
static VALUE snapshot_size0(VALUE self)
{
Snapshot *ss;
TypedData_Get_Struct(self, Snapshot, &snapshot_type, ss);
return LONG2FIX(RSTRING_LENINT(ss->blob));
}
|
#stop ⇒ Object
1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1373
static VALUE context_stop(VALUE self)
{
Context *c;
// does not grab |mtx| because Context.stop can be called from another
// thread and then we deadlock if e.g. the V8 thread busy-loops in JS
TypedData_Get_Struct(self, Context, &context_type, c);
if (atomic_load(&c->quit))
rb_raise(context_disposed_error, "disposed context");
v8_terminate_execution(c->pst);
return Qnil;
}
|
#warmup!(arg) ⇒ Object
1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 |
# File 'ext/mini_racer_extension/mini_racer_extension.c', line 1731
static VALUE snapshot_warmup(VALUE self, VALUE arg)
{
VALUE a, e, cv;
Snapshot *ss;
Context *c;
DesCtx d;
Ser s;
TypedData_Get_Struct(self, Snapshot, &snapshot_type, ss);
Check_Type(arg, T_STRING);
cv = context_alloc(context_class);
context_initialize(0, NULL, cv);
TypedData_Get_Struct(cv, Context, &context_type, c);
// request is (W)armup, [snapshot, "warmup code"]
ser_init1(&s, 'W');
ser_array_begin(&s, 2);
ser_string8(&s, (const uint8_t *)RSTRING_PTR(ss->blob), RSTRING_LENINT(ss->blob));
add_string(&s, arg);
ser_array_end(&s, 2);
// response is [arraybuffer, error]
DesCtx_init(&d);
d.transcode_latin1 = 0; // don't mangle snapshot binary data
a = rendezvous1(c, &s.b, &d);
e = rb_ary_pop(a);
context_dispose(cv);
if (*RSTRING_PTR(e))
rb_raise(snapshot_error, "%s", RSTRING_PTR(e)+1);
ss->blob = rb_ary_pop(a);
return self;
}
|