-
Notifications
You must be signed in to change notification settings - Fork 0
/
ps2_kbd.s
640 lines (600 loc) · 13.4 KB
/
ps2_kbd.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
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
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
/*****************************************************
* STM32eForth version 7.20
* Adapted to beyond Jupiter board by Picatout
* date: 2020-11-22
* IMPLEMENTATION NOTES:
* Use USART1 for console I/O
* port config: 115200 8N1
* TX on PA9, RX on PA10
*
* eForth is executed from flash, not copied to RAM
* eForth use main stack R13 as return stack (thread stack not used)
*
* Forth return stack is at end of RAM (addr=0x200005000) and reserve 512 bytes
* a 128 bytes flwr_buffer is reserved below rstack for flash row writing
* a 128 bytes tib is reserved below flwr_buffer
* Forth dstack is below tib and reserve 512 bytes
*
******************************************************/
/***************************************
PS2 KEYBOARD INTERFACE
***************************************/
.syntax unified
.cpu cortex-m4
.fpu softvfp
.thumb
.include "stm32f411ce.inc"
.section .text, "ax", %progbits
.include "mcSaite.inc"
/**********************************
keyboard structure
struct {
byte bitcntr; received bit counter
byte rxshift; shiftin keycode
byte flags; flags
byte parity; count parity bits
}
flags
:0 -> parity error flags
**********************************/
// keyboard state flags
.equ KBD_F_CTGL,(1<<0) // capslock was toggled
.equ KBD_TX,(1<<1) // transmit character to keyboard
.equ KBD_F_CAPS,(1<<2) // capslock
.equ KBD_F_SHIFT,(1<<3) // shift down
.equ KBD_F_CTRL,(1<<4) // ctrl down
.equ KBD_F_ALT,(1<<5) // alt down
.equ KBD_F_XT,(1<<6) // extended key
.equ KBD_F_REL,(1<<7) // key released flag
// structure members offset
.equ KBD_FLAGS,KBD_STRUCT+2
.equ KBD_SHIFTER,KBD_STRUCT+1
.equ KBD_BITCNTR,KBD_STRUCT
.equ KBD_PARITY,KBD_STRUCT+3
.equ KBD_DATA_PIN, 14
.equ KBD_CLOCK_PIN, 15
.equ KBD_GPIO,GPIOC_BASE_ADR
/**********************************
kbd_isr
interrupt service routine
EXTI0 connected to keyboard
clock signal and triggered
on falling edge
**********************************/
_GBL_FUNC kbd_isr
_MOV32 r2,EXTI_BASE_ADR
mov r0,#(1<<KBD_CLOCK_PIN)
str r0,[r2,#EXTI_PR] // reset pending flag
_MOV32 r3,KBD_GPIO
ldr r0,[UP,#KBD_FLAGS]
tst r0,#KBD_TX
bne send_bit
ldrh r0,[r3,#GPIO_IDR]
ldrb r1,[UP,#KBD_BITCNTR]
add r2,r1,#1
strb r2,[UP,#KBD_BITCNTR]
cmp r1,#0
beq start_bit
cmp r1,#9
beq parity_bit
cmp r1,#10
beq stop_bit
// data bit
ldrb r2,[UP,#KBD_SHIFTER]
lsr r2,#1
tst r0,#(1<<KBD_DATA_PIN) // data bit
beq 1f
orr r2,#(1<<7)
ldrb r0,[UP,#KBD_PARITY]
add r0,#1
strb r0,[UP,#KBD_PARITY]
1: strb r2,[UP,#KBD_SHIFTER]
b 9f
start_bit:
tst r0,#(1<<KBD_DATA_PIN)
bne 9f // not a start bit
eor r0,r0
strb r0,[UP,#KBD_SHIFTER]
strb r0,[UP,#KBD_PARITY]
ldrb r0,[UP,#KBD_FLAGS]
mvn r1,#1
and r0,r1 // clear error flag
strb r0,[UP,#KBD_FLAGS]
b 9f
parity_bit:
ldrb r1,[UP,#KBD_PARITY]
tst r0,#(1<<KBD_DATA_PIN)
beq 9f
add r1,#1
strb r1,[UP,#KBD_PARITY]
b 9f
stop_bit:
tst r0,#(1<<KBD_DATA_PIN)
beq 8f // error stop bit expected
ldrb r1,[UP,#KBD_PARITY]
tst r1,#1
beq 8f // error parity
ldrb r0,[UP,#KBD_SHIFTER]
ldrb r1,[UP,#KBD_FLAGS]
cmp r0,#XT_KEY
bne 1f
orr r1,#KBD_F_XT
strb r1,[UP,#KBD_FLAGS]
b 8f
1: tst r1,#KBD_F_REL
beq store_code
cmp r0,#SC_CAPS
bne 1f
eor r1,#KBD_F_CAPS
orr r1,#KBD_F_CTGL
b 2f
1: _CALL do_async_key
ldrb r1,[UP,#KBD_FLAGS]
2: mvn r2,#(KBD_F_REL+KBD_F_XT)
and r1,r2
strb r1,[UP,#KBD_FLAGS]
b 8f
// store code in queue
store_code:
cmp r0,#KEY_REL
bne 1f
// set release flags
orr r1,#KBD_F_REL
strb r1,[UP,#KBD_FLAGS]
b 8f
1: mov r1,r0
_CALL do_async_key
bne 8f // was async key
ldr r0,[UP,#KBD_QTAIL]
add r2,UP,#KBD_QUEUE
strb r1,[r2,r0]
add r0,#1
and r0,#KBD_QUEUE_SIZE-1
strb r0,[UP,#KBD_QTAIL]
8: eor r0,r0
strh r0,[UP,#KBD_BITCNTR]
9: _RET
/* send bit to keyboard
registers usage:
r0 bit shifter
r1 bit counter
r2 output bit
r3 KBD_GPIO
*/
send_bit:
ldrb r1,[UP,#KBD_BITCNTR]
add r0,r1,#1
strb r0,[UP,#KBD_BITCNTR]
ldrb r0,[UP,#KBD_SHIFTER]
mov r2,#(1<<KBD_DATA_PIN)
// cbz r1,9f
1: cmp r1,#8
beq send_parity
cmp r1,#9
beq send_stop
cmp r1,#10
beq rx_ack_bit
// data bits
tst r0,#1
lsr r0,#1
strb r0,[UP,#KBD_SHIFTER]
bne 1f
lsl r2,#16
b 2f
1: ldrb r0,[UP,#KBD_PARITY]
add r0,#1
strb r0,[UP,#KBD_PARITY]
2: str r2,[R3,#GPIO_BSRR]
b 9f
send_parity:
ldrb r0,[UP,#KBD_PARITY]
tst r0,#1
beq 1f
lsl r2,#16
1: str r2,[r3,#GPIO_BSRR]
b 9f
send_stop:
// str r2,[r3,#GPIO_BSRR]
// release data pin
ldr r1,[r3,#GPIO_MODER]
bic r1,#(3<<(2*KBD_DATA_PIN))
str r1,[r3,#GPIO_MODER]
b 9f
rx_ack_bit:
ldrb r0,[UP,#KBD_FLAGS]
mvn r1,#KBD_TX
and r0,r1
ldrh r1,[r3,#GPIO_IDR]
tst r1,#(1<<KBD_DATA_PIN)
strb r0,[UP,#KBD_FLAGS]
eor r0,r0
strb r0,[UP,#KBD_BITCNTR]
9: _RET
/*************************************
check if it is an asynchronous key
input:
r0 virtual code
output:
r0 code order | 255
*************************************/
is_async_key:
push {r1}
ldr r1,=async_keys
_CALL table_scan
pop {r1}
_RET
/***************************
check if async key
and process it
input:
r0 code
output:
r0 0|-1
Z flag set->not async, reset->async key
****************************/
do_async_key:
_CALL is_async_key
cmp r0,#255
bne set_async_key
movs r0,#0
_RET
// asynchornous key, set/reset flag
set_async_key:
push {r1,r2}
ldrb r2,[UP,#KBD_FLAGS]
ldr r1,=async_jump
tbb [r1,r0]
shift_key:
mov r0,#KBD_F_SHIFT
b set_reset
ctrl_key:
mov r0,#KBD_F_CTRL
b set_reset
alt_key:
mov r0,#KBD_F_ALT
set_reset:
tst r2,#KBD_F_REL
beq 1f
mvn r0,r0
and r2,r0
b 2f
1: orr r2,r0
2: strb r2,[UP,#KBD_FLAGS]
movs r0,#-1
9: pop {r1,r2}
_RET
// asynchronous key table
async_keys:
.byte SC_LSHIFT,0 // left shift
.byte SC_RSHIFT,0 // right shift
.byte SC_LCTRL,1 // left control
.byte SC_RCTRL,1 // right control
.byte SC_LALT,2 // left alt
.byte SC_RALT,2 // right alt (alt char)
.byte 0,255
async_jump: // tbb table for async keys
.byte 0 // shift key
.byte (ctrl_key-shift_key)/2
.byte (alt_key-shift_key)/2
/**********************************
kbd_init
initialize keyboard
PS2 clock on PA11
PS2 data on PA12
**********************************/
_GBL_FUNC kbd_init
// clock and data pins as INPUT_FLOAT
_MOV32 r3,KBD_GPIO
mov r0,r3
mov r1,#KBD_CLOCK_PIN
mov r2,#INPUT_FLOAT
_CALL gpio_config
mov r0,r3
mov r1,#KBD_DATA_PIN
mov r2,#INPUT_FLOAT
_CALL gpio_config
// map EXTI15 on PC15 i.e. kbd clock pin
_MOV32 r2,SYSCFG_BASE_ADR
mov r0,#(2<<12)
strh r0,[r2,#SYSCFG_EXTICR4]
// interrupt triggered on falling edge
_MOV32 r2,EXTI_BASE_ADR
mov r0,#(1<<KBD_CLOCK_PIN)
str r0,[r2,#EXTI_IMR] // enable EXTI15
str r0,[r2,#EXTI_FTSR] // on falling edge
eor r0,r0
str r0,[UP,#KBD_QHEAD]
str r0,[UP,#KBD_QTAIL]
// enable interrupt EXTI15_10_IRQ in NVIC
mov r0,#EXTI15_10_IRQ
mov r1,#1
_CALL nvic_set_priority
mov r0,#EXTI15_10_IRQ
_CALL nvic_enable_irq
_RET
// KEY-ASYNC ( -- n )
// return async key flags
_HEADER KEY_ASYNC,9,"KEY-ASYNC"
_PUSH
ldrb TOS,[UP,#KBD_FLAGS]
and TOS,#0xFC
_NEXT
// KEYCODE
// extract keyboard scancode from queue.
// output:
// T0 keycode | 0
keycode:
push {T1,T2,T3}
eor T0,T0
add T3,UP,#KBD_QUEUE
ldr T1,[UP,#KBD_QHEAD]
ldr T2,[UP,#KBD_QTAIL]
cmp T1,T2
beq 2f
ldrb T0,[T3,T1]
add T1,#1
and T1,#KBD_QUEUE_SIZE-1
str T1,[UP,#KBD_QHEAD]
2: pop {T1,T2,T3}
_RET
wait_code:
_CALL keycode
movs T0,T0
beq wait_code
_RET
// translation table scan
// input:
// T0 target code
// T1 table pointer
// output:
// T0 0 | code
// Z flag
table_scan:
push {T2}
1: ldrb T2,[T1],#1
cbz T2,2f
cmp T2,T0
beq 2f
add T1,#1
b 1b
2: ldrb T0,[T1]
movs T0,T0 // set/reset zero flag
9: pop {T2}
_RET
/**********************************
PS2-KEY? ( -- key -1 | 0 )
get a character from keyboard
don't wait for it.
*********************************/
_HEADER PS2_QKEY,8,"PS2-KEY?"
_PUSH
eor TOS,TOS
ldr T1,=sc_ascii // translation table
ldrb T0,[UP,#KBD_FLAGS]
mov T2,#KBD_F_XT
tst T0,T2
beq 1f
ldr T1,=extended // extended code translation
1: _CALL keycode
cbz T0,inkey_exit
cmp T0,#XT2_KEY // pause
beq pause_key
_CALL table_scan
mov TOS,T0
_CALL do_modifiers
cmp TOS,#VK_CTRL_C
beq user_reboot
_PUSH
mov TOS,#-1
inkey_exit:
_NEXT
pause_key: // discard next 7 codes
mov T1,#7
1: _CALL wait_code
subs T1,#1
bne 1b
_NEXT
// check for modifiers flags
// and process it.
do_modifiers:
ldrb T0,[UP,#KBD_FLAGS]
tst T0,#KBD_F_SHIFT
bne shift_down
tst T0,#KBD_F_ALT
bne altchar_down
tst T0,#KBD_F_CTRL
bne ctrl_down
b 9f
shift_down:
mov T0,TOS
ldr T1,=shifted
b 8f
altchar_down:
mov T0,TOS
ldr T1,=altchar
b 8f
ctrl_down:
mov T0,TOS
ldr T1,=controls
8: _CALL table_scan
mov TOS,T0
9: _CALL do_capslock
_RET
do_capslock:
ldrb T0,[UP,#KBD_FLAGS]
tst T0,#KBD_F_CAPS
beq 9f
cmp TOS,#'A'
bmi 9f
cmp TOS,#'Z'+1
bmi 3f
cmp TOS,#'a'
bmi 9f
cmp TOS,#'z'+1
bpl 9f
3: mov T0,#(1<<5)
eor TOS,T0
9: _RET
/***************************
send byte do keyboard
input:
r0 byte to send
use:
r1,r2 temp
r3 KBD_GPIO
***************************/
kbd_send:
push {r0,r1,r2,r3}
// wait pre-video phase
// for least video output disturbance
1: ldr r0,[UP,#VID_STATE]
cmp r0,ST_PREVID
bne 1b
// disable video interrupt
mov r0,#TIM3_IRQ
_CALL nvic_disable_irq
// take control of keyboard clock line
_MOV32 r3,KBD_GPIO
mov r0,r3
mov r1,#KBD_CLOCK_PIN
mov r2,#OUTPUT_OD
_CALL gpio_config
mov r0,r3
mov r1,#KBD_CLOCK_PIN
eor r2,r2
_CALL gpio_out
// delay to hold clock line to 0 for 150µsec
mov r0,#150*48
1: subs r0,#1
bne 1b
pop {r0}
strb r0,[UP,#KBD_SHIFTER]
ldr r0,[UP,#KBD_FLAGS]
orr r0,#KBD_TX
strb r0,[UP,#KBD_FLAGS]
eor r0,r0
strb r0,[UP,#KBD_BITCNTR]
strb r0,[UP,#KBD_PARITY]
// take control of data line
// and put it to 0 for start bit.
mov r0,r3
mov r1,#KBD_DATA_PIN
mov r2,#OUTPUT_OD
_CALL gpio_config
mov r0,r3
mov r1,#KBD_DATA_PIN
eor r2,r2
_CALL gpio_out
// release clock line
mov r0,r3
mov r1,#KBD_CLOCK_PIN
mov r2,#INPUT_FLOAT
_CALL gpio_config
// wait send completed
// expire after 10 msec
mov r0,#10
str r0,[UP,#CD_TIMER]
1: ldr r0,[UP,#CD_TIMER]
cbz r0, 4f
2: ldrb r0,[UP,#KBD_FLAGS]
tst r0,#KBD_TX
bne 1b
// enable video interrupt
4: mov r0,#TIM3_IRQ
_CALL nvic_enable_irq
ldrb r0,[UP,#KBD_FLAGS]
bic r1,r0,#KBD_TX
strb r1,[UP,#KBD_FLAGS]
and r0,#KBD_TX
cbz r0, 5f
ldr r1,[r3,#GPIO_MODER]
bic r1,r1,#(3<<(2*KBD_DATA_PIN))
str r1,[r3,#GPIO_MODER]
eor r0,r0
strb r0,[UP,#KBD_BITCNTR]
mvn r0,r0
5: pop {r1,r2,r3}
_RET
// flush keyboard queue
kbd_clear_queue:
eor T0,T0
str T0,[UP,#KBD_QHEAD]
str T0,[UP,#KBD_QTAIL]
ldrb T0,[UP,#KBD_FLAGS]
mvn T1,#3
and T0,T1
strb T0,[UP,#KBD_FLAGS]
_RET
/**********************************
KBD-RST ( -- c )
send a reset command to keyboard
**********************************/
_HEADER KBD_RST,7,"KBD-RST"
1: mov T0,#KBD_CMD_RESET
_CALL kbd_send
cbnz T0,3f // keyboard not dectected
_CALL kbd_clear_queue
_CALL wait_code
cmp r0,KBD_CMD_RESEND
beq 1b
mov T0,#500
str T0,[UP,#CD_TIMER]
2: _CALL keycode
cbnz T0,3f
ldr T0,[UP,#CD_TIMER]
cmp T0,#0
bne 2b
3: _PUSH
mov TOS,T0
_NEXT
/*****************************
KBD-LED ( c -- )
send command to control
keyboard LEDS
*****************************/
_HEADER KBD_LED,7,"KBD-LED"
1: _CALL kbd_clear_queue
mov T0,#KBD_CMD_LED
_CALL kbd_send
2: _CALL wait_code
cmp T0,#KBD_CMD_RESEND
beq 1b
cmp T0,#KBD_ACK
bne 2b
2: mov T0,TOS
and T0,#7
_CALL kbd_send
3: _CALL wait_code
cmp T0,#KBD_CMD_RESEND
beq 2b
cmp T0,#KBD_ACK
bne 3b
_POP
_NEXT
/**************************
CAPS-LED ( -- )
synch capslock LED
to KBD_F_CAPS
**************************/
_HEADER CAPS_LED,8,"CAPS-LED"
ldrb T0,[UP,#KBD_FLAGS]
tst T0,#1
bne 1f
_NEXT
1: _PUSH
mvn T1,#1
and T0,T1
strb T0,[UP,#KBD_FLAGS]
and TOS,T0,#KBD_F_CAPS
_CALL_COLWORD KBD_LED
/************************
WAIT-KEY ( -- c )
wait for keyboard key
*************************/
_HEADER WKEY,8,"WAIT-KEY"
_NEST
1: _ADR CAPS_LED
_ADR PS2_QKEY
_ADR QDUP
_QBRAN 1b
_UNNEST