зеркало из https://github.com/mozilla/pluotsorbet.git
222 строки
5.8 KiB
JavaScript
222 строки
5.8 KiB
JavaScript
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
|
|
|
'use strict';
|
|
|
|
var util = (function () {
|
|
var Utf8TextDecoder;
|
|
|
|
function decodeUtf8(arrayBuffer) {
|
|
if (!Utf8TextDecoder) {
|
|
Utf8TextDecoder = new TextDecoder("utf-8");
|
|
}
|
|
return Utf8TextDecoder.decode(new Uint8Array(arrayBuffer));
|
|
}
|
|
|
|
/**
|
|
* Provides a UTF-8 decoder that will throw an exception on error rather
|
|
* than silently sanitizing the output.
|
|
*/
|
|
var fallibleUtf8Decoder = new TextDecoder("utf-8", { fatal: true });
|
|
|
|
/**
|
|
* Decodes a UTF-8 string stored in an ArrayBufferView.
|
|
*
|
|
* @param arr An ArrayBufferView to decode (such as a Uint8Array).
|
|
* @returns The decoded string.
|
|
* @throws An invalid enoding is encountered, see
|
|
* TextDecoder.prototype.decode().
|
|
*/
|
|
function decodeUtf8Array(arr) {
|
|
return fallibleUtf8Decoder.decode(arr);
|
|
}
|
|
|
|
// Decode Java's modified UTF-8 (JVM specs, $ 4.4.7)
|
|
// http://docs.oracle.com/javase/specs/jvms/se5.0/html/ClassFile.doc.html#7963
|
|
function javaUTF8Decode(arr) {
|
|
var str = '';
|
|
var i = 0;
|
|
while (i < arr.length) {
|
|
var x = arr[i++];
|
|
|
|
if (x <= 0x7f) {
|
|
// Code points in the range '\u0001' to '\u007F' are represented by a
|
|
// single byte.
|
|
// The 7 bits of data in the byte give the value of the code point
|
|
// represented.
|
|
str += String.fromCharCode(x);
|
|
} else if (x <= 0xdf) {
|
|
// The null code point ('\u0000') and code points in the range '\u0080'
|
|
// to '\u07FF' are represented by a pair of bytes x and y.
|
|
var y = arr[i++];
|
|
str += String.fromCharCode(((x & 0x1f) << 6) + (y & 0x3f));
|
|
} else {
|
|
// Code points in the range '\u0800' to '\uFFFF' are represented by 3
|
|
// bytes x, y, and z.
|
|
var y = arr[i++];
|
|
var z = arr[i++];
|
|
str += String.fromCharCode(((x & 0xf) << 12) + ((y & 0x3f) << 6) + (z & 0x3f));
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function defaultValue(type) {
|
|
if (type === 'J')
|
|
return Long.ZERO;
|
|
if (type[0] === '[' || type[0] === 'L')
|
|
return null;
|
|
return 0;
|
|
}
|
|
|
|
var INT_MAX = Math.pow(2, 31) - 1;
|
|
var INT_MIN = -INT_MAX - 1;
|
|
|
|
function double2int(d) {
|
|
if (d > INT_MAX)
|
|
return INT_MAX;
|
|
if (d < INT_MIN)
|
|
return INT_MIN;
|
|
return d|0;
|
|
}
|
|
|
|
function double2long(d) {
|
|
if (d === Number.POSITIVE_INFINITY)
|
|
return Long.MAX_VALUE;
|
|
if (d === Number.NEGATIVE_INFINITY)
|
|
return Long.MIN_VALUE;
|
|
return Long.fromNumber(d);
|
|
}
|
|
|
|
var jStringEncoder = new TextEncoder('utf-16');
|
|
var jStringDecoder = new TextDecoder('utf-16');
|
|
|
|
function fromJavaChars(chars, offset, count) {
|
|
if (!chars) {
|
|
return null;
|
|
}
|
|
if (typeof count !== 'number')
|
|
count = chars.length;
|
|
if (typeof offset !== 'number')
|
|
offset = 0;
|
|
return jStringDecoder.decode(chars.subarray(offset, offset + count));
|
|
}
|
|
|
|
function fromJavaString(jStr) {
|
|
return J2ME.fromJavaString(jStr);
|
|
}
|
|
|
|
function newPrimitiveArray(type, size) {
|
|
// !!! ask mbx should this be using newArray instead?
|
|
var constructor = J2ME.getArrayConstructor(type);
|
|
return new constructor(size);
|
|
}
|
|
|
|
|
|
// Remove
|
|
function newArray(classInfo, size) {
|
|
return J2ME.newArray(classInfo.klass, size);
|
|
}
|
|
|
|
function newMultiArray(classInfo, lengths) {
|
|
var length = lengths[0];
|
|
var array = newArray(classInfo.elementClass, length);
|
|
if (lengths.length > 1) {
|
|
lengths = lengths.slice(1);
|
|
for (var i=0; i<length; i++)
|
|
array[i] = newMultiArray(classInfo.elementClass, lengths);
|
|
}
|
|
return array;
|
|
}
|
|
|
|
// Remove
|
|
function newObject(classInfo) {
|
|
return J2ME.newObject(classInfo.klass);
|
|
}
|
|
|
|
function newString(s) {
|
|
return J2ME.newString(s);
|
|
}
|
|
|
|
/**
|
|
* Returns an ArrayBufferView of the underlying code points
|
|
* represented by the given Java string.
|
|
*
|
|
* NOTE: Do not modify the ArrayBuffer; it may be shared between
|
|
* multiple string instances.
|
|
*/
|
|
function stringToCharArray(str) {
|
|
return new Uint16Array(jStringEncoder.encode(str).buffer);
|
|
}
|
|
|
|
var id = (function() {
|
|
var gen = 0;
|
|
return function() {
|
|
return ++gen;
|
|
}
|
|
})();
|
|
|
|
function tag(obj) {
|
|
if (!obj.tag)
|
|
obj.tag = id();
|
|
return obj.tag;
|
|
}
|
|
|
|
/**
|
|
* Compare two typed arrays, returning *true* if they have the same length
|
|
* and values, *false* otherwise. Note that we compare by value, not by byte,
|
|
* so:
|
|
* compareTypedArrays(new Uint8Array([0x00, 0xFF]), new Uint8Array[0x00, 0xFF])
|
|
* returns *true*;
|
|
*
|
|
* and:
|
|
* compareTypedArrays(new Uint8Array([0x00, 0xFF]), new Uint32Array[0x00000000, 0x000000FF])
|
|
* also returns *true*;
|
|
*
|
|
* but:
|
|
* compareTypedArrays(new Uint8Array([0x00, 0xFF]), new Uint16Array([0x00FF]))
|
|
* returns *false*.
|
|
*/
|
|
function compareTypedArrays(ary1, ary2) {
|
|
if (ary1.length != ary2.length) {
|
|
return false;
|
|
}
|
|
|
|
for (var i = 0; i < ary1.length; i++) {
|
|
if (ary1[i] !== ary2[i]) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function pad(num, len) {
|
|
return "0".repeat(len - num.toString().length) + num;
|
|
}
|
|
|
|
return {
|
|
INT_MAX: INT_MAX,
|
|
INT_MIN: INT_MIN,
|
|
decodeUtf8: decodeUtf8,
|
|
decodeUtf8Array: decodeUtf8Array,
|
|
javaUTF8Decode: javaUTF8Decode,
|
|
defaultValue: defaultValue,
|
|
double2int: double2int,
|
|
double2long: double2long,
|
|
fromJavaChars: fromJavaChars,
|
|
fromJavaString: fromJavaString,
|
|
newPrimitiveArray: newPrimitiveArray,
|
|
newArray: newArray,
|
|
newMultiArray: newMultiArray,
|
|
newObject: newObject,
|
|
newString: newString,
|
|
stringToCharArray: stringToCharArray,
|
|
id: id,
|
|
tag: tag,
|
|
compareTypedArrays: compareTypedArrays,
|
|
pad: pad,
|
|
};
|
|
})();
|