-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdetect.c
434 lines (333 loc) · 19.4 KB
/
detect.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
/*
* detect.c
* "shoutouts to chironb on github, who i """"borrowed"""" this code from, because i sure as hell wouldn't have been able to do this wizardry, especially not the drive section (see drive.c)" - Korbo
*/
#include <6502.h>
#include <peekpoke.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "detect.h"
unsigned char detect_sid(void) {
// **********
// DETECT SID
// **********
// Taken from the work published here:
// https://codebase64.org/doku.php?id=base:detecting_sid_type_-_safe_method
// ;SID DETECTION ROUTINE
// ;By SounDemon - Based on a tip from Dag Lem.
// ;Put together by FTC after SounDemons instructions
// ;...and tested by Rambones and Jeff.
// ; - Don't run this routine on a badline
// ; - Won't work in VICE (always detects 6581) unless resid-fp emulation is enabled
// sei ;No disturbing interrupts
// lda #$ff
// cmp $d012 ;Don't run it on a badline.
// bne *-3
// ;Detection itself starts here
// lda #$ff ;Set frequency in voice 3 to $ffff
// sta $d412 ;...and set testbit (other bits doesn't matter) in $d012 to disable oscillator
// sta $d40e
// sta $d40f
// lda #$20 ;Sawtooth wave and gatebit OFF to start oscillator again.
// sta $d412
// lda $d41b ;Accu now has different value depending on sid model (6581=3/8580=2)
// lsr ;...that is: Carry flag is set for 6581, and clear for 8580.
// bcc model_8580
// model_6581:
// [...]
// model_8580:
// [...]
register unsigned char i = 0; // For looping
unsigned char D41B_before = 0;
unsigned char D41B_after = 0;
unsigned char one_D41B_after_value = 0;
unsigned char one_D41B_after_frequency = 0;
unsigned char two_D41B_after_value = 0;
unsigned char two_D41B_after_frequency = 0;
unsigned char three_D41B_after_value = 0;
unsigned char three_D41B_after_frequency = 0;
unsigned char likely_8580 = 0;
unsigned char likely_6581 = 0;
D41B_before = PEEK(0xD41B); // Record iniital value
for (i = 0 ; i < 255 ; ++i) {
POKE(0xD011, 0x0B); // Load 0x0B into 0xD011(53265) - Disable VIC-II
POKE(0xD412, 0xFF); // Load 0xFF into 0xD412 - Set frequency in voice 3 to $ffff
POKE(0xD40E, 0xFF); // Load 0xFF into 0xD40E - ...and set testbit (other bits doesn't matter) in $d012 to disable oscillator
POKE(0xD40F, 0xFF); // Load 0xFF into 0xD40F
POKE(0xD412, 0x20); // Load 0x20 into 0xD412 - Sawtooth wave and gatebit OFF to start oscillator again.
POKE(0xD011, 0x1B); // Load 0x1B into 0xD011(53265) - Enable VIC-II
D41B_after = PEEK(0xD41B); // Record current value
// check curent value to see if it's sotred
if ( D41B_after == one_D41B_after_value ) { // does the current value equal slot one's value
++one_D41B_after_frequency; // if so, update the count for this slot
} else if ( D41B_after == two_D41B_after_value ) { // does the current value equal slot two's value
++two_D41B_after_frequency; // if so, update the count for this slot
} else if ( D41B_after == three_D41B_after_value ) { // does the current value equal slot three's value
++three_D41B_after_frequency; // if so, update the count for this slot
// if it doesn't match any slots, try to allocate one
} else if ( one_D41B_after_frequency == 0 ) { // if this slot one empty?
one_D41B_after_value = D41B_after; // if so, update the value for this slot and...
++one_D41B_after_frequency; // ...update the count for this slot
} else if ( two_D41B_after_frequency == 0 ) { // if this slot one empty?
two_D41B_after_value = D41B_after; // if so, update the value for this slot and...
++two_D41B_after_frequency; // ...update the count for this slot
} else if ( three_D41B_after_frequency == 0 ) { // if this slot one empty?
three_D41B_after_value = D41B_after; // if so, update the value for this slot and...
++three_D41B_after_frequency; // ...update the count for this slot
} else {
D41B_after = 0;
one_D41B_after_value = 0;
one_D41B_after_frequency = 0;
two_D41B_after_value = 0;
two_D41B_after_frequency = 0;
three_D41B_after_value = 0;
three_D41B_after_frequency = 0;
};
};
// Scan for 8580 --> 0x08 & any of 0x33 0x32
if (one_D41B_after_value == 0x08 || two_D41B_after_value == 0x08 || three_D41B_after_value == 0x08) {
++likely_8580;
};
if (one_D41B_after_value == 0x32 || two_D41B_after_value == 0x32 || three_D41B_after_value == 0x32) {
++likely_8580;
};
// Scan for 6581 --> 0x09 & any of 0x34 0x33
if (one_D41B_after_value == 0x09 || two_D41B_after_value == 0x09 || three_D41B_after_value == 0x09) {
++likely_6581;
};
if (one_D41B_after_value == 0x34 || two_D41B_after_value == 0x34 || three_D41B_after_value == 0x34) {
++likely_6581;
};
// Codes:
// 1 - 6581
// 2 - 8580
// 0 - Undetermined
if (likely_6581 > likely_8580) {
return(1);
} else if (likely_8580 > likely_6581) {
return(2);
} else {
return(0);
};
};//end func
void fifteen_nops(void) {
/* 10 */ __asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");
/* 10 */ __asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");
/* 10 */ __asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");
/* 10 */ __asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");
/* 5 */ __asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");__asm__ ("NOP");
};//end-func
unsigned char raster_scan_line;
unsigned long int estimated_original_hz;
unsigned char estimated_mhz_only;
unsigned long int estimated_remaining_hz;
// // Using this for the fastest loops below for CPU MHz detection.
// // https://cc65.github.io/doc/cc65.html#register-vars
// #pragma register-vars(on)
void detect_speed(unsigned char display_mhz, unsigned char detected_cpu, unsigned char ntscpal, char * input_string) {
register unsigned char cycles_to_burn; // This is a zero page variable so it's super fast. https://cc65.github.io/doc/cc65.html#register-vars
unsigned char i;
unsigned char ntsc_status = ntscpal < 2;
unsigned long int scaling_factor;
unsigned char mhz_string[6];
// 1.02 MHz NTSC / 0.985 MHz PAL
// The clock speed doubles when 80-column display modes are in use (2.02 MHz for NTSC, 1.97 MHz for PAL)
// C64 CLOCK=1022730 for NTSC systems and CLOCK= 985250 for PAL systems.
// C128 CLOCK=2045460 for NTSC systems and CLOCK=1970500 for PAL systems.
// TODO: FUCKING PAL! I FUCKING HATE PAL!!!
// TODO: I need to make a function full of 10 NOP's
// TODO: And then call that to save some space.
// TODO: And then I need to make PAL versions
// TODO: For the C64/C128 and the SuperCPU.
// TODO: FUCK YOU PAL!
estimated_mhz_only = 0;
estimated_remaining_hz = 0;
estimated_original_hz = 0;
raster_scan_line = 0;
i = 0;
scaling_factor = 0;
cycles_to_burn = 0;
SEI(); // Disable interupts.
if ( detected_cpu == CPU_65816 ) {
// Display mode $D011 Bit #7: Read: Current raster line (bit #8). C64 generates 262 (0-255 0) scanlines total on NTSC, 312 for Pal
// TODO: Do teh proper bit shift thing here!
while(1){
if ( PEEK(0xD012) == 0 && PEEK(0xD011) == 27 ) break;
};//end-while
for (i = 0; i < 7; ++i) ; // *THIS IS HERE INTENTIONALLY!* This is because this slows down the SuperCPU and somehow this makes the timing math work out perfectly!
for (i = 1; i <= 60; ++i) {
fifteen_nops();
};//end-for
raster_scan_line = PEEK(0xD012); // Record the current raster scan line.
if (ntsc_status) {
scaling_factor = 122727600 ; // This is the NTSC scaling factor for 6502 based systems like the C64, C128 and Ultimate 64.
} else {
scaling_factor = 120200500 ; // This is the PAL >:-| scaling factor for 6502 based systems like the C64, C128 and Ultimate 64.
};//end-if
} else { // 6510 CPU
POKE(0xD030,253); // Put C128 into fast mode. This comes *BEFORE* we start the timing so it doesn't affect the timing. However, the math works out better above with the SuperCPU.
// Display mode $D011 Bit #7: Read: Current raster line (bit #8). C64 generates 262 (0-255 0) scanlines total on NTSC, 312 for Pal
// TODO: Do teh proper bit shift thing here!
while(1){
if ( PEEK(0xD012) == 0 && PEEK(0xD011) == 27 ) break;
};//end-while
while(cycles_to_burn != 50) { // 50 worked but doesn't scale up preciscly.
fifteen_nops();
++cycles_to_burn; // Do literally nothing... one more time. Makes the math easier.
};//end-while
raster_scan_line = PEEK(0xD012); // Record the current raster scan line.
if (ntsc_status) {
scaling_factor = 94091160 ; // This is the NTSC scaling factor for 6502 based systems like the C64, C128 and Ultimate 64.
} else {
scaling_factor = 93598750 ; // This is the PAL >:-| scaling factor for 6502 based systems like the C64, C128 and Ultimate 64.
};//end-if
};//end-if
POKE(0xD030,252); // Put C128 into C64 *normal* mode.
CLI(); // Enable interupts.
// if (raster_scan_line == 0) return; // If for some reason this completes in less than a raster line of time, and to avoid dividing by zero, exti without printing anything.
// custom shit goes here
// else do the math below
// This is prolly chearting prolly...
if ( detected_cpu == CPU_65816 ) {
switch(raster_scan_line) {
case 5 :
case 6 :
case 7 : strcpy(mhz_string,"20.45"); break; // TODO: This doesn't really work until I do a better job of determining NTSC/PAL by counting rasterlines. // if (ntsc_status==TRUE) strcpy(mhz_string,"20.45"); else strcpy(mhz_string,"20.03"); break; // added 27 bytes
case 118 :
case 119 :
case 120 :
case 121 :
case 122 : strcpy(mhz_string,"1.02"); break;
default : goto NON_STANDARD_RESULT;
};//end-switch
} else {
switch(raster_scan_line) {
case 0 :
case 1 : strcpy(mhz_string,"> 41"); break;
case 2 : strcpy(mhz_string,"40.91"); break;
case 3 : strcpy(mhz_string,"30.68"); break;
case 4 : strcpy(mhz_string,"20.45"); break;
case 5 : strcpy(mhz_string,"17.90"); break;
case 6 : strcpy(mhz_string,"15.34"); break;
case 7 : strcpy(mhz_string,"12.78"); break;
case 8 : strcpy(mhz_string,"10.23"); break;
case 44 :
case 45 :
case 46 :
case 47 :
case 48 : if (ntsc_status) strcpy(mhz_string,"2.04"); else strcpy(mhz_string,"1.97"); break;
case 91 :
case 92 :
case 93 : strcpy(mhz_string,"1.02"); break;
default : goto NON_STANDARD_RESULT;
};//end-switch
};//end-if
goto STANDARD_RESULT;
NON_STANDARD_RESULT:;
estimated_original_hz = scaling_factor / raster_scan_line; // This is the scaling factor. C64/C128/Ultimate=171818640 and SuperCPU=225000600.
estimated_mhz_only = estimated_original_hz / 1000000; // Get only the MHz.
estimated_remaining_hz = estimated_original_hz - (estimated_mhz_only*1000000); // Get the rest of the hz
estimated_remaining_hz = estimated_remaining_hz / 10000; // We only want the lazt 2 decimals.
if (display_mhz) sprintf(input_string, "%u.%02lu MHz", estimated_mhz_only, estimated_remaining_hz ); // Print the result if it's been told to.
goto END_SPEED;
STANDARD_RESULT:;
if (display_mhz) sprintf(input_string, "%s MHz", mhz_string);
END_SPEED:;
};//end-func
unsigned char detect_cpu(unsigned char sid_detected) {
unsigned char gotten_cpu = 0;
unsigned char peek0 = PEEK(0);
// cc65 has a getcpu() function that only tells you if it's a 6502 or others, but not the Commodore specific ones.
// The cc65 online documentation is super old. The GitHub has more up-to-date information.
// https://github.com/cc65/cc65/blob/master/include/6502.h
//
// /* Possible returns of getcpu() */
// #define CPU_6502 0
// #define CPU_65C02 1
// #define CPU_65816 2
// #define CPU_4510 3
// #define CPU_65SC02 4
// #define CPU_65CE02 5
// #define CPU_HUC6280 6
// #define CPU_2A0x 7
// #define CPU_45GS02 8
// Detect 6510/8500/8502 --> Address 0x00 should be 47, whereas regular 6502's seem to have 76 there.
// Detect SID to guess that it's a Commodore 64C or not.
// Detect a Commodore 128 --> Address 0xD030 (53296) is always 255 on a non-Commodore 128 6502 CPU.
gotten_cpu = getcpu();
if ( gotten_cpu == 0 ) {
if (PEEK(0) == 47 && PEEK(0xD030) != 255) { return(9 ); // 9 MOS 8502 (Commodore 128)
} else if (PEEK(0) == 47 && sid_detected == 2 ) { return(10); // 10 MOS 8500 (Commodore 64C)
} else if (PEEK(0) == 47 && sid_detected == 1 ) { return(11); // 11 MOS 6510 (Commodore 64)
} else { return(0); // 0 MOS 6502 ???
};
} else { return(gotten_cpu); // 255 Unknown
};
};//end func
unsigned char detect_kernal(void) {
// I have discovered a *NEW* byte to look at in any kernal and figure out what it is. I need to re-write this to use this, becuase it'll save a bunch of code!
// REVIEW - E4AC (58540) - Each Kernal:
// - 901227-01 --> 0x2B
// - 901227-02 --> 0x5C
// - 901227-03 --> 0x81
// - 251104-01 --> 0x00
// - 251104-04 --> 0xB3
// - 901246-01 --> 0x63
// C64 R3 58497 _77_ && 58498 79 && 58677 14
// SX R3 58497 _83_ && 58498 88 && 58677 6
// J64 R3 58497 _68_ && 58498 79 && 58677 14
// JSX R3 58497 _68_ && 58498 79 && 58677 6
unsigned char kernal_value = 0;
kernal_value = PEEK(0xE4AC);
switch (kernal_value) {
case 0x2B : return(1); break; // 901227-01 - R1 C64 First Kernal
case 0x5C : return(2); break; // 901227-02 - R2 C64 Early Kernal
case 0x81 :
switch ( PEEK(58497L) ) {
case 68 :
switch (PEEK(58677L)) {
case 14 : return(10); break; // R3 C64 JiffyDOS
case 6 : return(11); break; // R3 SX-64 JiffyDOS
};//end-switch
break;
};//end-switch
return(3); // 901227-03 - R3 C64 Kernal
break;
case 0xB3 : return(4); break; // 251104-04 - R4 SX-64 Common Latest Kernal
case 0x00 : return(5); break; // 251104-01 - R1 SX-64 Rare First Kernal
case 0x63 : return(6); break; // 901246-01 - 4064 Educator 64 --> ? Fair to say this is most similar to a stock R3 901227-03 kernal. From here: "Note that some patches of 901227-03 are included." http://www.zimmers.net/anonftp/pub/cbm/firmware/computers/c64/revisions.txt
default : return(0); break; // Default Unknown Kernal
};//end switch
}; // end func
unsigned char detect_model(unsigned char ntscpal, unsigned char sid_detected, unsigned char kernal_detected, unsigned char detected_cpu, char * input_string) {
unsigned char ntscpal_detected = 0;
unsigned char detected_1571 = 0;
unsigned char result = 0;
// TODO: !!!!!!!!!!!!!!! DELETE THE LINE BELOW! IT'S NOT USED AND IS STUPID!!!
// unsigned char disk_buffer[40] = " "; // TOODO: Look up how big this string can actually get in Commodore DOS / 1541 stuff...
unsigned char read_bytes = 0;
// First, why am I doing this twice? This whole function is a waste and needs to be tighten up a lot
ntscpal_detected = ntscpal <= 2 ;;
if (detected_cpu == 9) {
detect_drive(8, 0);
detected_1571=(drive_detected_type[0]==0x71); // If we've got a 1571 on device 8 then maybe we've got a C128D
};//end-if
// SID Return Values: // CPU Return Values:
// 1 - 6581 // 9 - MOS 8502 Commodore 128
// 2 - 8580 // 10 - MOS 8500 Commodore 64C
// 0 - Undetermined // 11 - MOS 6510 Commodore 64
if (kernal_detected == 4 || kernal_detected == 5 || kernal_detected == 11) { strcpy(input_string, "SX-64"); return(5); // SX-64 251104-01 or 251104-04 Kernal ROM
} else if (kernal_detected == 6 ) { strcpy(input_string, "Educator 64"); return(11); // Educator 64
// TODO: THIS CANNOT BE 0 SINCE THERE"S NO ELS CONDITION!!!!! Need to rework! it's a SID problem!
} else if (detected_cpu == 9 && sid_detected == 1 && detected_1571 == 1) { strcpy(input_string, "Commodore 128D"); return(14); // 128D CPU:8502 GPU:8564 NTSC or 8566/69 PAL SID:6581 Drive: Must have 1571
} else if (detected_cpu == 9 && sid_detected == 1 ) { strcpy(input_string, "Commodore 128"); return(13); // 128 CPU:8502 GPU:8564 NTSC or 8566/69 PAL SID:6581 Drive: Anything other than 1571
} else if (detected_cpu == 9 && sid_detected == 2 ) { strcpy(input_string, "Commodore 128DCR"); return(16); // 128DCR CPU:8502 GPU:8564 NTSC or 8566/69 PAL SID:8580 Don't need to worry about the drive, because the DCR is the only machine with the 8502 CPU and 8580 SID.
} else if (ntscpal_detected == 0 && sid_detected == 1 && kernal_detected == 1) { strcpy(input_string, "Commodore 64"); return(12); // 64 (Early) NTSC + 6581 + KERNAL R1 901227-01 --> C64 (Early) --> VIC-II 6567
} else if (ntscpal_detected == 0 && sid_detected == 1 ) { strcpy(input_string, "Commodore 64"); return(12); // 64 NTSC + 6581 + KERNAL R2 901227-02 --> C64 --> VIC-II 6567
} else if (ntscpal_detected == 0 && sid_detected == 2 ) { strcpy(input_string, "Commodore 64C"); return(13); // 64C NTSC + 8580 + Any --> C64C --> VIC-II 8562
} else if (ntscpal_detected == 1 && sid_detected == 1 ) { strcpy(input_string, "Commodore 64"); return(12); // 64 PAL + 6581 + Any --> C64 --> VIC-II 6569/6572/6573
} else if (ntscpal_detected == 1 && sid_detected == 2 ) { strcpy(input_string, "Commodore 64C"); return(13); // 64C PAL + 8580 + Any --> C64C --> VIC-II 8565
} else { strcpy(input_string, "Unknown"); return(7); // Couldn't detect a specific model!
};//end if
};//end func