mirror of
https://github.com/DPS2004/gba.git
synced 2025-01-23 12:57:38 -03:00
3687 lines
136 KiB
JavaScript
3687 lines
136 KiB
JavaScript
"use strict";
|
|
/*
|
|
Copyright (C) 2012-2015 Grant Galitz
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
function ARMInstructionSet(CPUCore) {
|
|
this.CPUCore = CPUCore;
|
|
this.initialize();
|
|
}
|
|
ARMInstructionSet.prototype.initialize = function () {
|
|
this.wait = this.CPUCore.wait;
|
|
this.registers = this.CPUCore.registers;
|
|
this.registersUSR = this.CPUCore.registersUSR;
|
|
this.branchFlags = this.CPUCore.branchFlags;
|
|
this.fetch = 0;
|
|
this.decode = 0;
|
|
this.execute = 0;
|
|
this.memory = this.CPUCore.memory;
|
|
}
|
|
ARMInstructionSet.prototype.executeIteration = function () {
|
|
//Push the new fetch access:
|
|
this.fetch = this.memory.memoryReadCPU32(this.readPC() | 0) | 0;
|
|
//Execute Conditional Instruction:
|
|
this.executeConditionalCode();
|
|
//Update the pipelining state:
|
|
this.execute = this.decode | 0;
|
|
this.decode = this.fetch | 0;
|
|
}
|
|
ARMInstructionSet.prototype.executeConditionalCode = function () {
|
|
//LSB of condition code is used to reverse the test logic:
|
|
if (((this.execute << 3) ^ this.branchFlags.checkConditionalCode(this.execute | 0)) >= 0) {
|
|
//Passed the condition code test, so execute:
|
|
this.executeDecoded();
|
|
}
|
|
else {
|
|
//Increment the program counter if we failed the test:
|
|
this.incrementProgramCounter();
|
|
}
|
|
}
|
|
|
|
ARMInstructionSet.prototype.executeBubble = function () {
|
|
//Push the new fetch access:
|
|
this.fetch = this.memory.memoryReadCPU32(this.readPC() | 0) | 0;
|
|
//Update the Program Counter:
|
|
this.incrementProgramCounter();
|
|
//Update the pipelining state:
|
|
this.execute = this.decode | 0;
|
|
this.decode = this.fetch | 0;
|
|
}
|
|
ARMInstructionSet.prototype.incrementProgramCounter = function () {
|
|
//Increment The Program Counter:
|
|
this.registers[15] = ((this.registers[15] | 0) + 4) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.getLR = function () {
|
|
return ((this.readPC() | 0) - 4) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.getIRQLR = function () {
|
|
return this.getLR() | 0;
|
|
}
|
|
ARMInstructionSet.prototype.getCurrentFetchValue = function () {
|
|
return this.fetch | 0;
|
|
}
|
|
ARMInstructionSet.prototype.getSWICode = function () {
|
|
return (this.execute >> 16) & 0xFF;
|
|
}
|
|
if (typeof Math.imul == "function") {
|
|
//Math.imul found, insert the optimized path in:
|
|
ARMInstructionSet.prototype.getPopCount = function () {
|
|
var temp = this.execute & 0xFFFF;
|
|
temp = ((temp | 0) - ((temp >> 1) & 0x5555)) | 0;
|
|
temp = ((temp & 0x3333) + ((temp >> 2) & 0x3333)) | 0;
|
|
temp = (((temp | 0) + (temp >> 4)) & 0xF0F) | 0;
|
|
temp = Math.imul(temp | 0, 0x1010101) >> 24;
|
|
return temp | 0;
|
|
}
|
|
}
|
|
else {
|
|
//Math.imul not found, use the compatibility method:
|
|
ARMInstructionSet.prototype.getPopCount = function () {
|
|
var temp = this.execute & 0xFFFF;
|
|
temp = ((temp | 0) - ((temp >> 1) & 0x5555)) | 0;
|
|
temp = ((temp & 0x3333) + ((temp >> 2) & 0x3333)) | 0;
|
|
temp = (((temp | 0) + (temp >> 4)) & 0xF0F) | 0;
|
|
temp = (temp * 0x1010101) >> 24;
|
|
return temp | 0;
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.getNegativeOffsetStartAddress = function (currentAddress) {
|
|
//Used for LDMD/STMD:
|
|
currentAddress = currentAddress | 0;
|
|
var offset = this.getPopCount() << 2;
|
|
currentAddress = ((currentAddress | 0) - (offset | 0)) | 0;
|
|
return currentAddress | 0;
|
|
}
|
|
ARMInstructionSet.prototype.getPositiveOffsetStartAddress = function (currentAddress) {
|
|
//Used for LDMD/STMD:
|
|
currentAddress = currentAddress | 0;
|
|
var offset = this.getPopCount() << 2;
|
|
currentAddress = ((currentAddress | 0) + (offset | 0)) | 0;
|
|
return currentAddress | 0;
|
|
}
|
|
ARMInstructionSet.prototype.writeRegister = function (address, data) {
|
|
//Unguarded non-pc register write:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
this.registers[address & 0xF] = data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.writeUserRegister = function (address, data) {
|
|
//Unguarded non-pc user mode register write:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
this.registersUSR[address & 0x7] = data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.guardRegisterWrite = function (address, data) {
|
|
//Guarded register write:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
if ((address | 0) < 0xF) {
|
|
//Non-PC Write:
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//We performed a branch:
|
|
this.CPUCore.branch(data & -4);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.multiplyGuard12OffsetRegisterWrite = function (data) {
|
|
//Writes to R15 ignored in the multiply instruction!
|
|
data = data | 0;
|
|
var address = (this.execute >> 0xC) & 0xF;
|
|
if ((address | 0) != 0xF) {
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.multiplyGuard16OffsetRegisterWrite = function (data) {
|
|
//Writes to R15 ignored in the multiply instruction!
|
|
data = data | 0;
|
|
var address = (this.execute >> 0x10) & 0xF;
|
|
this.incrementProgramCounter();
|
|
if ((address | 0) != 0xF) {
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.performMUL32 = function () {
|
|
var result = 0;
|
|
if (((this.execute >> 16) & 0xF) != (this.execute & 0xF)) {
|
|
/*
|
|
http://www.chiark.greenend.org.uk/~theom/riscos/docs/ultimate/a252armc.txt
|
|
|
|
Due to the way that Booth's algorithm has been implemented, certain
|
|
combinations of operand registers should be avoided. (The assembler will
|
|
issue a warning if these restrictions are overlooked.)
|
|
The destination register (Rd) should not be the same as the Rm operand
|
|
register, as Rd is used to hold intermediate values and Rm is used
|
|
repeatedly during the multiply. A MUL will give a zero result if Rm=Rd, and
|
|
a MLA will give a meaningless result.
|
|
*/
|
|
result = this.CPUCore.performMUL32(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0) | 0;
|
|
}
|
|
return result | 0;
|
|
}
|
|
ARMInstructionSet.prototype.performMUL32MLA = function () {
|
|
var result = 0;
|
|
if (((this.execute >> 16) & 0xF) != (this.execute & 0xF)) {
|
|
/*
|
|
http://www.chiark.greenend.org.uk/~theom/riscos/docs/ultimate/a252armc.txt
|
|
|
|
Due to the way that Booth's algorithm has been implemented, certain
|
|
combinations of operand registers should be avoided. (The assembler will
|
|
issue a warning if these restrictions are overlooked.)
|
|
The destination register (Rd) should not be the same as the Rm operand
|
|
register, as Rd is used to hold intermediate values and Rm is used
|
|
repeatedly during the multiply. A MUL will give a zero result if Rm=Rd, and
|
|
a MLA will give a meaningless result.
|
|
*/
|
|
result = this.CPUCore.performMUL32MLA(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0) | 0;
|
|
}
|
|
return result | 0;
|
|
}
|
|
ARMInstructionSet.prototype.guard12OffsetRegisterWrite = function (data) {
|
|
data = data | 0;
|
|
this.incrementProgramCounter();
|
|
this.guard12OffsetRegisterWrite2(data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guard12OffsetRegisterWrite2 = function (data) {
|
|
data = data | 0;
|
|
this.guardRegisterWrite((this.execute >> 0xC) & 0xF, data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guard16OffsetRegisterWrite = function (data) {
|
|
data = data | 0;
|
|
this.guardRegisterWrite((this.execute >> 0x10) & 0xF, data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guard16OffsetUserRegisterWrite = function (data) {
|
|
data = data | 0;
|
|
var address = (this.execute >> 0x10) & 0xF;
|
|
if ((address | 0) < 0xF) {
|
|
//Non-PC Write:
|
|
this.guardUserRegisterWrite(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//We performed a branch:
|
|
this.CPUCore.branch(data & -4);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.guardProgramCounterRegisterWriteCPSR = function (data) {
|
|
data = data | 0;
|
|
//Restore SPSR to CPSR:
|
|
data = data & (-4 >> (this.CPUCore.SPSRtoCPSR() >> 5));
|
|
//We performed a branch:
|
|
this.CPUCore.branch(data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guardRegisterWriteCPSR = function (address, data) {
|
|
//Guard for possible pc write with cpsr update:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
if ((address | 0) < 0xF) {
|
|
//Non-PC Write:
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//Restore SPSR to CPSR:
|
|
this.guardProgramCounterRegisterWriteCPSR(data | 0);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.guard12OffsetRegisterWriteCPSR = function (data) {
|
|
data = data | 0;
|
|
this.incrementProgramCounter();
|
|
this.guard12OffsetRegisterWriteCPSR2(data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guard12OffsetRegisterWriteCPSR2 = function (data) {
|
|
data = data | 0;
|
|
this.guardRegisterWriteCPSR((this.execute >> 0xC) & 0xF, data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guard16OffsetRegisterWriteCPSR = function (data) {
|
|
data = data | 0;
|
|
this.guardRegisterWriteCPSR((this.execute >> 0x10) & 0xF, data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guardUserRegisterWrite = function (address, data) {
|
|
//Guard only on user access, not PC!:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
switch (this.CPUCore.modeFlags & 0x1f) {
|
|
case 0x10:
|
|
case 0x1F:
|
|
this.writeRegister(address | 0, data | 0);
|
|
break;
|
|
case 0x11:
|
|
if ((address | 0) < 8) {
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//User-Mode Register Write Inside Non-User-Mode:
|
|
this.writeUserRegister(address | 0, data | 0);
|
|
}
|
|
break;
|
|
default:
|
|
if ((address | 0) < 13) {
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//User-Mode Register Write Inside Non-User-Mode:
|
|
this.writeUserRegister(address | 0, data | 0);
|
|
}
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.guardRegisterWriteLDM = function (address, data) {
|
|
//Proxy guarded register write for LDM:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
this.guardRegisterWrite(address | 0, data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.guardUserRegisterWriteLDM = function (address, data) {
|
|
//Proxy guarded user mode register write with PC guard for LDM:
|
|
address = address | 0;
|
|
data = data | 0;
|
|
if ((address | 0) < 0xF) {
|
|
if ((this.execute & 0x8000) != 0) {
|
|
//PC is in the list, don't do user-mode:
|
|
this.writeRegister(address | 0, data | 0);
|
|
}
|
|
else {
|
|
//PC isn't in the list, do user-mode:
|
|
this.guardUserRegisterWrite(address | 0, data | 0);
|
|
}
|
|
}
|
|
else {
|
|
this.guardProgramCounterRegisterWriteCPSR(data | 0);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.readPC = function () {
|
|
//PC register read:
|
|
return this.registers[0xF] | 0;
|
|
}
|
|
ARMInstructionSet.prototype.readRegister = function (address) {
|
|
//Unguarded register read:
|
|
address = address | 0;
|
|
return this.registers[address & 0xF] | 0;
|
|
}
|
|
ARMInstructionSet.prototype.readUserRegister = function (address) {
|
|
//Unguarded user mode register read:
|
|
address = address | 0;
|
|
var data = 0;
|
|
if ((address | 0) < 0xF) {
|
|
data = this.registersUSR[address & 0x7] | 0;
|
|
}
|
|
else {
|
|
//Get Special Case PC Read:
|
|
data = this.readPC() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.read0OffsetRegister = function () {
|
|
//Unguarded register read at position 0:
|
|
return this.readRegister(this.execute | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.read8OffsetRegister = function () {
|
|
//Unguarded register read at position 0x8:
|
|
return this.readRegister(this.execute >> 0x8) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.read12OffsetRegister = function () {
|
|
//Unguarded register read at position 0xC:
|
|
return this.readRegister(this.execute >> 0xC) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.read16OffsetRegister = function () {
|
|
//Unguarded register read at position 0x10:
|
|
return this.readRegister(this.execute >> 0x10) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.read16OffsetUserRegister = function () {
|
|
//Guarded register read at position 0x10:
|
|
return this.guardUserRegisterRead(this.execute >> 0x10) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.guard12OffsetRegisterRead = function () {
|
|
this.incrementProgramCounter();
|
|
return this.readRegister((this.execute >> 12) & 0xF) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.guardUserRegisterRead = function (address) {
|
|
//Guard only on user access, not PC!:
|
|
address = address | 0;
|
|
var data = 0;
|
|
switch (this.CPUCore.modeFlags & 0x1F) {
|
|
case 0x10:
|
|
case 0x1F:
|
|
data = this.readRegister(address | 0) | 0;
|
|
break;
|
|
case 0x11:
|
|
if ((address | 0) < 8) {
|
|
data = this.readRegister(address | 0) | 0;
|
|
}
|
|
else {
|
|
//User-Mode Register Read Inside Non-User-Mode:
|
|
data = this.readUserRegister(address | 0) | 0;
|
|
}
|
|
break;
|
|
default:
|
|
if ((address | 0) < 13) {
|
|
data = this.readRegister(address | 0) | 0;
|
|
}
|
|
else {
|
|
//User-Mode Register Read Inside Non-User-Mode:
|
|
data = this.readUserRegister(address | 0) | 0;
|
|
}
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.BX = function () {
|
|
//Branch & eXchange:
|
|
var address = this.read0OffsetRegister() | 0;
|
|
if ((address & 0x1) == 0) {
|
|
//Stay in ARM mode:
|
|
this.CPUCore.branch(address & -4);
|
|
}
|
|
else {
|
|
//Enter THUMB mode:
|
|
this.CPUCore.enterTHUMB();
|
|
this.CPUCore.branch(address & -2);
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.B = function () {
|
|
//Branch:
|
|
this.CPUCore.branch(((this.readPC() | 0) + ((this.execute << 8) >> 6)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.BL = function () {
|
|
//Branch with Link:
|
|
this.writeRegister(0xE, this.getLR() | 0);
|
|
this.B();
|
|
}
|
|
ARMInstructionSet.prototype.AND = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform bitwise AND:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(operand1 & operand2);
|
|
}
|
|
ARMInstructionSet.prototype.AND2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform bitwise AND:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(operand1 & operand2);
|
|
}
|
|
ARMInstructionSet.prototype.ANDS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ANDS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.EOR = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform bitwise EOR:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(operand1 ^ operand2);
|
|
}
|
|
ARMInstructionSet.prototype.EOR2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform bitwise EOR:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(operand1 ^ operand2);
|
|
}
|
|
ARMInstructionSet.prototype.EORS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform bitwise EOR:
|
|
var result = operand1 ^ operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.EORS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform bitwise EOR:
|
|
var result = operand1 ^ operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SUB = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Subtraction:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(((operand1 | 0) - (operand2 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SUB2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Subtraction:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(((operand1 | 0) - (operand2 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SUBS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setSUBFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SUBS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setSUBFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSB = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Subtraction:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(((operand2 | 0) - (operand1 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSB2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Subtraction:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(((operand2 | 0) - (operand1 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSBS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setSUBFlags(operand2 | 0, operand1 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSBS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setSUBFlags(operand2 | 0, operand1 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADD = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Addition:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(((operand1 | 0) + (operand2 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADD2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Addition:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(((operand1 | 0) + (operand2 | 0)) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADDS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setADDFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADDS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setADDFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADC = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Addition w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand1 | 0) + (operand2 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) + (this.branchFlags.getCarry() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADC2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Addition w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand1 | 0) + (operand2 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) + (this.branchFlags.getCarry() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite2(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADCS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setADCFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ADCS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setADCFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SBC = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Subtraction w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand1 | 0) - (operand2 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) - (this.branchFlags.getCarryReverse() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SBC2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Subtraction w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand1 | 0) - (operand2 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) - (this.branchFlags.getCarryReverse() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite2(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SBCS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setSBCFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SBCS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setSBCFlags(operand1 | 0, operand2 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSC = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform Reverse Subtraction w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand2 | 0) - (operand1 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) - (this.branchFlags.getCarryReverse() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSC2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform Reverse Subtraction w/ Carry:
|
|
//Update destination register:
|
|
operand1 = ((operand2 | 0) - (operand1 | 0)) | 0;
|
|
operand1 = ((operand1 | 0) - (this.branchFlags.getCarryReverse() >>> 31)) | 0;
|
|
this.guard12OffsetRegisterWrite2(operand1 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSCS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR(this.branchFlags.setSBCFlags(operand2 | 0, operand1 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.RSCS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWriteCPSR2(this.branchFlags.setSBCFlags(operand2 | 0, operand1 | 0) | 0);
|
|
}
|
|
ARMInstructionSet.prototype.TSTS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
}
|
|
ARMInstructionSet.prototype.TSTS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.TEQS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform bitwise EOR:
|
|
var result = operand1 ^ operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
}
|
|
ARMInstructionSet.prototype.TEQS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform bitwise EOR:
|
|
var result = operand1 ^ operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.CMPS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
this.branchFlags.setCMPFlags(operand1 | 0, operand2 | 0);
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
}
|
|
ARMInstructionSet.prototype.CMPS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
this.branchFlags.setCMPFlags(operand1 | 0, operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.CMNS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1();
|
|
this.branchFlags.setCMNFlags(operand1 | 0, operand2 | 0);
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
}
|
|
ARMInstructionSet.prototype.CMNS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3();
|
|
this.branchFlags.setCMNFlags(operand1 | 0, operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ORR = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing1() | 0;
|
|
//Perform bitwise OR:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(operand1 | operand2);
|
|
}
|
|
ARMInstructionSet.prototype.ORR2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing3() | 0;
|
|
//Perform bitwise OR:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(operand1 | operand2);
|
|
}
|
|
ARMInstructionSet.prototype.ORRS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform bitwise OR:
|
|
var result = operand1 | operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.ORRS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform bitwise OR:
|
|
var result = operand1 | operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MOV = function () {
|
|
//Perform move:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(this.operand2OP_DataProcessing1() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MOV2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
//Perform move:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(this.operand2OP_DataProcessing3() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MOVS = function () {
|
|
var operand2 = this.operand2OP_DataProcessing2() | 0;
|
|
//Perform move:
|
|
this.branchFlags.setNZInt(operand2 | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MOVS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand2 = this.operand2OP_DataProcessing4() | 0;
|
|
//Perform move:
|
|
this.branchFlags.setNZInt(operand2 | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.BIC = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
//NOT operand 2:
|
|
var operand2 = ~this.operand2OP_DataProcessing1();
|
|
//Perform bitwise AND:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(operand1 & operand2);
|
|
}
|
|
ARMInstructionSet.prototype.BIC2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
//NOT operand 2:
|
|
var operand2 = ~this.operand2OP_DataProcessing3();
|
|
//Perform bitwise AND:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(operand1 & operand2);
|
|
}
|
|
ARMInstructionSet.prototype.BICS = function () {
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
//NOT operand 2:
|
|
var operand2 = ~this.operand2OP_DataProcessing2();
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.BICS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand1 = this.read16OffsetRegister() | 0;
|
|
//NOT operand 2:
|
|
var operand2 = ~this.operand2OP_DataProcessing4();
|
|
//Perform bitwise AND:
|
|
var result = operand1 & operand2;
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MVN = function () {
|
|
//Perform move negative:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite(~this.operand2OP_DataProcessing1());
|
|
}
|
|
ARMInstructionSet.prototype.MVN2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
//Perform move negative:
|
|
//Update destination register:
|
|
this.guard12OffsetRegisterWrite2(~this.operand2OP_DataProcessing3());
|
|
}
|
|
ARMInstructionSet.prototype.MVNS = function () {
|
|
var operand2 = ~this.operand2OP_DataProcessing2();
|
|
//Perform move negative:
|
|
this.branchFlags.setNZInt(operand2 | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR(operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MVNS2 = function () {
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
var operand2 = ~this.operand2OP_DataProcessing4();
|
|
//Perform move negative:
|
|
this.branchFlags.setNZInt(operand2 | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.guard12OffsetRegisterWriteCPSR2(operand2 | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MRS = function () {
|
|
//Transfer PSR to Register:
|
|
var psr = 0;
|
|
if ((this.execute & 0x400000) == 0) {
|
|
//CPSR->Register
|
|
psr = this.rc() | 0;
|
|
}
|
|
else {
|
|
//SPSR->Register
|
|
psr = this.rs() | 0;
|
|
}
|
|
this.guard12OffsetRegisterWrite(psr | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MSR = function () {
|
|
switch (this.execute & 0x2400000) {
|
|
case 0:
|
|
//Reg->CPSR
|
|
this.MSR1();
|
|
break;
|
|
case 0x400000:
|
|
//Reg->SPSR
|
|
this.MSR2();
|
|
break;
|
|
case 0x2000000:
|
|
//Immediate->CPSR
|
|
this.MSR3();
|
|
break;
|
|
default:
|
|
//Immediate->SPSR
|
|
this.MSR4();
|
|
}
|
|
//Increment PC:
|
|
this.incrementProgramCounter();
|
|
}
|
|
ARMInstructionSet.prototype.MSR1 = function () {
|
|
var newcpsr = this.read0OffsetRegister() | 0;
|
|
this.branchFlags.setNZCV(newcpsr | 0);
|
|
if ((this.execute & 0x10000) != 0 && (this.CPUCore.modeFlags & 0x1F) != 0x10) {
|
|
this.CPUCore.switchRegisterBank(newcpsr & 0x1F);
|
|
this.CPUCore.modeFlags = newcpsr & 0xDF;
|
|
this.CPUCore.assertIRQ();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.MSR2 = function () {
|
|
var operand = this.read0OffsetRegister() | 0;
|
|
var bank = 1;
|
|
switch (this.CPUCore.modeFlags & 0x1F) {
|
|
case 0x12: //IRQ
|
|
break;
|
|
case 0x13: //Supervisor
|
|
bank = 2;
|
|
break;
|
|
case 0x11: //FIQ
|
|
bank = 0;
|
|
break;
|
|
case 0x17: //Abort
|
|
bank = 3;
|
|
break;
|
|
case 0x1B: //Undefined
|
|
bank = 4;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
var spsr = (operand >> 20) & 0xF00;
|
|
if ((this.execute & 0x10000) != 0) {
|
|
spsr = spsr | (operand & 0xFF);
|
|
}
|
|
else {
|
|
spsr = spsr | (this.CPUCore.SPSR[bank | 0] & 0xFF);
|
|
}
|
|
this.CPUCore.SPSR[bank | 0] = spsr | 0;
|
|
}
|
|
ARMInstructionSet.prototype.MSR3 = function () {
|
|
var operand = this.imm() | 0;
|
|
this.branchFlags.setNZCV(operand | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MSR4 = function () {
|
|
var operand = this.imm() >> 20;
|
|
var bank = 1;
|
|
switch (this.CPUCore.modeFlags & 0x1F) {
|
|
case 0x12: //IRQ
|
|
break;
|
|
case 0x13: //Supervisor
|
|
bank = 2;
|
|
break;
|
|
case 0x11: //FIQ
|
|
bank = 0;
|
|
break;
|
|
case 0x17: //Abort
|
|
bank = 3;
|
|
break;
|
|
case 0x1B: //Undefined
|
|
bank = 4;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
var spsr = this.CPUCore.SPSR[bank | 0] & 0xFF;
|
|
this.CPUCore.SPSR[bank | 0] = spsr | (operand & 0xF00);
|
|
}
|
|
ARMInstructionSet.prototype.MUL = function () {
|
|
//Perform multiplication:
|
|
var result = this.performMUL32() | 0;
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MULS = function () {
|
|
//Perform multiplication:
|
|
var result = this.performMUL32() | 0;
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MLA = function () {
|
|
//Perform multiplication:
|
|
var result = this.performMUL32MLA() | 0;
|
|
//Perform addition:
|
|
result = ((result | 0) + (this.read12OffsetRegister() | 0)) | 0;
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.MLAS = function () {
|
|
//Perform multiplication:
|
|
var result = this.performMUL32MLA() | 0;
|
|
//Perform addition:
|
|
result = ((result | 0) + (this.read12OffsetRegister() | 0)) | 0;
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNZInt(result | 0);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(result | 0);
|
|
}
|
|
ARMInstructionSet.prototype.UMULL = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performUMUL64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0);
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.UMULLS = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performUMUL64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0);
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNegative(this.CPUCore.mul64ResultHigh | 0);
|
|
this.branchFlags.setZero(this.CPUCore.mul64ResultHigh | this.CPUCore.mul64ResultLow);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.UMLAL = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performUMLA64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0, this.read16OffsetRegister() | 0, this.read12OffsetRegister() | 0);
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.UMLALS = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performUMLA64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0, this.read16OffsetRegister() | 0, this.read12OffsetRegister() | 0);
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNegative(this.CPUCore.mul64ResultHigh | 0);
|
|
this.branchFlags.setZero(this.CPUCore.mul64ResultHigh | this.CPUCore.mul64ResultLow);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SMULL = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performMUL64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0);
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SMULLS = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performMUL64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0);
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNegative(this.CPUCore.mul64ResultHigh | 0);
|
|
this.branchFlags.setZero(this.CPUCore.mul64ResultHigh | this.CPUCore.mul64ResultLow);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SMLAL = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performMLA64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0, this.read16OffsetRegister() | 0, this.read12OffsetRegister() | 0);
|
|
//Update destination register:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SMLALS = function () {
|
|
//Perform multiplication:
|
|
this.CPUCore.performMLA64(this.read0OffsetRegister() | 0, this.read8OffsetRegister() | 0, this.read16OffsetRegister() | 0, this.read12OffsetRegister() | 0);
|
|
this.branchFlags.setCarryFalse();
|
|
this.branchFlags.setNegative(this.CPUCore.mul64ResultHigh | 0);
|
|
this.branchFlags.setZero(this.CPUCore.mul64ResultHigh | this.CPUCore.mul64ResultLow);
|
|
//Update destination register and guard CPSR for PC:
|
|
this.multiplyGuard16OffsetRegisterWrite(this.CPUCore.mul64ResultHigh | 0);
|
|
this.multiplyGuard12OffsetRegisterWrite(this.CPUCore.mul64ResultLow | 0);
|
|
}
|
|
ARMInstructionSet.prototype.STRH = function () {
|
|
//Perform halfword store calculations:
|
|
var address = this.operand2OP_LoadStore1() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write16(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRH = function () {
|
|
//Perform halfword load calculations:
|
|
var address = this.operand2OP_LoadStore1() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read16(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDRSH = function () {
|
|
//Perform signed halfword load calculations:
|
|
var address = this.operand2OP_LoadStore1() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite((this.CPUCore.read16(address | 0) << 16) >> 16);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDRSB = function () {
|
|
//Perform signed byte load calculations:
|
|
var address = this.operand2OP_LoadStore1() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite((this.CPUCore.read8(address | 0) << 24) >> 24);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRH2 = function () {
|
|
//Perform halfword store calculations:
|
|
var address = this.operand2OP_LoadStore2() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write16(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRH2 = function () {
|
|
//Perform halfword load calculations:
|
|
var address = this.operand2OP_LoadStore2() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read16(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDRSH2 = function () {
|
|
//Perform signed halfword load calculations:
|
|
var address = this.operand2OP_LoadStore2() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite((this.CPUCore.read16(address | 0) << 16) >> 16);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDRSB2 = function () {
|
|
//Perform signed byte load calculations:
|
|
var address = this.operand2OP_LoadStore2() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite((this.CPUCore.read8(address | 0) << 24) >> 24);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STR = function () {
|
|
//Perform word store calculations:
|
|
var address = this.operand2OP_LoadStore3Normal() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDR = function () {
|
|
//Perform word load calculations:
|
|
var address = this.operand2OP_LoadStore3Normal() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRB = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore3Normal() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRB = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore3Normal() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STR4 = function () {
|
|
//Perform word store calculations:
|
|
var address = this.operand2OP_LoadStore4() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDR4 = function () {
|
|
//Perform word load calculations:
|
|
var address = this.operand2OP_LoadStore4() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRB4 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore4() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRB4 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore4() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRT = function () {
|
|
//Perform word store calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore3User() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRT = function () {
|
|
//Perform word load calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore3User() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRBT = function () {
|
|
//Perform byte store calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore3User() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRBT = function () {
|
|
//Perform byte load calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore3User() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STR2 = function () {
|
|
//Perform word store calculations:
|
|
var address = this.operand2OP_LoadStore5Normal() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDR2 = function () {
|
|
//Perform word load calculations:
|
|
var address = this.operand2OP_LoadStore5Normal() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRB2 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore5Normal() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRB2 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore5Normal() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRT2 = function () {
|
|
//Perform word store calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore5User() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRT2 = function () {
|
|
//Perform word load calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore5User() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRBT2 = function () {
|
|
//Perform byte store calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore5User() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRBT2 = function () {
|
|
//Perform byte load calculations (forced user-mode):
|
|
var address = this.operand2OP_LoadStore5User() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STR3 = function () {
|
|
//Perform word store calculations:
|
|
var address = this.operand2OP_LoadStore6() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write32(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDR3 = function () {
|
|
//Perform word load calculations:
|
|
var address = this.operand2OP_LoadStore6() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read32(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STRB3 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore6() | 0;
|
|
//Write to memory location:
|
|
this.CPUCore.write8(address | 0, this.guard12OffsetRegisterRead() | 0);
|
|
}
|
|
ARMInstructionSet.prototype.LDRB3 = function () {
|
|
//Perform byte store calculations:
|
|
var address = this.operand2OP_LoadStore6() | 0;
|
|
//Read from memory location:
|
|
this.guard12OffsetRegisterWrite(this.CPUCore.read8(address | 0) | 0);
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.STMIA = function () {
|
|
//Only initialize the STMIA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIAW = function () {
|
|
//Only initialize the STMIA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
var finalAddress = this.getPositiveOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDA = function () {
|
|
//Only initialize the STMDA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDAW = function () {
|
|
//Only initialize the STMDA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var finalAddress = currentAddress | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIB = function () {
|
|
//Only initialize the STMIB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIBW = function () {
|
|
//Only initialize the STMIB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
var finalAddress = this.getPositiveOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDB = function () {
|
|
//Only initialize the STMDB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDBW = function () {
|
|
//Only initialize the STMDB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var finalAddress = currentAddress | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.readRegister(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIAG = function () {
|
|
//Only initialize the STMIA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIAWG = function () {
|
|
//Only initialize the STMIA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
var finalAddress = this.getPositiveOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDAG = function () {
|
|
//Only initialize the STMDA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDAWG = function () {
|
|
//Only initialize the STMDA sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var finalAddress = currentAddress | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIBG = function () {
|
|
//Only initialize the STMIB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMIBWG = function () {
|
|
//Only initialize the STMIB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
var finalAddress = this.getPositiveOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDBG = function () {
|
|
//Only initialize the STMDB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.STMDBWG = function () {
|
|
//Only initialize the STMDB sequence if the register list is non-empty:
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get offset start address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var finalAddress = currentAddress | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Push register(s) into memory:
|
|
var count = 0;
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Push a register into memory:
|
|
this.memory.memoryWrite32(currentAddress | 0, this.guardUserRegisterRead(rListPosition | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
//Compute writeback immediately after the first register load:
|
|
if ((count | 0) == 0) {
|
|
count = 1;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(finalAddress | 0);
|
|
}
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.LDMIA = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIAW = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 0x40) | 0;
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDA = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDAW = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var writebackAddress = currentAddress | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(writebackAddress | 0);
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) - 0x40) | 0;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIB = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIBW = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) + 0x40) | 0;
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDB = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) - 4) | 0;
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDBW = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var writebackAddress = currentAddress | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(writebackAddress | 0);
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) - 0x40) | 0;
|
|
this.guardRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIAG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIAWG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 0x40) | 0;
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDAG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardUserRegisterWriteLDM(0xF, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDAWG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var writebackAddress = currentAddress | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(writebackAddress | 0);
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) - 0x40) | 0;
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIBG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMIBWG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) + 0x40) | 0;
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDBG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; (rListPosition | 0) < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) - 4) | 0;
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LDMDBWG = function () {
|
|
//Get the base address:
|
|
var currentAddress = this.read16OffsetRegister() | 0;
|
|
//Updating the address bus away from PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
if ((this.execute & 0xFFFF) > 0) {
|
|
//Get the offset address:
|
|
currentAddress = this.getNegativeOffsetStartAddress(currentAddress | 0) | 0;
|
|
var writebackAddress = currentAddress | 0;
|
|
//Load register(s) from memory:
|
|
for (var rListPosition = 0; rListPosition < 0x10; rListPosition = ((rListPosition | 0) + 1) | 0) {
|
|
if ((this.execute & (1 << rListPosition)) != 0) {
|
|
//Load a register from memory:
|
|
this.guardUserRegisterWriteLDM(rListPosition | 0, this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
currentAddress = ((currentAddress | 0) + 4) | 0;
|
|
}
|
|
}
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(writebackAddress | 0);
|
|
}
|
|
else {
|
|
//Empty reglist loads PC:
|
|
currentAddress = ((currentAddress | 0) - 0x40) | 0;
|
|
this.guardProgramCounterRegisterWriteCPSR(this.memory.memoryRead32(currentAddress | 0) | 0);
|
|
//Store the updated base address back into register:
|
|
this.guard16OffsetRegisterWrite(currentAddress | 0);
|
|
}
|
|
//Updating the address bus back to PC fetch:
|
|
this.wait.NonSequentialBroadcast();
|
|
//Internal Cycle:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
}
|
|
ARMInstructionSet.prototype.LoadStoreMultiple = function () {
|
|
this.incrementProgramCounter();
|
|
switch ((this.execute >> 20) & 0x1F) {
|
|
case 0:
|
|
this.STMDA();
|
|
break;
|
|
case 0x1:
|
|
this.LDMDA();
|
|
break;
|
|
case 0x2:
|
|
this.STMDAW();
|
|
break;
|
|
case 0x3:
|
|
this.LDMDAW();
|
|
break;
|
|
case 0x4:
|
|
this.STMDAG();
|
|
break;
|
|
case 0x5:
|
|
this.LDMDAG();
|
|
break;
|
|
case 0x6:
|
|
this.STMDAWG();
|
|
break;
|
|
case 0x7:
|
|
this.LDMDAWG();
|
|
break;
|
|
case 0x8:
|
|
this.STMIA();
|
|
break;
|
|
case 0x9:
|
|
this.LDMIA();
|
|
break;
|
|
case 0xA:
|
|
this.STMIAW();
|
|
break;
|
|
case 0xB:
|
|
this.LDMIAW();
|
|
break;
|
|
case 0xC:
|
|
this.STMIAG();
|
|
break;
|
|
case 0xD:
|
|
this.LDMIAG();
|
|
break;
|
|
case 0xE:
|
|
this.STMIAWG();
|
|
break;
|
|
case 0xF:
|
|
this.LDMIAWG();
|
|
break;
|
|
case 0x10:
|
|
this.STMDB();
|
|
break;
|
|
case 0x11:
|
|
this.LDMDB();
|
|
break;
|
|
case 0x12:
|
|
this.STMDBW();
|
|
break;
|
|
case 0x13:
|
|
this.LDMDBW();
|
|
break;
|
|
case 0x14:
|
|
this.STMDBG();
|
|
break;
|
|
case 0x15:
|
|
this.LDMDBG();
|
|
break;
|
|
case 0x16:
|
|
this.STMDBWG();
|
|
break;
|
|
case 0x17:
|
|
this.LDMDBWG();
|
|
break;
|
|
case 0x18:
|
|
this.STMIB();
|
|
break;
|
|
case 0x19:
|
|
this.LDMIB();
|
|
break;
|
|
case 0x1A:
|
|
this.STMIBW();
|
|
break;
|
|
case 0x1B:
|
|
this.LDMIBW();
|
|
break;
|
|
case 0x1C:
|
|
this.STMIBG();
|
|
break;
|
|
case 0x1D:
|
|
this.LDMIBG();
|
|
break;
|
|
case 0x1E:
|
|
this.STMIBWG();
|
|
break;
|
|
default:
|
|
this.LDMIBWG();
|
|
}
|
|
}
|
|
ARMInstructionSet.prototype.SWP = function () {
|
|
var base = this.read16OffsetRegister() | 0;
|
|
var data = this.CPUCore.read32(base | 0) | 0;
|
|
//Clock a cycle for the processing delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
this.CPUCore.write32(base | 0, this.read0OffsetRegister() | 0);
|
|
this.guard12OffsetRegisterWrite(data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SWPB = function () {
|
|
var base = this.read16OffsetRegister() | 0;
|
|
var data = this.CPUCore.read8(base | 0) | 0;
|
|
//Clock a cycle for the processing delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
this.CPUCore.write8(base | 0, this.read0OffsetRegister() | 0);
|
|
this.guard12OffsetRegisterWrite(data | 0);
|
|
}
|
|
ARMInstructionSet.prototype.SWI = function () {
|
|
//Software Interrupt:
|
|
this.CPUCore.SWI();
|
|
}
|
|
ARMInstructionSet.prototype.UNDEFINED = function () {
|
|
//Undefined Exception:
|
|
this.CPUCore.UNDEFINED();
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_DataProcessing1 = function () {
|
|
var data = 0;
|
|
switch ((this.execute & 0x2000060) >> 5) {
|
|
case 0:
|
|
data = this.lli() | 0;
|
|
break;
|
|
case 1:
|
|
data = this.lri() | 0;
|
|
break;
|
|
case 2:
|
|
data = this.ari() | 0;
|
|
break;
|
|
case 3:
|
|
data = this.rri() | 0;
|
|
break;
|
|
default:
|
|
data = this.imm() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_DataProcessing2 = function () {
|
|
var data = 0;
|
|
switch ((this.execute & 0x2000060) >> 5) {
|
|
case 0:
|
|
data = this.llis() | 0;
|
|
break;
|
|
case 1:
|
|
data = this.lris() | 0;
|
|
break;
|
|
case 2:
|
|
data = this.aris() | 0;
|
|
break;
|
|
case 3:
|
|
data = this.rris() | 0;
|
|
break;
|
|
default:
|
|
data = this.imms() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_DataProcessing3 = function () {
|
|
var data = 0;
|
|
switch ((this.execute >> 5) & 0x3) {
|
|
case 0:
|
|
data = this.llr() | 0;
|
|
break;
|
|
case 1:
|
|
data = this.lrr() | 0;
|
|
break;
|
|
case 2:
|
|
data = this.arr() | 0;
|
|
break;
|
|
default:
|
|
data = this.rrr() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_DataProcessing4 = function () {
|
|
var data = 0;
|
|
switch ((this.execute >> 5) & 0x3) {
|
|
case 0:
|
|
data = this.llrs() | 0;
|
|
break;
|
|
case 1:
|
|
data = this.lrrs() | 0;
|
|
break;
|
|
case 2:
|
|
data = this.arrs() | 0;
|
|
break;
|
|
default:
|
|
data = this.rrrs() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStoreOffsetGen = function () {
|
|
var data = 0;
|
|
switch ((this.execute >> 5) & 0x3) {
|
|
case 0:
|
|
data = this.lli() | 0;
|
|
break;
|
|
case 1:
|
|
data = this.lri() | 0;
|
|
break;
|
|
case 2:
|
|
data = this.ari() | 0;
|
|
break;
|
|
default:
|
|
data = this.rri() | 0;
|
|
}
|
|
return data | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStoreOperandDetermine = function () {
|
|
var offset = 0;
|
|
if ((this.execute & 0x400000) == 0) {
|
|
offset = this.read0OffsetRegister() | 0;
|
|
}
|
|
else {
|
|
offset = ((this.execute & 0xF00) >> 4) | (this.execute & 0xF);
|
|
}
|
|
return offset | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStorePostTUser = function (offset) {
|
|
offset = offset | 0;
|
|
var base = this.read16OffsetUserRegister() | 0;
|
|
if ((this.execute & 0x800000) == 0) {
|
|
offset = ((base | 0) - (offset | 0)) | 0;
|
|
}
|
|
else {
|
|
offset = ((base | 0) + (offset | 0)) | 0;
|
|
}
|
|
this.guard16OffsetUserRegisterWrite(offset | 0);
|
|
return base | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStorePostTNormal = function (offset) {
|
|
offset = offset | 0;
|
|
var base = this.read16OffsetRegister() | 0;
|
|
if ((this.execute & 0x800000) == 0) {
|
|
offset = ((base | 0) - (offset | 0)) | 0;
|
|
}
|
|
else {
|
|
offset = ((base | 0) + (offset | 0)) | 0;
|
|
}
|
|
this.guard16OffsetRegisterWrite(offset | 0);
|
|
return base | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStoreNotT = function (offset) {
|
|
offset = offset | 0;
|
|
var base = this.read16OffsetRegister() | 0;
|
|
if ((this.execute & 0x800000) == 0) {
|
|
offset = ((base | 0) - (offset | 0)) | 0;
|
|
}
|
|
else {
|
|
offset = ((base | 0) + (offset | 0)) | 0;
|
|
}
|
|
if ((this.execute & 0x200000) != 0) {
|
|
this.guard16OffsetRegisterWrite(offset | 0);
|
|
}
|
|
return offset | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore1 = function () {
|
|
return this.operand2OP_LoadStorePostTNormal(this.operand2OP_LoadStoreOperandDetermine() | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore2 = function () {
|
|
return this.operand2OP_LoadStoreNotT(this.operand2OP_LoadStoreOperandDetermine() | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore3Normal = function () {
|
|
return this.operand2OP_LoadStorePostTNormal(this.execute & 0xFFF) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore3User = function () {
|
|
return this.operand2OP_LoadStorePostTUser(this.execute & 0xFFF) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore4 = function () {
|
|
return this.operand2OP_LoadStoreNotT(this.execute & 0xFFF) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore5Normal = function () {
|
|
return this.operand2OP_LoadStorePostTNormal(this.operand2OP_LoadStoreOffsetGen() | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore5User = function () {
|
|
return this.operand2OP_LoadStorePostTUser(this.operand2OP_LoadStoreOffsetGen() | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.operand2OP_LoadStore6 = function () {
|
|
return this.operand2OP_LoadStoreNotT(this.operand2OP_LoadStoreOffsetGen() | 0) | 0;
|
|
}
|
|
ARMInstructionSet.prototype.lli = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Shift the register data left:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
return register << (shifter | 0);
|
|
}
|
|
ARMInstructionSet.prototype.llis = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Get the shift amount:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
this.branchFlags.setCarry(register << (((shifter | 0) - 1) | 0));
|
|
}
|
|
//Shift the register data left:
|
|
return register << (shifter | 0);
|
|
}
|
|
ARMInstructionSet.prototype.llr = function () {
|
|
//Logical Left Shift with Register:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Shift the register data left:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
if ((shifter | 0) < 0x20) {
|
|
register = register << (shifter | 0);
|
|
}
|
|
else {
|
|
register = 0;
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.llrs = function () {
|
|
//Logical Left Shift with Register and CPSR:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Get the shift amount:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
if ((shifter | 0) < 0x20) {
|
|
//Shift the register data left:
|
|
this.branchFlags.setCarry(register << (((shifter | 0) - 1) | 0));
|
|
register = register << (shifter | 0);
|
|
}
|
|
else {
|
|
if ((shifter | 0) == 0x20) {
|
|
//Shift bit 0 into carry:
|
|
this.branchFlags.setCarry(register << 31);
|
|
}
|
|
else {
|
|
//Everything Zero'd:
|
|
this.branchFlags.setCarryFalse();
|
|
}
|
|
register = 0;
|
|
}
|
|
}
|
|
//If shift is 0, just return the register without mod:
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.lri = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Shift the register data right logically:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
if ((shifter | 0) == 0) {
|
|
//Return 0:
|
|
register = 0;
|
|
}
|
|
else {
|
|
register = (register >>> (shifter | 0)) | 0;
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.lris = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Get the shift amount:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
//Shift the register data right logically:
|
|
register = (register >>> (shifter | 0)) | 0;
|
|
}
|
|
else {
|
|
this.branchFlags.setCarry(register | 0);
|
|
//Return 0:
|
|
register = 0;
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.lrr = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Shift the register data right logically:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
if ((shifter | 0) < 0x20) {
|
|
register = (register >>> (shifter | 0)) | 0;
|
|
}
|
|
else {
|
|
register = 0;
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.lrrs = function () {
|
|
//Logical Right Shift with Register and CPSR:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Get the shift amount:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
if ((shifter | 0) < 0x20) {
|
|
//Shift the register data right logically:
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
register = (register >>> (shifter | 0)) | 0;
|
|
}
|
|
else {
|
|
if ((shifter | 0) == 0x20) {
|
|
//Shift bit 31 into carry:
|
|
this.branchFlags.setCarry(register | 0);
|
|
}
|
|
else {
|
|
//Everything Zero'd:
|
|
this.branchFlags.setCarryFalse();
|
|
}
|
|
register = 0;
|
|
}
|
|
}
|
|
//If shift is 0, just return the register without mod:
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.ari = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Get the shift amount:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
if ((shifter | 0) == 0) {
|
|
//Shift full length if shifter is zero:
|
|
shifter = 0x1F;
|
|
}
|
|
//Shift the register data right:
|
|
return register >> (shifter | 0);
|
|
}
|
|
ARMInstructionSet.prototype.aris = function () {
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Get the shift amount:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
}
|
|
else {
|
|
//Shift full length if shifter is zero:
|
|
shifter = 0x1F;
|
|
this.branchFlags.setCarry(register | 0);
|
|
}
|
|
//Shift the register data right:
|
|
return register >> (shifter | 0);
|
|
}
|
|
ARMInstructionSet.prototype.arr = function () {
|
|
//Arithmetic Right Shift with Register:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Shift the register data right:
|
|
return register >> Math.min(this.read8OffsetRegister() & 0xFF, 0x1F);
|
|
}
|
|
ARMInstructionSet.prototype.arrs = function () {
|
|
//Arithmetic Right Shift with Register and CPSR:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Get the shift amount:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
//Check to see if we need to update CPSR:
|
|
if ((shifter | 0) > 0) {
|
|
if ((shifter | 0) < 0x20) {
|
|
//Shift the register data right arithmetically:
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
register = register >> (shifter | 0);
|
|
}
|
|
else {
|
|
//Set all bits with bit 31:
|
|
this.branchFlags.setCarry(register | 0);
|
|
register = register >> 0x1F;
|
|
}
|
|
}
|
|
//If shift is 0, just return the register without mod:
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.rri = function () {
|
|
//Rotate Right with Immediate:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Rotate the register right:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
if ((shifter | 0) > 0) {
|
|
//ROR
|
|
register = (register << (0x20 - (shifter | 0))) | (register >>> (shifter | 0));
|
|
}
|
|
else {
|
|
//RRX
|
|
register = (this.branchFlags.getCarry() & 0x80000000) | (register >>> 0x1);
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.rris = function () {
|
|
//Rotate Right with Immediate and CPSR:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Rotate the register right:
|
|
var shifter = (this.execute >> 7) & 0x1F;
|
|
if ((shifter | 0) > 0) {
|
|
//ROR
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
register = (register << (0x20 - (shifter | 0))) | (register >>> (shifter | 0));
|
|
}
|
|
else {
|
|
//RRX
|
|
var rrxValue = (this.branchFlags.getCarry() & 0x80000000) | (register >>> 0x1);
|
|
this.branchFlags.setCarry(register << 31);
|
|
register = rrxValue | 0;
|
|
}
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.rrr = function () {
|
|
//Rotate Right with Register:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Rotate the register right:
|
|
var shifter = this.read8OffsetRegister() & 0x1F;
|
|
if ((shifter | 0) > 0) {
|
|
//ROR
|
|
register = (register << (0x20 - (shifter | 0))) | (register >>> (shifter | 0));
|
|
}
|
|
//If shift is 0, just return the register without mod:
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.rrrs = function () {
|
|
//Rotate Right with Register and CPSR:
|
|
//Get the register data to be shifted:
|
|
var register = this.read0OffsetRegister() | 0;
|
|
//Clock a cycle for the shift delaying the CPU:
|
|
this.wait.CPUInternalSingleCyclePrefetch();
|
|
//Rotate the register right:
|
|
var shifter = this.read8OffsetRegister() & 0xFF;
|
|
if ((shifter | 0) > 0) {
|
|
shifter = shifter & 0x1F;
|
|
if ((shifter | 0) > 0) {
|
|
//ROR
|
|
this.branchFlags.setCarry((register >> (((shifter | 0) - 1) | 0)) << 31);
|
|
register = (register << (0x20 - (shifter | 0))) | (register >>> (shifter | 0));
|
|
}
|
|
else {
|
|
//No shift, but make carry set to bit 31:
|
|
this.branchFlags.setCarry(register | 0);
|
|
}
|
|
}
|
|
//If shift is 0, just return the register without mod:
|
|
return register | 0;
|
|
}
|
|
ARMInstructionSet.prototype.imm = function () {
|
|
//Get the immediate data to be shifted:
|
|
var immediate = this.execute & 0xFF;
|
|
//Rotate the immediate right:
|
|
var shifter = (this.execute >> 7) & 0x1E;
|
|
if ((shifter | 0) > 0) {
|
|
immediate = (immediate << (0x20 - (shifter | 0))) | (immediate >>> (shifter | 0));
|
|
}
|
|
return immediate | 0;
|
|
}
|
|
ARMInstructionSet.prototype.imms = function () {
|
|
//Get the immediate data to be shifted:
|
|
var immediate = this.execute & 0xFF;
|
|
//Rotate the immediate right:
|
|
var shifter = (this.execute >> 7) & 0x1E;
|
|
if ((shifter | 0) > 0) {
|
|
immediate = (immediate << (0x20 - (shifter | 0))) | (immediate >>> (shifter | 0));
|
|
this.branchFlags.setCarry(immediate | 0);
|
|
}
|
|
return immediate | 0;
|
|
}
|
|
ARMInstructionSet.prototype.rc = function () {
|
|
return (this.branchFlags.getNZCV() | this.CPUCore.modeFlags);
|
|
}
|
|
ARMInstructionSet.prototype.rs = function () {
|
|
var spsr = 0;
|
|
switch (this.CPUCore.modeFlags & 0x1f) {
|
|
case 0x12: //IRQ
|
|
spsr = this.CPUCore.SPSR[1] | 0;
|
|
break;
|
|
case 0x13: //Supervisor
|
|
spsr = this.CPUCore.SPSR[2] | 0;
|
|
break;
|
|
case 0x11: //FIQ
|
|
spsr = this.CPUCore.SPSR[0] | 0;
|
|
break;
|
|
case 0x17: //Abort
|
|
spsr = this.CPUCore.SPSR[3] | 0;
|
|
break;
|
|
case 0x1B: //Undefined
|
|
spsr = this.CPUCore.SPSR[4] | 0;
|
|
break;
|
|
default:
|
|
//Instruction hit an invalid SPSR request:
|
|
return this.rc() | 0;
|
|
}
|
|
return ((spsr & 0xF00) << 20) | (spsr & 0xFF);
|
|
}
|
|
function compileARMInstructionDecodeMap() {
|
|
var pseudoCodes = [
|
|
"LDRH",
|
|
"MOVS2",
|
|
"MUL",
|
|
"MSR",
|
|
"LDRSH",
|
|
"MVN2",
|
|
"SMLAL",
|
|
"RSCS",
|
|
"CMPS",
|
|
"MRS",
|
|
"RSBS2",
|
|
"ADDS",
|
|
"SUBS",
|
|
"RSB",
|
|
"SUBS2",
|
|
"MULS",
|
|
"SMLALS",
|
|
"STRB",
|
|
"CMNS2",
|
|
"UMLALS",
|
|
"ORR2",
|
|
"BX",
|
|
"RSBS",
|
|
"LDRSB",
|
|
"LoadStoreMultiple",
|
|
"ANDS2",
|
|
"BIC",
|
|
"ADD2",
|
|
"SBC2",
|
|
"AND",
|
|
"TSTS2",
|
|
"MOV2",
|
|
"MOVS",
|
|
"EOR",
|
|
"ORRS2",
|
|
"RSC",
|
|
"LDR2",
|
|
"SMULLS",
|
|
"LDRSB2",
|
|
"LDRB4",
|
|
"BL",
|
|
"LDRB3",
|
|
"SBCS2",
|
|
"MVNS2",
|
|
"MLAS",
|
|
"MVN",
|
|
"BICS2",
|
|
"UMLAL",
|
|
"CMPS2",
|
|
"LDRB",
|
|
"RSC2",
|
|
"ADC2",
|
|
"LDRSH2",
|
|
"ORR",
|
|
"ADDS2",
|
|
"EOR2",
|
|
"STR3",
|
|
"UMULL",
|
|
"ADD",
|
|
"LDRH2",
|
|
"STRB4",
|
|
"LDR4",
|
|
"EORS",
|
|
"ORRS",
|
|
"BICS",
|
|
"SMULL",
|
|
"EORS2",
|
|
"B",
|
|
"STR",
|
|
"STRH",
|
|
"TEQS",
|
|
"STR2",
|
|
"STRH2",
|
|
"AND2",
|
|
"SUB",
|
|
"MVNS",
|
|
"ADC",
|
|
"ADCS",
|
|
"MOV",
|
|
"CMNS",
|
|
"ADCS2",
|
|
"TSTS",
|
|
"RSCS2",
|
|
"ANDS",
|
|
"STRB3",
|
|
"SBC",
|
|
"STR4",
|
|
"LDR",
|
|
"LDR3",
|
|
"SUB2",
|
|
"STRB2",
|
|
"SWP",
|
|
"TEQS2",
|
|
"UMULLS",
|
|
"BIC2",
|
|
"MLA",
|
|
"SWI",
|
|
"LDRB2",
|
|
"SBCS",
|
|
"RSB2",
|
|
"SWPB",
|
|
"STRT",
|
|
"LDRT",
|
|
"STRBT",
|
|
"LDRBT",
|
|
"STRT2",
|
|
"LDRT2",
|
|
"STRBT2",
|
|
"LDRBT2"
|
|
];
|
|
function compileARMInstructionDecodeOpcodeMap(codeMap) {
|
|
var opcodeIndice = 0;
|
|
var instructionMap = getUint8Array(4096);
|
|
function generateMap1(instruction) {
|
|
for (var index = 0; index < 0x10; ++index) {
|
|
instructionMap[opcodeIndice++] = codeMap[instruction[index]];
|
|
}
|
|
}
|
|
function generateMap2(instruction) {
|
|
var translatedOpcode = codeMap[instruction];
|
|
for (var index = 0; index < 0x10; ++index) {
|
|
instructionMap[opcodeIndice++] = translatedOpcode;
|
|
}
|
|
}
|
|
function generateMap3(instruction) {
|
|
var translatedOpcode = codeMap[instruction];
|
|
for (var index = 0; index < 0x100; ++index) {
|
|
instructionMap[opcodeIndice++] = translatedOpcode;
|
|
}
|
|
}
|
|
function generateMap4(instruction) {
|
|
var translatedOpcode = codeMap[instruction];
|
|
for (var index = 0; index < 0x200; ++index) {
|
|
instructionMap[opcodeIndice++] = translatedOpcode;
|
|
}
|
|
}
|
|
function generateMap5(instruction) {
|
|
var translatedOpcode = codeMap[instruction];
|
|
for (var index = 0; index < 0x300; ++index) {
|
|
instructionMap[opcodeIndice++] = translatedOpcode;
|
|
}
|
|
}
|
|
function generateStoreLoadInstructionSector1() {
|
|
var instrMap = [
|
|
"STR2",
|
|
"LDR2",
|
|
"STRT2",
|
|
"LDRT2",
|
|
"STRB2",
|
|
"LDRB2",
|
|
"STRBT2",
|
|
"LDRBT2"
|
|
];
|
|
for (var instrIndex = 0; instrIndex < 0x10; ++instrIndex) {
|
|
for (var dataIndex = 0; dataIndex < 0x10; ++dataIndex) {
|
|
if ((dataIndex & 0x1) == 0) {
|
|
instructionMap[opcodeIndice++] = codeMap[instrMap[instrIndex & 0x7]];
|
|
}
|
|
else {
|
|
instructionMap[opcodeIndice++] = codeMap["UNDEFINED"];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
function generateStoreLoadInstructionSector2() {
|
|
var instrMap = [
|
|
"STR3",
|
|
"LDR3",
|
|
"STRB3",
|
|
"LDRB3"
|
|
];
|
|
for (var instrIndex = 0; instrIndex < 0x10; ++instrIndex) {
|
|
for (var dataIndex = 0; dataIndex < 0x10; ++dataIndex) {
|
|
if ((dataIndex & 0x1) == 0) {
|
|
instructionMap[opcodeIndice++] = codeMap[instrMap[((instrIndex >> 1) & 0x2) | (instrIndex & 0x1)]];
|
|
}
|
|
else {
|
|
instructionMap[opcodeIndice++] = codeMap["UNDEFINED"];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//0
|
|
generateMap1([
|
|
"AND",
|
|
"AND2",
|
|
"AND",
|
|
"AND2",
|
|
"AND",
|
|
"AND2",
|
|
"AND",
|
|
"AND2",
|
|
"AND",
|
|
"MUL",
|
|
"AND",
|
|
"STRH",
|
|
"AND",
|
|
"UNDEFINED",
|
|
"AND",
|
|
"UNDEFINED"
|
|
]);
|
|
//1
|
|
generateMap1([
|
|
"ANDS",
|
|
"ANDS2",
|
|
"ANDS",
|
|
"ANDS2",
|
|
"ANDS",
|
|
"ANDS2",
|
|
"ANDS",
|
|
"ANDS2",
|
|
"ANDS",
|
|
"MULS",
|
|
"ANDS",
|
|
"LDRH",
|
|
"ANDS",
|
|
"LDRSB",
|
|
"ANDS",
|
|
"LDRSH"
|
|
]);
|
|
//2
|
|
generateMap1([
|
|
"EOR",
|
|
"EOR2",
|
|
"EOR",
|
|
"EOR2",
|
|
"EOR",
|
|
"EOR2",
|
|
"EOR",
|
|
"EOR2",
|
|
"EOR",
|
|
"MLA",
|
|
"EOR",
|
|
"STRH",
|
|
"EOR",
|
|
"UNDEFINED",
|
|
"EOR",
|
|
"UNDEFINED"
|
|
]);
|
|
//3
|
|
generateMap1([
|
|
"EORS",
|
|
"EORS2",
|
|
"EORS",
|
|
"EORS2",
|
|
"EORS",
|
|
"EORS2",
|
|
"EORS",
|
|
"EORS2",
|
|
"EORS",
|
|
"MLAS",
|
|
"EORS",
|
|
"LDRH",
|
|
"EORS",
|
|
"LDRSB",
|
|
"EORS",
|
|
"LDRSH"
|
|
]);
|
|
//4
|
|
generateMap1([
|
|
"SUB",
|
|
"SUB2",
|
|
"SUB",
|
|
"SUB2",
|
|
"SUB",
|
|
"SUB2",
|
|
"SUB",
|
|
"SUB2",
|
|
"SUB",
|
|
"UNDEFINED",
|
|
"SUB",
|
|
"STRH",
|
|
"SUB",
|
|
"UNDEFINED",
|
|
"SUB",
|
|
"UNDEFINED"
|
|
]);
|
|
//5
|
|
generateMap1([
|
|
"SUBS",
|
|
"SUBS2",
|
|
"SUBS",
|
|
"SUBS2",
|
|
"SUBS",
|
|
"SUBS2",
|
|
"SUBS",
|
|
"SUBS2",
|
|
"SUBS",
|
|
"UNDEFINED",
|
|
"SUBS",
|
|
"LDRH",
|
|
"SUBS",
|
|
"LDRSB",
|
|
"SUBS",
|
|
"LDRSH"
|
|
]);
|
|
//6
|
|
generateMap1([
|
|
"RSB",
|
|
"RSB2",
|
|
"RSB",
|
|
"RSB2",
|
|
"RSB",
|
|
"RSB2",
|
|
"RSB",
|
|
"RSB2",
|
|
"RSB",
|
|
"UNDEFINED",
|
|
"RSB",
|
|
"STRH",
|
|
"RSB",
|
|
"UNDEFINED",
|
|
"RSB",
|
|
"UNDEFINED"
|
|
]);
|
|
//7
|
|
generateMap1([
|
|
"RSBS",
|
|
"RSBS2",
|
|
"RSBS",
|
|
"RSBS2",
|
|
"RSBS",
|
|
"RSBS2",
|
|
"RSBS",
|
|
"RSBS2",
|
|
"RSBS",
|
|
"UNDEFINED",
|
|
"RSBS",
|
|
"LDRH",
|
|
"RSBS",
|
|
"LDRSB",
|
|
"RSBS",
|
|
"LDRSH"
|
|
]);
|
|
//8
|
|
generateMap1([
|
|
"ADD",
|
|
"ADD2",
|
|
"ADD",
|
|
"ADD2",
|
|
"ADD",
|
|
"ADD2",
|
|
"ADD",
|
|
"ADD2",
|
|
"ADD",
|
|
"UMULL",
|
|
"ADD",
|
|
"STRH",
|
|
"ADD",
|
|
"UNDEFINED",
|
|
"ADD",
|
|
"UNDEFINED"
|
|
]);
|
|
//9
|
|
generateMap1([
|
|
"ADDS",
|
|
"ADDS2",
|
|
"ADDS",
|
|
"ADDS2",
|
|
"ADDS",
|
|
"ADDS2",
|
|
"ADDS",
|
|
"ADDS2",
|
|
"ADDS",
|
|
"UMULLS",
|
|
"ADDS",
|
|
"LDRH",
|
|
"ADDS",
|
|
"LDRSB",
|
|
"ADDS",
|
|
"LDRSH"
|
|
]);
|
|
//A
|
|
generateMap1([
|
|
"ADC",
|
|
"ADC2",
|
|
"ADC",
|
|
"ADC2",
|
|
"ADC",
|
|
"ADC2",
|
|
"ADC",
|
|
"ADC2",
|
|
"ADC",
|
|
"UMLAL",
|
|
"ADC",
|
|
"STRH",
|
|
"ADC",
|
|
"UNDEFINED",
|
|
"ADC",
|
|
"UNDEFINED"
|
|
]);
|
|
//B
|
|
generateMap1([
|
|
"ADCS",
|
|
"ADCS2",
|
|
"ADCS",
|
|
"ADCS2",
|
|
"ADCS",
|
|
"ADCS2",
|
|
"ADCS",
|
|
"ADCS2",
|
|
"ADCS",
|
|
"UMLALS",
|
|
"ADCS",
|
|
"LDRH",
|
|
"ADCS",
|
|
"LDRSB",
|
|
"ADCS",
|
|
"LDRSH"
|
|
]);
|
|
//C
|
|
generateMap1([
|
|
"SBC",
|
|
"SBC2",
|
|
"SBC",
|
|
"SBC2",
|
|
"SBC",
|
|
"SBC2",
|
|
"SBC",
|
|
"SBC2",
|
|
"SBC",
|
|
"SMULL",
|
|
"SBC",
|
|
"STRH",
|
|
"SBC",
|
|
"UNDEFINED",
|
|
"SBC",
|
|
"UNDEFINED"
|
|
]);
|
|
//D
|
|
generateMap1([
|
|
"SBCS",
|
|
"SBCS2",
|
|
"SBCS",
|
|
"SBCS2",
|
|
"SBCS",
|
|
"SBCS2",
|
|
"SBCS",
|
|
"SBCS2",
|
|
"SBCS",
|
|
"SMULLS",
|
|
"SBCS",
|
|
"LDRH",
|
|
"SBCS",
|
|
"LDRSB",
|
|
"SBCS",
|
|
"LDRSH"
|
|
]);
|
|
//E
|
|
generateMap1([
|
|
"RSC",
|
|
"RSC2",
|
|
"RSC",
|
|
"RSC2",
|
|
"RSC",
|
|
"RSC2",
|
|
"RSC",
|
|
"RSC2",
|
|
"RSC",
|
|
"SMLAL",
|
|
"RSC",
|
|
"STRH",
|
|
"RSC",
|
|
"UNDEFINED",
|
|
"RSC",
|
|
"UNDEFINED"
|
|
]);
|
|
//F
|
|
generateMap1([
|
|
"RSCS",
|
|
"RSCS2",
|
|
"RSCS",
|
|
"RSCS2",
|
|
"RSCS",
|
|
"RSCS2",
|
|
"RSCS",
|
|
"RSCS2",
|
|
"RSCS",
|
|
"SMLALS",
|
|
"RSCS",
|
|
"LDRH",
|
|
"RSCS",
|
|
"LDRSB",
|
|
"RSCS",
|
|
"LDRSH"
|
|
]);
|
|
//10
|
|
generateMap1([
|
|
"MRS",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"SWP",
|
|
"UNDEFINED",
|
|
"STRH2",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED"
|
|
]);
|
|
//11
|
|
generateMap1([
|
|
"TSTS",
|
|
"TSTS2",
|
|
"TSTS",
|
|
"TSTS2",
|
|
"TSTS",
|
|
"TSTS2",
|
|
"TSTS",
|
|
"TSTS2",
|
|
"TSTS",
|
|
"UNDEFINED",
|
|
"TSTS",
|
|
"LDRH2",
|
|
"TSTS",
|
|
"LDRSB2",
|
|
"TSTS",
|
|
"LDRSH2"
|
|
]);
|
|
//12
|
|
generateMap1([
|
|
"MSR",
|
|
"BX",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"STRH2",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED"
|
|
]);
|
|
//13
|
|
generateMap1([
|
|
"TEQS",
|
|
"TEQS2",
|
|
"TEQS",
|
|
"TEQS2",
|
|
"TEQS",
|
|
"TEQS2",
|
|
"TEQS",
|
|
"TEQS2",
|
|
"TEQS",
|
|
"UNDEFINED",
|
|
"TEQS",
|
|
"LDRH2",
|
|
"TEQS",
|
|
"LDRSB2",
|
|
"TEQS",
|
|
"LDRSH2"
|
|
]);
|
|
//14
|
|
generateMap1([
|
|
"MRS",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"SWPB",
|
|
"UNDEFINED",
|
|
"STRH2",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED"
|
|
]);
|
|
//15
|
|
generateMap1([
|
|
"CMPS",
|
|
"CMPS2",
|
|
"CMPS",
|
|
"CMPS2",
|
|
"CMPS",
|
|
"CMPS2",
|
|
"CMPS",
|
|
"CMPS2",
|
|
"CMPS",
|
|
"UNDEFINED",
|
|
"CMPS",
|
|
"LDRH2",
|
|
"CMPS",
|
|
"LDRSB2",
|
|
"CMPS",
|
|
"LDRSH2"
|
|
]);
|
|
//16
|
|
generateMap1([
|
|
"MSR",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"STRH2",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED",
|
|
"UNDEFINED"
|
|
]);
|
|
//17
|
|
generateMap1([
|
|
"CMNS",
|
|
"CMNS2",
|
|
"CMNS",
|
|
"CMNS2",
|
|
"CMNS",
|
|
"CMNS2",
|
|
"CMNS",
|
|
"CMNS2",
|
|
"CMNS",
|
|
"UNDEFINED",
|
|
"CMNS",
|
|
"LDRH2",
|
|
"CMNS",
|
|
"LDRSB2",
|
|
"CMNS",
|
|
"LDRSH2"
|
|
]);
|
|
//18
|
|
generateMap1([
|
|
"ORR",
|
|
"ORR2",
|
|
"ORR",
|
|
"ORR2",
|
|
"ORR",
|
|
"ORR2",
|
|
"ORR",
|
|
"ORR2",
|
|
"ORR",
|
|
"UNDEFINED",
|
|
"ORR",
|
|
"STRH2",
|
|
"ORR",
|
|
"UNDEFINED",
|
|
"ORR",
|
|
"UNDEFINED"
|
|
]);
|
|
//19
|
|
generateMap1([
|
|
"ORRS",
|
|
"ORRS2",
|
|
"ORRS",
|
|
"ORRS2",
|
|
"ORRS",
|
|
"ORRS2",
|
|
"ORRS",
|
|
"ORRS2",
|
|
"ORRS",
|
|
"UNDEFINED",
|
|
"ORRS",
|
|
"LDRH2",
|
|
"ORRS",
|
|
"LDRSB2",
|
|
"ORRS",
|
|
"LDRSH2"
|
|
]);
|
|
//1A
|
|
generateMap1([
|
|
"MOV",
|
|
"MOV2",
|
|
"MOV",
|
|
"MOV2",
|
|
"MOV",
|
|
"MOV2",
|
|
"MOV",
|
|
"MOV2",
|
|
"MOV",
|
|
"UNDEFINED",
|
|
"MOV",
|
|
"STRH2",
|
|
"MOV",
|
|
"UNDEFINED",
|
|
"MOV",
|
|
"UNDEFINED"
|
|
]);
|
|
//1B
|
|
generateMap1([
|
|
"MOVS",
|
|
"MOVS2",
|
|
"MOVS",
|
|
"MOVS2",
|
|
"MOVS",
|
|
"MOVS2",
|
|
"MOVS",
|
|
"MOVS2",
|
|
"MOVS",
|
|
"UNDEFINED",
|
|
"MOVS",
|
|
"LDRH2",
|
|
"MOVS",
|
|
"LDRSB2",
|
|
"MOVS",
|
|
"LDRSH2"
|
|
]);
|
|
//1C
|
|
generateMap1([
|
|
"BIC",
|
|
"BIC2",
|
|
"BIC",
|
|
"BIC2",
|
|
"BIC",
|
|
"BIC2",
|
|
"BIC",
|
|
"BIC2",
|
|
"BIC",
|
|
"UNDEFINED",
|
|
"BIC",
|
|
"STRH2",
|
|
"BIC",
|
|
"UNDEFINED",
|
|
"BIC",
|
|
"UNDEFINED"
|
|
]);
|
|
//1D
|
|
generateMap1([
|
|
"BICS",
|
|
"BICS2",
|
|
"BICS",
|
|
"BICS2",
|
|
"BICS",
|
|
"BICS2",
|
|
"BICS",
|
|
"BICS2",
|
|
"BICS",
|
|
"UNDEFINED",
|
|
"BICS",
|
|
"LDRH2",
|
|
"BICS",
|
|
"LDRSB2",
|
|
"BICS",
|
|
"LDRSH2"
|
|
]);
|
|
//1E
|
|
generateMap1([
|
|
"MVN",
|
|
"MVN2",
|
|
"MVN",
|
|
"MVN2",
|
|
"MVN",
|
|
"MVN2",
|
|
"MVN",
|
|
"MVN2",
|
|
"MVN",
|
|
"UNDEFINED",
|
|
"MVN",
|
|
"STRH2",
|
|
"MVN",
|
|
"UNDEFINED",
|
|
"MVN",
|
|
"UNDEFINED"
|
|
]);
|
|
//1F
|
|
generateMap1([
|
|
"MVNS",
|
|
"MVNS2",
|
|
"MVNS",
|
|
"MVNS2",
|
|
"MVNS",
|
|
"MVNS2",
|
|
"MVNS",
|
|
"MVNS2",
|
|
"MVNS",
|
|
"UNDEFINED",
|
|
"MVNS",
|
|
"LDRH2",
|
|
"MVNS",
|
|
"LDRSB2",
|
|
"MVNS",
|
|
"LDRSH2"
|
|
]);
|
|
//20
|
|
generateMap2("AND");
|
|
//21
|
|
generateMap2("ANDS");
|
|
//22
|
|
generateMap2("EOR");
|
|
//23
|
|
generateMap2("EORS");
|
|
//24
|
|
generateMap2("SUB");
|
|
//25
|
|
generateMap2("SUBS");
|
|
//26
|
|
generateMap2("RSB");
|
|
//27
|
|
generateMap2("RSBS");
|
|
//28
|
|
generateMap2("ADD");
|
|
//29
|
|
generateMap2("ADDS");
|
|
//2A
|
|
generateMap2("ADC");
|
|
//2B
|
|
generateMap2("ADCS");
|
|
//2C
|
|
generateMap2("SBC");
|
|
//2D
|
|
generateMap2("SBCS");
|
|
//2E
|
|
generateMap2("RSC");
|
|
//2F
|
|
generateMap2("RSCS");
|
|
//30
|
|
generateMap2("UNDEFINED");
|
|
//31
|
|
generateMap2("TSTS");
|
|
//32
|
|
generateMap2("MSR");
|
|
//33
|
|
generateMap2("TEQS");
|
|
//34
|
|
generateMap2("UNDEFINED");
|
|
//35
|
|
generateMap2("CMPS");
|
|
//36
|
|
generateMap2("MSR");
|
|
//37
|
|
generateMap2("CMNS");
|
|
//38
|
|
generateMap2("ORR");
|
|
//39
|
|
generateMap2("ORRS");
|
|
//3A
|
|
generateMap2("MOV");
|
|
//3B
|
|
generateMap2("MOVS");
|
|
//3C
|
|
generateMap2("BIC");
|
|
//3D
|
|
generateMap2("BICS");
|
|
//3E
|
|
generateMap2("MVN");
|
|
//3F
|
|
generateMap2("MVNS");
|
|
//40
|
|
generateMap2("STR");
|
|
//41
|
|
generateMap2("LDR");
|
|
//42
|
|
generateMap2("STRT");
|
|
//43
|
|
generateMap2("LDRT");
|
|
//44
|
|
generateMap2("STRB");
|
|
//45
|
|
generateMap2("LDRB");
|
|
//46
|
|
generateMap2("STRBT");
|
|
//47
|
|
generateMap2("LDRBT");
|
|
//48
|
|
generateMap2("STR");
|
|
//49
|
|
generateMap2("LDR");
|
|
//4A
|
|
generateMap2("STRT");
|
|
//4B
|
|
generateMap2("LDRT");
|
|
//4C
|
|
generateMap2("STRB");
|
|
//4D
|
|
generateMap2("LDRB");
|
|
//4E
|
|
generateMap2("STRBT");
|
|
//4F
|
|
generateMap2("LDRBT");
|
|
//50
|
|
generateMap2("STR4");
|
|
//51
|
|
generateMap2("LDR4");
|
|
//52
|
|
generateMap2("STR4");
|
|
//53
|
|
generateMap2("LDR4");
|
|
//54
|
|
generateMap2("STRB4");
|
|
//55
|
|
generateMap2("LDRB4");
|
|
//56
|
|
generateMap2("STRB4");
|
|
//57
|
|
generateMap2("LDRB4");
|
|
//58
|
|
generateMap2("STR4");
|
|
//59
|
|
generateMap2("LDR4");
|
|
//5A
|
|
generateMap2("STR4");
|
|
//5B
|
|
generateMap2("LDR4");
|
|
//5C
|
|
generateMap2("STRB4");
|
|
//5D
|
|
generateMap2("LDRB4");
|
|
//5E
|
|
generateMap2("STRB4");
|
|
//5F
|
|
generateMap2("LDRB4");
|
|
//60-6F
|
|
generateStoreLoadInstructionSector1();
|
|
//70-7F
|
|
generateStoreLoadInstructionSector2();
|
|
//80-9F
|
|
generateMap4("LoadStoreMultiple");
|
|
//A0-AF
|
|
generateMap3("B");
|
|
//B0-BF
|
|
generateMap3("BL");
|
|
//C0-EF
|
|
generateMap5("UNDEFINED");
|
|
//F0-FF
|
|
generateMap3("SWI");
|
|
//Set to prototype:
|
|
ARMInstructionSet.prototype.instructionMap = instructionMap;
|
|
}
|
|
function compileARMInstructionDecodeOpcodeSwitch() {
|
|
var opcodeNameMap = {};
|
|
//ARMInstructionSet.prototype.opcodeCount = [];
|
|
var opcodeCount = pseudoCodes.length | 0;
|
|
var code = "switch (this.instructionMap[((this.execute >> 16) & 0xFF0) | ((this.execute >> 4) & 0xF)] & 0xFF) {";
|
|
for (var opcodeNumber = 0; (opcodeNumber | 0) < (opcodeCount | 0); opcodeNumber = ((opcodeNumber | 0) + 1) | 0) {
|
|
var opcodeName = pseudoCodes[opcodeNumber | 0];
|
|
opcodeNameMap[opcodeName] = opcodeNumber | 0;
|
|
/*code += "case " + (opcodeNumber | 0) + ":{this." + opcodeName + "();++ARMInstructionSet.prototype.opcodeCount[" + opcodeNumber + "][1];break};";
|
|
ARMInstructionSet.prototype.opcodeCount[opcodeNumber | 0] = [opcodeName, 0];*/
|
|
code += "case " + (opcodeNumber | 0) + ":{this." + opcodeName + "();break};";
|
|
}
|
|
code += "default:{this.UNDEFINED()}}";
|
|
opcodeNameMap["UNDEFINED"] = opcodeNumber | 0;
|
|
ARMInstructionSet.prototype.executeDecoded = Function(code);
|
|
return opcodeNameMap;
|
|
}
|
|
compileARMInstructionDecodeOpcodeMap(compileARMInstructionDecodeOpcodeSwitch());
|
|
}
|
|
compileARMInstructionDecodeMap();
|
|
/*function sortPriority() {
|
|
var sorted = ARMInstructionSet.prototype.opcodeCount.sort(function(a, b) {
|
|
return b[1] - a[1];
|
|
});
|
|
var output = "[\n";
|
|
var length = sorted.length - 1;
|
|
for (var index = 0; index <= length; index++) {
|
|
output += "\t\"" + sorted[index][0] + "\"" + ((index < length) ? "," : "") + "\n";
|
|
}
|
|
output += "];";
|
|
console.log(output);
|
|
}*/
|