Serial Wombat Arduino Library
Loading...
Searching...
No Matches
SerialWombatPS2Keyboard.h
Go to the documentation of this file.
1#pragma once
2
3/*
4Copyright 2023 Broadwell Consulting Inc.
5
6"Serial Wombat" is a registered trademark of Broadwell Consulting Inc. in
7the United States. See SerialWombat.com for usage guidance.
8
9Permission is hereby granted, free of charge, to any person obtaining a
10 * copy of this software and associated documentation files (the "Software"),
11 * to deal in the Software without restriction, including without limitation
12 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
13 * and/or sell copies of the Software, and to permit persons to whom the
14 * Software is furnished to do so, subject to the following conditions:
15
16The above copyright notice and this permission notice shall be included in
17 * all copies or substantial portions of the Software.
18
19THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
22 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
23 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
24 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
25 * OTHER DEALINGS IN THE SOFTWARE.
26*/
27
28#include <stdint.h>
29#include "Stream.h"
30#include "SerialWombat.h"
33
34
35typedef enum
36{
38 SCANCODE_1 = 0x16,
39 SCANCODE_2 = 0x1E,
40 SCANCODE_3 = 0x26,
41 SCANCODE_4 = 0x25,
42 SCANCODE_5 = 0x2E,
43 SCANCODE_6 = 0x36,
44 SCANCODE_7 = 0x3D,
45 SCANCODE_8 = 0x3E,
46 SCANCODE_9 = 0x46,
47 SCANCODE_0 = 0x45,
52 SCANCODE_Q = 0x15,
53 SCANCODE_W = 0x1D,
54 SCANCODE_E = 0x24,
55 SCANCODE_R = 0x2D,
56 SCANCODE_T = 0x2C,
57 SCANCODE_Y = 0x35,
58 SCANCODE_U = 0x3C,
59 SCANCODE_I = 0x43,
60 SCANCODE_O = 0x44,
61 SCANCODE_P = 0x4D,
65 SCANCODE_A = 0x1C,
66 SCANCODE_S = 0x1B,
67 SCANCODE_D = 0x23,
68 SCANCODE_G = 0x34,
69 SCANCODE_H = 0x33,
70 SCANCODE_J = 0x3B,
71 SCANCODE_K = 0x42,
72 SCANCODE_L = 0x4B,
77 SCANCODE_Z = 0x1A,
78 SCANCODE_X = 0x22,
79 SCANCODE_C = 0x21,
80 SCANCODE_V = 0x2A,
81 SCANCODE_B = 0x32,
82 SCANCODE_N = 0x31,
83 SCANCODE_M = 0x3A,
137
138
180
182 public Stream, public SerialWombatPin
183{
184public:
199 int16_t begin(uint8_t clockPin, uint8_t dataPin, uint8_t bufferMode = 0, uint8_t queueMode = 0, uint16_t queueAddress = 0xFFFF, uint8_t pullUpDown = 0)
200 {
201 _pin = clockPin;
203
204 uint8_t tx[] = { (uint8_t)SerialWombatCommands::CONFIGURE_PIN_MODE0,
205 _pin,
206 (uint8_t)_pinMode ,
207 dataPin,
208 queueMode,
209 bufferMode,
210 0x55,
211 pullUpDown};
212 int16_t result = _sw.sendPacket(tx);
213 if (result < 0)
214 {
215 return result;
216 }
217
218 if (queueAddress != 0xFFFF)
219 {
220 uint8_t tx6[] = { (uint8_t)SerialWombatCommands::CONFIGURE_PIN_MODE6,
221 _pin,
222 _pinMode,
223 SW_LE16(queueAddress),
224 0x55,
225 0x55,
226 0x55 };
227
228 return _sw.sendPacket(tx6);
229 }
230 return (result);
231 }
232
233
234
240 {
241 uint8_t tx[8] = { 201, _pin, _pinMode, 0,0x55,0x55,0x55,0x55 };
242 uint8_t rx[8];
243 _sw.sendPacket(tx, rx);
244 return (rx[4]);
245 }
246
250 int read()
251 {
252 uint8_t tx[8] = { 202, _pin,_pinMode, 1,0x55,0x55,0x55,0x55 };
253 uint8_t rx[8];
254 if (_sw.sendPacket(tx, rx) < 0)
255 {
256 return -1;
257 }
258
259 if (rx[3] != 0)
260 {
261 return (rx[4]);
262 }
263 else
264 {
265 return (-1);
266 }
267 }
268
271 void flush()
272 {
273
274 }
275
280 int peek()
281 {
282 uint8_t tx[8] = { 203, _pin,_pinMode,0x55,0x55,0x55,0x55,0x55 };
283 uint8_t rx[8];
284 _sw.sendPacket(tx, rx);
285 if (rx[4] > 0)
286 {
287 return (rx[5]);
288 }
289 else
290 {
291 return (-1);
292 }
293 }
294
302 size_t write(uint8_t data)
303{
304
305 return (1);
306}
307
316 size_t write(const uint8_t* buffer, size_t size)
317{
318 return(size);
319}
320
326{
327 return(0);
328}
329
340 size_t readBytes(char* buffer, size_t length)
341 {
342 int index = 0;
343 int bytesAvailable = 0;
344 uint32_t timeoutMillis = millis() + timeout;
345 while (length > 0 && timeoutMillis > millis())
346 {
347 int bytecount = 4;
348 if (length < 4)
349 {
350 bytecount = length;
351 }
352 {
353
354 uint8_t tx[8] = { 202, _pin,_pinMode, (uint8_t)bytecount,0x55,0x55,0x55,0x55 };
355 uint8_t rx[8];
356 _sw.sendPacket(tx, rx);
357 bytesAvailable = rx[3];
358
359 if (bytesAvailable == 0)
360 {
361 continue;
362 }
363 else
364 {
365 timeoutMillis = millis() + timeout;
366 }
367 uint8_t bytesReturned = bytecount;
368 if (rx[3] < bytecount)
369 {
370 bytesReturned = rx[3];
371 }
372 for (int i = 0; i < bytesReturned; ++i)
373 {
374 buffer[index] = rx[i + 4];
375 ++index;
376 --bytesAvailable;
377 --length;
378
379 }
380 }
381
382 }
383 return (index);
384 }
385
386 int16_t readCurrentScanCodes(uint8_t* buffer, uint8_t startValue)
387 {
388 uint8_t tx[8] = { 207, _pin,_pinMode, startValue,0x55,0x55,0x55,0x55 };
389 uint8_t rx[8];
390
391 int16_t result = _sw.sendPacket(tx, rx);
392
393 if (result < 0)
394 {
395 return result;
396 }
397 int count = 0;
398 for (int i = 3; i < 8; ++i)
399 {
400 if (rx[i] != 0)
401 {
402 ++count;
403 }
404 buffer[i - 3] = rx[i];
405 }
406 return count;
407 }
408
412 void setTimeout(long timeout_mS)
413 {
414 if (timeout_mS == 0)
415 {
416 timeout = 0x80000000;
417 }
418 else
419 {
420 timeout = timeout_mS;
421 }
422 }
423
427 uint8_t scanCodeToAscii(uint8_t scanCode, bool shiftActive = false)
428 {
429
430 if (scanCode >= 0x80)
431 {
432 return 0;
433 }
434 //This array is used to convert PS2 Code set 2 codes to Ascii. The first column is unshfited data, the second if shift is active
435 const uint8_t KBSCSet2[][2] =
436 {
437 { ' ' , ' ' }, /* 0 */
438 { ' ' , ' ' }, /* 1 */
439 { ' ' , ' ' }, /* 2 */
440 { ' ' , ' ' }, /* 3 */
441 { ' ' , ' ' }, /* 4 */
442 { ' ' , ' ' }, /* 5 */
443 { ' ' , ' ' }, /* 6 */
444 { ' ' , ' ' }, /* 7 */
445 { ' ' , ' ' }, /* 8 */
446 { ' ' , ' ' }, /* 9 */
447 { ' ' , ' ' }, /* A */
448 { ' ' , ' ' }, /* B */
449 { ' ' , ' ' }, /* C */
450 { 9 , 9 }, /* D */
451 { '`' , '~' }, /* E */
452 { ' ' , ' ' }, /* F */
453 { ' ' , ' ' }, /* 10 */
454 { ' ' , ' ' }, /* 11 */
455 { 0 , 0 }, /* 12 */
456 { ' ' , ' ' }, /* 13 */
457 { ' ' , ' ' }, /* 14 */
458 { 'q' , 'Q' }, /* 15 */
459 { '1' , '!' }, /* 16 */
460 { ' ' , ' ' }, /* 17 */
461 { ' ' , ' ' }, /* 18 */
462 { ' ' , ' ' }, /* 19 */
463 { 'z' , 'Z' }, /* 1A */
464 { 's' , 'S' }, /* 1B */
465 { 'a' , 'A' }, /* 1C */
466 { 'w' , 'W' }, /* 1D */
467 { '2' , '@' }, /* 1E */
468 { ' ' , ' ' }, /* 1F */
469 { ' ' , ' ' }, /* 20 */
470 { 'c' , 'C' }, /* 21 */
471 { 'x' , 'X' }, /* 22 */
472 { 'd' , 'D' }, /* 23 */
473 { 'e' , 'E' }, /* 24 */
474 { '4' , '$' }, /* 25 */
475 { '3' , '#' }, /* 26 */
476 { ' ' , ' ' }, /* 27 */
477 { ' ' , ' ' }, /* 28 */
478 { ' ' , ' ' }, /* 29 - SPACE */
479 { 'v' , 'V' }, /* 2A */
480 { 'f' , 'F' }, /* 2B */
481 { 't' , 'T' }, /* 2C */
482 { 'r' , 'R' }, /* 2D */
483 { '5' , '%' }, /* 2E */
484 { ' ' , ' ' }, /* 2F */
485 { ' ' , ' ' }, /* 30 */
486 { 'n' , 'N' }, /* 31 */
487 { 'b' , 'B' }, /* 32 */
488 { 'h' , 'H' }, /* 33 */
489 { 'g' , 'G' }, /* 34 */
490 { 'y' , 'Y' }, /* 35 */
491 { '6' , '^' }, /* 36 */
492 { ' ' , ' ' }, /* 37 */
493 { ' ' , ' ' }, /* 38 */
494 { ' ' , ' ' }, /* 39 */
495 { 'm' , 'M' }, /* 3A */
496 { 'j' , 'J' }, /* 3B */
497 { 'u' , 'U' }, /* 3C */
498 { '7' , '&' }, /* 3D */
499 { '8' , '*' }, /* 3E */
500 { ' ' , ' ' }, /* 3F */
501 { ' ' , ' ' }, /* 40 */
502 { ',' , '>' }, /* 41 */
503 { 'k' , 'K' }, /* 42 */
504 { 'i' , 'I' }, /* 43 */
505 { 'o' , 'O' }, /* 44 */
506 { '0' , ')' }, /* 45 */
507 { '9' , '(' }, /* 46 */
508 { ' ' , ' ' }, /* 47 */
509 { ' ' , ' ' }, /* 48 */
510 { '.' , '<' }, /* 49 */
511 { '/' , '?' }, /* 4A */
512 { 'l' , 'L' }, /* 4B */
513 { ';' , ':' }, /* 4C */
514 { 'p' , 'P' }, /* 4D */
515 { '-' , '_' }, /* 4E */
516 { ' ' , ' ' }, /* 4F */
517 { ' ' , ' ' }, /* 50 */
518 { ' ' , ' ' }, /* 51 */
519 { '\'' , '"' }, /* 52 */
520 { ' ' , ' ' }, /* 53 */
521 { '[' , '{' }, /* 54 */
522 { '=' , '+' }, /* 55 */
523 { ' ' , ' ' }, /* 56 */
524 { ' ' , ' ' }, /* 57 */
525 { ' ' , ' ' }, /* 58 */
526 { 0 , 0 }, /* 59 */
527 { 0xD , 0xD }, /* 5A */
528 { ']' , '}' }, /* 5B */
529 { ' ' , ' ' }, /* 5C */
530 { '\\' , '|' }, /* 5D */
531 { ' ' , ' ' }, /* 5E */
532 { ' ' , ' ' }, /* 5F */
533 { ' ' , ' ' }, /* 60 */
534 { ' ' , ' ' }, /* 61 */
535 { ' ' , ' ' }, /* 62 */
536 { ' ' , ' ' }, /* 63 */
537 { ' ' , ' ' }, /* 64 */
538 { ' ' , ' ' }, /* 65 */
539 { 8 , 8 }, /* 66 */
540 { ' ' , ' ' }, /* 67 */
541 { ' ' , ' ' }, /* 68 */
542 { '1' , '1' }, /* 69 */
543 { ' ' , ' ' }, /* 6A */
544 { '4' , '4' }, /* 6B */
545 { '7' , '7' }, /* 6C */
546 { ' ' , ' ' }, /* 6D */
547 { ' ' , ' ' }, /* 6E */
548 { ' ' , ' ' }, /* 6F */
549 { '0' , '0' }, /* 70 */
550 { ' ' , ' ' }, /* 71 */
551 { '2' , '2' }, /* 72 */
552 { '5' , '5' }, /* 73 */
553 { '6' , '6' }, /* 74 */
554 { '8' , '8' }, /* 75 */
555 { 0x1B , 0x1B }, /* 76 */
556 { ' ' , ' ' }, /* 77 */
557 { ' ' , ' ' }, /* 78 */
558 { ' ' , ' ' }, /* 79 */
559 { '3' , '3' }, /* 7A */
560 { ' ' , ' ' }, /* 7B */
561 { ' ' , ' ' }, /* 7C */
562 { '9' , '9' }, /* 7D */
563 { ' ' , ' ' }, /* 7E */
564 { ' ' , ' ' }, /* 7F */
565 };
566
567 if (shiftActive)
568 {
569 return ( KBSCSet2[scanCode][ 1]);
570 }
571
572 return ( KBSCSet2[scanCode][0]);
573 }
574
580 bool isKeyPressed(uint8_t scanCode)
581{
582 uint8_t buffer[5];
583 int16_t result = readCurrentScanCodes(buffer, scanCode);
584
585 if (result < 0)
586 {
587 return false;
588 }
589
590 if (buffer[0] == scanCode)
591 {
592 return true;
593 }
594
595 return false;
596
597}
598
600{
601 return isKeyPressed((uint8_t) scanCode);
602}
603
604 uint8_t _pin = 255;
605protected:
606 uint32_t timeout = 1;
607};
608
609
610
611
#define SW_LE16(_a)
Convert a uint16_t to two bytes in little endian format for array initialization.
@ PIN_MODE_PS2KEYBOARD
(32)
@ SCANCODE_RIGHTALT
@ SCANCODE_SEMICOLON
@ SCANCODE_OPENSINGLEQUOTE
@ SCANCODE_RIGHTBRACKET
@ SCANCODE_RIGHTCTRL
@ SCANCODE_DOWNARROW
@ SCANCODE_SPACEBAR
@ SCANCODE_LEFTBRACKET
@ SCANCODE_KPASTERISK
@ SCANCODE_PRINTSCREEN
@ SCANCODE_LEFTSHIFT
@ SCANCODE_RIGHTSHIFT
@ SCANCODE_BACKSPACE
@ SCANCODE_CAPSLOCK
@ SCANCODE_SCROLLLOCK
@ SCANCODE_LEFTCTRL
@ SCANCODE_LEFTARROW
@ SCANCODE_RIGHTARROW
Class for a Serial Wombat chip. Each Serial Wombat chip on a project should have its own instance.
SerialWombatChip & _sw
SerialWombatPin(SerialWombatChip &serialWombatChip)
Instantiates a Serial Wombat Pin.
int peek()
Query the SerialWombatPS2Keyboard queue for the next avaialble byte, but don't remove it from the que...
bool isKeyPressed(uint8_t scanCode)
Check to see if a key is currently pressed (pin mode must be configured for bitfield mode)
int16_t readCurrentScanCodes(uint8_t *buffer, uint8_t startValue)
size_t write(uint8_t data)
Write a byte to the SerialWombatPS2Keyboard queue (Does Nothing)
int available()
Queries the SerialWombatPS2Keyboard for number bytes available to read.
size_t write(const uint8_t *buffer, size_t size)
Write bytes to the SerialWombatPS2Keyboard queue (Does nothing)
int read()
Reads a byte from the SerialWombatPS2Keyboard queue.
void setTimeout(long timeout_mS)
implemented to fulfill Stream requirement.
bool isKeyPressed(PS2KeyboardScanCode scanCode)
size_t readBytes(char *buffer, size_t length)
Reads a specified number of bytes from the SerialWombatPS2Keyboard queue queue.
int availableForWrite()
Number of bytes avaialble to write to SerialWombatPS2Keyboard queue. Returns 0.
void flush()
Discard all bytes from the SerialWombatPS2Keyboard queue.
uint8_t scanCodeToAscii(uint8_t scanCode, bool shiftActive=false)
convernts a set 2 Scan Code to Ascii
SerialWombatPS2Keyboard(SerialWombatChip &serialWombat)
Constructor for the SerialWombatPS2Keyboard class.
int16_t begin(uint8_t clockPin, uint8_t dataPin, uint8_t bufferMode=0, uint8_t queueMode=0, uint16_t queueAddress=0xFFFF, uint8_t pullUpDown=0)
Initalize the SerialWombatPS2Keyboard.