Fantasy 8Bit system (F8), is a fantasy 8bit console and a set of libraries for creating fantasy 8bit consoles.
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

966 行
27KB

  1. /*
  2. * Emulate a basic 6502 (MOS) chip.
  3. */
  4. const BITM = require('../../utils/bitman.js');
  5. var IMem = require('../../memory').IMem;
  6. // mode = 0 - Immediate
  7. // mode = 1 - Zero Page
  8. // mode = 2 - Zero Page, X
  9. // mode = 3 - Zero Page, Y
  10. // mode = 4 - Absolute
  11. // mode = 5 - Absolute, X
  12. // mode = 6 - Absolute, Y
  13. // mode = 7 - Indirect, X
  14. // mode = 8 - Indirect, Y
  15. function ProcessOp(cpu, mode){
  16. if (mode < 0 || mode > 9){return false;}
  17. switch(cpu.__cycle){
  18. case 0:
  19. if (mode === 0){
  20. cpu.__mem.address = cpu.__PC;
  21. PCUp(cpu, 1);
  22. cpu.__opv = cpu.__mem.byte;
  23. }
  24. return (mode === 0 || mode === 9);
  25. case 1:
  26. switch(mode){
  27. case 1: // Zero Page
  28. cpu.__mem.address = cpu.__opv;
  29. cpu.__opv = cpu.__mem.byte;
  30. return true;
  31. case 2: // Zero Page, X
  32. case 3: // Zero Page, Y
  33. cpu.__opv = (cpu.__opv + ((mode === 2) ? cpu.__XR : cpu.__YR)) & 0xFF; break;
  34. case 4: // Absolute
  35. case 5: // Absolute, X
  36. case 6: // Absolute, Y
  37. cpu.__mem.address = cpu.__PC;
  38. PCUp(cpu, 1);
  39. cpu.__opv |= cpu.__mem.byte << 8;
  40. break;
  41. case 7: // Indirect, X
  42. cpu.__opv = (cpu.__opv + cpu.__XR) & 0xFF; break;
  43. case 8: // Indirect, Y
  44. cpu.__mem.address = cpu.__opv;
  45. cpu.__opv = cpu.__mem.byte;
  46. break;
  47. } break;
  48. case 2:
  49. switch(mode){
  50. case 2: // Zero Page, X
  51. case 3: // Zero Page, Y
  52. case 4: // Absolute
  53. cpu.__mem.address = cpu.__opv;
  54. cpu.__opv = cpu.__mem.byte;
  55. return true;
  56. case 5: // Absolute, X
  57. case 6: // Absolute, Y
  58. let s = (mode === 5) ? cpu.__XR : cpu.__YR;
  59. let l = (cpu.__opv & 0xFF) + s;
  60. cpu.__opv = (cpu.__opv & 0xFF00) | (l & 0xFF);
  61. if (l < 255){
  62. cpu.__mem.address = cpu.__opv;
  63. cpu.__opv = cpu.__mem.byte;
  64. return true;
  65. }
  66. break;
  67. case 7: // Indirect, X
  68. cpu.__mem.address = cpu.__opv;
  69. cpu.__opv = cpu.__mem.byte;
  70. break;
  71. case 8: // Indirect, Y
  72. cpu.__mem.address += 1;
  73. cpu.__opv |= cpu.__mem.byte << 8;
  74. break;
  75. } break;
  76. case 3:
  77. switch(mode){
  78. case 5: // Absolute, X
  79. case 6: // Absolute, Y
  80. let h = (cpu.__opv >> 8) + 1;
  81. cpu.__mem.address = (cpu.__opv & 0xFF) | (h << 8);
  82. cpu.__opv = cpu.__mem.byte;
  83. return true;
  84. case 7: // Indirect, X
  85. cpu.__mem.address += 1;
  86. cpu.__opv |= cpu.__mem.byte << 8;
  87. break;
  88. case 8: // Indirect, Y
  89. let l = (cpu.__opv & 0xFF) + cpu.__YR;
  90. cpu.__opv = (cpu.__opv & 0xFF00) | (l & 0xFF);
  91. if (l <= 255){
  92. cpu.__mem.address = cpu.__opv;
  93. cpu.__opv = cpu.__mem.byte;
  94. return true;
  95. }
  96. break;
  97. } break;
  98. case 4:
  99. if (mode === 8){
  100. let h = (cpu.__opv >> 8) + 1;
  101. cpu.__opv = (cpu.__opv & 0x00FF) | (h << 8);
  102. }
  103. cpu.__mem.address = cpu.__opv;
  104. cpu.__opv = cpu.__mem.byte;
  105. return true;
  106. }
  107. cpu.__cycle += 1;
  108. return false;
  109. }
  110. function ADC(cpu){ // To be used by both the ADC and SBC op codes.
  111. let pmode = [0x69, 0x65, 0x75, null, 0x6D, 0x7D, 0x79, 0x61, 0x71, null].indexOf(cpu.__op);
  112. if (ProcessOp(cpu, pmode) === true){
  113. cpu.__op = -1;
  114. ALU(cpu, 0, cpu.__opv);
  115. }
  116. }
  117. function AND(cpu){
  118. let pmode = [0x29, 0x25, 0x35, null, 0x2D, 0x3D, 0x39, 0x21, 0x31, null].indexOf(cpu.__op);
  119. if (ProcessOp(cpu, pmode) === true){
  120. cpu.__op = -1;
  121. this.__AR &= this.__opv;
  122. cpu.Z = (this.__AR === 0);
  123. cpu.N = BITM.val(this.__AR, 7);
  124. }
  125. }
  126. function ASL(cpu){
  127. let pmode = [null, 0x06, 0x16, null, 0x0E, 0x1E, null, null, null, 0x0A].indexOf(cpu.__op);
  128. if (cpu.__ophold === 1){pmode = -1;}
  129. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  130. if (cpu.__ophold === 0){
  131. cpu.__opv = (mode === 9) ? cpu.__AR : cpu.__opv;
  132. cpu.C = BITM.val(cpu.__opv, 7);
  133. cpu.__opv = (cpu.__opv << 1) & 0xFF;
  134. cpu.__ophold = 1;
  135. } else {
  136. if (mode === 9){
  137. cpu.__AR = cpu.__opv;
  138. } else {
  139. cpu.__mem.byte = cpu.__opv;
  140. }
  141. cpu.Z = (this.__opv === 0);
  142. cpu.N = BITM.val(this.__opv, 7);
  143. cpu.__op = -1;
  144. cpu.__ophold = 0;
  145. }
  146. }
  147. }
  148. function BIT(cpu){
  149. let pmode = [null, 0x24, null, null, 0x2C, null, null, null, null, null].indexOf(cpu.__op);
  150. if (ProcessOp(cpu, pmode) === true){
  151. cpu.__op = -1;
  152. let v = this.__AR & this.__opv;
  153. cpu.Z = (v === 0);
  154. cpu.N = BITM.val(this.__opv, 7);
  155. cpu.V = BITM.val(this.__opv, 6);
  156. }
  157. }
  158. function BRANCH(cpu){
  159. switch(cpu.__cycle){
  160. case 0:
  161. let branch = false;
  162. switch(cpu.__op){
  163. case 0x10: // BPL
  164. branch = (cpu.N === 0); break;
  165. case 0x30: // BMI
  166. branch = (cpu.N === 1); break;
  167. case 0x50: // BVC
  168. branch = (cpu.V === 0); break;
  169. case 0x70: // BVS
  170. branch = (cpu.V === 1); break;
  171. case 0x90: // BCC
  172. branch = (cpu.C === 0); break;
  173. case 0xB0: // BCS
  174. branch = (cpu.C === 1); break;
  175. case 0xD0: // BNE
  176. branch = (cpu.Z === 0); break;
  177. case 0xF0: // BEQ
  178. branch = (cpu.Z === 1); break;
  179. }
  180. if (branch === false)
  181. PCUp(cpu, 1);
  182. case 1:
  183. if (cpu.__cycle === 1){ // TODO: Huh???
  184. cpu.__mem.address = this.__PC;
  185. let v = cpu.__mem.byte;
  186. if (v > 128){
  187. PCDown(cpu, 255 - v);
  188. } else {
  189. PCUp(cpu, v);
  190. }
  191. }
  192. cpu.__op = -1;
  193. }
  194. cpu.__cycle += 1;
  195. }
  196. function BRK(cpu, nmi){
  197. nmi = (nmi === true);
  198. switch(cpu.__cycle){
  199. case 0:
  200. PCUp(cpu, 1);
  201. case 1:
  202. StackPush(cpu, cpu.__PC >> 8); break;
  203. case 2:
  204. StackPush(cpu, cpu.__PC & 0xFF); break;
  205. case 3:
  206. StackPush(cpu, cpu.__SP); break;
  207. case 4:
  208. cpu.__mem.address = (nmi) ? 0xFFFC : 0xFFFE;
  209. cpu.__PC = cpu.__mem.byte;
  210. break;
  211. case 5:
  212. cpu.__mem.address = (nmi) ? 0xFFFD : 0xFFFF;
  213. cpu.__PC |= cpu.__mem.byte << 8;
  214. cpu.B = 1;
  215. cpu.__op = -1;
  216. break;
  217. }
  218. cpu.__cycle += 1;
  219. }
  220. function CMP(cpu){
  221. let pmode = [0xC9, 0xC5, 0xD5, null, 0xCD, 0xDD, 0xD9, 0xC1, 0xD1, null].indexOf(cpu.__op);
  222. if (ProcessOp(cpu, pmode) === true){
  223. cpu.__op = -1;
  224. cpu.C = (cpu.__AR >= cpu.__opv);
  225. cpu.Z = (this.__AR === cpu.__opv);
  226. let v = this.__AR - cpu.__opv;
  227. if (v < 0){v += 256;}
  228. cpu.N = BITM.val(v, 7);
  229. }
  230. }
  231. function CPX(cpu){
  232. let pmode = [0xE0, 0xE4, null, null, 0xEC, null, null, null, null, null].indexOf(cpu.__op);
  233. if (ProcessOp(cpu, pmode) === true){
  234. cpu.__op = -1;
  235. cpu.C = (cpu.__XR >= cpu.__opv);
  236. cpu.Z = (this.__XR === cpu.__opv);
  237. let v = this.__XR - cpu.__opv;
  238. if (v < 0){v += 256;}
  239. cpu.N = BITM.val(v, 7);
  240. }
  241. }
  242. function CPY(cpu){
  243. let pmode = [0xC0, 0xC4, null, null, 0xCC, null, null, null, null, null].indexOf(cpu.__op);
  244. if (ProcessOp(cpu, pmode) === true){
  245. cpu.__op = -1;
  246. cpu.C = (cpu.__YR >= cpu.__opv);
  247. cpu.Z = (this.__YR === cpu.__opv);
  248. let v = this.__YR - cpu.__opv;
  249. if (v < 0){v += 256;}
  250. cpu.N = BITM.val(v, 7);
  251. }
  252. }
  253. function DEC(cpu){
  254. let pmode = [null, 0xC6, 0xD6, null, 0xCE, 0xDE, null, null, null, null].indexOf(cpu.__op);
  255. if (cpu.__ophold === 1){pmode = -1;}
  256. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  257. if (cpu.__ophold === 0){
  258. cpu.__opv = ByteWrap(cpu.__opv, -1);
  259. cpu.__ophold = 1;
  260. } else {
  261. cpu.__mem.byte = cpu.__opv;
  262. cpu.Z = (this.__opv === 0);
  263. cpu.N = BITM.val(this.__opv, 7);
  264. cpu.__op = -1;
  265. cpu.__ophold = 0;
  266. }
  267. }
  268. }
  269. function EOR(cpu){
  270. let pmode = [0x49, 0x45, 0x55, null, 0x4D, 0x5D, 0x59, 0x41, 0x51, null].indexOf(cpu.__op);
  271. if (ProcessOp(cpu, pmode) === true){
  272. cpu.__op = -1;
  273. cpu.__AR ^= cpu.__opv;
  274. cpu.Z = (cpu.__AR === 0);
  275. cpu.N = BITM.val(cpu.__AR, 7);
  276. }
  277. }
  278. function FLAG(cpu){
  279. switch (cpu.__op){
  280. case 0x18: // CLC
  281. cpu.C = 0; break;
  282. case 0x38: // SEC
  283. cpu.C = 1; break;
  284. case 0x58: // CLI
  285. cpu.I = 0; break;
  286. case 0x78: // SEI
  287. cpu.I = 1; break;
  288. case 0xB8: // CLV
  289. cpu.V = 0; break;
  290. case 0xD8: // CLD
  291. cpu.D = 0; break;
  292. case 0xF8: // SED
  293. cpu.D = 1; break;
  294. }
  295. cpu.__op = -1;
  296. }
  297. function INC(cpu){
  298. let pmode = [null, 0xE6, 0xF6, null, 0xEE, 0xFE, null, null, null, null].indexOf(cpu.__op);
  299. if (cpu.__ophold === 1){pmode = -1;}
  300. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  301. if (cpu.__ophold === 0){
  302. cpu.__opv = ByteWrap(cpu.__opv, 1);
  303. cpu.__ophold = 1;
  304. } else {
  305. cpu.__mem.byte = cpu.__opv;
  306. cpu.Z = (this.__opv === 0);
  307. cpu.N = BITM.val(this.__opv, 7);
  308. cpu.__op = -1;
  309. cpu.__ophold = 0;
  310. }
  311. }
  312. }
  313. function JMP(cpu){
  314. switch(cpu.__cycle){
  315. case 0:
  316. cpu.__mem.address = cpu.__PC;
  317. cpu.__opv = cpu.__mem.byte;
  318. PCUp(cpu, 1); break;
  319. case 1:
  320. cpu.__mem.address = cpu.__PC;
  321. cpu.__opv |= cpu.__mem.byte << 8;
  322. if (cpu.__op === 0x4C){
  323. cpu.__PC = cpu.__opv;
  324. cpu.__op = -1;
  325. }
  326. break;
  327. case 2:
  328. cpu.__mem.address = cpu.__opv;
  329. cpu.__PC = cpu.__mem.byte;
  330. break;
  331. case 3:
  332. cpu.__mem.address += 1;
  333. cpu.__PC |= cpu.__mem.byte << 8;
  334. cpu.__op = -1;
  335. break;
  336. }
  337. cpu.__cycle += 1;
  338. }
  339. function JRS(cpu){
  340. switch(cpu.__cycle){
  341. case 0:
  342. cpu.__mem.address = cpu.__PC;
  343. cpu.__opv = cpu.__mem.byte;
  344. PCUp(cpu, 1); break;
  345. case 1:
  346. // Discard stack data (or... No Op)
  347. break;
  348. case 2:
  349. StackPush(cpu, cpu.__PC >> 8); break;
  350. case 3:
  351. StackPush(cpu, cpu.__PC & 0xFF); break;
  352. case 4:
  353. cpu.__mem.address = cpu.__PC;
  354. cpu.__PC = cpu.__opv | (cpu.__mem.byte << 8)
  355. cpu.__op = -1;
  356. break;
  357. }
  358. cpu.__cycle += 1;
  359. }
  360. function LDA(cpu){
  361. let pmode = [0xA9, 0xA5, 0xB5, null, 0xAD, 0xBD, 0xB9, 0xA1, 0xB1, null].indexOf(cpu.__op);
  362. if (ProcessOp(cpu, pmode) === true){
  363. cpu.__op = -1;
  364. cpu.__AR = cpu.__opv;
  365. cpu.Z = (cpu.__AR === 0);
  366. cpu.N = (cpu.__AR >= 0x80);
  367. }
  368. }
  369. function LDX(cpu){
  370. let pmode = [0xA2, 0xA6, null, 0xB6, 0xAE, null, 0xBE, null, null, null].indexOf(cpu.__op);
  371. if (ProcessOp(cpu, pmode) === true){
  372. cpu.__op = -1;
  373. cpu.__XR = cpu.__opv;
  374. cpu.Z = (cpu.__XR === 0);
  375. cpu.N = (cpu.__XR >= 0x80);
  376. }
  377. }
  378. function LDY(cpu){
  379. let pmode = [0xA0, 0xA4, 0xB4, null, 0xAC, 0xBC, null, null, null, null].indexOf(cpu.__op);
  380. if (ProcessOp(cpu, pmode) === true){
  381. cpu.__op = -1;
  382. cpu.__YR = cpu.__opv;
  383. cpu.Z = (cpu.__YR === 0);
  384. cpu.N = (cpu.__YR >= 0x80);
  385. }
  386. }
  387. function LSR(cpu){
  388. let pmode = [null, 0x46, 0x56, null, 0x4E, 0x5E, null, null, null, 0x4A].indexOf(cpu.__op);
  389. if (cpu.__ophold === 1){pmode = -1;}
  390. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  391. if (cpu.__ophold === 0){
  392. cpu.__opv = (mode === 9) ? cpu.__AR : cpu.__opv;
  393. cpu.C = BITM.val(cpu.__opv, 0);
  394. cpu.__opv = cpu.__opv >> 1;
  395. cpu.__ophold = 1;
  396. } else {
  397. if (mode === 9){
  398. cpu.__AR = cpu.__opv;
  399. } else {
  400. cpu.__mem.byte = cpu.__opv;
  401. }
  402. // TODO: Is Z and N effected by the A register, or by the resulting value? I think the latter.
  403. cpu.Z = (this.__opv === 0);
  404. cpu.N = BITM.val(this.__opv, 7);
  405. cpu.__op = -1;
  406. cpu.__ophold = 0;
  407. }
  408. }
  409. }
  410. function ORA(cpu){
  411. let pmode = [0x09, 0x05, 0x15, null, 0x0D, 0x1D, 0x19, 0x01, 0x11, null].indexOf(cpu.__op);
  412. if (ProcessOp(cpu, pmode) === true){
  413. cpu.__op = -1;
  414. cpu.__AR |= cpu.__opv;
  415. cpu.Z = (cpu.__AR === 0);
  416. cpu.N = BITM.val(cpu.__AR, 7);
  417. }
  418. }
  419. function REGISTER(cpu){
  420. let t = 0;
  421. switch(this.__op){
  422. case 0xAA: // TAX
  423. cpu.__XR = cpu.__AR;
  424. t = cpu.__XR;
  425. break;
  426. case 0x8A: // TXA
  427. cpu.__AR = cpu.__XR;
  428. t = cpu.__AR;
  429. break;
  430. case 0xCA: // DEX
  431. cpu.__XR = (cpu.__XR === 0) ? 255 : cpu.__XR - 1;
  432. t = cpu.__XR;
  433. break;
  434. case 0xE8: // INX
  435. cpu.__XR = (cpu.__XR === 255) ? 0 : cpu.__XR + 1;
  436. t = cpu.__XR;
  437. break;
  438. case 0xA8: // TAY
  439. cpu.__YR = cpu.__AR;
  440. t = cpu.__YR;
  441. break;
  442. case 0x98: // TYA
  443. cpu.__AR = cpu.__YR;
  444. t = cpu.__AR;
  445. break;
  446. case 0x88: // DEY
  447. cpu.__YR = (cpu.__YR === 0) ? 255 : cpu.__YR - 1;
  448. t = cpu.__YR;
  449. break;
  450. case 0xC8: // INY
  451. cpu.__YR = (cpu.__YR === 255) ? 0 : cpu.__YR + 1;
  452. t = cpu.__YR;
  453. break;
  454. }
  455. cpu.N = (t >= 0x80);
  456. cpu.Z = (t === 0);
  457. cpu.__op = -1;
  458. }
  459. function ROL(cpu){
  460. let pmode = [null, 0x26, 0x36, null, 0x2E, 0x3E, null, null, null, 0x2A].indexOf(cpu.__op);
  461. if (cpu.__ophold === 1){pmode = -1;}
  462. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  463. if (cpu.__ophold === 0){
  464. let v = cpu.C;
  465. cpu.__opv = (mode === 9) ? cpu.__AR : cpu.__opv;
  466. cpu.C = BITM.val(cpu.__opv, 7);
  467. cpu.__opv = ((cpu.__opv << 1) + v) & 0xFF;
  468. cpu.__ophold = 1;
  469. } else {
  470. if (mode === 9){
  471. cpu.__AR = cpu.__opv;
  472. } else {
  473. cpu.__mem.byte = cpu.__opv;
  474. }
  475. cpu.Z = (this.__opv === 0);
  476. cpu.N = BITM.val(this.__opv, 7);
  477. cpu.__op = -1;
  478. cpu.__ophold = 0;
  479. }
  480. }
  481. }
  482. function ROR(cpu){
  483. let pmode = [null, 0x66, 0x76, null, 0x6E, 0x7E, null, null, null, 0x6A].indexOf(cpu.__op);
  484. if (cpu.__ophold === 1){pmode = -1;}
  485. if (cpu.__ophold === 1 || ProcessOp(cpu, pmode) === true){
  486. if (cpu.__ophold === 0){
  487. let v = cpu.C;
  488. cpu.__opv = (mode === 9) ? cpu.__AR : cpu.__opv;
  489. cpu.C = BITM.val(cpu.__opv, 0);
  490. cpu.__opv = ((cpu.__opv >> 1) + (v * 0x80)) & 0xFF;
  491. cpu.__ophold = 1;
  492. } else {
  493. if (mode === 9){
  494. cpu.__AR = cpu.__opv;
  495. } else {
  496. cpu.__mem.byte = cpu.__opv;
  497. }
  498. cpu.Z = (this.__opv === 0);
  499. cpu.N = BITM.val(this.__opv, 7);
  500. cpu.__op = -1;
  501. cpu.__ophold = 0;
  502. }
  503. }
  504. }
  505. function RTI(cpu){
  506. switch(cpu.__cycle){
  507. case 0: break; // Discard PC
  508. case 1: // Discard Stack Pointer Data
  509. StackPop(cpu); break;
  510. case 2:
  511. cpu.__PR = StackPop(cpu); break;
  512. case 3:
  513. cpu.__PC = StackPop(cpu); break;
  514. case 4:
  515. cpu.__PC |= StackPop(cpu) << 8;
  516. cpu.__op = -1;
  517. break;
  518. }
  519. cpu.__cycle += 1;
  520. }
  521. function RTS(cpu){
  522. switch(cpu.__cycle){
  523. case 0: break; // Discard PC
  524. case 1:
  525. StackPop(cpu); break;
  526. case 2:
  527. cpu.__PC = StackPop(cpu); break;
  528. case 3:
  529. cpu.__PC |= StackPop(cpu) << 8; break;
  530. case 4:
  531. PCUp(cpu, 1);
  532. cpu.__op = -1;
  533. break;
  534. }
  535. cpu.__cycle += 1;
  536. }
  537. function SBC(cpu){
  538. let pmode = [0xE9, 0xE5, 0xF5, null, 0xED, 0xFD, 0xF9, 0xE1, 0xF1].indexOf(cpu.__op);
  539. if (ProcessOp(cpu, pmode) === true){
  540. cpu.__op = -1;
  541. ALU(cpu, 1, cpu.__opv);
  542. }
  543. }
  544. function STA(cpu){
  545. let pmode = [null, 0x85, 0x95, null, 0x8D, 0x9D, 0x99, 0x81, 0x91, null].indexOf(cpu.__op);
  546. if (ProcessOp(cpu, pmode) === true){
  547. cpu.__op = -1;
  548. cpu.__mem.address = cpu.__opv;
  549. cpu.__mem.byte = cpu.__AR;
  550. }
  551. }
  552. function STACK(cpu){
  553. switch(cpu.__op){
  554. case 0x9A: // TXS
  555. cpu.__SP = cpu.__XR;
  556. cpu.__op = -1;
  557. break;
  558. case 0xBA: // TSX
  559. cpu.__XR = cpu.__SP;
  560. cpu.__op = -1;
  561. break;
  562. case 0x48: // PHA
  563. if (cpu.__cycle === 0){
  564. cpu.__mem.address = 0x0100 | cpu.__SP;
  565. } else if (cpu.__cycle === 1){
  566. cpu.__mem.byte = cpu.__AR;
  567. cpu.__SP = (cpu.__SP === 0) ? 255 : cpu.__SP - 1;
  568. cpu.__op = -1;
  569. }
  570. break;
  571. case 0x68: // PLA
  572. if (cpu.__cycle === 0){
  573. cpu.__mem.address = 0x0100 | cpu.__SP;
  574. } else if (cpu.__cycle === 1){
  575. cpu.__AR = cpu.__mem.byte;
  576. cpu.Z = (cpu.__AR === 0);
  577. cpu.N = BITM.isOn(cpu.__AR, 7);
  578. } else if (cpu.__step === 2){
  579. cpu.__SP = (cpu.__SP === 255) ? 0 : cpu.__SP + 1;
  580. cpu.__op = -1;
  581. }
  582. break;
  583. case 0x08: // PHP
  584. if (cpu.__cycle === 0){
  585. cpu.__mem.address = 0x0100 | cpu.__SP;
  586. } else if (cpu.__cycle === 1){
  587. cpu.__mem.byte = cpu.__PR;
  588. cpu.__SP = (cpu.__SP === 0) ? 255 : cpu.__SP - 1;
  589. cpu.__op = -1;
  590. }
  591. break;
  592. case 0x28: // PLP
  593. if (cpu.__cycle === 0){
  594. cpu.__mem.address = 0x0100 | cpu.__SP;
  595. } else if (cpu.__cycle === 1){
  596. cpu.__SP = cpu.__mem.byte;
  597. } else if (cpu.__cycle === 2){
  598. cpu.__SP = (cpu.__SP === 255) ? 0 : cpu.__SP + 1;
  599. cpu.__op = -1;
  600. }
  601. break;
  602. }
  603. cpu.__cycle += 1;
  604. }
  605. function STX(cpu){
  606. let pmode = [null, 0x86, null, 0x96, 0x8E, null, null, null, null, null].indexOf(cpu.__op);
  607. if (ProcessOp(cpu, pmode) === true){
  608. cpu.__op = -1;
  609. cpu.__mem.address = cpu.__opv;
  610. cpu.__mem.byte = cpu.__XR;
  611. }
  612. }
  613. function STY(cpu){
  614. let pmode = [null, 0x84, 0x94, null, 0x8C, null, null, null, null, null].indexOf(cpu.__op);
  615. if (ProcessOp(cpu, pmode) === true){
  616. cpu.__op = -1;
  617. cpu.__mem.address = cpu.__opv;
  618. cpu.__mem.byte = cpu.__YR;
  619. }
  620. }
  621. // --------------------------------------------------------------------------------------------
  622. // Test to see if both a and b's high byte is the same... same page.
  623. function SamePage(a, b){
  624. return ((a >> 8) == (b >> 8));
  625. }
  626. function ByteWrap(v, a){
  627. v += a;
  628. if (v < 0){v += 256;}
  629. else if (v > 255){v -= 256;}
  630. return v;
  631. }
  632. function StackPush(cpu, v){
  633. cpu.__mem.address = 0x0100 | cpu.__SP;
  634. cpu.__mem.byte = v & 0xFF;
  635. cpu.__SP = ByteWrap(cpu.__SP, -1);
  636. }
  637. function StackPop(cpu){
  638. cpu.__mem.address = 0x0100 | cpu.__SP;
  639. cpu.__SP = ByteWrap(cpu.__SP, 1);
  640. return cpu.__mem.byte;
  641. }
  642. function PCHI(cpu, b){
  643. cpu.__PC = (cpu.__PC & 0x00FF) | (b << 8);
  644. }
  645. function PCLOW(cpu, b){
  646. cpu.__PC = (cpu.__PC & 0xFF00) | b;
  647. }
  648. function PCUp(cpu, amount){
  649. if (cpu.__pcc === 1){
  650. PCHI(cpu, ((cpu.__PC & 0xFF00) >> 8) + 1);
  651. cpu.__pcc = 0;
  652. } else if ((cpu.__PC & 0x00FF) + amount > 255){
  653. cpu.__pcc = 1;
  654. PCLOW(cpu, ((cpu.__PC & 0x00FF) + amount) - 256);
  655. } else {
  656. PCLOW(cpu, (cpu.__PC & 0x00FF) + amount);
  657. }
  658. }
  659. function PCDown(cpu, amount){
  660. if (cpu.__pcc === -1){
  661. PCHI(cpu, ((cpu.__PC & 0xFF00) >> 8) - 1);
  662. cpu.__pcc = 0;
  663. } else if ((cpu.__PC & 0x00FF) - amount < 0){
  664. cpu.__pcc = -1;
  665. PCLOW(cpu, ((cpu.__PC & 0x00FF) - amount) + 256);
  666. } else {
  667. PCLOW(cpu, (cpu.__PC & 0x00FF) - amount);
  668. }
  669. }
  670. function MemAddrFrom(cpu, addr){
  671. cpu.__mem.address = addr;
  672. let v = cpu.__mem.byte;
  673. cpu.__mem.address = (addr + 1) & 0xFF;
  674. v |= cpu.__mem.byte << 8;
  675. return v;
  676. }
  677. function ALU(cpu, m, b){
  678. switch(m){
  679. case 0: // Addition
  680. cpu.__AR = (cpu.__AR + b) + ((cpu.C === 1) ? 1 : 0);
  681. //cpu.C = (this.__AR >= 256);
  682. break;
  683. case 1: // Subtraction
  684. cpu.__AR = (cpu.__AR - b) - ((cpu.C === 0) ? 1 : 0);
  685. //cpu.C = (this.__AR >= 0);
  686. cpu.__AR += (cpu.__AR < 0) ? 256 : 0;
  687. break;
  688. }
  689. cpu.V = (BITM.isOn(cpu.__AR, 7) === BITM.isOn(b, 7)) && (BITM.val(cpu.__AR, 7) !== BITM.val(cpu.__AR, 7));
  690. cpu.C = (m === 0) ? (this.__AR >= 256) : (cpu.V === 0);
  691. cpu.N = BITM.val(cpu.__AR, 7);
  692. cpu.__AR &= 0xFF;
  693. cpu.Z = (cpu.__AR === 0);
  694. return cpu.__AR;
  695. }
  696. // --------------------------------------------------------------------------------------------
  697. // --------------------------------------------------------------------------------------------
  698. // --------------------------------------------------------------------------------------------
  699. class CPU{
  700. constructor(){
  701. // Registers
  702. this.__PC = 0; // Program Counter (16 bit)
  703. this.__SP = 255; // Stack Pointer (8 bit)
  704. this.__PR = 32; // Status Register (8 bit, bit 5 is always 1)
  705. this.__XR = 0; // X Register (8 bit)
  706. this.__YR = 0; // Y Register (8 bit)
  707. this.__AR = 0; // Accumulator Register (8 bit)
  708. // Variables to watch for Hardware Interrupts.
  709. this.__nmi = false;
  710. this.__irq = false;
  711. this.__rst = true;
  712. this.__iinit = false;
  713. // Variable for tracking tick operations.
  714. this.__op = -1;
  715. this.__ophold = 0;
  716. this.__opv = 0;
  717. this.__cycle = 0;
  718. this.__pcc = 0; // Program Counter Carry.
  719. // Memory module or controller.
  720. this.__mem = null; // Must be explicitly attached.
  721. // Hold any created CLK instances.
  722. this.__clkfn = null;
  723. }
  724. // ----------------------------------------
  725. // CPU Registers. Here for debug purposes.
  726. get PC(){return this.__PC;}
  727. get SP(){return this.__SP;}
  728. get P(){return this.__PR;}
  729. get X(){return this.__XR;}
  730. get Y(){return this.__YR;}
  731. get A(){return this.__AR;}
  732. // ----------------------------------------
  733. // Quick Flag Access
  734. get N(){return (BITM.isOn(this.__PR, 7)) ? 1 : 0;}
  735. set N(n){this.__PR = (n === true || n === 1) ? BITM.set(this.__PR, 7) : BITM.clear(this.__PR, 7);}
  736. get V(){return (BITM.isOn(this.__PR, 6)) ? 1 : 0;}
  737. set V(v){this.__PR = (v === true || v === 1) ? BITM.set(this.__PR, 6) : BITM.clear(this.__PR, 6);}
  738. get B(){return (BITM.isOn(this.__PR, 4)) ? 1 : 0;}
  739. set B(b){this.__PR = (b === true || b === 1) ? BITM.set(this.__PR, 4) : BITM.clear(this.__PR, 4);}
  740. get D(){return (BITM.isOn(this.__PR, 3)) ? 1 : 0;}
  741. set D(d){this.__PR = (d === true || d === 1) ? BITM.set(this.__PR, 3) : BITM.clear(this.__PR, 3);}
  742. get I(){return (BITM.isOn(this.__PR, 2)) ? 1 : 0;}
  743. set I(i){this.__PR = (i === true || i === 1) ? BITM.set(this.__PR, 2) : BITM.clear(this.__PR, 2);}
  744. get Z(){return (BITM.isOn(this.__PR, 1)) ? 1 : 0;}
  745. set Z(z){this.__PR = (z === true || z === 1) ? BITM.set(this.__PR, 1) : BITM.clear(this.__PR, 1);}
  746. get C(){return (BITM.isOn(this.__PR, 0)) ? 1 : 0;}
  747. set C(c){this.__PR = (c === true || c === 1) ? BITM.set(this.__PR, 0) : BITM.clear(this.__PR, 0);}
  748. // ----------------------------------------
  749. // Hardware interrupt triggers. Settable only.
  750. set NMI(n){
  751. this.__nmi = (n === true);
  752. this.__iinit = true;
  753. }
  754. set IRQ(q){
  755. if (BITM.val(this.__PR, 2) === 0 && this.__nmi === false){
  756. this.__irq = (q === true);
  757. this.__iinit = true;
  758. }
  759. }
  760. set reset(r){
  761. this.__rst = (r === true);
  762. }
  763. // -----------------------------------------
  764. // Set and Get Memory property.
  765. get memory(){return this.__mem;}
  766. set memory(m){
  767. if (!(m instanceof IMem))
  768. throw new ValueError("Expected Memory instance object.");
  769. this.__mem = m;
  770. }
  771. // -----------------------------------------
  772. clk(){
  773. if (this.__clkfn === null){
  774. this.__clkfn = (function(){
  775. if (this.__pcc !== 0){
  776. if (this.__pcc > 0) {
  777. PCUp(this, 1);
  778. } else {
  779. PCDown(this, 1);
  780. }
  781. } else if (this.__op < 0){
  782. if (this.__rst){
  783. this.__rst = false;
  784. this.__mem.address = 0xFFFC;
  785. this.__PC = this.__mem.byte;
  786. this.__mem.address = 0xFFFD;
  787. this.__PC |= this.__mem.byte << 8;
  788. // Disabling the IRQ interrupts is the ONLY flag that must be set
  789. // during reset. The others are random.
  790. this.I = 1;
  791. } else if (this.__nmi) {
  792. if (this.__iinit){
  793. this.__cycle = 0;
  794. this.__iinit = false;
  795. }
  796. BRK(this, true); // BRK does all of the interrupt work. The 'true' tells BRK to use the nmi vectors.
  797. if (this.__cycle === 6)
  798. this.__nmi = false;
  799. } else if (this.__irq) {
  800. if (this.I === 0){
  801. if (this.__iinit){
  802. this.__cycle = 0;
  803. this.__iinit = false;
  804. }
  805. BRK(this); // Because both the BRK opcode and an IRQ interrupt do the same things!
  806. if (this.__cycle === 6)
  807. this.__irq = false;
  808. } else {
  809. // NOPE... try again!
  810. this.__irq = false;
  811. this.__clkfn();
  812. }
  813. } else {
  814. this.__cycle = 0;
  815. this.__mem.address = this.__PC;
  816. this.__op = this.__mem.byte;
  817. PCUp(this, 1);
  818. }
  819. } else {
  820. switch(this.__op){
  821. case 0x69: case 0x65: case 0x75: case 0x6D: case 0x7D: case 0x79: case 0x61: case 0x71:
  822. ADC(this); break;
  823. case 0x29: case 0x25: case 0x35: case 0x2D: case 0x3D: case 0x39: case 0x21: case 0x31:
  824. AND(this); break;
  825. case 0x0A: case 0x06: case 0x16: case 0x0E: case 0x1E:
  826. ASL(this); break;
  827. case 0x24: case 0x2C:
  828. BIT(this); break;
  829. case 0x10: case 0x30: case 0x50: case 0x70: case 0x90: case 0xB0: case 0xD0: case 0xF0:
  830. BRANCH(this); break;
  831. case 0x00:
  832. BRK(this); break;
  833. case 0xC9: case 0xC5: case 0xD5: case 0xCD: case 0xDD: case 0xD9: case 0xC1: case 0xD1:
  834. CMP(this); break;
  835. case 0xE0: case 0xE4: case 0xEC:
  836. CPX(this); break;
  837. case 0xC0: case 0xC4: case 0xCC:
  838. CPY(this); break;
  839. case 0xC6: case 0xD6: case 0xCE: case 0xDE:
  840. DEC(this); break;
  841. case 0x49: case 0x45: case 0x55: case 0x4D: case 0x5D: case 0x59: case 0x41: case 0x51:
  842. EOR(this); break;
  843. case 0x18: case 0x38: case 0x58: case 0x78: case 0xB8: case 0xD8: case 0xF8:
  844. FLAG(this); break;
  845. case 0xE6: case 0xF6: case 0xEE: case 0xFE:
  846. INC(this); break;
  847. case 0x4C: case 0x6C:
  848. JMP(this); break;
  849. case 0x20:
  850. JSR(this); break;
  851. case 0xA9: case 0xA5: case 0xB5: case 0xAD: case 0xBD: case 0xB9: case 0xA1: case 0xB1:
  852. LDA(this); break;
  853. case 0xA2: case 0xA6: case 0xB6: case 0xAE: case 0xBE:
  854. LDX(this); break;
  855. case 0xA0: case 0xA4: case 0xB4: case 0xAC: case 0xBC:
  856. LDY(this); break;
  857. case 0x4A: case 0x46: case 0x56: case 0x4E: case 0x5E:
  858. LSR(this); break;
  859. case 0xEA:
  860. // NOP
  861. if (this.__step == 1){
  862. this.__op = -1;
  863. } else {this.__step += 1;}
  864. break;
  865. case 0x09: case 0x05: case 0x15: case 0x0D: case 0x1D: case 0x19: case 0x01: case 0x11:
  866. ORA(this); break;
  867. case 0xAA: case 0x8A: case 0xCA: case 0xE8: case 0xA8: case 0x98: case 0x88: case 0xC8:
  868. REGISTER(this); break;
  869. case 0x2A: case 0x26: case 0x36: case 0x2E: case 0x3E:
  870. ROL(this); break;
  871. case 0x6A: case 0x66: case 0x76: case 0x6E: case 0x7E:
  872. ROR(this); break;
  873. case 0x40:
  874. RTI(this); break;
  875. case 0x60:
  876. RTS(this); break;
  877. case 0xE9: case 0xE5: case 0xF5: case 0xED: case 0xFD: case 0xF9: case 0xE1: case 0xF1:
  878. SBC(this); break;
  879. case 0x85: case 0x95: case 0x8D: case 0x9D: case 0x99: case 0x81: case 0x91:
  880. STA(this); break;
  881. case 0x9A: case 0xBA: case 0x48: case 0x68: case 0x08: case 0x28:
  882. STACK(this); break;
  883. case 0x86: case 0x96: case 0x8E:
  884. STX(this); break;
  885. case 0x84: case 0x94: case 0x8C:
  886. STY(this); break;
  887. }
  888. }
  889. }).bind(this);
  890. }
  891. return this.__clkfn;
  892. }
  893. }
  894. module.exports = CPU;