Fantasy 8Bit system (F8), is a fantasy 8bit console and a set of libraries for creating fantasy 8bit consoles.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

105 lines
2.6KB

  1. var Memory = require('src/memory');
  2. class Switch{
  3. constructor(mem){
  4. this.__mem = [mem];
  5. this.__idx = 0;
  6. }
  7. get mem(){return this.__mem[this.__idx];}
  8. get idx(){return this.__idx;}
  9. set idx(i){
  10. if (i >= 0 && i < this.__mem.length)
  11. this.__idx = i;
  12. }
  13. addMemModule(mem){
  14. if (!(mem instanceof Memory))
  15. throw new ValueError("Only Memory instances can be added to MMC Banks.");
  16. if (this.__mem.length >= 4)
  17. throw new RangeError("Bank handling maximum memory modules.");
  18. if (this.__mem.length > 0 && mem.size !== this.__mem[0].size)
  19. throw new RangeError("Memory module does not match size of already connected memory modules on this bank.");
  20. this.__mem.push(mem);
  21. }
  22. }
  23. class MMC extends Memory{
  24. constructor(){
  25. super();
  26. this.__switches = [];
  27. this.__addr = 0;
  28. this.__sidx = 0;
  29. }
  30. get size(){
  31. return this.__switches.reduce((acc, s)=>{
  32. acc += s.mem.size;
  33. }, 0);
  34. }
  35. get switches(){return this.__switches.length;}
  36. get address(){return this.__addr;}
  37. set address(a){
  38. if (a >= 0 && a < this.size){
  39. this.__addr = a;
  40. offset = 0;
  41. for (let s=0; s < this.__switches.length; s++){
  42. if (a >= offset && a < offset + this.__switches[s].mem.size){
  43. this.__sidx = b;
  44. this.__switches[s].mem.address = a - offset;
  45. break;
  46. } else {
  47. offset += this.__switches[s].mem.size;
  48. }
  49. }
  50. }
  51. }
  52. get byte(){return (this.__switches.length > 0) ? this.__switches[this.__sidx].mem.byte : -1;}
  53. set byte(b){
  54. if (this.__switches.length > 0){
  55. this.__switches[this.__sidx].mem.byte = b;
  56. }
  57. }
  58. connectMemory(mem, addroff){
  59. addroff = (typeof(addroff) === 'number' && addroff >= 0) ? addroff : -1;
  60. if (addroff < 0 || addroff === this.size){
  61. this.__switches.push(new Switch(mem));
  62. } else {
  63. offset = 0;
  64. for (let s=0; s < this.__switches.length; s++){
  65. if (addroff === offset){
  66. if (this.__switches[s].mem.size !== mem.size)
  67. throw new RangeError("Memory modules assigned to the same bank must be the same byte size.");
  68. this.__switches[s].addMemModule(mem);
  69. offset = -1;
  70. break;
  71. } else {
  72. offset += this.__switches[s].mem.size;
  73. }
  74. }
  75. if (offset >= 0)
  76. throw new RangeError("Cannot align memory module to bank at address " + addroff);
  77. }
  78. return this;
  79. }
  80. mmSwitchRegister(){
  81. return (function(byte){
  82. let mmcidx = (byte & 0xF0) >> 4;
  83. if (mmcidx < this.__switches.length){
  84. this.__switches[mmcidx].idx = (byte & 0x0F);
  85. }
  86. }).bind(this);
  87. }
  88. }
  89. module.exports = MMC;