-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathinput.c
159 lines (119 loc) · 3.44 KB
/
input.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
/*
BootMii - a Free Software replacement for the Nintendo/BroadOn bootloader.
Requires mini.
Input-handling functions for BootMii. Inspired by GC_PAD.c from GCLIB.
Copyright (C) 2008, 2009 Haxx Enterprises <bushing@gmail.com>
Copyright (C) 2009 Andre Heider "dhewg" <dhewg@wiibrew.org>
Copyright (C) 2009 John Kelley <wiidev@kelley.ca>
Copyright (C) 2009 bLAStY <blasty@bootmii.org>
# This code is licensed to you under the terms of the GNU GPL, version 2;
# see file COPYING or http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt
*/
#include "powerpc.h"
#include "input.h"
#include "string.h"
#include "console.h"
#define PADREG(x) (0xCD006400 + (x)*4)
static GC_Pad _pad;
static void gcpad_init(void) {
write32(PADREG(0), 0x00400300); // read pad on channel 1
write32(PADREG(3), 0x00400300);
write32(PADREG(6), 0x00400300);
write32(PADREG(8), 0x00400300);
write32(PADREG(12), 0x000701f0); // enable poll chan 1, X = 7, Y = 1
write32(PADREG(14), 0x80000000); // transfer all buffer
}
static void gpio_init(void) {
// setup power and eject button hollywood IRQ for PPC
mask32(0x0d8000fc, 0, 0x41); // set GPIO owner to PPC
mask32(0x0d80003c, 1<<10, 0);
mask32(0x0d800034, 0, 1<<10);
mask32(0x0d8000d4, 0, 0x41);
mask32(0x0d8000cc, 0, 0x41);
}
void input_init(void) {
memset(&_pad, 0, sizeof(GC_Pad));
gpio_init();
gcpad_init();
// Check for any pending GPIO irq's, which should be ACK'd so we don't get ghost presses later.
// Try over and over again, until we are out of them.
while (read32(0x0d800030) & (1<<10)) {
if (read32(0x0d8000f0) & 1) {
while(read32(0x0d8000c8) & 1);
write32(0x0d8000d0, 1);
} else if (read32(0x0d8000f0) & 0x40) {
write32(0x0d8000d0, 0x40);
}
write32(0x0d800030, 1<<10);
}
// No IRQ's left to be ACK'd, continue our business.
}
u16 pad_read(GC_Pad *pad, int chan) {
u32 pdata = read32(PADREG(3 * chan + 1));
u32 pdata2 = read32(PADREG(3 * chan + 2));
u16 btns = pdata >> 16;
if (pad) {
u16 prev = pad->btns_held;
pad->btns_held = btns;
pad->btns_up = prev & ~btns;
pad->btns_down = btns & (btns ^ prev);
pad->x = 128 + ((pdata >> 8) & 0xff);
pad->y = 128 - (pdata & 0xff);
pad->cx = 128 + (pdata2 >> 24);
pad->cy = 128 - ((pdata2 >> 16) & 0xff);
pad->l = (pdata2 >> 8) & 0xff;
pad->r = pdata2 & 0xff;
return pad->btns_down;
}
return btns;
}
#define LONG_PRESS_USEC 800000 // 0.7 seconds
#define RST_DOWN !((read32(0x0C003000) >> 16) & 1)
u16 gpio_read(void) {
u16 res = 0;
u32 irq_flag = 0;
u64 press_start;
// while reset is signalled
if (RST_DOWN) {
res |= GPIO_RESET;
press_start = mftb_usec();
int toggle = 0;
// wait for user to release the button
while (RST_DOWN) {
if (res & GPIO_RESET_LP) {
toggle = !toggle;
console_set_blinker(toggle);
} else {
if (mftb_usec() - press_start >= LONG_PRESS_USEC) {
res |= GPIO_RESET_LP;
}
}
}
console_set_blinker(0);
}
if (read32(0x0d800030) & (1<<10)) {
irq_flag = read32(0x0d8000f0);
if (irq_flag & 1) {
res |= GPIO_POWER;
while(read32(0x0d8000c8) & 1);
write32(0x0d8000d0, 1);
} else if (irq_flag & 0x40) {
res |= GPIO_EJECT;
while(read32(0x0d8000c8) & 0x40);
write32(0x0d8000d0, 0x40);
}
write32(0x0d800030, 1<<10); // ack GPIO irq
}
return res;
}
u16 input_read(void) {
return pad_read(&_pad, 0) | gpio_read();
}
u16 input_wait(void) {
u16 res;
do {
usleep(INPUT_WAIT_CYCLE_DELAY);
res = input_read();
} while (!res);
return res;
}