Fantasy 8Bit system (F8), is a fantasy 8bit console and a set of libraries for creating fantasy 8bit consoles.
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

425 lines
11KB

  1. const IO = require('./common/io.js');
  2. const BCD = require('./utils/bcd.js');
  3. const BITM = require('./utils/bitman.js');
  4. function EnableICBit(cia, pos){
  5. v = 1 << pos;
  6. if ((cia.__ICMask & v) === 0){
  7. cia.__IC = cia.__IC | v;
  8. // Enabling any bit enables bit 7 as well!
  9. cia.__IC = cia.__IC | 0x80;
  10. }
  11. }
  12. function TODTick(cia){
  13. if (cia.__TODTick < 0){return;}
  14. cia.__TODTick += 1;
  15. if (cia.__TODTick === 6){
  16. cia.__TODTick = (cia.__CTA & 0x80) >> 7;
  17. if (cia.__TOD[0] == 0x09){
  18. cia.__TOD[0] = 0;
  19. if (cia.__TOD[1] == 0x59){
  20. cia.__TOD[1] = 0;
  21. if (cia.__TOD[2] == 0x59){
  22. cia.__TOD[2] = 0;
  23. if (cia.__TOD[3] == 0x11 || cia.__TOD[3] == 0x91){
  24. cia.__TOD[3] = (cia.__TOD[3] == 0x91) ? 0x00 : 0x80;
  25. } else {
  26. let pm = cia.__TOD[3] & 0x80;
  27. cia.__TOD[3] = BCD.add(cia.__TOD[3] & 0x7F, 0x01, 2) | pm;
  28. }
  29. } else {cia.__TOD[2] = BCD.add(cia.__TOD[2], 0x01, 2);}
  30. } else {cia.__TOD[1] = BCD.add(cia.__TOD[1], 0x01, 2);}
  31. } else {cia.__TOD[0] = BCD.add(cia.__TOD[0], 0x01, 2);}
  32. }
  33. if (cia.__TOD[0] == cia.__ALTOD[0] && cia.__TOD[1] == cia.__ALTOD[1] && cia.__TOD[2] == cia.__ALTOD[2] && cia.__TOD[3] == cia.__ALTOD[3]){
  34. EnableICBit(cia, 2);
  35. }
  36. }
  37. function TODLatch(cia){
  38. if (cia.__TODLatch === null){
  39. cia.__TODLatch = [
  40. cia.__TOD[0],
  41. cia.__TOD[1],
  42. cia.__TOD[2],
  43. cia.__TOD[3]
  44. ];
  45. }
  46. }
  47. function TimerATick(cia){
  48. if (cia.__timerA > 0){
  49. cia.__timerA -= 1;
  50. } else {
  51. // Check if Timer A is continuous
  52. if (BITM.val(cia.__CTA, 3) === 0){
  53. cia.__timerA = cia.__timerALatch;
  54. } else {
  55. // Not continuous, so clear bit 0 (stop timer).
  56. cia.__CTA = BITM.clear(cia.__CTA, 0);
  57. }
  58. // Setting the interrupt bit for timer A
  59. EnableICBit(cia, 0);
  60. //cia.__IC = BITM.set(cia.__IC, 0);
  61. // Check to see if timer modifies the PORT B bit 6.
  62. // If so, determine if it's a pulse or an toggle.
  63. if (BITM.isOn(cia.__CTA, 1))
  64. cia.__PDB = (BITM.isOn(cia.__CTA, 2) === 0) ? BITM.set(cia.__PDB, 6) : BITM.toggle(cia.__PDB, 6);
  65. // If timer B is running, and timer B tracks timer A underflows
  66. // tick timer B (or, tick it if CNT is positive)
  67. if (BITM.isOn(cia.__CTB, 0)){
  68. let cs = (this.__CTB & 0x60) >> 5;
  69. if (cs === 2 || (cs === 3 && cia.__CNT === 1))
  70. TimerBTick(cia);
  71. }
  72. }
  73. }
  74. function TimerBTick(cia){
  75. if (cia.__timerB > 0){
  76. cia.__timerB -= 1;
  77. } else {
  78. if (BITM.val(cia.__CTB, 3) === 0){
  79. cia.__timerB = cia.__timerBLatch;
  80. } else {
  81. cia.__CTB = BITM.clear(cia.__CTB, 0);
  82. }
  83. EnableICBit(cia, 1);
  84. //cia.__IC = BITM.set(cia.__IC, 1);
  85. if (BITM.isOn(cia.__CTB, 1))
  86. cia.__PDB = (BITM.isOn(cia.__CTB, 2) === 0) ? BITM.set(cia.__PDB, 7) : BITM.toggle(cia.__PDB, 7);
  87. }
  88. }
  89. function Tick(cia){
  90. if (BITM.isOn(cia.__CTA, 0)){
  91. if (BITM.isOn(cia.__CTA, 1) && !BITM.isOn(cia.__CTA, 2))
  92. cia.__PDB = BITM.clear(cia.__PDB, 6);
  93. if (!BITM.isOn(cia.__CTA, 5)){
  94. TimerATick(cia);
  95. }
  96. }
  97. if (BITM.isOn(cia.__CTB, 0)){
  98. if (BITM.isOn(cia.__CTB, 1) && !BITM.isOn(cia.__CTB, 2))
  99. cia.__PDB = BITM.clear(cia.__PDB, 7);
  100. let cs = (cia.__CTB & 0x60) >> 5;
  101. if (cs === 0)
  102. TimerBTick(cia);
  103. }
  104. if (cia.__CNT === 1){
  105. if (BITM.isOn(cia.__CTA, 0) && BITM.isOn(cia.__CTA, 5))
  106. TimerATick(cia);
  107. if (BITM.isOn(cia.__CTB, 0)){
  108. let cs = (cia.__CTB & 0x60) >> 5;
  109. if (cs === 1)
  110. TimerBTick(cia);
  111. }
  112. if (BITM.isOn(cia.__CTA, 6)){
  113. if (cia.__SPi > 0){
  114. cia.__SP = BITM.val(cia.__SData, 7);
  115. cia.__SData = (cia.__SData << 1) & 0xFF;
  116. cia.__SPi -= 1;
  117. if (cia.__SPi === 0)
  118. EnableICBit(cia, 3);
  119. //cia.__IC = BITM.set(cia.__IC, 3);
  120. }
  121. } else {
  122. if (cia.__SPi < 8){
  123. cia.__SData = (cia.__SData << 1) | cia.__SP;
  124. cia.__SPi += 1;
  125. if (cia.__SPi === 8)
  126. EnableICBit(cia, 3);
  127. //cia.__IC = BITM.set(cia.__IC, 3);
  128. }
  129. }
  130. }
  131. cia.__CNT = 0;
  132. }
  133. class MOSCIA{
  134. constructor(){
  135. this.__io = new IO(["PDA", "PDB"], ["PDA", "PDB"]);
  136. this.__regsel = 0;
  137. this.__RW = 0;
  138. this.__CNT = 0;
  139. this.__SP = 0;
  140. this.__SPi = 0;
  141. this.__SData = 0;
  142. this.__PDA = 0;
  143. this.__PDB = 0;
  144. this.__DDA = 0;
  145. this.__DDB = 0;
  146. this.__CTA = 0;
  147. this.__CTB = 0;
  148. this.__IC = 0;
  149. this.__ICMask = 0;
  150. this.__timerA = 0;
  151. this.__timerALatch = 1;
  152. this.__timerB = 0;
  153. this.__timerBLatch = 1;
  154. // Time Of Day
  155. this.__TODTick = 0;
  156. this.__TODLatch = null;
  157. this.__TOD = [0,0,0,0];
  158. this.__ALTOD = [0,0,0,0];
  159. }
  160. get RS(){return this.__regsel;}
  161. set RS(rs){this.__regsel = rs & 0x0F;}
  162. get RW(){return this.__RW;}
  163. set RW(rw){this.__RW = (rw === true || rw === 1) ? 1 : 0;}
  164. get DATA(){
  165. let val = 0;
  166. switch(this.__regsel){
  167. case 0x00: // Peripheral Data A
  168. this.__io.triggerRead("PDA", this.__DDA);
  169. val = this.__PDA; break;
  170. case 0x01: // Peripheral Data B
  171. this.__io.triggerRead("PDB", this.__DDB);
  172. val = this.__PDB; break;
  173. case 0x02: // Data Direction Reg A
  174. val = this.__DDA; break;
  175. case 0x03: // Data Direction Reg B
  176. val = this.__DDB; break;
  177. case 0x04: // Timer A Low
  178. val = (this.__timerA & 0x00FF); break;
  179. case 0x05: // Timer A High
  180. val = ((this.__timerA & 0xFF00) >> 8); break;
  181. case 0x06: // Timer B Low
  182. val = (this.__timerB & 0x00FF); break;
  183. case 0x07: // Timer B High
  184. val = ((this.__timerB & 0xFF00) >> 8); break;
  185. case 0x08: // 10th of Sec Reg
  186. val = (this.__TODLatch !== null) ? this.__TODLatch[0] : this.__TOD[0];
  187. this.__TODLatch = null;
  188. break;
  189. case 0x09: // Seconds Reg
  190. val = (this.__TODLatch !== null) ? this.__TODLatch[1] : this.__TOD[1]; break;
  191. case 0x0A: // Minutes Reg
  192. val = (this.__TODLatch !== null) ? this.__TODLatch[2] : this.__TOD[2]; break;
  193. case 0x0B: // Hours AM/PM Reg
  194. TODLatch(this);
  195. val = this.__TODLatch[3]; break;
  196. case 0x0C: // Serial Data
  197. val = this.__SData; break;
  198. case 0x0D: // Interrupt Control
  199. let v = this.__IC;
  200. this.__IC = 0;
  201. return v; break;
  202. case 0x0E: // Control Timer Reg A
  203. val = this.__CTA; break;
  204. case 0x0F: // Control Timer Reg B
  205. val = this.__CTB; break;
  206. }
  207. return val;
  208. }
  209. set DATA(d){
  210. if (this.__RW === 1){return;}
  211. let tod = 0;
  212. switch(this.__regsel){
  213. case 0x00: // Peripheral Data A
  214. this.__io.triggerWrite("PDA", this.__PDA, d & 0xFF);
  215. this.__PDA = (this.__PDA & (~this.__DDA)) | ((d & 0xFF) & this.__DDA);
  216. break;
  217. case 0x01: // Peripheral Data B
  218. this.__io.triggerWrite("PDB", this.__PDB, d & 0xFF);
  219. this.__PDB = (this.__PDB & (~this.__DDB)) | ((d & 0xFF) & this.__DDB);
  220. break;
  221. case 0x02: // Data Direction Reg A
  222. this.__DDA = d & 0xFF;
  223. break;
  224. case 0x03: // Data Direction Reg B
  225. this.__DDB = d & 0xFF;
  226. break;
  227. case 0x04: // Timer A Low
  228. this.__timerALatch = (this.__timerALatch & 0xFF00) | (d & 0xFF);
  229. break;
  230. case 0x05: // Timer A High
  231. this.__timerALatch = (this.__timerALatch & 0x00FF) | ((d & 0xFF) << 8);
  232. break
  233. case 0x06: // Timer B Low
  234. this.__timerBLatch = (this.__timerBLatch & 0xFF00) | (d & 0xFF);
  235. break;
  236. case 0x07: // Timer B High
  237. this.__timerBLatch = (this.__timerBLatch & 0x00FF) | ((d & 0xFF) << 8);
  238. break;
  239. case 0x08: // 10th of Sec Reg
  240. if (this.__TODTick < 0){
  241. tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
  242. tod[0] = d & 0xFF;
  243. this.__TODTick = (this.__CTA & 0x80) >> 7; // Restarts TOD cycling.
  244. }
  245. break;
  246. case 0x09: // Seconds Reg
  247. if (this.__TODTick < 0){
  248. tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
  249. tod[1] = d & 0xFF;
  250. }
  251. break;
  252. case 0x0A: // Minutes Reg
  253. if (this.__TODTick < 0){
  254. tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
  255. tod[2] = d & 0xFF;
  256. }
  257. break;
  258. case 0x0B: // Hours AM/PM Reg
  259. this.__TODTick = -1; // This will prevent TOD from cycling.
  260. tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
  261. tod[3] = d & 0xFF;
  262. break;
  263. case 0x0C: // Serial Data
  264. break;
  265. case 0x0D: // Interrupt Control
  266. if (BITM.isOn(d & 0xFF, 7)){
  267. for (let i=0; i < 5; i++){
  268. if (BITM.isOn(d & 0xFF, i))
  269. this.__ICMask = BITM.set(this.__ICMask, i);
  270. }
  271. } else {
  272. for (let i=0; i < 5; i++){
  273. if (BITM.isOn(d & 0xFF, i))
  274. this.__ICMask = BITM.clear(this.__ICMask, i);
  275. }
  276. }
  277. break;
  278. case 0x0E: // Control Timer Reg A
  279. this.__CTA = d & 0xFF;
  280. if (BITM.isOn(this.__CTA, 4))
  281. this.__timerA = this.__timerALatch;
  282. // If __TODTick is less than 0, TOD is in write mode.
  283. // As such, the __TODTick counter will be reset once 10ths is
  284. // written to... so there's no need to adjust the ticks here...
  285. // Besides, doing so will break the system.
  286. if (this.__TODTick >= 0){
  287. if (BITM.isOn(this.__CTA, 7)){
  288. this.__TODTick = (this.__TODTick === 5) ? 0 : this.__TODTick + 1;
  289. } else {
  290. this.__TODTick -= (this.__TODTick === 5) ? this.__TODTick : 0;
  291. }
  292. }
  293. break;
  294. case 0x0F: // Control Timer Reg B
  295. this.__CTB = d & 0xFF;
  296. if (BITM.isOn(this.__CTB, 4))
  297. this.__timerB = this.__timerBLatch;
  298. break;
  299. }
  300. }
  301. get CNT(){return this.__CNT;}
  302. set CNT(c){
  303. this.__CNT = (c >= 1) ? 1 : 0;
  304. }
  305. get SP(){return this.__SP;}
  306. set SP(sp){
  307. this.__SP = (sp === true || sp === 1) ? 1 : 0;
  308. }
  309. set FLAG(f){
  310. if (f === true || f === 1)
  311. EnableICBit(this, 4);
  312. }
  313. get TOD(){return 0;}
  314. set TOD(b){
  315. if (b > 0 || this.__TODLatch < 2)
  316. TODTick(this);
  317. }
  318. get phi2(){return 0;}
  319. set phi2(p){
  320. if (p === true || p === 1){
  321. Tick(this);
  322. }
  323. }
  324. onRead(name, fn){
  325. this.__io.onRead(name, fn);
  326. return this;
  327. }
  328. onWrite(name, fn){
  329. this.__io.onWrite(name, fn);
  330. return this;
  331. }
  332. setPDA(v){
  333. this.__PDA = (this.__PDA & this.__DDA) | ((v & 0xFF) & (~this.__DDA));
  334. return this;
  335. }
  336. setPDB(v){
  337. this.__PDB = (this.__PDB & this.__DDB) | ((v & 0xFF) & (~this.__DDB));
  338. return this;
  339. }
  340. setTOD(h,m,s,t){
  341. // This method primarily exists for testing purposes. This should NOT
  342. // be used in a true emulation.
  343. this.__TOD[0] = t & 0xFF;
  344. this.__TOD[1] = s & 0xFF;
  345. this.__TOD[2] = m & 0xFF;
  346. this.__TOD[3] = h & 0xFF;
  347. }
  348. reset(){
  349. this.__CNT = 0;
  350. this.__PDA = 0;
  351. this.__PDB = 0;
  352. this.__DDA = 0;
  353. this.__DDB = 0;
  354. this.__CTA = 0;
  355. this.__CTB = 0;
  356. this.__IC = 0;
  357. this.__ICMask = 0;
  358. this.__timerA = 0;
  359. this.__timerALatch = 1;
  360. this.__timerB = 0;
  361. this.__timerBLatch = 1;
  362. // Time Of Day
  363. this.__TODLatch = 0;
  364. this.__TODTick = 0;
  365. this.__TOD = [0,0,0,0];
  366. this.__LTOD = [0,0,0,0];
  367. this.__ALTOD = [0,0,0,0];
  368. return this;
  369. }
  370. }
  371. module.exports = MOSCIA;