-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbrowser-client.js
178 lines (148 loc) · 5.24 KB
/
browser-client.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
/*
Visual rendering of abstract CPU parts for browsers.
NB. This module is extremely ugly & temporary; this is just for quickly testing/controlling the abstract model
*/
import { ControlUnit } from './cpu-sim.js'
import { Bitstring } from './binary-encoding.js'
const { document } = window // Browser deps imported from global to module scope
let memContainerEl = document.querySelector(`.memory`)
let regContainerEl = document.querySelector(`.register`)
let irEl = document.querySelector(`.instructionRegister`)
let pcEl = document.querySelector(`.programCounter`)
let vm = {
translateToHex: false,
simulationDelayMs: 1000
}
let regs = Array(0x10).fill(Bitstring.fromHex(`00`))
let mem = Array(0x100).fill(Bitstring.fromHex(`00`)).map((val, i) => i % 2 === 0 ? Bitstring.fromHex(`00`) : Bitstring.fromHex(`0f`))
// Load instruction into memory
let loadIns = (startAddr, hex) => {
mem[startAddr] = Bitstring.fromHex(hex.slice(0,2))
mem[startAddr + 1] = Bitstring.fromHex(hex.slice(2))
}
loadIns(0, '20ff') // Load into reg0 pattern 0xff
loadIns(2, '2109') // Load into reg1 pattern 0x09
loadIns(4, '2208') // Load into reg2 pattern 0x08
loadIns(6, '5312') // Add into reg3 result of reg1 + reg2
loadIns(8, '4034') // "Move" copy) reg3 to reg4
loadIns(10, '7541') // reg5 = reg4 OR reg1
loadIns(12, '8641') // reg6 = reg4 AND reg1
loadIns(14, '9741') // reg7 = reg4 XOR reg1
loadIns(16, 'a305') // rotateRight reg3 by 5
loadIns(18, 'a303') // rotateRight reg3 by 3 (return to original)
loadIns(20, 'a309') // rotateRight reg3 by 9 (same as 1)
loadIns(22, '3440') // store reg4 into mem0x40
loadIns(24, '1840') // load mem0x40 into reg8
loadIns(26, 'b830') // if (reg8===reg0) jump to mem0x30
loadIns(28, '4080') // copy reg8 over reg0
loadIns(30, 'b830') // if (reg8===reg0) jump to mem0x30
loadIns(50, 'c000') // halt
// TODO - test addFloat
export let renderView = (mem, reg, pc, ir)=>{
let lastExecutedEls = memContainerEl.querySelectorAll(`.Cell_lastActive`)
mem.forEach((val, i)=>{
let sequence = mem[i]
let cellEl = memContainerEl.children[i]
let toWrite = sequence || `undef`
if (sequence && vm.translateToHex) toWrite = toWrite.toHex()
if (!cellEl){
cellEl = document.createElement(`li`)
cellEl.classList.add('Cell')
memContainerEl.appendChild(cellEl)
let numberEl = document.createElement(`span`)
numberEl.classList.add(`Cell_number`)
numberEl.innerText = i + `:`
let textNode = document.createTextNode(toWrite)
cellEl.appendChild(numberEl)
cellEl.appendChild(textNode)
} else {
let textNode = cellEl.childNodes[1]
textNode.textContent = toWrite
}
if (pc.toDec() === i || pc.toDec() === i - 1){ // These cells represents the memory address(es) last executed
if (lastExecutedEls.length > 0) lastExecutedEls.forEach(el => {
el.classList.add('Cell_executed')
el.classList.remove(`Cell_lastActive`)
})
cellEl.classList.add(`Cell_lastActive`)
}
})
for (let i in reg){
i = Number(i)
let sequence = reg[i]
let registerEl = regContainerEl.children[i]
if (!registerEl){
registerEl = document.createElement(`li`)
registerEl.classList.add('Cell')
regContainerEl.appendChild(registerEl)
}
registerEl.innerText = sequence
}
irEl.innerText = ir.toHex()
pcEl.innerText = pc.toDec()
// Visual "tick" indicator
pcEl.style.transition = 'background-color 50ms'
pcEl.style.backgroundColor = 'hsla(55, 100%, 53%, 0.51)'
setTimeout(()=>{
pcEl.style.backgroundColor = ''
}, 50)
}
let cu = new ControlUnit({
mem, regs,
waitBetweenCyclesMs: vm.simulationDelayMs,
afterCycleFn: (mem, regs, pc, ir)=>{
// console.debug(pc.toHex(), ir.toHex(), regs, mem)
renderView(mem, regs, pc, ir)
}
})
/*
Bind user inputs
*/
let bootBtnEl = document.querySelector(`.js-BootBtn`)
let interruptBtnEl = document.querySelector(`.js-InterruptBtn`)
let resetBtnEl = document.querySelector(`.js-ResetBtn`)
let bootAddrEl = document.querySelector(`.js-BootAddr`)
let translateHexEl = document.querySelector(`input[name="translateToHex"]`)
let simDelayEl = document.querySelector(`input[name="simulationDelay"]`)
let memAddrToChangeEl = document.querySelector(`.js-MemAddrToChange`)
let memValToChangeEl = document.querySelector(`.js-MemValToChange`)
bootBtnEl.addEventListener(`click`, e => {
memContainerEl.innerHTML = ``
regContainerEl.innerHTML = ``
cu.boot()
interruptBtnEl.focus()
})
bootBtnEl.focus()
interruptBtnEl.addEventListener(`click`, e => {
cu.interrupt = 1
})
resetBtnEl.addEventListener(`click`, e => {
cu.reset()
})
bootAddrEl.addEventListener(`change`, function(){
cu.pc = Bitstring.fromHex(this.value)
})
translateHexEl.addEventListener(`change`, function(){
vm.translateToHex = this.checked
})
simDelayEl.addEventListener(`change`, function(){
vm.simulationDelayMs = Number(this.value)
cu.cfg.waitBetweenCyclesMs = vm.simulationDelayMs
})
simDelayEl.value = vm.simulationDelayMs
memValToChangeEl.addEventListener(`change`, function(){
let cellAddress = Bitstring.fromHex(memAddrToChangeEl.value)
let cellValue = Bitstring.fromHex(this.value)
cu.mem[cellAddress.toDec()] = cellValue
})
/*
Bindings for the browser REPL to access
*/
Object.assign(window, {
Bitstring, cu, mem,
bs: Bitstring.fromDec(64),
bs1: new Bitstring(`11111111`),
bs0: new Bitstring(`00000000`),
a: new ArrayBuffer(32),
v: new Int8Array()
})