merge mapper code from FCEUX
[fceu.git] / boards / fk23c.c
CommitLineData
386f5371 1/* FCE Ultra - NES/Famicom Emulator\r
2 *\r
3 * Copyright notice for this file:\r
4 * Copyright (C) 2006 CaH4e3\r
5 *\r
6 * This program is free software; you can redistribute it and/or modify\r
7 * it under the terms of the GNU General Public License as published by\r
8 * the Free Software Foundation; either version 2 of the License, or\r
9 * (at your option) any later version.\r
10 *\r
11 * This program is distributed in the hope that it will be useful,\r
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of\r
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
14 * GNU General Public License for more details.\r
15 *\r
16 * You should have received a copy of the GNU General Public License\r
17 * along with this program; if not, write to the Free Software\r
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\r
19 */\r
20\r
21#include "mapinc.h"\r
22#include "mmc3.h"\r
23#include "../ines.h"\r
24\r
25static bool is_BMCFK23CA;\r
26static uint8 unromchr;\r
27static uint32 dipswitch;\r
28static uint8 *CHRRAM=NULL;\r
29static uint32 CHRRAMSize;\r
30\r
31static void BMCFK23CCW(uint32 A, uint8 V)\r
32{\r
33 if(EXPREGS[0]&0x40)\r
34 setchr8(EXPREGS[2]|unromchr);\r
35 else if(EXPREGS[0]&0x20) {\r
36 setchr1r(0x10, A, V);\r
37 }\r
38 else\r
39 {\r
40 uint16 base=(EXPREGS[2]&0x7F)<<3;\r
41 if(EXPREGS[3]&2)\r
42 {\r
43 int cbase=(MMC3_cmd&0x80)<<5;\r
44 setchr1(A,V|base);\r
45 setchr1(0x0000^cbase,DRegBuf[0]|base);\r
46 setchr1(0x0400^cbase,EXPREGS[6]|base);\r
47 setchr1(0x0800^cbase,DRegBuf[1]|base);\r
48 setchr1(0x0c00^cbase,EXPREGS[7]|base);\r
49 }\r
50 else\r
51 setchr1(A,V|base);\r
52 }\r
53}\r
54\r
55//some games are wired differently, and this will need to be changed.\r
56//all the WXN games require prg_bonus = 1, and cah4e3's multicarts require prg_bonus = 0\r
57//we'll populate this from a game database\r
58static int prg_bonus;\r
59static int prg_mask;\r
60\r
61//prg_bonus = 0\r
62//4-in-1 (FK23C8021)[p1][!].nes\r
63//4-in-1 (FK23C8033)[p1][!].nes\r
64//4-in-1 (FK23C8043)[p1][!].nes\r
65//4-in-1 (FK23Cxxxx, S-0210A PCB)[p1][!].nes\r
66\r
67//prg_bonus = 1\r
68//[m176]大富翁2-上海大亨.wxn.nes\r
69//[m176]宠物翡翠.fix.nes\r
70//[m176]格兰帝亚.wxn.nes\r
71//[m176]梦幻之星.wxn.nes\r
72//[m176]水浒神兽.fix.nes\r
73//[m176]西楚霸王.fix.nes\r
74//[m176]超级大富翁.wxn.nes\r
75//[m176]雄霸天下.wxn.nes\r
76\r
77//works as-is under virtuanes m176\r
78//[m176]三侠五义.wxn.nes\r
79//[m176]口袋金.fix.nes\r
80//[m176]爆笑三国.fix.nes\r
81\r
82//needs other tweaks\r
83//[m176]三国忠烈传.wxn.nes\r
84//[m176]破釜沉舟.fix.nes\r
85\r
86//PRG wrapper\r
87static void BMCFK23CPW(uint32 A, uint8 V)\r
88{\r
89 uint32 bank = (EXPREGS[1] & 0x1F);\r
90 uint32 hiblock = ((EXPREGS[0] & 8) << 4)|((EXPREGS[0] & 0x80) << 1)|(UNIFchrrama?((EXPREGS[2] & 0x40)<<3):0);\r
91 uint32 block = (EXPREGS[1] & 0x60) | hiblock;\r
92 uint32 extra = (EXPREGS[3] & 2);\r
93\r
94 if((EXPREGS[0]&7)==4)\r
95 setprg32(0x8000,EXPREGS[1]>>1);\r
96 else if ((EXPREGS[0]&7)==3)\r
97 {\r
98 setprg16(0x8000,EXPREGS[1]);\r
99 setprg16(0xC000,EXPREGS[1]);\r
100 } \r
101 else\r
102 { \r
103 if(EXPREGS[0]&3)\r
104 {\r
105 uint32 blocksize = (6)-(EXPREGS[0]&3);\r
106 uint32 mask = (1<<blocksize)-1;\r
107 V &= mask;\r
108 //V &= 63; //? is this a good idea?\r
109 V |= (EXPREGS[1]<<1);\r
110 setprg8(A,V);\r
111 }\r
112 else\r
113 setprg8(A,V & prg_mask);\r
114\r
115 if(EXPREGS[3]&2)\r
116 {\r
117 setprg8(0xC000,EXPREGS[4]);\r
118 setprg8(0xE000,EXPREGS[5]);\r
119 }\r
120 }\r
121 setprg8r(0x10,0x6000,A001B&3);\r
122}\r
123\r
124//PRG handler ($8000-$FFFF)\r
125static DECLFW(BMCFK23CHiWrite)\r
126{\r
127 if(EXPREGS[0]&0x40)\r
128 {\r
129 if(EXPREGS[0]&0x30)\r
130 unromchr=0;\r
131 else\r
132 {\r
133 unromchr=V&3;\r
134 FixMMC3CHR(MMC3_cmd);\r
135 }\r
136 }\r
137 else\r
138 {\r
139 if((A==0x8001)&&(EXPREGS[3]&2&&MMC3_cmd&8))\r
140 {\r
141 EXPREGS[4|(MMC3_cmd&3)]=V;\r
142 FixMMC3PRG(MMC3_cmd);\r
143 FixMMC3CHR(MMC3_cmd);\r
144 }\r
145 else\r
146 if(A<0xC000) {\r
147 if(UNIFchrrama) { // hacky... strange behaviour, must be bit scramble due to pcb layot restrictions\r
148 // check if it not interfer with other dumps\r
149 if((A==0x8000)&&(V==0x46))\r
150 V=0x47;\r
151 else if((A==0x8000)&&(V==0x47))\r
152 V=0x46;\r
153 }\r
154 MMC3_CMDWrite(A,V);\r
155 FixMMC3PRG(MMC3_cmd);\r
156 }\r
157 else\r
158 MMC3_IRQWrite(A,V);\r
159 }\r
160}\r
161\r
162//EXP handler ($5000-$5FFF)\r
163static DECLFW(BMCFK23CWrite)\r
164{\r
165 if(A&(1<<(dipswitch+4)))\r
166 {\r
167 //printf("+ ");\r
168 EXPREGS[A&3]=V;\r
169\r
170 bool remap = false;\r
171\r
172 //sometimes writing to reg0 causes remappings to occur. we think the 2 signifies this. \r
173 //if not, 0x24 is a value that is known to work\r
174 //however, the low 4 bits are known to control the mapping mode, so 0x20 is more likely to be the immediate remap flag\r
175 remap |= ((EXPREGS[0]&0xF0)==0x20); \r
176\r
177 //this is an actual mapping reg. i think reg0 controls what happens when reg1 is written. anyway, we have to immediately remap these\r
178 remap |= (A&3)==1; \r
179 //this too.\r
180 remap |= (A&3)==2; \r
181\r
182 if(remap)\r
183 {\r
184 FixMMC3PRG(MMC3_cmd);\r
185 FixMMC3CHR(MMC3_cmd);\r
186 }\r
187 }\r
188\r
189 if(is_BMCFK23CA)\r
190 {\r
191 if(EXPREGS[3]&2)\r
192 EXPREGS[0] &= ~7; // hacky hacky! if someone wants extra banking, then for sure doesn't want mode 4 for it! (allow to run A version boards on normal mapper)\r
193 }\r
194\r
195 //printf("%04X = $%02X\n",A,V);\r
196 //printf("%02X %02X %02X %02X\n",EXPREGS[0],EXPREGS[1],EXPREGS[2],EXPREGS[3]);\r
197}\r
198\r
199static void BMCFK23CReset(void)\r
200{\r
201 //NOT NECESSARY ANYMORE\r
202 //this little hack makes sure that we try all the dip switch settings eventually, if we reset enough\r
203 // dipswitch++;\r
204 // dipswitch&=7;\r
205 //printf("BMCFK23C dipswitch set to %d\n",dipswitch);\r
206\r
207 EXPREGS[0]=EXPREGS[1]=EXPREGS[2]=EXPREGS[3]=0;\r
208 EXPREGS[4]=EXPREGS[5]=EXPREGS[6]=EXPREGS[7]=0xFF;\r
209 MMC3RegReset();\r
210 FixMMC3PRG(MMC3_cmd);\r
211 FixMMC3CHR(MMC3_cmd);\r
212}\r
213\r
214static void BMCFK23CPower(void)\r
215{\r
216 dipswitch = 0;\r
217 GenMMC3Power();\r
218 EXPREGS[0]=EXPREGS[1]=EXPREGS[2]=EXPREGS[3]=0;\r
219 EXPREGS[4]=EXPREGS[5]=EXPREGS[6]=EXPREGS[7]=0xFF;\r
220 GenMMC3Power();\r
221 SetWriteHandler(0x5000,0x5fff,BMCFK23CWrite);\r
222 SetWriteHandler(0x8000,0xFFFF,BMCFK23CHiWrite);\r
223 FixMMC3PRG(MMC3_cmd);\r
224 FixMMC3CHR(MMC3_cmd);\r
225}\r
226\r
227static void BMCFK23CAPower(void)\r
228{\r
229 GenMMC3Power();\r
230 dipswitch = 0;\r
231 EXPREGS[0]=EXPREGS[1]=EXPREGS[2]=EXPREGS[3]=0;\r
232 EXPREGS[4]=EXPREGS[5]=EXPREGS[6]=EXPREGS[7]=0xFF;\r
233 SetWriteHandler(0x5000,0x5fff,BMCFK23CWrite);\r
234 SetWriteHandler(0x8000,0xFFFF,BMCFK23CHiWrite);\r
235 FixMMC3PRG(MMC3_cmd);\r
236 FixMMC3CHR(MMC3_cmd);\r
237}\r
238\r
239static void BMCFK23CAClose(void)\r
240{\r
241 if(CHRRAM)\r
242 FCEU_gfree(CHRRAM);\r
243 CHRRAM=NULL;\r
244}\r
245\r
246void BMCFK23C_Init(CartInfo *info)\r
247{\r
248 is_BMCFK23CA = false;\r
249\r
250 GenMMC3_Init(info, 512, 256, 8, 0);\r
251 cwrap=BMCFK23CCW;\r
252 pwrap=BMCFK23CPW;\r
253 info->Power=BMCFK23CPower;\r
254 info->Reset=BMCFK23CReset;\r
255 AddExState(EXPREGS, 8, 0, "EXPR");\r
256 AddExState(&unromchr, 1, 0, "UNCHR");\r
257 AddExState(&dipswitch, 1, 0, "DIPSW");\r
258\r
259 prg_bonus = 1;\r
260 if(MasterRomInfoParams.find("bonus") != MasterRomInfoParams.end())\r
261 prg_bonus = atoi(MasterRomInfoParams["bonus"].c_str());\r
262\r
263 prg_mask = 0x7F>>(prg_bonus);\r
264}\r
265\r
266void BMCFK23CA_Init(CartInfo *info)\r
267{\r
268 is_BMCFK23CA = true;\r
269\r
270 GenMMC3_Init(info, 512, 256, 8, 0);\r
271 cwrap=BMCFK23CCW;\r
272 pwrap=BMCFK23CPW;\r
273 info->Power=BMCFK23CAPower;\r
274 info->Reset=BMCFK23CReset;\r
275 info->Close=BMCFK23CAClose;\r
276\r
277 CHRRAMSize=8192;\r
278 CHRRAM=(uint8*)FCEU_gmalloc(CHRRAMSize);\r
279 SetupCartCHRMapping(0x10, CHRRAM, CHRRAMSize, 1);\r
280 AddExState(CHRRAM, CHRRAMSize, 0, "CRAM");\r
281\r
282 AddExState(EXPREGS, 8, 0, "EXPR");\r
283 AddExState(&unromchr, 1, 0, "UNCHR");\r
284 AddExState(&dipswitch, 1, 0, "DIPSW");\r
285\r
286 prg_bonus = 1;\r
287 if(MasterRomInfoParams.find("bonus") != MasterRomInfoParams.end())\r
288 prg_bonus = atoi(MasterRomInfoParams["bonus"].c_str());\r
289 prg_mask = 0x7F>>(prg_bonus);\r
290}\r