From 17f65b3a8da672e70ae54781a967e3fbaf063152 Mon Sep 17 00:00:00 2001 From: Ruslan Bukin Date: Fri, 30 May 2014 06:45:50 +0000 Subject: [PATCH] o Make keyboard-related properties to be compatible with vendor standard o Allow setting keymap in FDT, use hardcoded one by default o Represent fallback keymap as a list rather than directly usable M*N array Submitted by: Maxim Ignatenko --- sys/arm/samsung/exynos/chrome_kb.c | 161 +++++++++++++++--- sys/arm/samsung/exynos/chrome_kb.h | 160 +++++++++-------- .../dts/arm/exynos5250-chromebook-snow.dts | 7 +- .../dts/arm/exynos5250-chromebook-spring.dts | 7 +- 4 files changed, 222 insertions(+), 113 deletions(-) diff --git a/sys/arm/samsung/exynos/chrome_kb.c b/sys/arm/samsung/exynos/chrome_kb.c index 4b3d20298ebe..d5d55e9233df 100644 --- a/sys/arm/samsung/exynos/chrome_kb.c +++ b/sys/arm/samsung/exynos/chrome_kb.c @@ -105,9 +105,6 @@ __FBSDID("$FreeBSD$"); #define CKB_FLAG_POLLING 0x2 #define KBD_DRIVER_NAME "ckbd" -/* TODO: take interrupt from DTS */ -#define KB_GPIO_INT 146 - struct ckb_softc { keyboard_t sc_kbd; keymap_t sc_keymap; @@ -130,9 +127,11 @@ struct ckb_softc { int flag; int rows; int cols; + int gpio; device_t dev; device_t gpio_dev; struct thread *sc_poll_thread; + uint16_t *keymap; uint8_t *scan_local; uint8_t *scan; @@ -199,7 +198,7 @@ static int ckb_intr(keyboard_t *kbd, void *arg) { - return (0); + return (0); } /* lock the access to the keyboard, not used */ @@ -207,7 +206,7 @@ static int ckb_lock(keyboard_t *kbd, int lock) { - return (1); + return (1); } /* clear the internal state of the keyboard */ @@ -309,20 +308,33 @@ ckb_read(keyboard_t *kbd, int wait) return (0); } -int scantokey(int i, int j); - -int -scantokey(int i, int j) +static uint16_t +keymap_read(struct ckb_softc *sc, int col, int row) { - int k; - for (k = 0; k < KEYMAP_LEN; k++) - if ((keymap[k].col == i) && (keymap[k].row == j)) - return (keymap[k].key); + KASSERT(sc->keymap != NULL, "keymap_read: no keymap"); + if (col >= 0 && col < sc->cols && + row >= 0 && row < sc->rows) { + return sc->keymap[row * sc->cols + col]; + } return (0); } +static int +keymap_write(struct ckb_softc *sc, int col, int row, uint16_t key) +{ + + KASSERT(sc->keymap != NULL, "keymap_write: no keymap"); + if (col >= 0 && col < sc->cols && + row >= 0 && row < sc->rows) { + sc->keymap[row * sc->cols + col] = key; + return (0); + } + + return (-1); +} + /* read char from the keyboard */ static uint32_t ckb_read_char_locked(keyboard_t *kbd, int wait) @@ -350,9 +362,10 @@ ckb_read_char_locked(keyboard_t *kbd, int wait) if (sc->sc_flags & CKB_FLAG_POLLING) { for (;;) { - GPIO_PIN_GET(sc->gpio_dev, KB_GPIO_INT, &status); + GPIO_PIN_GET(sc->gpio_dev, sc->gpio, &status); if (status == 0) { - if (ec_command(EC_CMD_MKBP_STATE, sc->scan, sc->cols, + if (ec_command(EC_CMD_MKBP_STATE, sc->scan, + sc->cols, sc->scan, sc->cols)) { return (NOKEY); } @@ -373,7 +386,7 @@ ckb_read_char_locked(keyboard_t *kbd, int wait) if (oldbit == newbit) continue; - key = scantokey(i,j); + key = keymap_read(sc, i, j); if (key == 0) { continue; }; @@ -666,27 +679,109 @@ dummy_kbd_configure(int flags) KEYBOARD_DRIVER(ckbd, ckbdsw, dummy_kbd_configure); +/* + * Parses 'keymap' into sc->keymap. + * Requires sc->cols and sc->rows to be set. + */ +static int +parse_keymap(struct ckb_softc *sc, pcell_t *keymap, size_t len) +{ + int i; + + sc->keymap = malloc(sc->cols * sc->rows * sizeof(sc->keymap[0]), + M_DEVBUF, M_NOWAIT | M_ZERO); + if (sc->keymap == NULL) { + return (ENOMEM); + } + + for (i = 0; i < len; i++) { + /* + * Return value is ignored, we just write whatever fits into + * specified number of rows and columns and silently ignore + * everything else. + * Keymap entries follow this format: 0xRRCCKKKK + * RR - row number, CC - column number, KKKK - key code + */ + keymap_write(sc, (keymap[i] >> 16) & 0xff, + (keymap[i] >> 24) & 0xff, + keymap[i] & 0xffff); + } + + return (0); +} + +/* Allocates a new array for keymap and returns it in 'keymap'. */ +static int +read_keymap(phandle_t node, const char *prop, pcell_t **keymap, size_t *len) +{ + + if ((*len = OF_getproplen(node, prop)) <= 0) { + return (ENXIO); + } + if ((*keymap = malloc(*len, M_DEVBUF, M_NOWAIT)) == NULL) { + return (ENOMEM); + } + if (OF_getencprop(node, prop, *keymap, *len) != *len) { + return (ENXIO); + } + return (0); +} + static int parse_dts(struct ckb_softc *sc) { phandle_t node; pcell_t dts_value; - int len; + pcell_t *keymap; + int len, ret; + const char *keymap_prop = NULL; if ((node = ofw_bus_get_node(sc->dev)) == -1) return (ENXIO); - if ((len = OF_getproplen(node, "keypad,num-rows")) <= 0) + if ((len = OF_getproplen(node, "google,key-rows")) <= 0) return (ENXIO); - OF_getprop(node, "keypad,num-rows", &dts_value, len); + OF_getprop(node, "google,key-rows", &dts_value, len); sc->rows = fdt32_to_cpu(dts_value); - if ((len = OF_getproplen(node, "keypad,num-columns")) <= 0) + if ((len = OF_getproplen(node, "google,key-columns")) <= 0) return (ENXIO); - OF_getprop(node, "keypad,num-columns", &dts_value, len); + OF_getprop(node, "google,key-columns", &dts_value, len); sc->cols = fdt32_to_cpu(dts_value); - if ((sc->rows == 0) || (sc->cols == 0)) + if ((len = OF_getproplen(node, "freebsd,intr-gpio")) <= 0) + return (ENXIO); + OF_getprop(node, "freebsd,intr-gpio", &dts_value, len); + sc->gpio = fdt32_to_cpu(dts_value); + + if (OF_hasprop(node, "freebsd,keymap")) { + keymap_prop = "freebsd,keymap"; + device_printf(sc->dev, "using FreeBSD-specific keymap from FDT\n"); + } else if (OF_hasprop(node, "linux,keymap")) { + keymap_prop = "linux,keymap"; + device_printf(sc->dev, "using Linux keymap from FDT\n"); + } else { + device_printf(sc->dev, "using built-in keymap\n"); + } + + if (keymap_prop != NULL) { + if ((ret = read_keymap(node, keymap_prop, &keymap, &len))) { + device_printf(sc->dev, + "failed to read keymap from FDT: %d\n", ret); + return (ret); + } + ret = parse_keymap(sc, keymap, len); + free(keymap, M_DEVBUF); + if (ret) { + return (ret); + } + } else { + if ((ret = parse_keymap(sc, default_keymap, KEYMAP_LEN))) { + return (ret); + } + } + + if ((sc->rows == 0) || (sc->cols == 0) || (sc->gpio == 0)) return (ENXIO); return (0); @@ -721,6 +816,7 @@ chrome_kb_attach(device_t dev) sc = device_get_softc(dev); sc->dev = dev; + sc->keymap = NULL; if ((error = parse_dts(sc)) != 0) return error; @@ -736,8 +832,7 @@ chrome_kb_attach(device_t dev) sc->cols, sc->rows); #endif - /* TODO: take interrupt from DTS */ - pad_setup_intr(KB_GPIO_INT, ckb_ec_intr, sc); + pad_setup_intr(sc->gpio, ckb_ec_intr, sc); kbd = &sc->sc_kbd; rid = 0; @@ -786,7 +881,8 @@ chrome_kb_probe(device_t dev) if (!ofw_bus_status_okay(dev)) return (ENXIO); - if (ofw_bus_is_compatible(dev, "google,cros-ec-keyb")) { + if (ofw_bus_is_compatible(dev, "google,cros-ec-keyb") || + ofw_bus_is_compatible(dev, "google,mkbp-keyb")) { device_set_desc(dev, "Chrome EC Keyboard"); return (BUS_PROBE_DEFAULT); } @@ -794,9 +890,24 @@ chrome_kb_probe(device_t dev) return (ENXIO); } +static int +chrome_kb_detach(device_t dev) +{ + struct ckb_softc *sc; + + sc = device_get_softc(dev); + + if (sc->keymap != NULL) { + free(sc->keymap, M_DEVBUF); + } + + return 0; +} + static device_method_t chrome_kb_methods[] = { DEVMETHOD(device_probe, chrome_kb_probe), DEVMETHOD(device_attach, chrome_kb_attach), + DEVMETHOD(device_detach, chrome_kb_detach), { 0, 0 } }; diff --git a/sys/arm/samsung/exynos/chrome_kb.h b/sys/arm/samsung/exynos/chrome_kb.h index 3903a9be830c..6ff03861b323 100644 --- a/sys/arm/samsung/exynos/chrome_kb.h +++ b/sys/arm/samsung/exynos/chrome_kb.h @@ -26,97 +26,93 @@ * $FreeBSD$ */ -void ckb_ec_intr(void *); +#include -struct key { - uint8_t row; - uint8_t col; - uint8_t key; -}; +void ckb_ec_intr(void *); #define KEYMAP_LEN 75 -struct key keymap[KEYMAP_LEN] = { - { 0x00, 0x01, 0x7d }, /* lmeta */ - { 0x00, 0x02, 0x3b }, /* F1 */ - { 0x00, 0x03, 0x30 }, /* B */ - { 0x00, 0x04, 0x44 }, /* F10 */ - { 0x00, 0x06, 0x31 }, /* N */ - { 0x00, 0x08, 0x0d }, /* = */ - { 0x00, 0x0a, 0x64 }, /* ralt */ +pcell_t default_keymap[KEYMAP_LEN] = { + 0x0001007d, /* lmeta */ + 0x0002003b, /* F1 */ + 0x00030030, /* B */ + 0x00040044, /* F10 */ + 0x00060031, /* N */ + 0x0008000d, /* = */ + 0x000a0064, /* ralt */ - { 0x01, 0x01, 0x01 }, /* escape */ - { 0x01, 0x02, 0x3e }, /* F4 */ - { 0x01, 0x03, 0x22 }, /* G */ - { 0x01, 0x04, 0x41 }, /* F7 */ - { 0x01, 0x06, 0x23 }, /* H */ - { 0x01, 0x08, 0x28 }, /* ' */ - { 0x01, 0x09, 0x43 }, /* F9 */ - { 0x01, 0x0b, 0x0e }, /* backspace */ + 0x01010001, /* escape */ + 0x0102003e, /* F4 */ + 0x01030022, /* G */ + 0x01040041, /* F7 */ + 0x01060023, /* H */ + 0x01080028, /* ' */ + 0x01090043, /* F9 */ + 0x010b000e, /* backspace */ - { 0x02, 0x00, 0x1d }, /* lctrl */ - { 0x02, 0x01, 0x0f }, /* tab */ - { 0x02, 0x02, 0x3d }, /* F3 */ - { 0x02, 0x03, 0x14 }, /* t */ - { 0x02, 0x04, 0x40 }, /* F6 */ - { 0x02, 0x05, 0x1b }, /* ] */ - { 0x02, 0x06, 0x15 }, /* y */ - { 0x02, 0x07, 0x56 }, /* 102nd */ - { 0x02, 0x08, 0x1a }, /* [ */ - { 0x02, 0x09, 0x42 }, /* F8 */ + 0x0200001d, /* lctrl */ + 0x0201000f, /* tab */ + 0x0202003d, /* F3 */ + 0x02030014, /* t */ + 0x02040040, /* F6 */ + 0x0205001b, /* ] */ + 0x02060015, /* y */ + 0x02070056, /* 102nd */ + 0x0208001a, /* [ */ + 0x02090042, /* F8 */ - { 0x03, 0x01, 0x29 }, /* grave */ - { 0x03, 0x02, 0x3c }, /* F2 */ - { 0x03, 0x03, 0x06 }, /* 5 */ - { 0x03, 0x04, 0x3f }, /* F5 */ - { 0x03, 0x06, 0x07 }, /* 6 */ - { 0x03, 0x08, 0x0c }, /* - */ - { 0x03, 0x0b, 0x2b }, /* \ */ + 0x03010029, /* grave */ + 0x0302003c, /* F2 */ + 0x03030006, /* 5 */ + 0x0304003f, /* F5 */ + 0x03060007, /* 6 */ + 0x0308000c, /* - */ + 0x030b002b, /* \ */ - { 0x04, 0x00, 0x61 }, /* rctrl */ - { 0x04, 0x01, 0x1e }, /* a */ - { 0x04, 0x02, 0x20 }, /* d */ - { 0x04, 0x03, 0x21 }, /* f */ - { 0x04, 0x04, 0x1f }, /* s */ - { 0x04, 0x05, 0x25 }, /* k */ - { 0x04, 0x06, 0x24 }, /* j */ - { 0x04, 0x08, 0x27 }, /* ; */ - { 0x04, 0x09, 0x26 }, /* l */ - { 0x04, 0x0a, 0x2b }, /* \ */ - { 0x04, 0x0b, 0x1c }, /* enter */ + 0x04000061, /* rctrl */ + 0x0401001e, /* a */ + 0x04020020, /* d */ + 0x04030021, /* f */ + 0x0404001f, /* s */ + 0x04050025, /* k */ + 0x04060024, /* j */ + 0x04080027, /* ; */ + 0x04090026, /* l */ + 0x040a002b, /* \ */ + 0x040b001c, /* enter */ - { 0x05, 0x01, 0x2c }, /* z */ - { 0x05, 0x02, 0x2e }, /* c */ - { 0x05, 0x03, 0x2f }, /* v */ - { 0x05, 0x04, 0x2d }, /* x */ - { 0x05, 0x05, 0x33 }, /* , */ - { 0x05, 0x06, 0x32 }, /* m */ - { 0x05, 0x07, 0x2a }, /* lsh */ - { 0x05, 0x08, 0x35 }, /* / */ - { 0x05, 0x09, 0x34 }, /* . */ - { 0x05, 0x0B, 0x39 }, /* space */ + 0x0501002c, /* z */ + 0x0502002e, /* c */ + 0x0503002f, /* v */ + 0x0504002d, /* x */ + 0x05050033, /* , */ + 0x05060032, /* m */ + 0x0507002a, /* lsh */ + 0x05080035, /* / */ + 0x05090034, /* . */ + 0x050B0039, /* space */ - { 0x06, 0x01, 0x02 }, /* 1 */ - { 0x06, 0x02, 0x04 }, /* 3 */ - { 0x06, 0x03, 0x05 }, /* 4 */ - { 0x06, 0x04, 0x03 }, /* 2 */ - { 0x06, 0x05, 0x09 }, /* 8 */ - { 0x06, 0x06, 0x08 }, /* 7 */ - { 0x06, 0x08, 0x0b }, /* 0 */ - { 0x06, 0x09, 0x0a }, /* 9 */ - { 0x06, 0x0a, 0x38 }, /* lalt */ - { 0x06, 0x0b, 0x64 }, /* down */ - { 0x06, 0x0c, 0x62 }, /* right */ + 0x06010002, /* 1 */ + 0x06020004, /* 3 */ + 0x06030005, /* 4 */ + 0x06040003, /* 2 */ + 0x06050009, /* 8 */ + 0x06060008, /* 7 */ + 0x0608000b, /* 0 */ + 0x0609000a, /* 9 */ + 0x060a0038, /* lalt */ + 0x060b0064, /* down */ + 0x060c0062, /* right */ - { 0x07, 0x01, 0x10 }, /* q */ - { 0x07, 0x02, 0x12 }, /* e */ - { 0x07, 0x03, 0x13 }, /* r */ - { 0x07, 0x04, 0x11 }, /* w */ - { 0x07, 0x05, 0x17 }, /* i */ - { 0x07, 0x06, 0x16 }, /* u */ - { 0x07, 0x07, 0x36 }, /* rsh */ - { 0x07, 0x08, 0x19 }, /* p */ - { 0x07, 0x09, 0x18 }, /* o */ - { 0x07, 0x0b, 0x5F }, /* up */ - { 0x07, 0x0c, 0x61 }, /* left */ + 0x07010010, /* q */ + 0x07020012, /* e */ + 0x07030013, /* r */ + 0x07040011, /* w */ + 0x07050017, /* i */ + 0x07060016, /* u */ + 0x07070036, /* rsh */ + 0x07080019, /* p */ + 0x07090018, /* o */ + 0x070b005F, /* up */ + 0x070c0061, /* left */ }; diff --git a/sys/boot/fdt/dts/arm/exynos5250-chromebook-snow.dts b/sys/boot/fdt/dts/arm/exynos5250-chromebook-snow.dts index f7dc3f98f5cc..c8ef54aea044 100644 --- a/sys/boot/fdt/dts/arm/exynos5250-chromebook-snow.dts +++ b/sys/boot/fdt/dts/arm/exynos5250-chromebook-snow.dts @@ -59,9 +59,10 @@ }; keyboard-controller { - compatible = "google,cros-ec-keyb"; - keypad,num-rows = <8>; - keypad,num-columns = <13>; + compatible = "google,mkbp-keyb"; + google,key-rows = <8>; + google,key-columns = <13>; + freebsd,intr-gpio = <146>; }; }; diff --git a/sys/boot/fdt/dts/arm/exynos5250-chromebook-spring.dts b/sys/boot/fdt/dts/arm/exynos5250-chromebook-spring.dts index ef1aa092414b..9f7c1faa4450 100644 --- a/sys/boot/fdt/dts/arm/exynos5250-chromebook-spring.dts +++ b/sys/boot/fdt/dts/arm/exynos5250-chromebook-spring.dts @@ -59,9 +59,10 @@ }; keyboard-controller { - compatible = "google,cros-ec-keyb"; - keypad,num-rows = <8>; - keypad,num-columns = <13>; + compatible = "google,mkbp-keyb"; + google,key-rows = <8>; + google,key-columns = <13>; + freebsd,intr-gpio = <146>; }; };