1 |
/* |
2 |
* VIC_SC.cpp - 6569R5 emulation (cycle based) |
3 |
* |
4 |
* Frodo (C) 1994-1997,2002-2005 Christian Bauer |
5 |
* |
6 |
* This program is free software; you can redistribute it and/or modify |
7 |
* it under the terms of the GNU General Public License as published by |
8 |
* the Free Software Foundation; either version 2 of the License, or |
9 |
* (at your option) any later version. |
10 |
* |
11 |
* This program is distributed in the hope that it will be useful, |
12 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
13 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
14 |
* GNU General Public License for more details. |
15 |
* |
16 |
* You should have received a copy of the GNU General Public License |
17 |
* along with this program; if not, write to the Free Software |
18 |
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
19 |
*/ |
20 |
|
21 |
/* |
22 |
* Incompatibilities: |
23 |
* ------------------ |
24 |
* |
25 |
* - Color of $ff bytes read when BA is low and AEC is high |
26 |
* is not correct |
27 |
* - Changes to border/background color are visible 7 pixels |
28 |
* too late |
29 |
* - Sprite data access doesn't respect BA |
30 |
* - Sprite collisions are only detected within the visible |
31 |
* screen area (excluding borders) |
32 |
* - Sprites are only drawn if they completely fit within the |
33 |
* left/right limits of the chunky bitmap |
34 |
*/ |
35 |
|
36 |
#include "sysdeps.h" |
37 |
|
38 |
#include "VIC.h" |
39 |
#include "C64.h" |
40 |
#include "CPUC64.h" |
41 |
#include "Display.h" |
42 |
#include "Prefs.h" |
43 |
|
44 |
|
45 |
// First and last displayed line |
46 |
const int FIRST_DISP_LINE = 0x10; |
47 |
const int LAST_DISP_LINE = 0x11f; |
48 |
|
49 |
// First and last possible line for Bad Lines |
50 |
const int FIRST_DMA_LINE = 0x30; |
51 |
const int LAST_DMA_LINE = 0xf7; |
52 |
|
53 |
// Display window coordinates |
54 |
const int ROW25_YSTART = 0x33; |
55 |
const int ROW25_YSTOP = 0xfb; |
56 |
const int ROW24_YSTART = 0x37; |
57 |
const int ROW24_YSTOP = 0xf7; |
58 |
|
59 |
#if defined(SMALL_DISPLAY) |
60 |
/* This does not work yet, the sprite code doesn't know about it. */ |
61 |
const int COL40_XSTART = 0x14; |
62 |
const int COL40_XSTOP = 0x154; |
63 |
const int COL38_XSTART = 0x1B; |
64 |
const int COL38_XSTOP = 0x14B; |
65 |
#else |
66 |
const int COL40_XSTART = 0x20; |
67 |
const int COL40_XSTOP = 0x160; |
68 |
const int COL38_XSTART = 0x27; |
69 |
const int COL38_XSTOP = 0x157; |
70 |
#endif |
71 |
|
72 |
|
73 |
// Tables for sprite X expansion |
74 |
uint16 ExpTable[256] = { |
75 |
0x0000, 0x0003, 0x000C, 0x000F, 0x0030, 0x0033, 0x003C, 0x003F, |
76 |
0x00C0, 0x00C3, 0x00CC, 0x00CF, 0x00F0, 0x00F3, 0x00FC, 0x00FF, |
77 |
0x0300, 0x0303, 0x030C, 0x030F, 0x0330, 0x0333, 0x033C, 0x033F, |
78 |
0x03C0, 0x03C3, 0x03CC, 0x03CF, 0x03F0, 0x03F3, 0x03FC, 0x03FF, |
79 |
0x0C00, 0x0C03, 0x0C0C, 0x0C0F, 0x0C30, 0x0C33, 0x0C3C, 0x0C3F, |
80 |
0x0CC0, 0x0CC3, 0x0CCC, 0x0CCF, 0x0CF0, 0x0CF3, 0x0CFC, 0x0CFF, |
81 |
0x0F00, 0x0F03, 0x0F0C, 0x0F0F, 0x0F30, 0x0F33, 0x0F3C, 0x0F3F, |
82 |
0x0FC0, 0x0FC3, 0x0FCC, 0x0FCF, 0x0FF0, 0x0FF3, 0x0FFC, 0x0FFF, |
83 |
0x3000, 0x3003, 0x300C, 0x300F, 0x3030, 0x3033, 0x303C, 0x303F, |
84 |
0x30C0, 0x30C3, 0x30CC, 0x30CF, 0x30F0, 0x30F3, 0x30FC, 0x30FF, |
85 |
0x3300, 0x3303, 0x330C, 0x330F, 0x3330, 0x3333, 0x333C, 0x333F, |
86 |
0x33C0, 0x33C3, 0x33CC, 0x33CF, 0x33F0, 0x33F3, 0x33FC, 0x33FF, |
87 |
0x3C00, 0x3C03, 0x3C0C, 0x3C0F, 0x3C30, 0x3C33, 0x3C3C, 0x3C3F, |
88 |
0x3CC0, 0x3CC3, 0x3CCC, 0x3CCF, 0x3CF0, 0x3CF3, 0x3CFC, 0x3CFF, |
89 |
0x3F00, 0x3F03, 0x3F0C, 0x3F0F, 0x3F30, 0x3F33, 0x3F3C, 0x3F3F, |
90 |
0x3FC0, 0x3FC3, 0x3FCC, 0x3FCF, 0x3FF0, 0x3FF3, 0x3FFC, 0x3FFF, |
91 |
0xC000, 0xC003, 0xC00C, 0xC00F, 0xC030, 0xC033, 0xC03C, 0xC03F, |
92 |
0xC0C0, 0xC0C3, 0xC0CC, 0xC0CF, 0xC0F0, 0xC0F3, 0xC0FC, 0xC0FF, |
93 |
0xC300, 0xC303, 0xC30C, 0xC30F, 0xC330, 0xC333, 0xC33C, 0xC33F, |
94 |
0xC3C0, 0xC3C3, 0xC3CC, 0xC3CF, 0xC3F0, 0xC3F3, 0xC3FC, 0xC3FF, |
95 |
0xCC00, 0xCC03, 0xCC0C, 0xCC0F, 0xCC30, 0xCC33, 0xCC3C, 0xCC3F, |
96 |
0xCCC0, 0xCCC3, 0xCCCC, 0xCCCF, 0xCCF0, 0xCCF3, 0xCCFC, 0xCCFF, |
97 |
0xCF00, 0xCF03, 0xCF0C, 0xCF0F, 0xCF30, 0xCF33, 0xCF3C, 0xCF3F, |
98 |
0xCFC0, 0xCFC3, 0xCFCC, 0xCFCF, 0xCFF0, 0xCFF3, 0xCFFC, 0xCFFF, |
99 |
0xF000, 0xF003, 0xF00C, 0xF00F, 0xF030, 0xF033, 0xF03C, 0xF03F, |
100 |
0xF0C0, 0xF0C3, 0xF0CC, 0xF0CF, 0xF0F0, 0xF0F3, 0xF0FC, 0xF0FF, |
101 |
0xF300, 0xF303, 0xF30C, 0xF30F, 0xF330, 0xF333, 0xF33C, 0xF33F, |
102 |
0xF3C0, 0xF3C3, 0xF3CC, 0xF3CF, 0xF3F0, 0xF3F3, 0xF3FC, 0xF3FF, |
103 |
0xFC00, 0xFC03, 0xFC0C, 0xFC0F, 0xFC30, 0xFC33, 0xFC3C, 0xFC3F, |
104 |
0xFCC0, 0xFCC3, 0xFCCC, 0xFCCF, 0xFCF0, 0xFCF3, 0xFCFC, 0xFCFF, |
105 |
0xFF00, 0xFF03, 0xFF0C, 0xFF0F, 0xFF30, 0xFF33, 0xFF3C, 0xFF3F, |
106 |
0xFFC0, 0xFFC3, 0xFFCC, 0xFFCF, 0xFFF0, 0xFFF3, 0xFFFC, 0xFFFF |
107 |
}; |
108 |
|
109 |
uint16 MultiExpTable[256] = { |
110 |
0x0000, 0x0005, 0x000A, 0x000F, 0x0050, 0x0055, 0x005A, 0x005F, |
111 |
0x00A0, 0x00A5, 0x00AA, 0x00AF, 0x00F0, 0x00F5, 0x00FA, 0x00FF, |
112 |
0x0500, 0x0505, 0x050A, 0x050F, 0x0550, 0x0555, 0x055A, 0x055F, |
113 |
0x05A0, 0x05A5, 0x05AA, 0x05AF, 0x05F0, 0x05F5, 0x05FA, 0x05FF, |
114 |
0x0A00, 0x0A05, 0x0A0A, 0x0A0F, 0x0A50, 0x0A55, 0x0A5A, 0x0A5F, |
115 |
0x0AA0, 0x0AA5, 0x0AAA, 0x0AAF, 0x0AF0, 0x0AF5, 0x0AFA, 0x0AFF, |
116 |
0x0F00, 0x0F05, 0x0F0A, 0x0F0F, 0x0F50, 0x0F55, 0x0F5A, 0x0F5F, |
117 |
0x0FA0, 0x0FA5, 0x0FAA, 0x0FAF, 0x0FF0, 0x0FF5, 0x0FFA, 0x0FFF, |
118 |
0x5000, 0x5005, 0x500A, 0x500F, 0x5050, 0x5055, 0x505A, 0x505F, |
119 |
0x50A0, 0x50A5, 0x50AA, 0x50AF, 0x50F0, 0x50F5, 0x50FA, 0x50FF, |
120 |
0x5500, 0x5505, 0x550A, 0x550F, 0x5550, 0x5555, 0x555A, 0x555F, |
121 |
0x55A0, 0x55A5, 0x55AA, 0x55AF, 0x55F0, 0x55F5, 0x55FA, 0x55FF, |
122 |
0x5A00, 0x5A05, 0x5A0A, 0x5A0F, 0x5A50, 0x5A55, 0x5A5A, 0x5A5F, |
123 |
0x5AA0, 0x5AA5, 0x5AAA, 0x5AAF, 0x5AF0, 0x5AF5, 0x5AFA, 0x5AFF, |
124 |
0x5F00, 0x5F05, 0x5F0A, 0x5F0F, 0x5F50, 0x5F55, 0x5F5A, 0x5F5F, |
125 |
0x5FA0, 0x5FA5, 0x5FAA, 0x5FAF, 0x5FF0, 0x5FF5, 0x5FFA, 0x5FFF, |
126 |
0xA000, 0xA005, 0xA00A, 0xA00F, 0xA050, 0xA055, 0xA05A, 0xA05F, |
127 |
0xA0A0, 0xA0A5, 0xA0AA, 0xA0AF, 0xA0F0, 0xA0F5, 0xA0FA, 0xA0FF, |
128 |
0xA500, 0xA505, 0xA50A, 0xA50F, 0xA550, 0xA555, 0xA55A, 0xA55F, |
129 |
0xA5A0, 0xA5A5, 0xA5AA, 0xA5AF, 0xA5F0, 0xA5F5, 0xA5FA, 0xA5FF, |
130 |
0xAA00, 0xAA05, 0xAA0A, 0xAA0F, 0xAA50, 0xAA55, 0xAA5A, 0xAA5F, |
131 |
0xAAA0, 0xAAA5, 0xAAAA, 0xAAAF, 0xAAF0, 0xAAF5, 0xAAFA, 0xAAFF, |
132 |
0xAF00, 0xAF05, 0xAF0A, 0xAF0F, 0xAF50, 0xAF55, 0xAF5A, 0xAF5F, |
133 |
0xAFA0, 0xAFA5, 0xAFAA, 0xAFAF, 0xAFF0, 0xAFF5, 0xAFFA, 0xAFFF, |
134 |
0xF000, 0xF005, 0xF00A, 0xF00F, 0xF050, 0xF055, 0xF05A, 0xF05F, |
135 |
0xF0A0, 0xF0A5, 0xF0AA, 0xF0AF, 0xF0F0, 0xF0F5, 0xF0FA, 0xF0FF, |
136 |
0xF500, 0xF505, 0xF50A, 0xF50F, 0xF550, 0xF555, 0xF55A, 0xF55F, |
137 |
0xF5A0, 0xF5A5, 0xF5AA, 0xF5AF, 0xF5F0, 0xF5F5, 0xF5FA, 0xF5FF, |
138 |
0xFA00, 0xFA05, 0xFA0A, 0xFA0F, 0xFA50, 0xFA55, 0xFA5A, 0xFA5F, |
139 |
0xFAA0, 0xFAA5, 0xFAAA, 0xFAAF, 0xFAF0, 0xFAF5, 0xFAFA, 0xFAFF, |
140 |
0xFF00, 0xFF05, 0xFF0A, 0xFF0F, 0xFF50, 0xFF55, 0xFF5A, 0xFF5F, |
141 |
0xFFA0, 0xFFA5, 0xFFAA, 0xFFAF, 0xFFF0, 0xFFF5, 0xFFFA, 0xFFFF |
142 |
}; |
143 |
|
144 |
#ifdef GLOBAL_VARS |
145 |
static uint16 mx[8]; // VIC registers |
146 |
static uint8 my[8]; |
147 |
static uint8 mx8; |
148 |
static uint8 ctrl1, ctrl2; |
149 |
static uint8 lpx, lpy; |
150 |
static uint8 me, mxe, mye, mdp, mmc; |
151 |
static uint8 vbase; |
152 |
static uint8 irq_flag, irq_mask; |
153 |
static uint8 clx_spr, clx_bgr; |
154 |
static uint8 ec, b0c, b1c, b2c, b3c, mm0, mm1; |
155 |
static uint8 sc[8]; |
156 |
|
157 |
static uint8 *ram, *char_rom, *color_ram; // Pointers to RAM and ROM |
158 |
static C64 *the_c64; // Pointer to C64 |
159 |
static C64Display *the_display; // Pointer to C64Display |
160 |
static MOS6510 *the_cpu; // Pointer to 6510 |
161 |
static MOS6569 *the_vic; // Pointer to self |
162 |
|
163 |
static uint8 colors[256]; // Indices of the 16 C64 colors (16 times mirrored to avoid "& 0x0f") |
164 |
|
165 |
static uint8 ec_color, b0c_color, b1c_color, b2c_color, b3c_color; // Indices for exterior/background colors |
166 |
static uint8 mm0_color, mm1_color; // Indices for MOB multicolors |
167 |
static uint8 spr_color[8]; // Indices for MOB colors |
168 |
|
169 |
static uint8 matrix_line[40]; // Buffer for video line, read in Bad Lines |
170 |
static uint8 color_line[40]; // Buffer for color line, read in Bad Lines |
171 |
|
172 |
#ifdef __POWERPC__ |
173 |
static double chunky_tmp[DISPLAY_X/8]; // Temporary line buffer for GameKit speedup |
174 |
#endif |
175 |
static uint8 *chunky_ptr; // Pointer in chunky bitmap buffer |
176 |
static uint8 *chunky_line_start; // Pointer to start of current line in bitmap buffer |
177 |
static uint8 *fore_mask_ptr; // Pointer in fore_mask_buf |
178 |
static int xmod; // Number of bytes per row |
179 |
|
180 |
static uint16 raster_x; // Current raster x position |
181 |
static uint16 raster_y; // Current raster line |
182 |
static uint16 irq_raster; // Interrupt raster line |
183 |
static uint16 dy_start; // Comparison values for border logic |
184 |
static uint16 dy_stop; |
185 |
static uint16 rc; // Row counter |
186 |
static uint16 vc; // Video counter |
187 |
static uint16 vc_base; // Video counter base |
188 |
static uint16 x_scroll; // X scroll value |
189 |
static uint16 y_scroll; // Y scroll value |
190 |
static uint16 cia_vabase; // CIA VA14/15 video base |
191 |
|
192 |
static int cycle; // Current cycle in line (1..63) |
193 |
|
194 |
static int display_idx; // Index of current display mode |
195 |
static int ml_index; // Index in matrix/color_line[] |
196 |
static int skip_counter; // Counter for frame-skipping |
197 |
|
198 |
static uint16 mc[8]; // Sprite data counters |
199 |
static uint16 mc_base[8]; // Sprite data counter bases |
200 |
|
201 |
static uint8 spr_coll_buf[0x180]; // Buffer for sprite-sprite collisions and priorities |
202 |
static uint8 fore_mask_buf[0x180/8]; // Foreground mask for sprite-graphics collisions and priorities |
203 |
|
204 |
static bool display_state; // true: Display state, false: Idle state |
205 |
static bool border_on; // Flag: Upper/lower border on |
206 |
static bool frame_skipped; // Flag: Frame is being skipped |
207 |
static bool bad_lines_enabled; // Flag: Bad Lines enabled for this frame |
208 |
static bool lp_triggered; // Flag: Lightpen was triggered in this frame |
209 |
static bool is_bad_line; // Flag: Current line is Bad Line |
210 |
static bool draw_this_line; // Flag: This line is drawn on the screen |
211 |
static bool ud_border_on; // Flag: Upper/lower border on |
212 |
static bool vblanking; // Flag: VBlank in next cycle |
213 |
|
214 |
static bool border_on_sample[5]; // Samples of border state at different cycles (1, 17, 18, 56, 57) |
215 |
static uint8 border_color_sample[DISPLAY_X/8]; // Samples of border color at each "displayed" cycle |
216 |
|
217 |
static uint16 matrix_base; // Video matrix base |
218 |
static uint16 char_base; // Character generator base |
219 |
static uint16 bitmap_base; // Bitmap base |
220 |
|
221 |
static uint8 ref_cnt; // Refresh counter |
222 |
static uint8 spr_exp_y; // 8 sprite y expansion flipflops |
223 |
static uint8 spr_dma_on; // 8 flags: Sprite DMA active |
224 |
static uint8 spr_disp_on; // 8 flags: Sprite display active |
225 |
static uint8 spr_draw; // 8 flags: Draw sprite in this line |
226 |
static uint16 spr_ptr[8]; // Sprite data pointers |
227 |
|
228 |
static uint8 gfx_data, char_data, color_data, last_char_data; |
229 |
static uint8 spr_data[8][4]; // Sprite data read |
230 |
static uint8 spr_draw_data[8][4]; // Sprite data for drawing |
231 |
|
232 |
static uint32 first_ba_cycle; // Cycle when BA first went low |
233 |
#endif |
234 |
|
235 |
|
236 |
/* |
237 |
* Constructor: Initialize variables |
238 |
*/ |
239 |
|
240 |
MOS6569::MOS6569(C64 *c64, C64Display *disp, MOS6510 *CPU, uint8 *RAM, uint8 *Char, uint8 *Color) |
241 |
#ifndef GLOBAL_VARS |
242 |
: ram(RAM), char_rom(Char), color_ram(Color), the_c64(c64), the_display(disp), the_cpu(CPU) |
243 |
#endif |
244 |
{ |
245 |
int i; |
246 |
|
247 |
// Set pointers |
248 |
#ifdef GLOBAL_VARS |
249 |
the_vic = this; |
250 |
the_c64 = c64; |
251 |
the_display = disp; |
252 |
the_cpu = CPU; |
253 |
ram = RAM; |
254 |
char_rom = Char; |
255 |
color_ram = Color; |
256 |
#endif |
257 |
matrix_base = 0; |
258 |
char_base = 0; |
259 |
bitmap_base = 0; |
260 |
|
261 |
// Get bitmap info |
262 |
chunky_ptr = chunky_line_start = disp->BitmapBase(); |
263 |
xmod = disp->BitmapXMod(); |
264 |
|
265 |
// Initialize VIC registers |
266 |
mx8 = 0; |
267 |
ctrl1 = ctrl2 = 0; |
268 |
lpx = lpy = 0; |
269 |
me = mxe = mye = mdp = mmc = 0; |
270 |
vbase = irq_flag = irq_mask = 0; |
271 |
clx_spr = clx_bgr = 0; |
272 |
cia_vabase = 0; |
273 |
ec = b0c = b1c = b2c = b3c = mm0 = mm1 = 0; |
274 |
for (i=0; i<8; i++) mx[i] = my[i] = sc[i] = 0; |
275 |
|
276 |
// Initialize other variables |
277 |
raster_y = TOTAL_RASTERS - 1; |
278 |
rc = 7; |
279 |
irq_raster = vc = vc_base = x_scroll = y_scroll = 0; |
280 |
dy_start = ROW24_YSTART; |
281 |
dy_stop = ROW24_YSTOP; |
282 |
ml_index = 0; |
283 |
|
284 |
cycle = 1; |
285 |
display_idx = 0; |
286 |
display_state = false; |
287 |
border_on = ud_border_on = vblanking = false; |
288 |
lp_triggered = draw_this_line = false; |
289 |
|
290 |
spr_dma_on = spr_disp_on = 0; |
291 |
for (i=0; i<8; i++) { |
292 |
mc[i] = 63; |
293 |
spr_ptr[i] = 0; |
294 |
} |
295 |
|
296 |
frame_skipped = false; |
297 |
skip_counter = 1; |
298 |
|
299 |
memset(spr_coll_buf, 0, 0x180); |
300 |
memset(fore_mask_buf, 0, 0x180/8); |
301 |
|
302 |
// Preset colors to black |
303 |
disp->InitColors(colors); |
304 |
ec_color = b0c_color = b1c_color = b2c_color = b3c_color = mm0_color = mm1_color = colors[0]; |
305 |
for (i=0; i<8; i++) spr_color[i] = colors[0]; |
306 |
} |
307 |
|
308 |
|
309 |
/* |
310 |
* Reinitialize the colors table for when the palette has changed |
311 |
*/ |
312 |
|
313 |
void MOS6569::ReInitColors(void) |
314 |
{ |
315 |
int i; |
316 |
|
317 |
// Build inverse color table. |
318 |
uint8 xlate_colors[256]; |
319 |
memset(xlate_colors, 0, sizeof(xlate_colors)); |
320 |
for (i=0; i<16; i++) |
321 |
xlate_colors[colors[i]] = i; |
322 |
|
323 |
// Get the new colors. |
324 |
the_display->InitColors(colors); |
325 |
|
326 |
// Build color translation table. |
327 |
for (i=0; i<256; i++) |
328 |
xlate_colors[i] = colors[xlate_colors[i]]; |
329 |
|
330 |
// Translate all the old colors variables. |
331 |
ec_color = colors[ec]; |
332 |
b0c_color = colors[b0c]; |
333 |
b1c_color = colors[b1c]; |
334 |
b2c_color = colors[b2c]; |
335 |
b3c_color = colors[b3c]; |
336 |
mm0_color = colors[mm0]; |
337 |
mm1_color = colors[mm1]; |
338 |
for (i=0; i<8; i++) |
339 |
spr_color[i] = colors[sc[i]]; |
340 |
|
341 |
// Translate the border color sample buffer. |
342 |
for (unsigned x = 0; x < sizeof(border_color_sample); x++) |
343 |
border_color_sample[x] = xlate_colors[border_color_sample[x]]; |
344 |
|
345 |
// Translate the chunky buffer. |
346 |
uint8 *scanline = the_display->BitmapBase(); |
347 |
for (int y=0; y<DISPLAY_Y; y++) { |
348 |
for (int x=0; x<DISPLAY_X; x++) |
349 |
scanline[x] = xlate_colors[scanline[x]]; |
350 |
scanline += xmod; |
351 |
} |
352 |
} |
353 |
|
354 |
|
355 |
/* |
356 |
* Get VIC state |
357 |
*/ |
358 |
|
359 |
void MOS6569::GetState(MOS6569State *vd) |
360 |
{ |
361 |
int i; |
362 |
|
363 |
vd->m0x = mx[0] & 0xff; vd->m0y = my[0]; |
364 |
vd->m1x = mx[1] & 0xff; vd->m1y = my[1]; |
365 |
vd->m2x = mx[2] & 0xff; vd->m2y = my[2]; |
366 |
vd->m3x = mx[3] & 0xff; vd->m3y = my[3]; |
367 |
vd->m4x = mx[4] & 0xff; vd->m4y = my[4]; |
368 |
vd->m5x = mx[5] & 0xff; vd->m5y = my[5]; |
369 |
vd->m6x = mx[6] & 0xff; vd->m6y = my[6]; |
370 |
vd->m7x = mx[7] & 0xff; vd->m7y = my[7]; |
371 |
vd->mx8 = mx8; |
372 |
|
373 |
vd->ctrl1 = (ctrl1 & 0x7f) | ((raster_y & 0x100) >> 1); |
374 |
vd->raster = raster_y & 0xff; |
375 |
vd->lpx = lpx; vd->lpy = lpy; |
376 |
vd->ctrl2 = ctrl2; |
377 |
vd->vbase = vbase; |
378 |
vd->irq_flag = irq_flag; |
379 |
vd->irq_mask = irq_mask; |
380 |
|
381 |
vd->me = me; vd->mxe = mxe; vd->mye = mye; vd->mdp = mdp; vd->mmc = mmc; |
382 |
vd->mm = clx_spr; vd->md = clx_bgr; |
383 |
|
384 |
vd->ec = ec; |
385 |
vd->b0c = b0c; vd->b1c = b1c; vd->b2c = b2c; vd->b3c = b3c; |
386 |
vd->mm0 = mm0; vd->mm1 = mm1; |
387 |
vd->m0c = sc[0]; |
388 |
vd->m1c = sc[1]; |
389 |
vd->m2c = sc[2]; |
390 |
vd->m3c = sc[3]; |
391 |
vd->m4c = sc[4]; |
392 |
vd->m5c = sc[5]; |
393 |
vd->m6c = sc[6]; |
394 |
vd->m7c = sc[7]; |
395 |
|
396 |
vd->pad0 = 0; |
397 |
vd->irq_raster = irq_raster; |
398 |
vd->vc = vc; |
399 |
vd->vc_base = vc_base; |
400 |
vd->rc = rc; |
401 |
vd->spr_dma = spr_dma_on; |
402 |
vd->spr_disp = spr_disp_on; |
403 |
for (i=0; i<8; i++) { |
404 |
vd->mc[i] = mc[i]; |
405 |
vd->mc_base[i] = mc_base[i]; |
406 |
} |
407 |
vd->display_state = display_state; |
408 |
vd->bad_line = raster_y >= FIRST_DMA_LINE && raster_y <= LAST_DMA_LINE && ((raster_y & 7) == y_scroll) && bad_lines_enabled; |
409 |
vd->bad_line_enable = bad_lines_enabled; |
410 |
vd->lp_triggered = lp_triggered; |
411 |
vd->border_on = border_on; |
412 |
|
413 |
vd->bank_base = cia_vabase; |
414 |
vd->matrix_base = ((vbase & 0xf0) << 6) | cia_vabase; |
415 |
vd->char_base = ((vbase & 0x0e) << 10) | cia_vabase; |
416 |
vd->bitmap_base = ((vbase & 0x08) << 10) | cia_vabase; |
417 |
for (i=0; i<8; i++) |
418 |
vd->sprite_base[i] = spr_ptr[i] | cia_vabase; |
419 |
|
420 |
vd->cycle = cycle; |
421 |
vd->raster_x = raster_x; |
422 |
vd->ml_index = ml_index; |
423 |
vd->ref_cnt = ref_cnt; |
424 |
vd->last_vic_byte = LastVICByte; |
425 |
vd->ud_border_on = ud_border_on; |
426 |
} |
427 |
|
428 |
|
429 |
/* |
430 |
* Set VIC state (only works if in VBlank) |
431 |
*/ |
432 |
|
433 |
void MOS6569::SetState(MOS6569State *vd) |
434 |
{ |
435 |
int i, j; |
436 |
|
437 |
mx[0] = vd->m0x; my[0] = vd->m0y; |
438 |
mx[1] = vd->m1x; my[1] = vd->m1y; |
439 |
mx[2] = vd->m2x; my[2] = vd->m2y; |
440 |
mx[3] = vd->m3x; my[3] = vd->m3y; |
441 |
mx[4] = vd->m4x; my[4] = vd->m4y; |
442 |
mx[5] = vd->m5x; my[5] = vd->m5y; |
443 |
mx[6] = vd->m6x; my[6] = vd->m6y; |
444 |
mx[7] = vd->m7x; my[7] = vd->m7y; |
445 |
mx8 = vd->mx8; |
446 |
for (i=0, j=1; i<8; i++, j<<=1) { |
447 |
if (mx8 & j) |
448 |
mx[i] |= 0x100; |
449 |
else |
450 |
mx[i] &= 0xff; |
451 |
} |
452 |
|
453 |
ctrl1 = vd->ctrl1; |
454 |
ctrl2 = vd->ctrl2; |
455 |
x_scroll = ctrl2 & 7; |
456 |
y_scroll = ctrl1 & 7; |
457 |
if (ctrl1 & 8) { |
458 |
dy_start = ROW25_YSTART; |
459 |
dy_stop = ROW25_YSTOP; |
460 |
} else { |
461 |
dy_start = ROW24_YSTART; |
462 |
dy_stop = ROW24_YSTOP; |
463 |
} |
464 |
display_idx = ((ctrl1 & 0x60) | (ctrl2 & 0x10)) >> 4; |
465 |
|
466 |
raster_y = 0; |
467 |
lpx = vd->lpx; lpy = vd->lpy; |
468 |
|
469 |
vbase = vd->vbase; |
470 |
cia_vabase = vd->bank_base; |
471 |
matrix_base = (vbase & 0xf0) << 6; |
472 |
char_base = (vbase & 0x0e) << 10; |
473 |
bitmap_base = (vbase & 0x08) << 10; |
474 |
|
475 |
irq_flag = vd->irq_flag; |
476 |
irq_mask = vd->irq_mask; |
477 |
|
478 |
me = vd->me; mxe = vd->mxe; mye = vd->mye; mdp = vd->mdp; mmc = vd->mmc; |
479 |
clx_spr = vd->mm; clx_bgr = vd->md; |
480 |
|
481 |
ec = vd->ec; |
482 |
ec_color = colors[ec]; |
483 |
|
484 |
b0c = vd->b0c; b1c = vd->b1c; b2c = vd->b2c; b3c = vd->b3c; |
485 |
b0c_color = colors[b0c]; |
486 |
b1c_color = colors[b1c]; |
487 |
b2c_color = colors[b2c]; |
488 |
b3c_color = colors[b3c]; |
489 |
|
490 |
mm0 = vd->mm0; mm1 = vd->mm1; |
491 |
mm0_color = colors[mm0]; |
492 |
mm1_color = colors[mm1]; |
493 |
|
494 |
sc[0] = vd->m0c; sc[1] = vd->m1c; |
495 |
sc[2] = vd->m2c; sc[3] = vd->m3c; |
496 |
sc[4] = vd->m4c; sc[5] = vd->m5c; |
497 |
sc[6] = vd->m6c; sc[7] = vd->m7c; |
498 |
for (i=0; i<8; i++) |
499 |
spr_color[i] = colors[sc[i]]; |
500 |
|
501 |
irq_raster = vd->irq_raster; |
502 |
vc = vd->vc; |
503 |
vc_base = vd->vc_base; |
504 |
rc = vd->rc; |
505 |
spr_dma_on = vd->spr_dma; |
506 |
spr_disp_on = vd->spr_disp; |
507 |
for (i=0; i<8; i++) { |
508 |
mc[i] = vd->mc[i]; |
509 |
mc_base[i] = vd->mc_base[i]; |
510 |
spr_ptr[i] = vd->sprite_base[i] & 0x3fff; |
511 |
} |
512 |
display_state = vd->display_state; |
513 |
bad_lines_enabled = vd->bad_line_enable; |
514 |
lp_triggered = vd->lp_triggered; |
515 |
border_on = vd->border_on; |
516 |
|
517 |
cycle = vd->cycle; |
518 |
raster_x = vd->raster_x; |
519 |
ml_index = vd->ml_index; |
520 |
ref_cnt = vd->ref_cnt; |
521 |
LastVICByte = vd->last_vic_byte; |
522 |
ud_border_on = vd->ud_border_on; |
523 |
} |
524 |
|
525 |
|
526 |
/* |
527 |
* Trigger raster IRQ |
528 |
*/ |
529 |
|
530 |
#ifdef GLOBAL_VARS |
531 |
static inline void raster_irq(void) |
532 |
#else |
533 |
inline void MOS6569::raster_irq(void) |
534 |
#endif |
535 |
{ |
536 |
irq_flag |= 0x01; |
537 |
if (irq_mask & 0x01) { |
538 |
irq_flag |= 0x80; |
539 |
the_cpu->TriggerVICIRQ(); |
540 |
} |
541 |
} |
542 |
|
543 |
|
544 |
/* |
545 |
* Read from VIC register |
546 |
*/ |
547 |
|
548 |
uint8 MOS6569::ReadRegister(uint16 adr) |
549 |
{ |
550 |
switch (adr) { |
551 |
case 0x00: case 0x02: case 0x04: case 0x06: |
552 |
case 0x08: case 0x0a: case 0x0c: case 0x0e: |
553 |
return mx[adr >> 1]; |
554 |
|
555 |
case 0x01: case 0x03: case 0x05: case 0x07: |
556 |
case 0x09: case 0x0b: case 0x0d: case 0x0f: |
557 |
return my[adr >> 1]; |
558 |
|
559 |
case 0x10: // Sprite X position MSB |
560 |
return mx8; |
561 |
|
562 |
case 0x11: // Control register 1 |
563 |
return (ctrl1 & 0x7f) | ((raster_y & 0x100) >> 1); |
564 |
|
565 |
case 0x12: // Raster counter |
566 |
return raster_y; |
567 |
|
568 |
case 0x13: // Light pen X |
569 |
return lpx; |
570 |
|
571 |
case 0x14: // Light pen Y |
572 |
return lpy; |
573 |
|
574 |
case 0x15: // Sprite enable |
575 |
return me; |
576 |
|
577 |
case 0x16: // Control register 2 |
578 |
return ctrl2 | 0xc0; |
579 |
|
580 |
case 0x17: // Sprite Y expansion |
581 |
return mye; |
582 |
|
583 |
case 0x18: // Memory pointers |
584 |
return vbase | 0x01; |
585 |
|
586 |
case 0x19: // IRQ flags |
587 |
return irq_flag | 0x70; |
588 |
|
589 |
case 0x1a: // IRQ mask |
590 |
return irq_mask | 0xf0; |
591 |
|
592 |
case 0x1b: // Sprite data priority |
593 |
return mdp; |
594 |
|
595 |
case 0x1c: // Sprite multicolor |
596 |
return mmc; |
597 |
|
598 |
case 0x1d: // Sprite X expansion |
599 |
return mxe; |
600 |
|
601 |
case 0x1e:{ // Sprite-sprite collision |
602 |
uint8 ret = clx_spr; |
603 |
clx_spr = 0; // Read and clear |
604 |
return ret; |
605 |
} |
606 |
|
607 |
case 0x1f:{ // Sprite-background collision |
608 |
uint8 ret = clx_bgr; |
609 |
clx_bgr = 0; // Read and clear |
610 |
return ret; |
611 |
} |
612 |
|
613 |
case 0x20: return ec | 0xf0; |
614 |
case 0x21: return b0c | 0xf0; |
615 |
case 0x22: return b1c | 0xf0; |
616 |
case 0x23: return b2c | 0xf0; |
617 |
case 0x24: return b3c | 0xf0; |
618 |
case 0x25: return mm0 | 0xf0; |
619 |
case 0x26: return mm1 | 0xf0; |
620 |
|
621 |
case 0x27: case 0x28: case 0x29: case 0x2a: |
622 |
case 0x2b: case 0x2c: case 0x2d: case 0x2e: |
623 |
return sc[adr - 0x27] | 0xf0; |
624 |
|
625 |
default: |
626 |
return 0xff; |
627 |
} |
628 |
} |
629 |
|
630 |
|
631 |
/* |
632 |
* Write to VIC register |
633 |
*/ |
634 |
|
635 |
void MOS6569::WriteRegister(uint16 adr, uint8 byte) |
636 |
{ |
637 |
switch (adr) { |
638 |
case 0x00: case 0x02: case 0x04: case 0x06: |
639 |
case 0x08: case 0x0a: case 0x0c: case 0x0e: |
640 |
mx[adr >> 1] = (mx[adr >> 1] & 0xff00) | byte; |
641 |
break; |
642 |
|
643 |
case 0x10:{ |
644 |
int i, j; |
645 |
mx8 = byte; |
646 |
for (i=0, j=1; i<8; i++, j<<=1) { |
647 |
if (mx8 & j) |
648 |
mx[i] |= 0x100; |
649 |
else |
650 |
mx[i] &= 0xff; |
651 |
} |
652 |
break; |
653 |
} |
654 |
|
655 |
case 0x01: case 0x03: case 0x05: case 0x07: |
656 |
case 0x09: case 0x0b: case 0x0d: case 0x0f: |
657 |
my[adr >> 1] = byte; |
658 |
break; |
659 |
|
660 |
case 0x11:{ // Control register 1 |
661 |
ctrl1 = byte; |
662 |
y_scroll = byte & 7; |
663 |
|
664 |
uint16 new_irq_raster = (irq_raster & 0xff) | ((byte & 0x80) << 1); |
665 |
if (irq_raster != new_irq_raster && raster_y == new_irq_raster) |
666 |
raster_irq(); |
667 |
irq_raster = new_irq_raster; |
668 |
|
669 |
if (byte & 8) { |
670 |
dy_start = ROW25_YSTART; |
671 |
dy_stop = ROW25_YSTOP; |
672 |
} else { |
673 |
dy_start = ROW24_YSTART; |
674 |
dy_stop = ROW24_YSTOP; |
675 |
} |
676 |
|
677 |
// In line $30, the DEN bit controls if Bad Lines can occur |
678 |
if (raster_y == 0x30 && byte & 0x10) |
679 |
bad_lines_enabled = true; |
680 |
|
681 |
// Bad Line condition? |
682 |
is_bad_line = (raster_y >= FIRST_DMA_LINE && raster_y <= LAST_DMA_LINE && ((raster_y & 7) == y_scroll) && bad_lines_enabled); |
683 |
|
684 |
display_idx = ((ctrl1 & 0x60) | (ctrl2 & 0x10)) >> 4; |
685 |
break; |
686 |
} |
687 |
|
688 |
case 0x12:{ // Raster counter |
689 |
uint16 new_irq_raster = (irq_raster & 0xff00) | byte; |
690 |
if (irq_raster != new_irq_raster && raster_y == new_irq_raster) |
691 |
raster_irq(); |
692 |
irq_raster = new_irq_raster; |
693 |
break; |
694 |
} |
695 |
|
696 |
case 0x15: // Sprite enable |
697 |
me = byte; |
698 |
break; |
699 |
|
700 |
case 0x16: // Control register 2 |
701 |
ctrl2 = byte; |
702 |
x_scroll = byte & 7; |
703 |
display_idx = ((ctrl1 & 0x60) | (ctrl2 & 0x10)) >> 4; |
704 |
break; |
705 |
|
706 |
case 0x17: // Sprite Y expansion |
707 |
mye = byte; |
708 |
spr_exp_y |= ~byte; |
709 |
break; |
710 |
|
711 |
case 0x18: // Memory pointers |
712 |
vbase = byte; |
713 |
matrix_base = (byte & 0xf0) << 6; |
714 |
char_base = (byte & 0x0e) << 10; |
715 |
bitmap_base = (byte & 0x08) << 10; |
716 |
break; |
717 |
|
718 |
case 0x19: // IRQ flags |
719 |
irq_flag = irq_flag & (~byte & 0x0f); |
720 |
if (irq_flag & irq_mask) // Set master bit if allowed interrupt still pending |
721 |
irq_flag |= 0x80; |
722 |
else |
723 |
the_cpu->ClearVICIRQ(); // Else clear interrupt |
724 |
break; |
725 |
|
726 |
case 0x1a: // IRQ mask |
727 |
irq_mask = byte & 0x0f; |
728 |
if (irq_flag & irq_mask) { // Trigger interrupt if pending and now allowed |
729 |
irq_flag |= 0x80; |
730 |
the_cpu->TriggerVICIRQ(); |
731 |
} else { |
732 |
irq_flag &= 0x7f; |
733 |
the_cpu->ClearVICIRQ(); |
734 |
} |
735 |
break; |
736 |
|
737 |
case 0x1b: // Sprite data priority |
738 |
mdp = byte; |
739 |
break; |
740 |
|
741 |
case 0x1c: // Sprite multicolor |
742 |
mmc = byte; |
743 |
break; |
744 |
|
745 |
case 0x1d: // Sprite X expansion |
746 |
mxe = byte; |
747 |
break; |
748 |
|
749 |
case 0x20: ec_color = colors[ec = byte]; break; |
750 |
case 0x21: b0c_color = colors[b0c = byte]; break; |
751 |
case 0x22: b1c_color = colors[b1c = byte]; break; |
752 |
case 0x23: b2c_color = colors[b2c = byte]; break; |
753 |
case 0x24: b3c_color = colors[b3c = byte]; break; |
754 |
case 0x25: mm0_color = colors[mm0 = byte]; break; |
755 |
case 0x26: mm1_color = colors[mm1 = byte]; break; |
756 |
|
757 |
case 0x27: case 0x28: case 0x29: case 0x2a: |
758 |
case 0x2b: case 0x2c: case 0x2d: case 0x2e: |
759 |
spr_color[adr - 0x27] = colors[sc[adr - 0x27] = byte]; |
760 |
break; |
761 |
} |
762 |
} |
763 |
|
764 |
|
765 |
/* |
766 |
* CIA VA14/15 has changed |
767 |
*/ |
768 |
|
769 |
void MOS6569::ChangedVA(uint16 new_va) |
770 |
{ |
771 |
cia_vabase = new_va << 14; |
772 |
WriteRegister(0x18, vbase); // Force update of memory pointers |
773 |
} |
774 |
|
775 |
|
776 |
/* |
777 |
* Trigger lightpen interrupt, latch lightpen coordinates |
778 |
*/ |
779 |
|
780 |
void MOS6569::TriggerLightpen(void) |
781 |
{ |
782 |
if (!lp_triggered) { // Lightpen triggers only once per frame |
783 |
lp_triggered = true; |
784 |
|
785 |
lpx = raster_x >> 1; // Latch current coordinates |
786 |
lpy = raster_y; |
787 |
|
788 |
irq_flag |= 0x08; // Trigger IRQ |
789 |
if (irq_mask & 0x08) { |
790 |
irq_flag |= 0x80; |
791 |
the_cpu->TriggerVICIRQ(); |
792 |
} |
793 |
} |
794 |
} |
795 |
|
796 |
|
797 |
/* |
798 |
* Read a byte from the VIC's address space |
799 |
*/ |
800 |
|
801 |
#ifdef GLOBAL_VARS |
802 |
static inline uint8 read_byte(uint16 adr) |
803 |
#else |
804 |
inline uint8 MOS6569::read_byte(uint16 adr) |
805 |
#endif |
806 |
{ |
807 |
uint16 va = adr | cia_vabase; |
808 |
if ((va & 0x7000) == 0x1000) |
809 |
#ifdef GLOBAL_VARS |
810 |
return the_vic->LastVICByte = char_rom[va & 0x0fff]; |
811 |
#else |
812 |
return LastVICByte = char_rom[va & 0x0fff]; |
813 |
#endif |
814 |
else |
815 |
#ifdef GLOBAL_VARS |
816 |
return the_vic->LastVICByte = ram[va]; |
817 |
#else |
818 |
return LastVICByte = ram[va]; |
819 |
#endif |
820 |
} |
821 |
|
822 |
|
823 |
/* |
824 |
* Quick memset of 8 bytes |
825 |
*/ |
826 |
|
827 |
inline void memset8(uint8 *p, uint8 c) |
828 |
{ |
829 |
p[0] = p[1] = p[2] = p[3] = p[4] = p[5] = p[6] = p[7] = c; |
830 |
} |
831 |
|
832 |
|
833 |
/* |
834 |
* Video matrix access |
835 |
*/ |
836 |
|
837 |
#ifdef __i386 |
838 |
inline |
839 |
#endif |
840 |
#ifdef GLOBAL_VARS |
841 |
static void matrix_access(void) |
842 |
#else |
843 |
void MOS6569::matrix_access(void) |
844 |
#endif |
845 |
{ |
846 |
if (the_cpu->BALow) { |
847 |
if (the_c64->CycleCounter-first_ba_cycle < 3) |
848 |
matrix_line[ml_index] = color_line[ml_index] = 0xff; |
849 |
else { |
850 |
uint16 adr = (vc & 0x03ff) | matrix_base; |
851 |
matrix_line[ml_index] = read_byte(adr); |
852 |
color_line[ml_index] = color_ram[adr & 0x03ff]; |
853 |
} |
854 |
} |
855 |
} |
856 |
|
857 |
|
858 |
/* |
859 |
* Graphics data access |
860 |
*/ |
861 |
|
862 |
#ifdef __i386 |
863 |
inline |
864 |
#endif |
865 |
#ifdef GLOBAL_VARS |
866 |
static void graphics_access(void) |
867 |
#else |
868 |
void MOS6569::graphics_access(void) |
869 |
#endif |
870 |
{ |
871 |
if (display_state) { |
872 |
|
873 |
uint16 adr; |
874 |
if (ctrl1 & 0x20) // Bitmap |
875 |
adr = ((vc & 0x03ff) << 3) | bitmap_base | rc; |
876 |
else // Text |
877 |
adr = (matrix_line[ml_index] << 3) | char_base | rc; |
878 |
if (ctrl1 & 0x40) // ECM |
879 |
adr &= 0xf9ff; |
880 |
gfx_data = read_byte(adr); |
881 |
char_data = matrix_line[ml_index]; |
882 |
color_data = color_line[ml_index]; |
883 |
ml_index++; |
884 |
vc++; |
885 |
|
886 |
} else { |
887 |
|
888 |
// Display is off |
889 |
gfx_data = read_byte(ctrl1 & 0x40 ? 0x39ff : 0x3fff); |
890 |
char_data = color_data = 0; |
891 |
} |
892 |
} |
893 |
|
894 |
|
895 |
/* |
896 |
* Background display (8 pixels) |
897 |
*/ |
898 |
|
899 |
#ifdef GLOBAL_VARS |
900 |
static void draw_background(void) |
901 |
#else |
902 |
void MOS6569::draw_background(void) |
903 |
#endif |
904 |
{ |
905 |
uint8 *p = chunky_ptr; |
906 |
uint8 c; |
907 |
|
908 |
if (!draw_this_line) |
909 |
return; |
910 |
|
911 |
switch (display_idx) { |
912 |
case 0: // Standard text |
913 |
case 1: // Multicolor text |
914 |
case 3: // Multicolor bitmap |
915 |
c = b0c_color; |
916 |
break; |
917 |
case 2: // Standard bitmap |
918 |
c = colors[last_char_data]; |
919 |
break; |
920 |
case 4: // ECM text |
921 |
if (last_char_data & 0x80) |
922 |
if (last_char_data & 0x40) |
923 |
c = b3c_color; |
924 |
else |
925 |
c = b2c_color; |
926 |
else |
927 |
if (last_char_data & 0x40) |
928 |
c = b1c_color; |
929 |
else |
930 |
c = b0c_color; |
931 |
break; |
932 |
default: |
933 |
c = colors[0]; |
934 |
break; |
935 |
} |
936 |
memset8(p, c); |
937 |
} |
938 |
|
939 |
|
940 |
/* |
941 |
* Graphics display (8 pixels) |
942 |
*/ |
943 |
|
944 |
#ifdef __i386 |
945 |
inline |
946 |
#endif |
947 |
#ifdef GLOBAL_VARS |
948 |
static void draw_graphics(void) |
949 |
#else |
950 |
void MOS6569::draw_graphics(void) |
951 |
#endif |
952 |
{ |
953 |
uint8 *p = chunky_ptr + x_scroll; |
954 |
uint8 c[4], data; |
955 |
|
956 |
if (!draw_this_line) |
957 |
return; |
958 |
if (ud_border_on) { |
959 |
draw_background(); |
960 |
return; |
961 |
} |
962 |
|
963 |
switch (display_idx) { |
964 |
|
965 |
case 0: // Standard text |
966 |
c[0] = b0c_color; |
967 |
c[1] = colors[color_data]; |
968 |
goto draw_std; |
969 |
|
970 |
case 1: // Multicolor text |
971 |
if (color_data & 8) { |
972 |
c[0] = b0c_color; |
973 |
c[1] = b1c_color; |
974 |
c[2] = b2c_color; |
975 |
c[3] = colors[color_data & 7]; |
976 |
goto draw_multi; |
977 |
} else { |
978 |
c[0] = b0c_color; |
979 |
c[1] = colors[color_data]; |
980 |
goto draw_std; |
981 |
} |
982 |
|
983 |
case 2: // Standard bitmap |
984 |
c[0] = colors[char_data]; |
985 |
c[1] = colors[char_data >> 4]; |
986 |
goto draw_std; |
987 |
|
988 |
case 3: // Multicolor bitmap |
989 |
c[0]= b0c_color; |
990 |
c[1] = colors[char_data >> 4]; |
991 |
c[2] = colors[char_data]; |
992 |
c[3] = colors[color_data]; |
993 |
goto draw_multi; |
994 |
|
995 |
case 4: // ECM text |
996 |
if (char_data & 0x80) |
997 |
if (char_data & 0x40) |
998 |
c[0] = b3c_color; |
999 |
else |
1000 |
c[0] = b2c_color; |
1001 |
else |
1002 |
if (char_data & 0x40) |
1003 |
c[0] = b1c_color; |
1004 |
else |
1005 |
c[0] = b0c_color; |
1006 |
c[1] = colors[color_data]; |
1007 |
goto draw_std; |
1008 |
|
1009 |
case 5: // Invalid multicolor text |
1010 |
memset8(p, colors[0]); |
1011 |
if (color_data & 8) { |
1012 |
fore_mask_ptr[0] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) >> x_scroll; |
1013 |
fore_mask_ptr[1] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) << (8-x_scroll); |
1014 |
} else { |
1015 |
fore_mask_ptr[0] |= gfx_data >> x_scroll; |
1016 |
fore_mask_ptr[1] |= gfx_data << (7-x_scroll); |
1017 |
} |
1018 |
return; |
1019 |
|
1020 |
case 6: // Invalid standard bitmap |
1021 |
memset8(p, colors[0]); |
1022 |
fore_mask_ptr[0] |= gfx_data >> x_scroll; |
1023 |
fore_mask_ptr[1] |= gfx_data << (7-x_scroll); |
1024 |
return; |
1025 |
|
1026 |
case 7: // Invalid multicolor bitmap |
1027 |
memset8(p, colors[0]); |
1028 |
fore_mask_ptr[0] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) >> x_scroll; |
1029 |
fore_mask_ptr[1] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) << (8-x_scroll); |
1030 |
return; |
1031 |
|
1032 |
default: // Can't happen |
1033 |
return; |
1034 |
} |
1035 |
|
1036 |
draw_std: |
1037 |
|
1038 |
fore_mask_ptr[0] |= gfx_data >> x_scroll; |
1039 |
fore_mask_ptr[1] |= gfx_data << (7-x_scroll); |
1040 |
|
1041 |
data = gfx_data; |
1042 |
p[7] = c[data & 1]; data >>= 1; |
1043 |
p[6] = c[data & 1]; data >>= 1; |
1044 |
p[5] = c[data & 1]; data >>= 1; |
1045 |
p[4] = c[data & 1]; data >>= 1; |
1046 |
p[3] = c[data & 1]; data >>= 1; |
1047 |
p[2] = c[data & 1]; data >>= 1; |
1048 |
p[1] = c[data & 1]; data >>= 1; |
1049 |
p[0] = c[data]; |
1050 |
return; |
1051 |
|
1052 |
draw_multi: |
1053 |
|
1054 |
fore_mask_ptr[0] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) >> x_scroll; |
1055 |
fore_mask_ptr[1] |= ((gfx_data & 0xaa) | (gfx_data & 0xaa) >> 1) << (8-x_scroll); |
1056 |
|
1057 |
data = gfx_data; |
1058 |
p[7] = p[6] = c[data & 3]; data >>= 2; |
1059 |
p[5] = p[4] = c[data & 3]; data >>= 2; |
1060 |
p[3] = p[2] = c[data & 3]; data >>= 2; |
1061 |
p[1] = p[0] = c[data]; |
1062 |
return; |
1063 |
} |
1064 |
|
1065 |
|
1066 |
/* |
1067 |
* Sprite display |
1068 |
*/ |
1069 |
|
1070 |
#ifdef GLOBAL_VARS |
1071 |
inline static void draw_sprites(void) |
1072 |
#else |
1073 |
inline void MOS6569::draw_sprites(void) |
1074 |
#endif |
1075 |
{ |
1076 |
int i; |
1077 |
int snum, sbit; // Sprite number/bit mask |
1078 |
int spr_coll=0, gfx_coll=0; |
1079 |
|
1080 |
// Clear sprite collision buffer |
1081 |
{ |
1082 |
uint32 *lp = (uint32 *)spr_coll_buf - 1; |
1083 |
for (i=0; i<DISPLAY_X/4; i++) |
1084 |
*++lp = 0; |
1085 |
} |
1086 |
|
1087 |
// Loop for all sprites |
1088 |
for (snum=0, sbit=1; snum<8; snum++, sbit<<=1) { |
1089 |
|
1090 |
// Is sprite visible? |
1091 |
if ((spr_draw & sbit) && mx[snum] <= DISPLAY_X-32) { |
1092 |
#ifdef __POWERPC__ |
1093 |
uint8 *p = (uint8 *)chunky_tmp + mx[snum] + 8; |
1094 |
#else |
1095 |
uint8 *p = chunky_line_start + mx[snum] + 8; |
1096 |
#endif |
1097 |
uint8 *q = spr_coll_buf + mx[snum] + 8; |
1098 |
uint8 color = spr_color[snum]; |
1099 |
|
1100 |
// Fetch sprite data and mask |
1101 |
uint32 sdata = (spr_draw_data[snum][0] << 24) | (spr_draw_data[snum][1] << 16) | (spr_draw_data[snum][2] << 8); |
1102 |
|
1103 |
int spr_mask_pos = mx[snum] + 8; // Sprite bit position in fore_mask_buf |
1104 |
|
1105 |
uint8 *fmbp = fore_mask_buf + (spr_mask_pos / 8); |
1106 |
int sshift = spr_mask_pos & 7; |
1107 |
uint32 fore_mask = (((*(fmbp+0) << 24) | (*(fmbp+1) << 16) | (*(fmbp+2) << 8) |
1108 |
| (*(fmbp+3))) << sshift) | (*(fmbp+4) >> (8-sshift)); |
1109 |
|
1110 |
if (mxe & sbit) { // X-expanded |
1111 |
if (mx[snum] > DISPLAY_X-56) |
1112 |
continue; |
1113 |
|
1114 |
uint32 sdata_l = 0, sdata_r = 0, fore_mask_r; |
1115 |
fore_mask_r = (((*(fmbp+4) << 24) | (*(fmbp+5) << 16) | (*(fmbp+6) << 8) |
1116 |
| (*(fmbp+7))) << sshift) | (*(fmbp+8) >> (8-sshift)); |
1117 |
|
1118 |
if (mmc & sbit) { // Multicolor mode |
1119 |
uint32 plane0_l, plane0_r, plane1_l, plane1_r; |
1120 |
|
1121 |
// Expand sprite data |
1122 |
sdata_l = MultiExpTable[sdata >> 24 & 0xff] << 16 | MultiExpTable[sdata >> 16 & 0xff]; |
1123 |
sdata_r = MultiExpTable[sdata >> 8 & 0xff] << 16; |
1124 |
|
1125 |
// Convert sprite chunky pixels to bitplanes |
1126 |
plane0_l = (sdata_l & 0x55555555) | (sdata_l & 0x55555555) << 1; |
1127 |
plane1_l = (sdata_l & 0xaaaaaaaa) | (sdata_l & 0xaaaaaaaa) >> 1; |
1128 |
plane0_r = (sdata_r & 0x55555555) | (sdata_r & 0x55555555) << 1; |
1129 |
plane1_r = (sdata_r & 0xaaaaaaaa) | (sdata_r & 0xaaaaaaaa) >> 1; |
1130 |
|
1131 |
// Collision with graphics? |
1132 |
if ((fore_mask & (plane0_l | plane1_l)) || (fore_mask_r & (plane0_r | plane1_r))) { |
1133 |
gfx_coll |= sbit; |
1134 |
if (mdp & sbit) { |
1135 |
plane0_l &= ~fore_mask; // Mask sprite if in background |
1136 |
plane1_l &= ~fore_mask; |
1137 |
plane0_r &= ~fore_mask_r; |
1138 |
plane1_r &= ~fore_mask_r; |
1139 |
} |
1140 |
} |
1141 |
|
1142 |
// Paint sprite |
1143 |
for (i=0; i<32; i++, plane0_l<<=1, plane1_l<<=1) { |
1144 |
uint8 col; |
1145 |
if (plane1_l & 0x80000000) { |
1146 |
if (plane0_l & 0x80000000) |
1147 |
col = mm1_color; |
1148 |
else |
1149 |
col = color; |
1150 |
} else { |
1151 |
if (plane0_l & 0x80000000) |
1152 |
col = mm0_color; |
1153 |
else |
1154 |
continue; |
1155 |
} |
1156 |
if (q[i]) |
1157 |
spr_coll |= q[i] | sbit; |
1158 |
else { |
1159 |
p[i] = col; |
1160 |
q[i] = sbit; |
1161 |
} |
1162 |
} |
1163 |
for (; i<48; i++, plane0_r<<=1, plane1_r<<=1) { |
1164 |
uint8 col; |
1165 |
if (plane1_r & 0x80000000) { |
1166 |
if (plane0_r & 0x80000000) |
1167 |
col = mm1_color; |
1168 |
else |
1169 |
col = color; |
1170 |
} else { |
1171 |
if (plane0_r & 0x80000000) |
1172 |
col = mm0_color; |
1173 |
else |
1174 |
continue; |
1175 |
} |
1176 |
if (q[i]) |
1177 |
spr_coll |= q[i] | sbit; |
1178 |
else { |
1179 |
p[i] = col; |
1180 |
q[i] = sbit; |
1181 |
} |
1182 |
} |
1183 |
|
1184 |
} else { // Standard mode |
1185 |
|
1186 |
// Expand sprite data |
1187 |
sdata_l = ExpTable[sdata >> 24 & 0xff] << 16 | ExpTable[sdata >> 16 & 0xff]; |
1188 |
sdata_r = ExpTable[sdata >> 8 & 0xff] << 16; |
1189 |
|
1190 |
// Collision with graphics? |
1191 |
if ((fore_mask & sdata_l) || (fore_mask_r & sdata_r)) { |
1192 |
gfx_coll |= sbit; |
1193 |
if (mdp & sbit) { |
1194 |
sdata_l &= ~fore_mask; // Mask sprite if in background |
1195 |
sdata_r &= ~fore_mask_r; |
1196 |
} |
1197 |
} |
1198 |
|
1199 |
// Paint sprite |
1200 |
for (i=0; i<32; i++, sdata_l<<=1) |
1201 |
if (sdata_l & 0x80000000) { |
1202 |
if (q[i]) // Collision with sprite? |
1203 |
spr_coll |= q[i] | sbit; |
1204 |
else { // Draw pixel if no collision |
1205 |
p[i] = color; |
1206 |
q[i] = sbit; |
1207 |
} |
1208 |
} |
1209 |
for (; i<48; i++, sdata_r<<=1) |
1210 |
if (sdata_r & 0x80000000) { |
1211 |
if (q[i]) // Collision with sprite? |
1212 |
spr_coll |= q[i] | sbit; |
1213 |
else { // Draw pixel if no collision |
1214 |
p[i] = color; |
1215 |
q[i] = sbit; |
1216 |
} |
1217 |
} |
1218 |
} |
1219 |
|
1220 |
} else { // Unexpanded |
1221 |
|
1222 |
if (mmc & sbit) { // Multicolor mode |
1223 |
uint32 plane0, plane1; |
1224 |
|
1225 |
// Convert sprite chunky pixels to bitplanes |
1226 |
plane0 = (sdata & 0x55555555) | (sdata & 0x55555555) << 1; |
1227 |
plane1 = (sdata & 0xaaaaaaaa) | (sdata & 0xaaaaaaaa) >> 1; |
1228 |
|
1229 |
// Collision with graphics? |
1230 |
if (fore_mask & (plane0 | plane1)) { |
1231 |
gfx_coll |= sbit; |
1232 |
if (mdp & sbit) { |
1233 |
plane0 &= ~fore_mask; // Mask sprite if in background |
1234 |
plane1 &= ~fore_mask; |
1235 |
} |
1236 |
} |
1237 |
|
1238 |
// Paint sprite |
1239 |
for (i=0; i<24; i++, plane0<<=1, plane1<<=1) { |
1240 |
uint8 col; |
1241 |
if (plane1 & 0x80000000) { |
1242 |
if (plane0 & 0x80000000) |
1243 |
col = mm1_color; |
1244 |
else |
1245 |
col = color; |
1246 |
} else { |
1247 |
if (plane0 & 0x80000000) |
1248 |
col = mm0_color; |
1249 |
else |
1250 |
continue; |
1251 |
} |
1252 |
if (q[i]) |
1253 |
spr_coll |= q[i] | sbit; |
1254 |
else { |
1255 |
p[i] = col; |
1256 |
q[i] = sbit; |
1257 |
} |
1258 |
} |
1259 |
|
1260 |
} else { // Standard mode |
1261 |
|
1262 |
// Collision with graphics? |
1263 |
if (fore_mask & sdata) { |
1264 |
gfx_coll |= sbit; |
1265 |
if (mdp & sbit) |
1266 |
sdata &= ~fore_mask; // Mask sprite if in background |
1267 |
} |
1268 |
|
1269 |
// Paint sprite |
1270 |
for (i=0; i<24; i++, sdata<<=1) |
1271 |
if (sdata & 0x80000000) { |
1272 |
if (q[i]) { // Collision with sprite? |
1273 |
spr_coll |= q[i] | sbit; |
1274 |
} else { // Draw pixel if no collision |
1275 |
p[i] = color; |
1276 |
q[i] = sbit; |
1277 |
} |
1278 |
} |
1279 |
} |
1280 |
} |
1281 |
} |
1282 |
} |
1283 |
|
1284 |
if (ThePrefs.SpriteCollisions) { |
1285 |
|
1286 |
// Check sprite-sprite collisions |
1287 |
if (clx_spr) |
1288 |
clx_spr |= spr_coll; |
1289 |
else { |
1290 |
clx_spr |= spr_coll; |
1291 |
irq_flag |= 0x04; |
1292 |
if (irq_mask & 0x04) { |
1293 |
irq_flag |= 0x80; |
1294 |
the_cpu->TriggerVICIRQ(); |
1295 |
} |
1296 |
} |
1297 |
|
1298 |
// Check sprite-background collisions |
1299 |
if (clx_bgr) |
1300 |
clx_bgr |= gfx_coll; |
1301 |
else { |
1302 |
clx_bgr |= gfx_coll; |
1303 |
irq_flag |= 0x02; |
1304 |
if (irq_mask & 0x02) { |
1305 |
irq_flag |= 0x80; |
1306 |
the_cpu->TriggerVICIRQ(); |
1307 |
} |
1308 |
} |
1309 |
} |
1310 |
} |
1311 |
|
1312 |
|
1313 |
#ifdef __POWERPC__ |
1314 |
static asm void fastcopy(register uchar *dst, register uchar *src); |
1315 |
static asm void fastcopy(register uchar *dst, register uchar *src) |
1316 |
{ |
1317 |
lfd fp0,0(src) |
1318 |
lfd fp1,8(src) |
1319 |
lfd fp2,16(src) |
1320 |
lfd fp3,24(src) |
1321 |
lfd fp4,32(src) |
1322 |
lfd fp5,40(src) |
1323 |
lfd fp6,48(src) |
1324 |
lfd fp7,56(src) |
1325 |
addi src,src,64 |
1326 |
stfd fp0,0(dst) |
1327 |
stfd fp1,8(dst) |
1328 |
stfd fp2,16(dst) |
1329 |
stfd fp3,24(dst) |
1330 |
stfd fp4,32(dst) |
1331 |
stfd fp5,40(dst) |
1332 |
stfd fp6,48(dst) |
1333 |
stfd fp7,56(dst) |
1334 |
addi dst,dst,64 |
1335 |
|
1336 |
lfd fp0,0(src) |
1337 |
lfd fp1,8(src) |
1338 |
lfd fp2,16(src) |
1339 |
lfd fp3,24(src) |
1340 |
lfd fp4,32(src) |
1341 |
lfd fp5,40(src) |
1342 |
lfd fp6,48(src) |
1343 |
lfd fp7,56(src) |
1344 |
addi src,src,64 |
1345 |
stfd fp0,0(dst) |
1346 |
stfd fp1,8(dst) |
1347 |
stfd fp2,16(dst) |
1348 |
stfd fp3,24(dst) |
1349 |
stfd fp4,32(dst) |
1350 |
stfd fp5,40(dst) |
1351 |
stfd fp6,48(dst) |
1352 |
stfd fp7,56(dst) |
1353 |
addi dst,dst,64 |
1354 |
|
1355 |
lfd fp0,0(src) |
1356 |
lfd fp1,8(src) |
1357 |
lfd fp2,16(src) |
1358 |
lfd fp3,24(src) |
1359 |
lfd fp4,32(src) |
1360 |
lfd fp5,40(src) |
1361 |
lfd fp6,48(src) |
1362 |
lfd fp7,56(src) |
1363 |
addi src,src,64 |
1364 |
stfd fp0,0(dst) |
1365 |
stfd fp1,8(dst) |
1366 |
stfd fp2,16(dst) |
1367 |
stfd fp3,24(dst) |
1368 |
stfd fp4,32(dst) |
1369 |
stfd fp5,40(dst) |
1370 |
stfd fp6,48(dst) |
1371 |
stfd fp7,56(dst) |
1372 |
addi dst,dst,64 |
1373 |
|
1374 |
lfd fp0,0(src) |
1375 |
lfd fp1,8(src) |
1376 |
lfd fp2,16(src) |
1377 |
lfd fp3,24(src) |
1378 |
lfd fp4,32(src) |
1379 |
lfd fp5,40(src) |
1380 |
lfd fp6,48(src) |
1381 |
lfd fp7,56(src) |
1382 |
addi src,src,64 |
1383 |
stfd fp0,0(dst) |
1384 |
stfd fp1,8(dst) |
1385 |
stfd fp2,16(dst) |
1386 |
stfd fp3,24(dst) |
1387 |
stfd fp4,32(dst) |
1388 |
stfd fp5,40(dst) |
1389 |
stfd fp6,48(dst) |
1390 |
stfd fp7,56(dst) |
1391 |
addi dst,dst,64 |
1392 |
|
1393 |
lfd fp0,0(src) |
1394 |
lfd fp1,8(src) |
1395 |
lfd fp2,16(src) |
1396 |
lfd fp3,24(src) |
1397 |
lfd fp4,32(src) |
1398 |
lfd fp5,40(src) |
1399 |
lfd fp6,48(src) |
1400 |
lfd fp7,56(src) |
1401 |
addi src,src,64 |
1402 |
stfd fp0,0(dst) |
1403 |
stfd fp1,8(dst) |
1404 |
stfd fp2,16(dst) |
1405 |
stfd fp3,24(dst) |
1406 |
stfd fp4,32(dst) |
1407 |
stfd fp5,40(dst) |
1408 |
stfd fp6,48(dst) |
1409 |
stfd fp7,56(dst) |
1410 |
addi dst,dst,64 |
1411 |
|
1412 |
lfd fp0,0(src) |
1413 |
lfd fp1,8(src) |
1414 |
lfd fp2,16(src) |
1415 |
lfd fp3,24(src) |
1416 |
lfd fp4,32(src) |
1417 |
lfd fp5,40(src) |
1418 |
lfd fp6,48(src) |
1419 |
lfd fp7,56(src) |
1420 |
addi src,src,64 |
1421 |
stfd fp0,0(dst) |
1422 |
stfd fp1,8(dst) |
1423 |
stfd fp2,16(dst) |
1424 |
stfd fp3,24(dst) |
1425 |
stfd fp4,32(dst) |
1426 |
stfd fp5,40(dst) |
1427 |
stfd fp6,48(dst) |
1428 |
stfd fp7,56(dst) |
1429 |
addi dst,dst,64 |
1430 |
blr |
1431 |
} |
1432 |
#endif |
1433 |
|
1434 |
|
1435 |
/* |
1436 |
* Emulate one clock cycle, returns true if new raster line has started |
1437 |
*/ |
1438 |
|
1439 |
// Set BA low |
1440 |
#define SetBALow \ |
1441 |
if (!the_cpu->BALow) { \ |
1442 |
first_ba_cycle = the_c64->CycleCounter; \ |
1443 |
the_cpu->BALow = true; \ |
1444 |
} |
1445 |
|
1446 |
// Turn on display if Bad Line |
1447 |
#define DisplayIfBadLine \ |
1448 |
if (is_bad_line) \ |
1449 |
display_state = true; |
1450 |
|
1451 |
// Turn on display and matrix access if Bad Line |
1452 |
#define FetchIfBadLine \ |
1453 |
if (is_bad_line) { \ |
1454 |
display_state = true; \ |
1455 |
SetBALow; \ |
1456 |
} |
1457 |
|
1458 |
// Turn on display and matrix access and reset RC if Bad Line |
1459 |
#define RCIfBadLine \ |
1460 |
if (is_bad_line) { \ |
1461 |
display_state = true; \ |
1462 |
rc = 0; \ |
1463 |
SetBALow; \ |
1464 |
} |
1465 |
|
1466 |
// Idle access |
1467 |
#define IdleAccess \ |
1468 |
read_byte(0x3fff) |
1469 |
|
1470 |
// Refresh access |
1471 |
#define RefreshAccess \ |
1472 |
read_byte(0x3f00 | ref_cnt--) |
1473 |
|
1474 |
// Turn on sprite DMA if necessary |
1475 |
#define CheckSpriteDMA \ |
1476 |
mask = 1; \ |
1477 |
for (i=0; i<8; i++, mask<<=1) \ |
1478 |
if ((me & mask) && (raster_y & 0xff) == my[i]) { \ |
1479 |
spr_dma_on |= mask; \ |
1480 |
mc_base[i] = 0; \ |
1481 |
if (mye & mask) \ |
1482 |
spr_exp_y &= ~mask; \ |
1483 |
} |
1484 |
|
1485 |
// Fetch sprite data pointer |
1486 |
#define SprPtrAccess(num) \ |
1487 |
spr_ptr[num] = read_byte(matrix_base | 0x03f8 | num) << 6; |
1488 |
|
1489 |
// Fetch sprite data, increment data counter |
1490 |
#define SprDataAccess(num, bytenum) \ |
1491 |
if (spr_dma_on & (1 << num)) { \ |
1492 |
spr_data[num][bytenum] = read_byte(mc[num] & 0x3f | spr_ptr[num]); \ |
1493 |
mc[num]++; \ |
1494 |
} else if (bytenum == 1) \ |
1495 |
IdleAccess; |
1496 |
|
1497 |
// Sample border color and increment chunky_ptr and fore_mask_ptr |
1498 |
#define SampleBorder \ |
1499 |
if (draw_this_line) { \ |
1500 |
if (border_on) \ |
1501 |
border_color_sample[cycle-13] = ec_color; \ |
1502 |
chunky_ptr += 8; \ |
1503 |
fore_mask_ptr++; \ |
1504 |
} |
1505 |
|
1506 |
|
1507 |
bool MOS6569::EmulateCycle(void) |
1508 |
{ |
1509 |
uint8 mask; |
1510 |
int i; |
1511 |
|
1512 |
switch (cycle) { |
1513 |
|
1514 |
// Fetch sprite pointer 3, increment raster counter, trigger raster IRQ, |
1515 |
// test for Bad Line, reset BA if sprites 3 and 4 off, read data of sprite 3 |
1516 |
case 1: |
1517 |
if (raster_y == TOTAL_RASTERS-1) |
1518 |
|
1519 |
// Trigger VBlank in cycle 2 |
1520 |
vblanking = true; |
1521 |
|
1522 |
else { |
1523 |
|
1524 |
// Increment raster counter |
1525 |
raster_y++; |
1526 |
|
1527 |
// Trigger raster IRQ if IRQ line reached |
1528 |
if (raster_y == irq_raster) |
1529 |
raster_irq(); |
1530 |
|
1531 |
// In line $30, the DEN bit controls if Bad Lines can occur |
1532 |
if (raster_y == 0x30) |
1533 |
bad_lines_enabled = ctrl1 & 0x10; |
1534 |
|
1535 |
// Bad Line condition? |
1536 |
is_bad_line = (raster_y >= FIRST_DMA_LINE && raster_y <= LAST_DMA_LINE && ((raster_y & 7) == y_scroll) && bad_lines_enabled); |
1537 |
|
1538 |
// Don't draw all lines, hide some at the top and bottom |
1539 |
draw_this_line = (raster_y >= FIRST_DISP_LINE && raster_y <= LAST_DISP_LINE && !frame_skipped); |
1540 |
} |
1541 |
|
1542 |
// First sample of border state |
1543 |
border_on_sample[0] = border_on; |
1544 |
|
1545 |
SprPtrAccess(3); |
1546 |
SprDataAccess(3, 0); |
1547 |
DisplayIfBadLine; |
1548 |
if (!(spr_dma_on & 0x18)) |
1549 |
the_cpu->BALow = false; |
1550 |
break; |
1551 |
|
1552 |
// Set BA for sprite 5, read data of sprite 3 |
1553 |
case 2: |
1554 |
if (vblanking) { |
1555 |
|
1556 |
// Vertical blank, reset counters |
1557 |
raster_y = vc_base = 0; |
1558 |
ref_cnt = 0xff; |
1559 |
lp_triggered = vblanking = false; |
1560 |
|
1561 |
if (!(frame_skipped = --skip_counter)) |
1562 |
skip_counter = ThePrefs.SkipFrames; |
1563 |
|
1564 |
the_c64->VBlank(!frame_skipped); |
1565 |
|
1566 |
// Get bitmap pointer for next frame. This must be done |
1567 |
// after calling the_c64->VBlank() because the preferences |
1568 |
// and screen configuration may have been changed there |
1569 |
chunky_line_start = the_display->BitmapBase(); |
1570 |
xmod = the_display->BitmapXMod(); |
1571 |
|
1572 |
// Trigger raster IRQ if IRQ in line 0 |
1573 |
if (irq_raster == 0) |
1574 |
raster_irq(); |
1575 |
|
1576 |
} |
1577 |
|
1578 |
// Our output goes here |
1579 |
#ifdef __POWERPC__ |
1580 |
chunky_ptr = (uint8 *)chunky_tmp; |
1581 |
#else |
1582 |
chunky_ptr = chunky_line_start; |
1583 |
#endif |
1584 |
|
1585 |
// Clear foreground mask |
1586 |
memset(fore_mask_buf, 0, DISPLAY_X/8); |
1587 |
fore_mask_ptr = fore_mask_buf; |
1588 |
|
1589 |
SprDataAccess(3,1); |
1590 |
SprDataAccess(3,2); |
1591 |
DisplayIfBadLine; |
1592 |
if (spr_dma_on & 0x20) |
1593 |
SetBALow; |
1594 |
break; |
1595 |
|
1596 |
// Fetch sprite pointer 4, reset BA is sprite 4 and 5 off |
1597 |
case 3: |
1598 |
SprPtrAccess(4); |
1599 |
SprDataAccess(4, 0); |
1600 |
DisplayIfBadLine; |
1601 |
if (!(spr_dma_on & 0x30)) |
1602 |
the_cpu->BALow = false; |
1603 |
break; |
1604 |
|
1605 |
// Set BA for sprite 6, read data of sprite 4 |
1606 |
case 4: |
1607 |
SprDataAccess(4, 1); |
1608 |
SprDataAccess(4, 2); |
1609 |
DisplayIfBadLine; |
1610 |
if (spr_dma_on & 0x40) |
1611 |
SetBALow; |
1612 |
break; |
1613 |
|
1614 |
// Fetch sprite pointer 5, reset BA if sprite 5 and 6 off |
1615 |
case 5: |
1616 |
SprPtrAccess(5); |
1617 |
SprDataAccess(5, 0); |
1618 |
DisplayIfBadLine; |
1619 |
if (!(spr_dma_on & 0x60)) |
1620 |
the_cpu->BALow = false; |
1621 |
break; |
1622 |
|
1623 |
// Set BA for sprite 7, read data of sprite 5 |
1624 |
case 6: |
1625 |
SprDataAccess(5, 1); |
1626 |
SprDataAccess(5, 2); |
1627 |
DisplayIfBadLine; |
1628 |
if (spr_dma_on & 0x80) |
1629 |
SetBALow; |
1630 |
break; |
1631 |
|
1632 |
// Fetch sprite pointer 6, reset BA if sprite 6 and 7 off |
1633 |
case 7: |
1634 |
SprPtrAccess(6); |
1635 |
SprDataAccess(6, 0); |
1636 |
DisplayIfBadLine; |
1637 |
if (!(spr_dma_on & 0xc0)) |
1638 |
the_cpu->BALow = false; |
1639 |
break; |
1640 |
|
1641 |
// Read data of sprite 6 |
1642 |
case 8: |
1643 |
SprDataAccess(6, 1); |
1644 |
SprDataAccess(6, 2); |
1645 |
DisplayIfBadLine; |
1646 |
break; |
1647 |
|
1648 |
// Fetch sprite pointer 7, reset BA if sprite 7 off |
1649 |
case 9: |
1650 |
SprPtrAccess(7); |
1651 |
SprDataAccess(7, 0); |
1652 |
DisplayIfBadLine; |
1653 |
if (!(spr_dma_on & 0x80)) |
1654 |
the_cpu->BALow = false; |
1655 |
break; |
1656 |
|
1657 |
// Read data of sprite 7 |
1658 |
case 10: |
1659 |
SprDataAccess(7, 1); |
1660 |
SprDataAccess(7, 2); |
1661 |
DisplayIfBadLine; |
1662 |
break; |
1663 |
|
1664 |
// Refresh, reset BA |
1665 |
case 11: |
1666 |
RefreshAccess; |
1667 |
DisplayIfBadLine; |
1668 |
the_cpu->BALow = false; |
1669 |
break; |
1670 |
|
1671 |
// Refresh, turn on matrix access if Bad Line |
1672 |
case 12: |
1673 |
RefreshAccess; |
1674 |
FetchIfBadLine; |
1675 |
break; |
1676 |
|
1677 |
// Refresh, turn on matrix access if Bad Line, reset raster_x, graphics display starts here |
1678 |
case 13: |
1679 |
draw_background(); |
1680 |
SampleBorder; |
1681 |
RefreshAccess; |
1682 |
FetchIfBadLine; |
1683 |
raster_x = 0xfffc; |
1684 |
break; |
1685 |
|
1686 |
// Refresh, VCBASE->VCCOUNT, turn on matrix access and reset RC if Bad Line |
1687 |
case 14: |
1688 |
draw_background(); |
1689 |
SampleBorder; |
1690 |
RefreshAccess; |
1691 |
RCIfBadLine; |
1692 |
vc = vc_base; |
1693 |
break; |
1694 |
|
1695 |
// Refresh and matrix access, increment mc_base by 2 if y expansion flipflop is set |
1696 |
case 15: |
1697 |
draw_background(); |
1698 |
SampleBorder; |
1699 |
RefreshAccess; |
1700 |
FetchIfBadLine; |
1701 |
|
1702 |
for (i=0; i<8; i++) |
1703 |
if (spr_exp_y & (1 << i)) |
1704 |
mc_base[i] += 2; |
1705 |
|
1706 |
ml_index = 0; |
1707 |
matrix_access(); |
1708 |
break; |
1709 |
|
1710 |
// Graphics and matrix access, increment mc_base by 1 if y expansion flipflop is set |
1711 |
// and check if sprite DMA can be turned off |
1712 |
case 16: |
1713 |
draw_background(); |
1714 |
SampleBorder; |
1715 |
graphics_access(); |
1716 |
FetchIfBadLine; |
1717 |
|
1718 |
mask = 1; |
1719 |
for (i=0; i<8; i++, mask<<=1) { |
1720 |
if (spr_exp_y & mask) |
1721 |
mc_base[i]++; |
1722 |
if ((mc_base[i] & 0x3f) == 0x3f) |
1723 |
spr_dma_on &= ~mask; |
1724 |
} |
1725 |
|
1726 |
matrix_access(); |
1727 |
break; |
1728 |
|
1729 |
// Graphics and matrix access, turn off border in 40 column mode, display window starts here |
1730 |
case 17: |
1731 |
if (ctrl2 & 8) { |
1732 |
if (raster_y == dy_stop) |
1733 |
ud_border_on = true; |
1734 |
else { |
1735 |
if (ctrl1 & 0x10) { |
1736 |
if (raster_y == dy_start) |
1737 |
border_on = ud_border_on = false; |
1738 |
else |
1739 |
if (!ud_border_on) |
1740 |
border_on = false; |
1741 |
} else |
1742 |
if (!ud_border_on) |
1743 |
border_on = false; |
1744 |
} |
1745 |
} |
1746 |
|
1747 |
// Second sample of border state |
1748 |
border_on_sample[1] = border_on; |
1749 |
|
1750 |
draw_background(); |
1751 |
draw_graphics(); |
1752 |
SampleBorder; |
1753 |
graphics_access(); |
1754 |
FetchIfBadLine; |
1755 |
matrix_access(); |
1756 |
break; |
1757 |
|
1758 |
// Turn off border in 38 column mode |
1759 |
case 18: |
1760 |
if (!(ctrl2 & 8)) { |
1761 |
if (raster_y == dy_stop) |
1762 |
ud_border_on = true; |
1763 |
else { |
1764 |
if (ctrl1 & 0x10) { |
1765 |
if (raster_y == dy_start) |
1766 |
border_on = ud_border_on = false; |
1767 |
else |
1768 |
if (!ud_border_on) |
1769 |
border_on = false; |
1770 |
} else |
1771 |
if (!ud_border_on) |
1772 |
border_on = false; |
1773 |
} |
1774 |
} |
1775 |
|
1776 |
// Third sample of border state |
1777 |
border_on_sample[2] = border_on; |
1778 |
|
1779 |
// Falls through |
1780 |
|
1781 |
// Graphics and matrix access |
1782 |
case 19: case 20: case 21: case 22: case 23: case 24: |
1783 |
case 25: case 26: case 27: case 28: case 29: case 30: |
1784 |
case 31: case 32: case 33: case 34: case 35: case 36: |
1785 |
case 37: case 38: case 39: case 40: case 41: case 42: |
1786 |
case 43: case 44: case 45: case 46: case 47: case 48: |
1787 |
case 49: case 50: case 51: case 52: case 53: case 54: // Gnagna... |
1788 |
draw_graphics(); |
1789 |
SampleBorder; |
1790 |
graphics_access(); |
1791 |
FetchIfBadLine; |
1792 |
matrix_access(); |
1793 |
last_char_data = char_data; |
1794 |
break; |
1795 |
|
1796 |
// Last graphics access, turn off matrix access, turn on sprite DMA if Y coordinate is |
1797 |
// right and sprite is enabled, handle sprite y expansion, set BA for sprite 0 |
1798 |
case 55: |
1799 |
draw_graphics(); |
1800 |
SampleBorder; |
1801 |
graphics_access(); |
1802 |
DisplayIfBadLine; |
1803 |
|
1804 |
// Invert y expansion flipflop if bit in MYE is set |
1805 |
mask = 1; |
1806 |
for (i=0; i<8; i++, mask<<=1) |
1807 |
if (mye & mask) |
1808 |
spr_exp_y ^= mask; |
1809 |
CheckSpriteDMA; |
1810 |
|
1811 |
if (spr_dma_on & 0x01) { // Don't remove these braces! |
1812 |
SetBALow; |
1813 |
} else |
1814 |
the_cpu->BALow = false; |
1815 |
break; |
1816 |
|
1817 |
// Turn on border in 38 column mode, turn on sprite DMA if Y coordinate is right and |
1818 |
// sprite is enabled, set BA for sprite 0, display window ends here |
1819 |
case 56: |
1820 |
if (!(ctrl2 & 8)) |
1821 |
border_on = true; |
1822 |
|
1823 |
// Fourth sample of border state |
1824 |
border_on_sample[3] = border_on; |
1825 |
|
1826 |
draw_graphics(); |
1827 |
SampleBorder; |
1828 |
IdleAccess; |
1829 |
DisplayIfBadLine; |
1830 |
CheckSpriteDMA; |
1831 |
|
1832 |
if (spr_dma_on & 0x01) |
1833 |
SetBALow; |
1834 |
break; |
1835 |
|
1836 |
// Turn on border in 40 column mode, set BA for sprite 1, paint sprites |
1837 |
case 57: |
1838 |
if (ctrl2 & 8) |
1839 |
border_on = true; |
1840 |
|
1841 |
// Fifth sample of border state |
1842 |
border_on_sample[4] = border_on; |
1843 |
|
1844 |
// Sample spr_disp_on and spr_data for sprite drawing |
1845 |
if ((spr_draw = spr_disp_on) != 0) |
1846 |
memcpy(spr_draw_data, spr_data, 8*4); |
1847 |
|
1848 |
// Turn off sprite display if DMA is off |
1849 |
mask = 1; |
1850 |
for (i=0; i<8; i++, mask<<=1) |
1851 |
if ((spr_disp_on & mask) && !(spr_dma_on & mask)) |
1852 |
spr_disp_on &= ~mask; |
1853 |
|
1854 |
draw_background(); |
1855 |
SampleBorder; |
1856 |
IdleAccess; |
1857 |
DisplayIfBadLine; |
1858 |
if (spr_dma_on & 0x02) |
1859 |
SetBALow; |
1860 |
break; |
1861 |
|
1862 |
// Fetch sprite pointer 0, mc_base->mc, turn on sprite display if necessary, |
1863 |
// turn off display if RC=7, read data of sprite 0 |
1864 |
case 58: |
1865 |
draw_background(); |
1866 |
SampleBorder; |
1867 |
|
1868 |
mask = 1; |
1869 |
for (i=0; i<8; i++, mask<<=1) { |
1870 |
mc[i] = mc_base[i]; |
1871 |
if ((spr_dma_on & mask) && (raster_y & 0xff) == my[i]) |
1872 |
spr_disp_on |= mask; |
1873 |
} |
1874 |
SprPtrAccess(0); |
1875 |
SprDataAccess(0, 0); |
1876 |
|
1877 |
if (rc == 7) { |
1878 |
vc_base = vc; |
1879 |
display_state = false; |
1880 |
} |
1881 |
if (is_bad_line || display_state) { |
1882 |
display_state = true; |
1883 |
rc = (rc + 1) & 7; |
1884 |
} |
1885 |
break; |
1886 |
|
1887 |
// Set BA for sprite 2, read data of sprite 0 |
1888 |
case 59: |
1889 |
draw_background(); |
1890 |
SampleBorder; |
1891 |
SprDataAccess(0, 1); |
1892 |
SprDataAccess(0, 2); |
1893 |
DisplayIfBadLine; |
1894 |
if (spr_dma_on & 0x04) |
1895 |
SetBALow; |
1896 |
break; |
1897 |
|
1898 |
// Fetch sprite pointer 1, reset BA if sprite 1 and 2 off, graphics display ends here |
1899 |
case 60: |
1900 |
draw_background(); |
1901 |
SampleBorder; |
1902 |
|
1903 |
if (draw_this_line) { |
1904 |
|
1905 |
// Draw sprites |
1906 |
if (spr_draw && ThePrefs.SpritesOn) |
1907 |
draw_sprites(); |
1908 |
|
1909 |
// Draw border |
1910 |
#ifdef __POWERPC__ |
1911 |
if (border_on_sample[0]) |
1912 |
for (i=0; i<4; i++) |
1913 |
memset8((uint8 *)chunky_tmp+i*8, border_color_sample[i]); |
1914 |
if (border_on_sample[1]) |
1915 |
memset8((uint8 *)chunky_tmp+4*8, border_color_sample[4]); |
1916 |
if (border_on_sample[2]) |
1917 |
for (i=5; i<43; i++) |
1918 |
memset8((uint8 *)chunky_tmp+i*8, border_color_sample[i]); |
1919 |
if (border_on_sample[3]) |
1920 |
memset8((uint8 *)chunky_tmp+43*8, border_color_sample[43]); |
1921 |
if (border_on_sample[4]) |
1922 |
for (i=44; i<DISPLAY_X/8; i++) |
1923 |
memset8((uint8 *)chunky_tmp+i*8, border_color_sample[i]); |
1924 |
#else |
1925 |
if (border_on_sample[0]) |
1926 |
for (i=0; i<4; i++) |
1927 |
memset8(chunky_line_start+i*8, border_color_sample[i]); |
1928 |
if (border_on_sample[1]) |
1929 |
memset8(chunky_line_start+4*8, border_color_sample[4]); |
1930 |
if (border_on_sample[2]) |
1931 |
for (i=5; i<43; i++) |
1932 |
memset8(chunky_line_start+i*8, border_color_sample[i]); |
1933 |
if (border_on_sample[3]) |
1934 |
memset8(chunky_line_start+43*8, border_color_sample[43]); |
1935 |
if (border_on_sample[4]) |
1936 |
for (i=44; i<DISPLAY_X/8; i++) |
1937 |
memset8(chunky_line_start+i*8, border_color_sample[i]); |
1938 |
#endif |
1939 |
|
1940 |
#ifdef __POWERPC__ |
1941 |
// Copy temporary buffer to bitmap |
1942 |
fastcopy(chunky_line_start, (uint8 *)chunky_tmp); |
1943 |
#endif |
1944 |
|
1945 |
// Increment pointer in chunky buffer |
1946 |
chunky_line_start += xmod; |
1947 |
} |
1948 |
|
1949 |
SprPtrAccess(1); |
1950 |
SprDataAccess(1, 0); |
1951 |
DisplayIfBadLine; |
1952 |
if (!(spr_dma_on & 0x06)) |
1953 |
the_cpu->BALow = false; |
1954 |
break; |
1955 |
|
1956 |
// Set BA for sprite 3, read data of sprite 1 |
1957 |
case 61: |
1958 |
SprDataAccess(1, 1); |
1959 |
SprDataAccess(1, 2); |
1960 |
DisplayIfBadLine; |
1961 |
if (spr_dma_on & 0x08) |
1962 |
SetBALow; |
1963 |
break; |
1964 |
|
1965 |
// Read sprite pointer 2, reset BA if sprite 2 and 3 off, read data of sprite 2 |
1966 |
case 62: |
1967 |
SprPtrAccess(2); |
1968 |
SprDataAccess(2, 0); |
1969 |
DisplayIfBadLine; |
1970 |
if (!(spr_dma_on & 0x0c)) |
1971 |
the_cpu->BALow = false; |
1972 |
break; |
1973 |
|
1974 |
// Set BA for sprite 4, read data of sprite 2 |
1975 |
case 63: |
1976 |
SprDataAccess(2, 1); |
1977 |
SprDataAccess(2, 2); |
1978 |
DisplayIfBadLine; |
1979 |
|
1980 |
if (raster_y == dy_stop) |
1981 |
ud_border_on = true; |
1982 |
else |
1983 |
if (ctrl1 & 0x10 && raster_y == dy_start) |
1984 |
ud_border_on = false; |
1985 |
|
1986 |
if (spr_dma_on & 0x10) |
1987 |
SetBALow; |
1988 |
|
1989 |
// Last cycle |
1990 |
raster_x += 8; |
1991 |
cycle = 1; |
1992 |
return true; |
1993 |
} |
1994 |
|
1995 |
// Next cycle |
1996 |
raster_x += 8; |
1997 |
cycle++; |
1998 |
return false; |
1999 |
} |