1
0
mirror of https://github.com/cc65/cc65.git synced 2025-01-11 11:30:13 +00:00
cc65/libsrc/common/getcpu.s

134 lines
4.8 KiB
ArmAsm
Raw Normal View History

;
; Ullrich von Bassewitz, 02.04.1999
;
; unsigned char getcpu (void);
;
2019-02-05 23:27:52 +01:00
.include "zeropage.inc"
.export _getcpu
; ---------------------------------------------------------------------------
; Subroutine to detect an 816. Returns
;
; - carry clear and 0 in A for a NMOS 6502 CPU
2018-03-01 14:57:42 -08:00
; - carry set and 1 in A for a 65C02
; - carry set and 2 in A for a 65816
2016-09-07 19:44:11 +02:00
; - carry set and 3 in A for a 4510
2018-03-01 14:57:42 -08:00
; - carry set and 4 in A for a 65SC02
; - carry set and 5 in A for a 65CE02
; - carry set and 6 in A for a HuC6280
; - carry clear and 7 in A for a 2a03/2a07
2018-12-27 10:50:25 +10:30
; - carry set and 8 in A for a 45GS02
;
; This function uses a $1A opcode which is a INA on the 816 and ignored
; (interpreted as a NOP) on a NMOS 6502. There are several CMOS versions
; of the 6502, but all of them interpret unknown opcodes as NOP so this is
; just what we want.
.p816 ; Enable 65816 instructions
_getcpu:
2018-12-27 10:50:25 +10:30
lda #0
2016-09-07 19:44:11 +02:00
inc a ; .byte $1A ; nop on nmos, inc on every cmos
cmp #1
2018-12-27 10:50:25 +10:30
bcc @IsNMOS
2018-03-01 14:57:42 -08:00
; This is at least a 65C02, check for a 65CE02/4510
2016-09-07 19:44:11 +02:00
2018-03-02 14:09:02 -08:00
.byte $42,$EA ; neg on 65CE02/4510, nop #$EA on 65C02, wdm $EA on 65816
2016-09-07 19:44:11 +02:00
cmp #1
2018-12-27 10:50:25 +10:30
beq @HasINCA
2018-03-01 14:57:42 -08:00
; This is at least a 65CE02, check for 4510
lda #5 ; CPU_65CE02 constant
2018-12-27 10:50:25 +10:30
ldx #0 ; to make sure MAP doesn't do anything, the upper nybl of X and Z must be clear
2018-03-02 14:09:02 -08:00
.byte $5C ; map on 4510, aug on 65CE02 (acts like 4 byte nop)
lda #3 ; CPU_4510 constant
nop
2018-12-27 10:50:25 +10:30
cmp #5
beq @LoadXAndReturn
; It is either a 4510 (C65) or a 45GS02 (MEGA65)
; 45GS02 supports 32-bit ZP indirect, so use that to check CPU type
; without requiring a functioning MEGA65 hypervisor.
; We setup a read of $200xx, then store a different value in $xx
; and then re-read $200xx to see if it is unchanged.
; Setup 32-bit pointer to $00020000+tmp1
lda #<$020000+tmp1
sta regsave
lda #>$020000+tmp1
sta regsave+1
sta regsave+3 ; also write to upper byte of pointer to save an extra LDA #$00
lda #^$020000+tmp1
sta regsave+2
2018-12-27 10:50:25 +10:30
; Prefixing LDA ($nn),Z with a NOP uses 32-bit ZP pointer on 45GS02,
; but normal 16-bit ZP pointer on 4510
; (We assume Z=$00, which will be the normal case)
nop ; prefix to tell next instruction to be 32-bit ZP
.byte $b2,regsave ; LDA (regsave),Z
2018-12-27 10:50:25 +10:30
eor #$ff ; change the value
sta tmp1 ; store in $xx
; now try again to load it: If the same, then 45GS02, as $200xx is unchanged
2018-12-27 10:50:25 +10:30
nop ; prefix to tell next instruction to be 32-bit ZP
.byte $b2,regsave ; LDA (regsave),Z
2018-12-29 22:19:51 +10:30
cmp tmp1 ; does the loaded value match what is in $xx?
bne @Is45GS02 ; $200xx and $xx have different values, so must be a MEGA65 45GS02
2018-12-27 10:50:25 +10:30
@Is4510:
lda #3 ; CPU_4510 constant
ldx #0 ; load high byte of word
rts
@Is45GS02:
lda #8 ; CPU_45GS02 constant
ldx #0 ; load high byte of word
rts
2018-03-01 14:57:42 -08:00
; 6502 type of cpu, check for a 2a03/2a07
2018-12-27 10:50:25 +10:30
@IsNMOS:
2018-03-01 14:57:42 -08:00
sed ; set decimal mode, no decimal mode on the 2a03/2a07
lda #9
clc
2018-03-02 14:09:02 -08:00
adc #1 ; $01+$09 = $10 on 6502, $01+$09 = $0A on 2a03/2a07
2018-03-01 14:57:42 -08:00
cld
2018-04-01 13:14:44 -07:00
cmp #$0a
2018-12-27 10:50:25 +10:30
beq @Is2a03
2018-03-01 14:57:42 -08:00
lda #0 ; CPU_6502 constant
2018-12-27 10:50:25 +10:30
beq @LoadXAndReturn
@Is2a03:
2018-03-01 14:57:42 -08:00
lda #7 ; CPU_2A0x constant
2018-12-27 10:50:25 +10:30
bne @LoadXAndReturn
2018-03-01 14:57:42 -08:00
; 65C02 cpu type, check for HuC6280
2018-12-27 10:50:25 +10:30
@CheckHuC6280:
ldx #6 ; CPU_HUC6280 constant
2018-03-02 14:09:02 -08:00
.byte $22,$EA ; sax nop on HuC6280 (A=$06, X=$01), nop #$EA on 65C02 (A=$01, X=$06)
2018-12-27 10:50:25 +10:30
bne @LoadXAndReturn
2018-03-01 14:57:42 -08:00
2018-02-27 11:44:34 -08:00
; Check for 65816/65802
2018-12-27 10:50:25 +10:30
@HasINCA:
xba ; .byte $EB, put $01 in B accu (nop on 65C02/65SC02)
2018-03-02 14:09:02 -08:00
dec a ; .byte $3A, A=$00
xba ; .byte $EB, A=$01 if 65816/65802 and A=$00 if 65C02/65SC02
inc a ; .byte $1A, A=$02 if 65816/65802 and A=$01 if 65C02/65SC02
2018-02-27 11:44:34 -08:00
cmp #2
2018-12-27 10:50:25 +10:30
beq @LoadXAndReturn
2018-02-27 11:44:34 -08:00
2018-03-01 14:57:42 -08:00
; check for 65SC02
2018-03-02 14:09:02 -08:00
ldy $F7
2018-02-27 11:44:34 -08:00
ldx #0
2018-03-02 14:09:02 -08:00
stx $F7
.byte $F7,$F7 ; nop nop on 65SC02, smb7 $F7 on 65C02
ldx $F7
sty $F7
cpx #$00
2018-12-27 10:50:25 +10:30
bne @CheckHuC6280
2018-02-27 11:44:34 -08:00
lda #4 ; CPU_65SC02 constant
2018-12-27 10:50:25 +10:30
@LoadXAndReturn:
ldx #0
rts