Fantasy 8Bit system (F8), is a fantasy 8bit console and a set of libraries for creating fantasy 8bit consoles.
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

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;