1009 lines
23 KiB
JavaScript
1009 lines
23 KiB
JavaScript
Number.prototype.hex = function(len)
|
|
{
|
|
len = (len || 8);
|
|
var str = (this >>> 0).toString(16).toUpperCase()
|
|
while (str.length < len)
|
|
{
|
|
str = "0" + str
|
|
}
|
|
return str
|
|
}
|
|
|
|
const u8 = 'u8', u16 = 'u16', u32 = 'u32',
|
|
s8 = 's8', s16 = 's16', s32 = 's32',
|
|
float = 'float', double = 'double'
|
|
|
|
const _typeSizes = {
|
|
u8: 1, u16: 2, u32: 4,
|
|
s8: 1, s16: 2, s32: 4,
|
|
float: 4, double: 8
|
|
}
|
|
|
|
const _regNums = {
|
|
r0: 0, at: 1, v0: 2, v1: 3,
|
|
a0: 4, a1: 5, a2: 6, a3: 7,
|
|
t0: 8, t1: 9, t2: 10, t3: 11,
|
|
t4: 12, t5: 13, t6: 14, t7: 15,
|
|
s0: 16, s1: 17, s2: 18, s3: 19,
|
|
s4: 20, s5: 21, s6: 22, s7: 23,
|
|
t8: 24, t9: 25, k0: 26, k1: 27,
|
|
gp: 28, sp: 29, fp: 30, ra: 31,
|
|
|
|
f0: 0, f1: 1, f2: 2, f3: 3,
|
|
f4: 4, f5: 5, f6: 6, f7: 7,
|
|
f8: 8, f9: 9, f10: 10, f11: 11,
|
|
f12: 12, f13: 13, f14: 14, f15: 15,
|
|
f16: 16, f17: 17, f18: 18, f19: 19,
|
|
f20: 20, f21: 21, f22: 22, f23: 23,
|
|
f24: 24, f25: 25, f26: 26, f27: 27,
|
|
f28: 28, f29: 29, f30: 30, f31: 31
|
|
}
|
|
|
|
const _gprNames = [
|
|
'r0', 'at', 'v0', 'v1', 'a0', 'a1', 'a2', 'a3',
|
|
't0', 't1', 't2', 't3', 't4', 't5', 't6', 't7',
|
|
's0', 's1', 's2', 's3', 's4', 's5', 's6', 's7',
|
|
't8', 't9', 'k0', 'k1', 'gp', 'sp', 'fp', 'ra'
|
|
]
|
|
|
|
|
|
// When we give callbacks or objects to native code, we need to make sure we keep a strong backing reference
|
|
// to them in Javascript so they don't get garbage collected before native code uses them.
|
|
const _strongBackingReferences = (function() {
|
|
const _references = [];
|
|
const noop = function () { };
|
|
|
|
return {
|
|
"add": function _strongBackingReferences_add(obj)
|
|
{
|
|
_references.push(obj);
|
|
},
|
|
"remove": function _strongBackingReferences_remove(obj)
|
|
{
|
|
const index = _references.indexOf(obj);
|
|
if (index !== -1)
|
|
{
|
|
_references.splice(index, 1);
|
|
}
|
|
},
|
|
"singleUseCallback": function _strongBackingReferences_singleUseCallback(callback)
|
|
{
|
|
if (callback === undefined)
|
|
{
|
|
return noop;
|
|
}
|
|
|
|
const singleUseCallback = function ()
|
|
{
|
|
callback.apply(undefined, arguments);
|
|
_strongBackingReferences.remove(callback);
|
|
};
|
|
|
|
_strongBackingReferences.add(singleUseCallback);
|
|
return singleUseCallback;
|
|
}
|
|
};
|
|
}) ();
|
|
|
|
|
|
const GPR_R0 = (1 << 0)
|
|
const GPR_AT = (1 << 1)
|
|
const GPR_V0 = (1 << 2)
|
|
const GPR_V1 = (1 << 3)
|
|
const GPR_A0 = (1 << 4)
|
|
const GPR_A1 = (1 << 5)
|
|
const GPR_A2 = (1 << 6)
|
|
const GPR_A3 = (1 << 7)
|
|
const GPR_T0 = (1 << 8)
|
|
const GPR_T1 = (1 << 9)
|
|
const GPR_T2 = (1 << 10)
|
|
const GPR_T3 = (1 << 11)
|
|
const GPR_T4 = (1 << 12)
|
|
const GPR_T5 = (1 << 13)
|
|
const GPR_T6 = (1 << 14)
|
|
const GPR_T7 = (1 << 15)
|
|
const GPR_S0 = (1 << 16)
|
|
const GPR_S1 = (1 << 17)
|
|
const GPR_S2 = (1 << 18)
|
|
const GPR_S3 = (1 << 19)
|
|
const GPR_S4 = (1 << 20)
|
|
const GPR_S5 = (1 << 21)
|
|
const GPR_S6 = (1 << 22)
|
|
const GPR_S7 = (1 << 23)
|
|
const GPR_T8 = (1 << 24)
|
|
const GPR_T9 = (1 << 25)
|
|
const GPR_K0 = (1 << 26)
|
|
const GPR_K1 = (1 << 27)
|
|
const GPR_GP = (1 << 28)
|
|
const GPR_SP = (1 << 29)
|
|
const GPR_FP = (1 << 30)
|
|
const GPR_RA = (1 << 31) >>> 0
|
|
|
|
const GPR_ANY_ARG = (GPR_A0 | GPR_A1 | GPR_A2 | GPR_A3)
|
|
const GPR_ANY_TEMP = (GPR_T0 | GPR_T1 | GPR_T2 | GPR_T3 | GPR_T4 | GPR_T5 | GPR_T6 | GPR_T7 | GPR_T8 | GPR_T9)
|
|
const GPR_ANY_SAVE = (GPR_S0 | GPR_S1 | GPR_S2 | GPR_S3 | GPR_S4 | GPR_S5 | GPR_S6 | GPR_S7)
|
|
const GPR_ANY = (GPR_R0 | GPR_AT | GPR_V0 | GPR_V1 | GPR_ANY_ARG | GPR_ANY_TEMP | GPR_ANY_SAVE | GPR_K0 | GPR_K1 | GPR_GP | GPR_SP | GPR_FP | GPR_RA) >>> 0
|
|
|
|
function AddressRange(start, end)
|
|
{
|
|
this.start = start >>> 0
|
|
this.end = end >>> 0
|
|
Object.freeze(this)
|
|
}
|
|
|
|
const ADDR_ANY = new AddressRange(0x00000000, 0xFFFFFFFF)
|
|
const ADDR_ANY_KUSEG = new AddressRange(0x00000000, 0x7FFFFFFF)
|
|
const ADDR_ANY_KSEG0 = new AddressRange(0x80000000, 0x9FFFFFFF)
|
|
const ADDR_ANY_KSEG1 = new AddressRange(0xA0000000, 0xBFFFFFFF)
|
|
const ADDR_ANY_KSEG2 = new AddressRange(0xC0000000, 0xFFFFFFFF)
|
|
const ADDR_ANY_RDRAM = new AddressRange(0x80000000, 0x807FFFFF)
|
|
const ADDR_ANY_RDRAM_UNC = new AddressRange(0xA0000000, 0xA07FFFFF)
|
|
const ADDR_ANY_CART_ROM = new AddressRange(0x90000000, 0x95FFFFFF)
|
|
const ADDR_ANY_CART_ROM_UNC = new AddressRange(0xB0000000, 0xB5FFFFFF)
|
|
|
|
const asm = {
|
|
gprname: function(num)
|
|
{
|
|
if(num in _gprNames)
|
|
{
|
|
return _gprNames[num]
|
|
}
|
|
return ""
|
|
}
|
|
}
|
|
|
|
const fs = {
|
|
open: function(path, mode)
|
|
{
|
|
if (["r", "rb", "w", "wb", "a", "ab", "r+", "rb+", "r+b", "w+", "wb+", "w+b", "a+", "ab+", "a+b"].indexOf(mode) == -1)
|
|
{
|
|
return false
|
|
}
|
|
return _native.fsOpen(path, mode)
|
|
},
|
|
close: function(fd)
|
|
{
|
|
return _native.fsClose(fd)
|
|
},
|
|
write: function(fd, buffer, offset, length, position)
|
|
{
|
|
return _native.fsWrite(fd, buffer, offset, length, position)
|
|
},
|
|
writeFile: function(path, buffer)
|
|
{
|
|
var fd = fs.open(path, 'wb')
|
|
|
|
if (fd == false)
|
|
{
|
|
return false
|
|
}
|
|
|
|
fs.write(fd, buffer)
|
|
fs.close(fd)
|
|
return true
|
|
},
|
|
read: function (fd, buffer, offset, length, position)
|
|
{
|
|
return _native.fsRead(fd, buffer, offset, length, position)
|
|
},
|
|
readFile: function(path)
|
|
{
|
|
var fd = fs.open(path, 'rb')
|
|
|
|
if(fd == false)
|
|
{
|
|
return false
|
|
}
|
|
|
|
var stats = fs.fstat(fd);
|
|
|
|
var buf = new Buffer(stats.size)
|
|
|
|
fs.read(fd, buf, 0, stats.size, 0)
|
|
|
|
return buf
|
|
},
|
|
fstat: function(fd)
|
|
{
|
|
var rawStats = _native.fsFStat(fd)
|
|
if (rawStats == false)
|
|
{
|
|
return false
|
|
}
|
|
return new fs.Stats(rawStats)
|
|
},
|
|
stat: function(path)
|
|
{
|
|
var rawStats = _native.fsStat(path)
|
|
if (rawStats == false)
|
|
{
|
|
return false
|
|
}
|
|
return new fs.Stats(rawStats)
|
|
},
|
|
mkdir: function(path)
|
|
{
|
|
return _native.fsMkDir(path)
|
|
},
|
|
rmdir: function(path)
|
|
{
|
|
return _native.fsRmDir(path)
|
|
},
|
|
readdir: function(path)
|
|
{
|
|
return _native.fsReadDir(path)
|
|
},
|
|
unlink: function(path)
|
|
{
|
|
return _native.fsUnlink(path)
|
|
},
|
|
Stats: function(rawstats)
|
|
{
|
|
this.dev = rawstats.dev
|
|
this.ino = rawstats.ino
|
|
this.mode = rawstats.mode
|
|
this.nlink = rawstats.nlink
|
|
this.uid = rawstats.uid
|
|
this.gid = rawstats.gid
|
|
this.rdev = rawstats.rdev
|
|
this.size = rawstats.size
|
|
this.atimeMs = rawstats.atimeMs
|
|
this.mtimeMs = rawstats.mtimeMs
|
|
this.ctimeMs = rawstats.ctimeMs
|
|
this.atime = new Date(this.atimeMs)
|
|
this.mtime = new Date(this.mtimeMs)
|
|
this.ctime = new Date(this.ctimeMs)
|
|
Object.freeze(this)
|
|
}
|
|
}
|
|
|
|
fs.Stats.S_IFMT = 0xF000
|
|
fs.Stats.S_IFDIR = 0x4000
|
|
fs.Stats.S_IFCHR = 0x2000
|
|
fs.Stats.S_IFIFO = 0x1000
|
|
fs.Stats.S_IFREG = 0x8000
|
|
fs.Stats.S_IREAD = 0x0100
|
|
fs.Stats.S_IWRITE = 0x0080
|
|
fs.Stats.S_IEXEC = 0x0040
|
|
|
|
fs.Stats.prototype.isDirectory = function()
|
|
{
|
|
return ((this.mode & fs.Stats.S_IFMT) == fs.Stats.S_IFDIR)
|
|
}
|
|
|
|
fs.Stats.prototype.isFile = function()
|
|
{
|
|
return ((this.mode & fs.Stats.S_IFMT) == fs.Stats.S_IFREG)
|
|
}
|
|
|
|
Object.freeze(fs.Stats)
|
|
Object.freeze(fs.Stats.prototype)
|
|
|
|
const system = {
|
|
pause: function()
|
|
{
|
|
_native.pause()
|
|
},
|
|
resume: function()
|
|
{
|
|
|
|
},
|
|
reset: function()
|
|
{
|
|
|
|
},
|
|
hardreset: function()
|
|
{
|
|
|
|
},
|
|
savestate: function()
|
|
{
|
|
|
|
},
|
|
loadstate: function()
|
|
{
|
|
|
|
},
|
|
setsaveslot: function(slot)
|
|
{
|
|
|
|
},
|
|
getsaveslot: function()
|
|
{
|
|
|
|
},
|
|
generatebitmap: function()
|
|
{
|
|
|
|
}
|
|
}
|
|
|
|
const debug = {
|
|
showmemory: function(address)
|
|
{
|
|
|
|
},
|
|
showcommands: function(address)
|
|
{
|
|
_native.showCommands(address)
|
|
},
|
|
breakhere: function()
|
|
{
|
|
_native.breakHere()
|
|
},
|
|
disasm: function()
|
|
{
|
|
|
|
}
|
|
}
|
|
|
|
const console = {
|
|
print: function(data)
|
|
{
|
|
_native.consolePrint(data);
|
|
},
|
|
log: function()
|
|
{
|
|
for(var i in arguments)
|
|
{
|
|
console.print(arguments[i].toString())
|
|
if(i < arguments.length - 1)
|
|
{
|
|
console.print(" ")
|
|
}
|
|
}
|
|
console.print("\r\n")
|
|
},
|
|
clear: function()
|
|
{
|
|
_native.consoleClear();
|
|
}
|
|
}
|
|
|
|
const screen = {
|
|
print: function(x, y, text)
|
|
{
|
|
_native.screenPrint(x, y, text)
|
|
}
|
|
}
|
|
|
|
const events = (function()
|
|
{
|
|
var callbacks = {};
|
|
var nextCallbackId = 0;
|
|
return {
|
|
on: function(hook, callback, param, param2, bOnce)
|
|
{
|
|
this._stashCallback(callback)
|
|
return _native.addCallback(hook, callback, param, param2, 0, 0, bOnce)
|
|
},
|
|
onexec: function(addr, callback)
|
|
{
|
|
var param = 0;
|
|
var param2 = 0;
|
|
|
|
if (typeof (addr) == "object")
|
|
{
|
|
param = addr.start;
|
|
param2 = addr.end;
|
|
}
|
|
else if (typeof (addr) == "number")
|
|
{
|
|
param = addr;
|
|
}
|
|
|
|
return events.on('exec', callback, param, param2)
|
|
},
|
|
onopcode: function (addr, value, arg3, arg4)
|
|
{
|
|
// onopcode(addr, value, callback)
|
|
// onopcode(addr, value, mask, callback)
|
|
|
|
var start = 0;
|
|
var end = 0;
|
|
var mask;
|
|
var callback;
|
|
|
|
if(typeof(addr) == "object")
|
|
{
|
|
start = addr.start;
|
|
end = addr.end;
|
|
}
|
|
else if (typeof (addr) == "number")
|
|
{
|
|
start = addr;
|
|
}
|
|
|
|
if (typeof (arg3) == "number")
|
|
{
|
|
mask = arg3;
|
|
callback = arg4;
|
|
}
|
|
else if (typeof (arg3) == "function")
|
|
{
|
|
mask = 0xFFFFFFFF;
|
|
callback = arg3;
|
|
}
|
|
|
|
this._stashCallback(callback);
|
|
return _native.addCallback('opcode', callback, start, end, value, mask)
|
|
},
|
|
ongprvalue: function(addr, registers, value, callback)
|
|
{
|
|
var start = 0;
|
|
var end = 0;
|
|
|
|
if(typeof(addr) == "object")
|
|
{
|
|
start = addr.start;
|
|
end = addr.end;
|
|
}
|
|
else if(typeof(addr) == "number")
|
|
{
|
|
start = addr;
|
|
}
|
|
|
|
this._stashCallback(callback);
|
|
return _native.addCallback('gprvalue', callback, start, end, registers, value)
|
|
},
|
|
onread: function(addr, callback)
|
|
{
|
|
var param = 0;
|
|
var param2 = 0;
|
|
|
|
if (typeof (addr) == "object")
|
|
{
|
|
param = addr.start;
|
|
param2 = addr.end;
|
|
}
|
|
else if (typeof (addr) == "number")
|
|
{
|
|
param = addr;
|
|
}
|
|
|
|
return events.on('read', callback, param, param2)
|
|
},
|
|
onwrite: function(addr, callback)
|
|
{
|
|
var param = 0;
|
|
var param2 = 0;
|
|
|
|
if (typeof (addr) == "object")
|
|
{
|
|
param = addr.start;
|
|
param2 = addr.end;
|
|
}
|
|
else if (typeof (addr) == "number")
|
|
{
|
|
param = addr;
|
|
}
|
|
|
|
return events.on('write', callback, param, param2)
|
|
},
|
|
ondraw: function(callback)
|
|
{
|
|
return events.on('draw', callback, 0)
|
|
},
|
|
remove: function(callbackId)
|
|
{
|
|
_native.removeCallback(callbackId)
|
|
},
|
|
clear: function(){},
|
|
_stashCallback: function(callback)
|
|
{
|
|
callbacks[nextCallbackId] = callback
|
|
return nextCallbackId++;
|
|
},
|
|
_unstashCallback: function()
|
|
{
|
|
|
|
},
|
|
}
|
|
})();
|
|
|
|
const gpr = new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
return _native.getGPRVal(prop)
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
return _native.getGPRVal(_regNums[prop])
|
|
}
|
|
switch(prop)
|
|
{
|
|
case 'pc': return _native.getPCVal(); break;
|
|
case 'hi': return _native.getHIVal(false); break;
|
|
case 'lo': return _native.getLOVal(false); break;
|
|
}
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
_native.setGPRVal(prop, false, val)
|
|
return
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
_native.setGPRVal(_regNums[prop], false, val)
|
|
return
|
|
}
|
|
switch(prop)
|
|
{
|
|
case 'pc': _native.setPCVal(val); break;
|
|
case 'hi': _native.setHIVal(false, val); break;
|
|
case 'lo': _native.setLOVal(false, val); break;
|
|
}
|
|
}
|
|
})
|
|
|
|
const ugpr = new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
return _native.getGPRVal(prop, true)
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
return _native.getGPRVal(_regNums[prop], true)
|
|
}
|
|
switch(prop)
|
|
{
|
|
case 'hi': return _native.getHIVal(true); break;
|
|
case 'lo': return _native.getLOVal(true); break;
|
|
}
|
|
},
|
|
set: function (obj, prop, val)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
_native.setGPRVal(prop, true, val)
|
|
return
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
_native.setGPRVal(_regNums[prop], true, val)
|
|
return
|
|
}
|
|
switch(prop)
|
|
{
|
|
case 'hi': _native.setHIVal(true, val); break;
|
|
case 'lo': _native.setLOVal(true, val); break;
|
|
}
|
|
}
|
|
})
|
|
|
|
const fpr = new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
if (typeof prop == 'number')
|
|
{
|
|
return _native.getFPRVal(prop)
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
return _native.getFPRVal(_regNums[prop])
|
|
}
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
_native.setFPRVal(prop, false, val)
|
|
return
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
_native.setFPRVal(_regNums[prop], false, val)
|
|
}
|
|
}
|
|
})
|
|
|
|
const dfpr = new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
if (typeof prop == 'number')
|
|
{
|
|
return _native.getFPRVal(prop, true)
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
return _native.getFPRVal(_regNums[prop], true)
|
|
}
|
|
},
|
|
set: function (obj, prop, val)
|
|
{
|
|
if (typeof prop == 'number' && prop < 32)
|
|
{
|
|
_native.setFPRVal(prop, true, val)
|
|
return
|
|
}
|
|
if (prop in _regNums)
|
|
{
|
|
_native.setFPRVal(_regNums[prop], true, val)
|
|
}
|
|
}
|
|
})
|
|
|
|
const cop0 = new Proxy({},
|
|
{
|
|
get: function (obj, prop) {
|
|
if (prop == "cause") {
|
|
return _native.getCauseVal();
|
|
}
|
|
},
|
|
set: function (obj, prop, val) {
|
|
if (prop == "cause") {
|
|
_native.setCauseVal(val);
|
|
}
|
|
}
|
|
})
|
|
|
|
const rom = {
|
|
u8: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 8, false)
|
|
}
|
|
}),
|
|
u16: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 16, false)
|
|
}
|
|
}),
|
|
u32: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 32, false)
|
|
}
|
|
}),
|
|
s8: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 8, true)
|
|
}
|
|
}),
|
|
s16: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 16, true)
|
|
}
|
|
}),
|
|
s32: new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMInt(prop, 32, true)
|
|
}
|
|
}),
|
|
'float': new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMFloat(prop)
|
|
},
|
|
set: function (obj, prop, val)
|
|
{
|
|
_native.setROMFloat(prop, val)
|
|
}
|
|
}),
|
|
'double': new Proxy({},
|
|
{
|
|
get: function (obj, prop)
|
|
{
|
|
return _native.getROMFloat(prop, true)
|
|
},
|
|
set: function (obj, prop, val)
|
|
{
|
|
_native.setROMFloat(prop, val, true)
|
|
}
|
|
}),
|
|
getblock: function (address, size)
|
|
{
|
|
return _native.getROMBlock(address, size)
|
|
},
|
|
getstring: function(address, maxLen)
|
|
{
|
|
return _native.getROMString(address, maxLen)
|
|
},
|
|
}
|
|
|
|
const mem = {
|
|
u8: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 8, false)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 8, val)
|
|
}
|
|
}),
|
|
u16: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 16, false)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 16, val)
|
|
}
|
|
}),
|
|
u32: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 32, false)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 32, val)
|
|
}
|
|
}),
|
|
s8: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 8, true)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 8, val)
|
|
}
|
|
}),
|
|
s16: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 16, true)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 16, val)
|
|
}
|
|
}),
|
|
s32: new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMInt(prop, 32, true)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMInt(prop, 32, val)
|
|
}
|
|
}),
|
|
'float': new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMFloat(prop)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMFloat(prop, val)
|
|
}
|
|
}),
|
|
'double': new Proxy({},
|
|
{
|
|
get: function(obj, prop)
|
|
{
|
|
return _native.getRDRAMFloat(prop, true)
|
|
},
|
|
set: function(obj, prop, val)
|
|
{
|
|
_native.setRDRAMFloat(prop, val, true)
|
|
}
|
|
}),
|
|
getblock: function(address, size)
|
|
{
|
|
return _native.getRDRAMBlock(address, size)
|
|
},
|
|
getstring: function(address, maxLen)
|
|
{
|
|
return _native.getRDRAMString(address, maxLen)
|
|
},
|
|
bindvar: function(obj, baseAddr, name, type)
|
|
{
|
|
Object.defineProperty(obj, name,
|
|
{
|
|
get: function()
|
|
{
|
|
return mem[type][baseAddr]
|
|
},
|
|
set: function(val)
|
|
{
|
|
mem[type][baseAddr] = val
|
|
}
|
|
})
|
|
return obj
|
|
},
|
|
bindvars: function(obj, list)
|
|
{
|
|
for(var i = 0; i < list.length; i++)
|
|
{
|
|
mem.bindvar(obj, list[i][0], list[i][1], list[i][2])
|
|
}
|
|
return obj
|
|
},
|
|
bindstruct: function(obj, baseAddr, props)
|
|
{
|
|
for (var name in props)
|
|
{
|
|
var type = props[name]
|
|
var size = _typeSizes[type]
|
|
mem.bindvar(obj, baseAddr, name, type)
|
|
baseAddr += size
|
|
}
|
|
return obj
|
|
},
|
|
typedef: function(props, proto)
|
|
{
|
|
var size = 0
|
|
for (var name in props)
|
|
{
|
|
size += _typeSizes[props[name]]
|
|
}
|
|
var StructClass = function(baseAddr)
|
|
{
|
|
mem.bindstruct(this, baseAddr, props)
|
|
}
|
|
StructClass.sizeof = function()
|
|
{
|
|
return size
|
|
}
|
|
/*if(proto)
|
|
{
|
|
StructClass.prototype = proto
|
|
}*/
|
|
return StructClass
|
|
}
|
|
}
|
|
|
|
function alert(text, caption){
|
|
caption = caption || ""
|
|
_native.msgBox(text, caption)
|
|
}
|
|
|
|
function Socket(fd)
|
|
{
|
|
var connected = false;
|
|
|
|
var _fd;
|
|
|
|
if(fd)
|
|
{
|
|
// assume this was constructed from Server
|
|
_fd = fd;
|
|
connected = true;
|
|
} else {
|
|
_fd = _native.sockCreate();
|
|
}
|
|
|
|
var _ondata = function(data){}
|
|
var _onclose = function(){}
|
|
var _onconnect = function(){}
|
|
|
|
var _onconnect_base = function(){
|
|
connected = 1;
|
|
if (_onconnect !== undefined)
|
|
{
|
|
_onconnect();
|
|
}
|
|
}
|
|
|
|
var _bufferSize = 2048
|
|
|
|
this.write = function(data, callback)
|
|
{
|
|
_native.write(_fd, data, _strongBackingReferences.singleUseCallback(callback))
|
|
}
|
|
|
|
this.close = function()
|
|
{
|
|
_native.close(_fd)
|
|
}
|
|
|
|
this.connect = function(settings, callback)
|
|
{
|
|
if(!connected)
|
|
{
|
|
_onconnect = callback;
|
|
_native.sockConnect(_fd, settings.host || '127.0.0.1', settings.port || 80, _strongBackingReferences.singleUseCallback(_onconnect_base));
|
|
}
|
|
}
|
|
|
|
var _read = function(data)
|
|
{
|
|
if(data == null)
|
|
{
|
|
connected = false;
|
|
_onclose();
|
|
return;
|
|
}
|
|
_native.read(_fd, _bufferSize, _strongBackingReferences.singleUseCallback(_read));
|
|
_ondata(data);
|
|
}
|
|
|
|
this.on = function(eventType, callback)
|
|
{
|
|
switch(eventType)
|
|
{
|
|
case 'data':
|
|
_ondata = callback
|
|
_native.read(_fd, _bufferSize, _strongBackingReferences.singleUseCallback(_read));
|
|
break;
|
|
case 'close':
|
|
// note: does nothing if ondata not set
|
|
_onclose = callback
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function Server(settings)
|
|
{
|
|
var _this = this;
|
|
var _fd = _native.sockCreate()
|
|
var _listening = false;
|
|
var _queued_accept = false;
|
|
|
|
var _onconnection = function(socket){}
|
|
|
|
this.listen = function (port) {
|
|
if (_native.sockListen(_fd, port || 80)) {
|
|
_listening = true;
|
|
} else {
|
|
throw new Error("failed to listen");
|
|
}
|
|
|
|
if (_queued_accept) {
|
|
_native.sockAccept(_fd, _strongBackingReferences.singleUseCallback(_acceptClient));
|
|
}
|
|
}
|
|
|
|
if(settings.port)
|
|
{
|
|
this.listen(settings.port || 80);
|
|
}
|
|
|
|
// Intermediate callback
|
|
// convert clientFd to Socket and accept next client
|
|
var _acceptClient = function(clientFd)
|
|
{
|
|
_onconnection(new Socket(clientFd))
|
|
_native.sockAccept(_fd, _strongBackingReferences.singleUseCallback(_acceptClient))
|
|
}
|
|
|
|
this.on = function(eventType, callback)
|
|
{
|
|
switch(eventType)
|
|
{
|
|
case 'connection':
|
|
_onconnection = callback;
|
|
if (_listening) {
|
|
_native.sockAccept(_fd, _strongBackingReferences.singleUseCallback(_acceptClient));
|
|
} else {
|
|
_queued_accept = true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|