-
-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathreadpad.S
156 lines (133 loc) · 5.45 KB
/
readpad.S
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
; ******************************************************************************
; * This file is part of MegaDrive++. *
; * *
; * Copyright (C) 2015-2019 by SukkoPera <[email protected]> *
; * *
; * MegaDrive++ is free software: you can redistribute it and/or modify *
; * it under the terms of the GNU General Public License as published by *
; * the Free Software Foundation, either version 3 of the License, or *
; * (at your option) any later version. *
; * *
; * MegaDrive++ is distributed in the hope that it will be useful, *
; * but WITHOUT ANY WARRANTY; without even the implied warranty of *
; * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
; * GNU General Public License for more details. *
; * *
; * You should have received a copy of the GNU General Public License *
; * along with MegaDrive++. If not, see <http://www.gnu.org/licenses/>. *
; ******************************************************************************
;
; MegaDrive++ - Universal Region mod, 50/60 Hz switch and In-Game-Reset (IGR)
; for Sega Mega Drive (AKA Genesis)
;
; Please refer to the GitHub page and wiki for any information:
; https://github.com/SukkoPera/MegaDrivePlusPlus
;
; General ISR writing tips: http://nerdralph.blogspot.it/2014/07/writing-avr-interrupt-service-routines.html
; Instruction reference: http://www.atmel.com/webdoc/avrassembler/avrassembler.wb_RJMP.html
; Register usage: https://gcc.gnu.org/wiki/avr-gcc#Register_Layout
#include <avr/io.h>
.extern g_buttons_1
.extern g_buttons_2
.extern g_buttons_3
.section .text
.global INT0_vect
#define PAD_IOPORT _SFR_IO_ADDR (PIND)
#define SEL_BIT PIND2
#define SIG_IOPORT _SFR_IO_ADDR (PORTB)
#define SIG_BIT DDB4
;~ #define ENABLE_SIGNALLING
#define PS_INIT 0 ; Initialization
#define PS_HI 1 ; Select is high (U/D/L/R/B/C)
#define PS_LO 2 ; Select is low (U/D/A/Start)
; The following states are only triggered by 6-button pads
#define PS_6BTN_XYZ 3 ; X/Y/Z/Mode
#define PS_6BTN_ALL_HI 4 ; Pins 1/2/3/4 are all HIGH
#define R_PORT r24
#define R_STATE r26
#define IO_TREG1 _SFR_IO_ADDR (GPIOR0)
#define IO_TREG2 _SFR_IO_ADDR (GPIOR1)
#define IO_TREG3 _SFR_IO_ADDR (GPIOR2)
INT0_vect: ; Save context
out IO_TREG1, R_STATE
in R_STATE, _SFR_IO_ADDR (SREG)
out IO_TREG2, R_STATE
in R_STATE, IO_TREG3 ; Load SM state
out IO_TREG3, R_PORT
#ifdef ENABLE_SIGNALLING
sbi SIG_IOPORT, SIG_BIT
#endif
; Read controller, all at once!
in R_PORT, PAD_IOPORT
#ifdef ENABLE_SIGNALLING
cbi SIG_IOPORT, SIG_BIT
sbi SIG_IOPORT, SIG_BIT
#endif
; State machine start
; Note that we do NOT implement a jump table with ijmp here, as it
; would actually be slower!
cpi R_STATE, PS_LO
breq ps_lo
cpi R_STATE, PS_6BTN_XYZ
breq ps_xyz
cpi R_STATE, PS_HI
breq ps_hi
cpi R_STATE, PS_INIT
breq ps_init
cpi R_STATE, PS_6BTN_ALL_HI
breq ps_all_hi
ps_init:
ldi R_STATE, PS_LO ; Assume select is HIGH, it will be low next time we're called
sbrs R_PORT, SEL_BIT
ldi R_STATE, PS_HI ; Whoops, it was LOW, so it will be high next time
rjmp sm_end
ps_hi:
sbrs R_PORT, SEL_BIT ; If SELECT is not HIGH...
rjmp sm_reset ; ... Reset SM
sts g_buttons_1, R_PORT ; Save port status to g_buttons_1
ldi R_STATE, PS_LO ; SELECT will be low next time we're called
rjmp sm_end
ps_lo:
sbrc R_PORT, SEL_BIT ; SELECT must be LOW
rjmp sm_reset
mov R_STATE, R_PORT ; Abuse R_STATE, we don't need it at this time
andi R_STATE, 0xF0 ; Check if U/D/L/R are all low at the same time (This sets SREG flags)
breq have6btn ; If == 0, U/D/L/R are all low, this means we have a 6-button pad
sts g_buttons_2, R_PORT ; Else, we have Start & A (and UP & DOWN, in case)
ldi R_STATE, PS_HI
rjmp sm_end
have6btn: ldi R_STATE, PS_6BTN_XYZ ; No useful data at this point with 6-button pads, just go to next state
rjmp sm_end
ps_xyz:
sbrs R_PORT, SEL_BIT ; SELECT must be HIGH
rjmp sm_reset
sts g_buttons_3, R_PORT
ldi R_STATE, PS_6BTN_ALL_HI
rjmp sm_end
ps_all_hi:
ldi R_STATE, PS_HI
sbrs R_PORT, SEL_BIT ; SELECT must be LOW
rjmp sm_end
; rjmp sm_reset
sm_reset: ; SELECT did not have the expected level, reset SM
#ifdef ENABLE_SIGNALLING
cbi SIG_IOPORT, SIG_BIT
sbi SIG_IOPORT, SIG_BIT
cbi SIG_IOPORT, SIG_BIT
sbi SIG_IOPORT, SIG_BIT
cbi SIG_IOPORT, SIG_BIT
sbi SIG_IOPORT, SIG_BIT
#endif
ldi R_STATE, PS_INIT
; rjmp sm_end
sm_end: ; End of state machine
#ifdef ENABLE_SIGNALLING
cbi SIG_IOPORT, SIG_BIT
#endif
; Restore context
in R_PORT, IO_TREG3
out IO_TREG3, R_STATE ; IO_TREG3 is no longer needed, store SM state
in R_STATE, IO_TREG2
out _SFR_IO_ADDR (SREG), R_STATE
in R_STATE, IO_TREG1
reti