/* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* vim: set shiftwidth=4 tabstop=4 autoindent cindent expandtab: */ 'use strict'; var Native = function() { } Native.prototype.JavaException = function(className, msg) { this.className = className; this.msg = msg; } Native.prototype.JavaException.prototype.toString = function() { return "Uncaught Java Exception " + this.className + " " + this.msg; } Native.prototype.invokeNative = function(caller, methodInfo) { function pushType(type, value) { if (type === "long" || type === "double") { caller.stack.push2(value); return; } caller.stack.push(value); } function popType(type) { return (type === "long" || type === "double") ? caller.stack.pop2() : caller.stack.pop(); } function popArgs(types) { var argc = types.length; if (!ACCESS_FLAGS.isStatic(methodInfo.access_flags)) ++argc; var args = Array(argc); for (var i=types.length-1, j=args.length-1; i >= 0; --i, --j) args[j] = popType(types[i].type); if (j >= 0) args[0] = caller.stack.pop(); return args; } var signature = Signature.parse(methodInfo.signature); var args = popArgs(signature.IN); if (!methodInfo.native) methodInfo.native = this.getNativeMethod(methodInfo); var result = methodInfo.native.apply(caller, args); if (signature.OUT.length) pushType(signature.OUT[0].type, result); } Native.prototype.getNativeMethod = function(methodInfo) { var classInfo = methodInfo.classInfo; var className = classInfo.className; var methodName = methodInfo.name; var signature = methodInfo.signature; return this[className + "." + methodName + "." + signature]; } Native.prototype["java/lang/System.arraycopy.(Ljava/lang/Object;ILjava/lang/Object;II)V"] = function(src, srcOffset, dst, dstOffset, length) { if (!src || !dst) { throw new JavaException("java/lang/NullPointerException", "Cannot copy to/from a null array."); return; } var proto = Object.getPrototypeOf(src); if (proto !== Int8Array.prototype && proto !== Int16Array.prototype && proto !== Int32Array.prototype && proto !== Uint16Array.prototype && proto !== Float32Array.prototype && proto !== Float64Array.prototype && proto !== Array.prototype) { throw new JavaException("java/lang/ArrayStoreException", "Can only copy to/from array types."); return; } if (proto !== Object.getPrototypeOf(dst)) { throw new JavaException("java/lang/ArrayStoreException", "Incompatible component types."); return; } if (srcOffset < 0 || (srcOffset+length) > src.length || dstOffset < 0 || (dstOffset+length) > dst.length || length < 0) { throw new JavaException("java/lang/ArrayIndexOutOfBoundsException", "Invalid index."); return; } if (proto === Array.prototype) { // TODO: check casting throw new JavaException("java/lang/ArrayStoreException", "Invalid element type."); return; } if (dst !== src || dstOffset < srcOffset) { for (var n = 0; n < length; ++n) dst[dstOffset++] = src[srcOffset++]; } else { dstOffset += length; srcOffset += length; for (var n = 0; n < length; ++n) dst[--dstOffset] = src[--srcOffset]; } } Native.prototype["java/lang/System.getProperty0.(Ljava/lang/String;)Ljava/lang/String;"] = function(key) { switch (util.fromJavaString(key)) { case "microedition.encoding": return CLASSES.newString("ISO-8859-1"); default: console.log("KEY: " + util.fromJavaString(key.value)); } } Native.prototype["java/lang/System.currentTimeMillis.()J"] = function() { return Long.fromNumber(Date.now()); } Native.prototype["com/sun/cldchi/jvm/JVM.unchecked_char_arraycopy.([CI[CII)V"] = function(src, srcOffset, dst, dstOffset, length) { if (dst !== src || dstOffset < srcOffset) { for (var n = 0; n < length; ++n) dst[dstOffset++] = src[srcOffset++]; } else { dstOffset += length; srcOffset += length; for (var n = 0; n < length; ++n) dst[--dstOffset] = src[--srcOffset]; } } Native.prototype["java/lang/Object.getClass.()Ljava/lang/Class;"] = function(obj) { return obj.class.getClassObject(); } Native.prototype["java/lang/Class.getName.()Ljava/lang/String;"] = function(obj) { return util.cache(obj, "getName", function () { return CLASSES.newString(obj.vmClass.className.replace("/", ".", "g")); }); } Native.prototype["java/lang/Class.forName.(Ljava/lang/String;)Ljava/lang/Class;"] = function(name) { var className = util.fromJavaString(name).replace(".", "/", "g"); var classInfo = (className[0] === "[") ? null : CLASSES.getClass(className); if (!classInfo) { throw new Native.JavaException("java/lang/ClassNotFoundException", "'" + className + "' not found."); } return classInfo.getClassObject(); } Native.prototype["java/lang/Class.newInstance.()Ljava/lang/Object;"] = function(classObject) { var classInfo = classObject.vmClass; CLASSES.initClass(classInfo); return new (classInfo.constructor)(); }; Native.prototype["java/lang/Class.isInterface.()Z"] = function(classObject) { var classInfo = classObject.vmClass; return ACCESS_FLAGS.isInterface(classInfo.access_flags) ? 1 : 0; } Native.prototype["java/lang/Class.isArray.()Z"] = function(classObject) { var classInfo = classObject.vmClass; return classInfo.isArrayClass ? 1 : 0; } Native.prototype["java/lang/Class.isAssignableFrom.(Ljava/lang/Class;)Z"] = function(classObject, fromClass) { if (!fromClass) { throw new JavaException("java/lang/NullPointerException"); return; } return fromClass.vmClass.isAssignableTo(classObject.vmClass) ? 1 : 0; } Native.prototype["java/lang/Class.isInstance.(Ljava/lang/Object;)Z"] = function(classObject, obj) { return obj && obj.class.isAssignableTo(classObject.vmClass) ? 1 : 0; } Native.prototype["java/lang/Float.floatToIntBits.(F)I"] = (function() { var fa = Float32Array(1); var ia = Int32Array(fa.buffer); return function(f) { fa[0] = f; return ia[0]; } })(); Native.prototype["java/lang/Double.doubleToLongBits.(D)J"] = (function() { var da = Float64Array(1); var ia = Int32Array(da.buffer); return function(d) { da[0] = d; return Long.fromBits(ia[0], ia[1]); } })(); Native.prototype["java/lang/Float.intBitsToFloat.(I)F"] = (function() { var fa = Float32Array(1); var ia = Int32Array(fa.buffer); return function(i) { ia[0] = i; return fa[0]; } })(); Native.prototype["java/lang/Double.longBitsToDouble.(J)D"] = (function() { var da = Float64Array(1); var ia = Int32Array(da.buffer); return function(l) { ia[0] = l.low; ia[1] = l.high; return da[0]; } })(); Native.prototype["java/lang/Throwable.fillInStackTrace.()V"] = (function() { }); Native.prototype["java/lang/Throwable.obtainBackTrace.()Ljava/lang/Object;"] = (function() { return null; }); Native.prototype["com/sun/cldchi/io/ConsoleOutputStream.write.(I)V"] = (function() { var s = ""; return function(obj, ch) { if (ch === 10) { document.getElementById("output").textContent += s + "\n"; s = ""; return; } s += String.fromCharCode(ch); } })(); Native.prototype["com/sun/cldc/io/ResourceInputStream.open.(Ljava/lang/String;)Ljava/lang/Object;"] = function(name) { var fileName = util.fromJavaString(name); var data = CLASSES.loadFile(fileName); if (!data) return null; var obj = CLASSES.newObject("java/lang/Object"); obj.data = Uint8Array(data); obj.pos = 0; return obj; }; Native.prototype["com/sun/cldc/io/ResourceInputStream.bytesRemain.(Ljava/lang/Object;)I"] = function(handle) { return handle.data.length - handle.pos; } Native.prototype["com/sun/cldc/io/ResourceInputStream.readByte.(Ljava/lang/Object;)I"] = function(handle) { return handle.data[handle.pos++]; } Native.prototype["com/sun/cldc/io/ResourceInputStream.readBytes.(Ljava/lang/Object;[BII)I"] = function(handle, b, off, len) { var data = handle.data; var remaining = data.length - handle.pos; if (remaining > len) len = remaining; for (var n = 0; n < len; ++n) b[off+n] = data[n]; handle.pos += len; return len; }