-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathEpson_CP.c
219 lines (199 loc) · 5.37 KB
/
Epson_CP.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
#include <stdio.h>
#include "pico/stdlib.h"
#include "pico/bootrom.h"
#include "pico/multicore.h"
#include "pico/stdio/driver.h"
#include "tusb.h"
#include "keymap.h"
#include "pins.h"
#include "video.h"
#include "font.h"
#include "font_hack.h"
// RCLK shifts to registers on rising edge;
// SH/LD' loads during LOW but needs to be high
// during shift/read. So strategy is:
//
//
// * shift in next row pattern to 595/shift out prev. column pattern
// * RCLK low to load columns
// * RCLK high to latch pattern
//
//
/// Initialize a GPIO pin as an output with the given level.
static inline void init_pin(pin_t pin, uint value) {
gpio_init(pin);
gpio_set_dir(pin, GPIO_OUT);
gpio_put(pin, value);
}
void initialize_gpio() {
gpio_init(SP_SER_IN);
init_pin(SP_CLK,0);
init_pin(SP_SER_OUT,0);
init_pin(SP_OE,1);
init_pin(SP_RCLK,0);
}
void delay_bit() {
sleep_us(4);
}
uint8_t scan_cols(uint8_t next_rows) {
uint8_t scan;
gpio_put(SP_RCLK,1); // Sample columns for T
delay_bit();
for (size_t i = 0; i < 8; i++) {
delay_bit();
gpio_put(SP_CLK,0);
delay_bit();
scan = (scan << 1) | (gpio_get(SP_SER_IN)?0:1);
gpio_put(SP_SER_OUT,(next_rows & (1 << (7-i) ))?1:0);
delay_bit();
gpio_put(SP_CLK,1);
delay_bit();
}
gpio_put(SP_RCLK,0);
delay_bit();
gpio_put(SP_RCLK,1); // Latch rowval for T+1
delay_bit();
gpio_put(SP_RCLK,0); // Load columns for T+1
return scan;
}
const size_t ROW_COUNT = 8;
typedef enum {
NONE,
DOWN,
UP,
} KeypressType;
typedef struct {
uint8_t row;
uint8_t col;
KeypressType type;
} Keypress;
Keypress last_kp;
Keypress scan() {
Keypress cur_kp = { 0, 0, NONE };
// do scan
scan_cols(0x01);
for (uint8_t i = 1; i <= 8; i++) {
uint8_t cols = scan_cols(1<<(i%8));
if (cols) {
cur_kp = (Keypress){ i-1, 31-__builtin_clz(cols), DOWN };
}
}
return cur_kp;
}
/*
void video_core() {
}
*/
uint8_t hexchar(int c) {
if (c >= '0' && c <= '9') { return c - '0'; }
if (c >= 'a' && c <= 'f') { return (c - 'a') + 10; }
if (c >= 'A' && c <= 'F') { return (c - 'A') + 10; }
return 0xff;
}
int read_from_usb(uint8_t* buf, int len, uint32_t timeout_us) {
uint32_t until = make_timeout_time_us(timeout_us);
int total_read = 0;
do {
if (1) { //(tud_cdc_connected() && tud_cdc_available()) {
//int read = tud_cdc_read(buf, len);
int read = stdio_usb.in_chars(buf, len);
if (read > 0) {
total_read += read;
len -= read;
buf += read;
if (len == 0) break;
}
}
} while (!time_reached(until));
return total_read;
}
#define TEXT_BUFFER_LEN 32
char text_buffer[TEXT_BUFFER_LEN];
#define FONT_BUFFER_LEN 4096
char font_buffer[FONT_BUFFER_LEN];
uint8_t* curfont = font_hack;
int main()
{
sleep_us(100);
stdio_init_all();
sleep_us(2000);
puts("Hello, world!");
initialize_gpio();
init_video();
//multicore_launch_core1(video_core);
gpio_put(SP_OE,0);
init_lcd();
while (true) {
int c = getchar_timeout_us(100);
if (c == PICO_ERROR_TIMEOUT) { // Scan for keypresses
Keypress kp = scan();
if ((kp.type != last_kp.type ||
kp.row != last_kp.row ||
kp.col != last_kp.col)
&& kp.type == DOWN) {
printf("Key down: %s (%d, %d)\n",keyname(keycode(kp.row,kp.col)),kp.row,kp.col);
}
last_kp = kp;
} else if (c == 'R') { // Reset Pi Pico into bootloader
printf("*** Resetting into bootloader ***");
reset_usb_boot(0,0);
} else if (c == 'C') { // Send arbitrary LCD command
// read next 6 chars as hex
uint8_t cmd[3] = { 0,0,0 };
for (uint8_t i = 0; i < 6; i++) {
c = getchar_timeout_us(8000000);
if (c == -1) { puts("command entry timeout\n"); break; }
uint8_t h = hexchar(c);
if (h >= 16) { c = -1; printf("non-hex char '%c'\n",c); break; }
cmd[i/2] |= h << ( (1-(i%2)) * 4 );
}
if (c != -1) {
printf("Sending command %x%x%x\n",cmd[0],cmd[1],cmd[2]);
send_command(cmd);
}
} else if (c == 'P') { // Load up a custom palette
int i = read_from_usb((uint8_t*)&palette[0],256*3,1000000);
printf("Got %d.\n",i);
} else if (c == 'V') { // Load framebuffer over USB
int count = read_from_usb(&framebuf[0][0],320*119,2000000);
printf("count %d of %d\n",count,320*119);
send_image();
} else if (c == 'v') { // Display current framebuffter
send_image();
} else if (c == 'E') { // erase video buffer
clear_buffer();
} else if (c == 'F') { // read in font buffer
c = getchar_timeout_us(3000);
uint16_t len = c;
c = getchar_timeout_us(3000);
len += c << 8;
if (len > FONT_BUFFER_LEN) {
printf("Font too big!\n");
} else {
int count = read_from_usb(&font_buffer[0],len,2000000);
printf("count %d of %d\n",count,len);
if (count == len) { curfont = font_buffer; }
}
} else if (c == 'T') {
// load and render text buffer
char* cbuf = text_buffer;
const char* cbend = text_buffer + TEXT_BUFFER_LEN - 1;
c = getchar_timeout_us(8000000);
while (c != '\r' && c != '\n' && c != -1 && cbuf != cbend) {
*(cbuf++) = c;
c = getchar_timeout_us(8000000);
}
if (c != -1) {
*cbuf = '\0';
render_text( 10, 20, text_buffer, curfont);
} else {
printf("Text enty timeout.\n");
}
} else if (c == '\n' || c == '\r') { // Test that Pico is responding
printf("Hello there.\n");
} else {
printf("Unrecognized data.\n");
}
}
return 0;
}