pluotsorbet/context.js

237 строки
5.9 KiB
JavaScript
Исходник Обычный вид История

2014-07-24 01:00:36 +04:00
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
'use strict';
function Context() {
2014-07-24 11:48:21 +04:00
this.frames = [];
}
Context.prototype.current = function() {
var frames = this.frames;
return frames[frames.length - 1];
}
Context.prototype.pushFrame = function(methodInfo, consumes) {
var caller = this.current();
var callee = new Frame(methodInfo);
callee.locals = caller.stack;
callee.localsBase = caller.stack.length - consumes;
this.frames.push(callee);
return callee;
}
Context.prototype.popFrame = function() {
var callee = this.frames.pop();
var caller = this.current();
if (callee.localsBase)
caller.stack.length = callee.localsBase;
return caller;
}
2014-07-26 00:09:49 +04:00
Context.prototype.pushClassInitFrame = function(classInfo) {
if (classInfo.initialized)
return;
2014-07-26 22:42:02 +04:00
// console.log("starting initialization of", classInfo.className, this.thread.toString());
classInfo.thread = this.thread;
var syntheticMethod = {
classInfo: {
constant_pool: [
null,
{ class_index: 2, name_and_type_index: 4 },
{ name_index: 3 },
{ bytes: "java/lang/Class" },
{ name_index: 5, signature_index: 6 },
{ bytes: "invoke_clinit" },
{ bytes: "()V" },
{ class_index: 2, name_and_type_index: 8 },
{ name_index: 9, signature_index: 10 },
{ bytes: "init9" },
{ bytes: "()V" },
],
},
code: [
0x2a, // aload_0
0x59, // dup
0xb7, 0x00, 0x01, // invokespecial <idx=1>
0xb7, 0x00, 0x07, // invokespecial <idx=7>
0xb1, // return
],
exception_table: [],
};
this.current().stack.push(classInfo.getClassObject());
this.pushFrame(syntheticMethod, 1);
}
Context.prototype.backTrace = function() {
var stack = [];
this.frames.forEach(function(frame) {
var methodInfo = frame.methodInfo;
2014-07-26 04:54:34 +04:00
if (!methodInfo || !methodInfo.name)
return;
var className = methodInfo.classInfo.className;
var methodName = methodInfo.name;
var signature = Signature.parse(methodInfo.signature);
var IN = signature.IN;
var args = [];
var lp = 0;
for (var n = 0; n < IN.length; ++n) {
var arg = frame.locals[frame.localsBase + lp];
++lp;
switch (IN[n].type) {
case "long":
case "double":
++lp;
break;
case "object":
if (arg === null)
arg = "null";
else if (arg.class.className === "java/lang/String")
arg = "'" + util.fromJavaString(arg) + "'";
else
arg = "<" + arg.class.className + ">";
}
args.push(arg);
}
2014-07-26 04:54:34 +04:00
stack.push(methodInfo.classInfo.className + "." + methodInfo.name + ":" + frame.ip +
"(" + args.join(",") + ")");
});
return stack.join("\n");
}
2014-07-26 00:09:49 +04:00
Context.prototype.raiseException = function(className, message) {
if (!message)
message = "";
message = "" + message;
var syntheticMethod = {
classInfo: {
constant_pool: [
2014-07-26 00:23:10 +04:00
null,
{ name_index: 2 },
{ bytes: className },
2014-07-26 00:23:10 +04:00
{ tag: TAGS.CONSTANT_String, string_index: 4 },
{ bytes: message },
2014-07-26 00:23:10 +04:00
{ class_index: 1, name_and_type_index: 6 },
{ name_index: 7, signature_index: 8 },
{ bytes: "<init>" },
{ bytes: "(Ljava/lang/String;)V" },
],
},
code: [
2014-07-26 00:24:30 +04:00
0xbb, 0x00, 0x01, // new <idx=1>
0x59, // dup
0x12, 0x03, // ldc <idx=2>
0xb7, 0x00, 0x05, // invokespecial <idx=5>
0xbf // athrow
],
exception_table: [],
};
this.pushFrame(syntheticMethod, 0);
2014-07-26 00:09:49 +04:00
throw VM.Yield;
}
Context.prototype.newString = function(s) {
var obj = CLASSES.newObject(CLASSES.java_lang_String);
var length = s.length;
var chars = CLASSES.newPrimitiveArray("C", length);
for (var n = 0; n < length; ++n)
chars[n] = s.charCodeAt(n);
obj["java/lang/String$value"] = chars;
obj["java/lang/String$offset"] = 0;
obj["java/lang/String$count"] = length;
return obj;
2014-07-24 01:28:32 +04:00
}
2014-07-26 00:09:49 +04:00
2014-07-26 02:24:14 +04:00
Context.prototype.execute = function(stopFrame) {
2014-07-26 00:09:49 +04:00
while (this.current() !== stopFrame) {
try {
VM.execute(this);
} catch (e) {
2014-07-26 04:54:34 +04:00
switch (e) {
case VM.Yield:
break;
case VM.Pause:
return;
default:
2014-07-26 00:09:49 +04:00
throw e;
2014-07-26 04:54:34 +04:00
}
2014-07-26 00:09:49 +04:00
}
}
}
2014-07-26 02:24:14 +04:00
Context.prototype.start = function(stopFrame) {
if (this.current() === stopFrame)
return;
var ctx = this;
2014-07-26 04:54:34 +04:00
ctx.stopFrame = stopFrame;
2014-07-26 02:24:14 +04:00
window.setZeroTimeout(function() {
try {
VM.execute(ctx);
} catch (e) {
2014-07-26 04:54:34 +04:00
switch (e) {
case VM.Yield:
break;
case VM.Pause:
return;
default:
2014-07-26 02:24:14 +04:00
throw e;
2014-07-26 04:54:34 +04:00
}
2014-07-26 02:24:14 +04:00
}
ctx.start(stopFrame);
});
}
2014-07-26 04:54:34 +04:00
Context.prototype.resume = function() {
this.start(this.stopFrame);
}
2014-07-26 22:42:02 +04:00
Context.prototype.wait = function(obj) {
if (!obj.waiters)
obj.waiters = [];
obj.waiters.push(this);
throw VM.Pause;
}
Context.prototype.notify = function(obj) {
2014-07-28 21:47:02 +04:00
if (obj.waiters && obj.waiters.length) {
if (!obj.ready)
obj.ready = [];
obj.ready.push(obj.waiters.pop());
}
if (!obj.ready || !obj.ready.length)
2014-07-26 22:42:02 +04:00
return;
2014-07-28 21:47:02 +04:00
var ctx = obj.ready.pop();
window.setZeroTimeout(VM.execute.bind(null, ctx));
2014-07-26 22:42:02 +04:00
}
Context.prototype.notifyAll = function(obj) {
2014-07-28 21:47:02 +04:00
while ((obj.waiters && obj.waiters.length) || (obj.ready && obj.ready.length))
2014-07-26 22:42:02 +04:00
this.notify(obj);
}
Context.prototype.monitorEnter = function(obj) {
var lock = obj.lock;
if (!lock) {
obj.lock = { thread: this.thread, count: 1 };
return;
}
if (lock.thread === this.thread) {
++lock.count;
return;
}
this.wait(obj);
}
Context.prototype.monitorExit = function(obj) {
var lock = obj.lock;
if (lock.thread !== this.thread) {
console.log("WARNING: thread tried to unlock a monitor it didn't own");
return;
}
if (--lock.count > 0) {
return;
}
obj.lock = null;
this.notifyAll(obj);
}