diff -r 0184f7e8d2fa sys/arch/evbarm/conf/GENERIC64 --- a/sys/arch/evbarm/conf/GENERIC64 Tue Dec 01 06:55:48 2020 +0000 +++ b/sys/arch/evbarm/conf/GENERIC64 Fri Dec 04 21:50:36 2020 +0000 @@ -317,6 +317,7 @@ pcf8563rtc* at iic? # PCF8563 RTC rkpmic* at iic? # Rockchip Power Management IC rkreg* at rkpmic? tcagpio* at iic? +goodixts* at iic? # Goodix touchscreen # Random number generators amdccp* at fdt? # AMD Cryptograhic Coprocessor RNG diff -r 0184f7e8d2fa sys/dev/i2c/axppmic.c --- a/sys/dev/i2c/axppmic.c Tue Dec 01 06:55:48 2020 +0000 +++ b/sys/dev/i2c/axppmic.c Fri Dec 04 21:50:36 2020 +0000 @@ -164,6 +164,8 @@ struct axppmic_ctrl { .c_enable_reg = (ereg), .c_enable_mask = (emask), \ .c_enable_val = (emask), .c_disable_val = 0 } +/* https://files.pine64.org/doc/datasheet/pine64/AXP803_Datasheet_V1.0.pdf */ + static const struct axppmic_ctrl axp803_ctrls[] = { AXP_CTRL("dldo1", 700, 3300, 100, 0x12, __BIT(3), 0x15, __BITS(4,0)), @@ -201,6 +203,10 @@ static const struct axppmic_ctrl axp803_ 0x13, __BIT(6), 0x29, __BITS(4,0)), AXP_CTRL("aldo3", 700, 3300, 100, 0x13, __BIT(7), 0x2a, __BITS(4,0)), + AXP_CTRL_IO("ldo-io0", 700, 3300, 100, + 0x90, __BITS(2,0), 0x3, 0x7, 0x91, __BITS(4,0)), + AXP_CTRL_IO("ldo-io1", 700, 3300, 100, + 0x92, __BITS(2,0), 0x3, 0x7, 0x93, __BITS(4,0)), }; static const struct axppmic_ctrl axp805_ctrls[] = { diff -r 0184f7e8d2fa sys/dev/i2c/files.i2c --- a/sys/dev/i2c/files.i2c Tue Dec 01 06:55:48 2020 +0000 +++ b/sys/dev/i2c/files.i2c Fri Dec 04 21:50:36 2020 +0000 @@ -400,3 +400,8 @@ file dev/i2c/pcagpio.c pcagpio device pcf8574io: leds, sysmon_envsys attach pcf8574io at iic file dev/i2c/pcf8574.c pcf8574io + +# Goodix touchscreen driver +device goodixts: fdt, wsmousedev, tpcalib +attach goodixts at iic +file dev/i2c/goodixts.c goodixts diff -r 0184f7e8d2fa sys/dev/i2c/goodixts.c --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/sys/dev/i2c/goodixts.c Fri Dec 04 21:50:36 2020 +0000 @@ -0,0 +1,725 @@ +/* $NetBSD$ */ + +/*- + * Copyright (c) 2020 The NetBSD Foundation, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS + * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +/* + * Goodix touchscreen driver + * + * https://files.pine64.org/doc/datasheet/pinephone/GT917S-Datasheet.pdf + * http://dl.linux-sunxi.org/touchscreen/GT928%20Datashet%20English.pdf + */ + +#include +__KERNEL_RCSID(0, "$NetBSD$"); + +#include +#include + +#include +#include +#include +#include +#include + +#include + +#include + +#include +#include +#include + +#define GOODIXTS_CMD 0x8040 +#define GOODIXTS_CMD_READ_COORD 0 +#define GOODIXTS_CMD_READ_DIFF_RAW 1 +#define GOODIXTS_CMD_SOFTWARE_RESET 2 +#define GOODIXTS_CMD_BASELINE_UPDATE 3 +#define GOODIXTS_CMD_BASELINE_CALIB 4 +#define GOODIXTS_CMD_SCREEN_OFF 5 +#define GOODIXTS_PROXIMITY_EN 0x8042 +#define GOODIXTS_GT9X_CONFIG 0x8047 +#define GOODIXTS_GT1X_CONFIG 0x8050 +#define GOODIXTS_CONFIG_LEN_GT917S 240 +#define GOODIXTS_CONFIG_MAX_LEN 240 +#define GOODIXTS_ID 0x8140 +#define GOODIXTS_ID_LEN 6 +#define GOODIXTS_REPORT 0x814e + +struct goodixts_config { + uint8_t version; + uint8_t x_max_le[2]; + uint8_t y_max_le[2]; + uint8_t touch_number; +#define TOUCH_NUMBER_TOUCHNO __BITS(0,3) + uint8_t module_switch1; +#define MODULE_SWITCH1_TRIGGER __BITS(0,1) +#define MODULE_SWITCH1_TRIGGER_RISING 0 +#define MODULE_SWITCH1_TRIGGER_FALLING 1 +#define MODULE_SWITCH1_TRIGGER_LOW 2 +#define MODULE_SWITCH1_TRIGGER_HIGH 3 +#define MODULE_SWITCH1_SITO __BIT(2) +#define MODULE_SWITCH1_X2Y __BIT(3) +#define MODULE_SWITCH1_STRETCH_RANK __BITS(3,4) + uint8_t module_switch2; +#define MODULE_SWITCH2_TOUCHKEY __BIT(0) + uint8_t shake_count; +#define SHAKE_COUNT_FINGERSHAKECOUNT __BITS(0,3) + uint8_t filter; +#define FILTER_NORMAL __BITS(0,5) +#define FILTER_FIRST __BITS(6,7) + uint8_t large_touch; + uint8_t noise_reduction; +#define NOISE_REDUCTION_VALUE __BITS(0,4) + uint8_t screen_touch_level; + uint8_t screen_leave_level; + uint8_t low_power_control; +#define LOW_POWER_CONTROL_TIME __BITS(0,3) + uint8_t refresh_rate; +#define REFRESH_RATE_P5MS __BITS(0,3) + uint8_t x_threshold_reserved; + uint8_t y_threshold_reserved; + uint8_t x_speed_limit_reserved; + uint8_t y_speed_limit_reserved; + uint8_t spacev; +#define SPACEV_BLANK_BOTTOM __BITS(0,3) +#define SPACEV_BLANK_TOP __BITS(4,7) + uint8_t spaceh; +#define SPACEH_BLANK_RIGHT __BITS(0,3) +#define SPACEH_BLANK_LEFT __BITS(4,7) + uint8_t stretch_rate_reserved; + uint8_t stretch_r0; + uint8_t stretch_r1; + uint8_t stretch_r2; + uint8_t stretch_rm; + uint8_t drv_groupA_num; +#define DRV_GROUPA_NUM_NUM __BITS(0,4) +#define DRV_GROUPA_NUM_ALL __BIT(7) + uint8_t drv_groupB_num; +#define DRV_GROUPB_NUM_NUM __BITS(0,4) +#define DRV_GROUPB_NUM_DUAL_FREQ __BIT(5) + uint8_t sensor_num; +#define SENSOR_NUM_GROUPA __BITS(0,3) +#define SENSOR_NUM_GROUPB __BITS(4,7) + uint8_t freqA_factor; + uint8_t freqB_factor; + uint8_t panel_bitfreq_le[2]; + uint8_t panel_sensor_time_reserved[2]; + uint8_t panel_tx_gain; +#define PANEL_TX_GAIN_DAC_GAIN __BITS(0,2) +#define PANEL_TX_GAIN_DRV_OUTPUT_R __BITS(3,4) + /* ok, I'm bored transcribing these, several pages more */ +}; +CTASSERT(sizeof(struct goodixts_config) == 0x806c - 0x8047); +CTASSERT(sizeof(struct goodixts_config) <= GOODIXTS_CONFIG_MAX_LEN); + +static const struct { + int type; + char name[12]; +} intr_trigger[] = { + [0] = { FDT_INTR_TYPE_POS_EDGE, "pos-edge" }, + [1] = { FDT_INTR_TYPE_NEG_EDGE, "neg-edge" }, + [2] = { FDT_INTR_TYPE_LOW_LEVEL, "low-level" }, + [3] = { FDT_INTR_TYPE_HIGH_LEVEL, "high-level" }, +}; + +struct goodixts_softc { + device_t sc_dev; + i2c_tag_t sc_i2c; + i2c_addr_t sc_addr; + int sc_phandle; + + struct fdtbus_regulator *sc_avdd28; + struct fdtbus_regulator *sc_vddio; + uint32_t sc_x, sc_y; + + char sc_id[4 + 1]; + uint16_t sc_version; + const struct goodixts_model *sc_model; + union { + struct goodixts_config c; + uint8_t buf[GOODIXTS_CONFIG_MAX_LEN]; + } sc_config; + + struct workqueue *sc_wq; + struct work sc_work; + volatile unsigned sc_work_pending; + + unsigned sc_intr_specifier[2]; + void *sc_ih; + + struct { + struct sysctllog *log; + const struct sysctlnode *root; + } sc_sysctl; + + struct tpcalib_softc sc_tpcalib; + device_t sc_wsmousedev; +}; + +static const struct goodixts_model { + const char gm_id[4 + 1]; + uint8_t gm_config_len; + uint16_t gm_config_reg; +} goodixts_models[] = { + { "917S", 240, GOODIXTS_GT9X_CONFIG }, +}; + +static int goodixts_match(device_t, cfdata_t, void *); +static void goodixts_attach(device_t, device_t, void *); +static int goodixts_detach(device_t, int); +static void goodixts_childdet(device_t, device_t); + +static int goodixts_read(struct goodixts_softc *, uint16_t, void *, size_t); +static int goodixts_write_8(struct goodixts_softc *, uint16_t, uint8_t); + +static int goodixts_reset(struct goodixts_softc *); + +static void goodixts_sysctl_attach(struct goodixts_softc *); +static void goodixts_sysctl_detach(struct goodixts_softc *); + +static int goodixts_enable(void *); +static void goodixts_disable(void *); +static int goodixts_ioctl(void *, unsigned long, void *, int, struct lwp *); + +static uint8_t goodixts_keycode(const uint8_t *, unsigned); + +static int goodixts_intr(void *); +static void goodixts_work(struct work *, void *); + +static const struct wsmouse_accessops goodixts_accessops = { + .enable = goodixts_enable, + .disable = goodixts_disable, + .ioctl = goodixts_ioctl, +}; + +CFATTACH_DECL2_NEW(goodixts, sizeof(struct goodixts_softc), + goodixts_match, goodixts_attach, goodixts_detach, NULL, NULL, + goodixts_childdet); + +static const struct device_compatible_entry compat_data[] = { + { "goodix,gt917s", 0 }, + { NULL, 0 }, +}; + +static int +goodixts_match(device_t parent, cfdata_t match, void *aux) +{ + const struct i2c_attach_args *ia = aux; + int match_result; + + if (iic_use_direct_match(ia, match, compat_data, &match_result)) + return match_result; + return 0; +} + +static void +goodixts_attach(device_t parent, device_t self, void *aux) +{ + struct goodixts_softc *sc = device_private(self); + struct i2c_attach_args *ia = aux; + struct wsmousedev_attach_args a; + uint8_t idbuf[12]; + char intrstr[128]; + unsigned i; + int error; + + sc->sc_dev = self; + sc->sc_i2c = ia->ia_tag; + sc->sc_addr = ia->ia_addr; + sc->sc_phandle = ia->ia_cookie; + + aprint_naive("\n"); + aprint_normal(": Goodix capacitative touchscreen\n"); + + /* Enable the analog power regulator. */ + sc->sc_avdd28 = fdtbus_regulator_acquire(sc->sc_phandle, + "AVDD28-supply"); + if (sc->sc_avdd28 == NULL) { + aprint_error_dev(self, "AVDD28-supply not found\n"); + return; + } + error = fdtbus_regulator_enable(sc->sc_avdd28); + if (sc->sc_avdd28 == NULL) { + aprint_error_dev(self, "AVDD28-supply disabled\n"); + return; + } + + /* Enable to GPIO supply voltage regulator. */ + sc->sc_vddio = fdtbus_regulator_acquire(sc->sc_phandle, + "VDDIO-supply"); + if (sc->sc_avdd28 == NULL) { + aprint_error_dev(self, "VDDIO-supply not found\n"); + return; + } + error = fdtbus_regulator_enable(sc->sc_vddio); + if (sc->sc_vddio == NULL) { + aprint_error_dev(self, "VDDIO-supply disabled\n"); + return; + } + + /* Get the touchscreen sizes. */ + of_getprop_uint32(sc->sc_phandle, "touchscreen-size-x", &sc->sc_x); + of_getprop_uint32(sc->sc_phandle, "touchscreen-size-y", &sc->sc_y); + + /* Reset the device and identify it. */ + error = goodixts_reset(sc); + if (error) { + aprint_error_dev(self, "failed to reset: %d\n", error); + return; + } + error = goodixts_read(sc, GOODIXTS_ID, idbuf, sizeof(idbuf)); + if (error) { + aprint_error_dev(self, "failed to read version: %d\n", error); + return; + } + strlcpy(sc->sc_id, idbuf, sizeof(sc->sc_id)); + sc->sc_version = le16dec(&idbuf[4]); + aprint_normal_dev(self, "%s version %04"PRIx16"\n", + sc->sc_id, sc->sc_version); + + /* Find the model or fail if we don't recognize it. */ + for (i = 0; i < __arraycount(goodixts_models); i++) { + if (strncmp(sc->sc_id, goodixts_models[i].gm_id, + sizeof(sc->sc_id)) == 0) { + sc->sc_model = &goodixts_models[i]; + break; + } + } + if (sc->sc_model == NULL) { + aprint_error_dev(self, "unknown model\n"); + return; + } + + /* Read the configuration. */ + KASSERT(sc->sc_model->gm_config_len <= sizeof(sc->sc_config)); + error = goodixts_read(sc, sc->sc_model->gm_config_reg, + sc->sc_config.buf, sc->sc_model->gm_config_len); + if (error) { + aprint_error_dev(self, "failed to read config: %d\n", error); + return; + } + + const unsigned trigger = __SHIFTOUT(sc->sc_config.c.module_switch1, + MODULE_SWITCH1_TRIGGER); + aprint_normal_dev(self, "intr trigger %s", intr_trigger[trigger].name); + + error = workqueue_create(&sc->sc_wq, device_xname(self), goodixts_work, + sc, PRI_NONE, IPL_NONE, WQ_MPSAFE); + if (error) { + aprint_error_dev(self, "can't create workqueue: %d\n", error); + return; + } + + if (!fdtbus_intr_str(sc->sc_phandle, 0, intrstr, sizeof(intrstr))) { + aprint_error_dev(self, "can't decode interrupt\n"); + return; + } + aprint_normal(", interrupting at %s\n", intrstr); + + /* XXX fdtbus_intr_establish_raw with trigger */ + /* XXX mask when not in use? */ +#if 0 + const unsigned specifier[3] = { + [0] = 7, /* interrupt type */ + [1] = 4, /* interrupt number */ + [2] = intr_trigger[trigger].type, + }; +#endif + sc->sc_ih = fdtbus_intr_establish(sc->sc_phandle, 0, IPL_TTY, + FDT_INTR_MPSAFE, goodixts_intr, sc); + if (sc->sc_ih == NULL) { + aprint_error_dev(self, "can't establish interrupt\n"); + return; + } + + /* Create sysctl nodes. */ + goodixts_sysctl_attach(sc); + + /* Attach wsmouse. */ + a.accessops = &goodixts_accessops; + a.accesscookie = sc; + sc->sc_wsmousedev = config_found_ia(self, "wsmousedev", &a, + wsmousedevprint); + + /* XXX wskbd for home `key'? */ +} + +static int +goodixts_detach(device_t self, int flags) +{ + struct goodixts_softc *sc = device_private(self); + int error; + + error = config_detach_children(self, flags); + if (error) + return error; + + goodixts_sysctl_detach(sc); + if (sc->sc_ih) + fdtbus_intr_disestablish(sc->sc_phandle, sc->sc_ih); + if (sc->sc_wq) + workqueue_destroy(sc->sc_wq); + if (sc->sc_vddio) { + fdtbus_regulator_disable(sc->sc_vddio); + fdtbus_regulator_release(sc->sc_vddio); + } + if (sc->sc_avdd28) { + fdtbus_regulator_disable(sc->sc_avdd28); + fdtbus_regulator_release(sc->sc_avdd28); + } + + return 0; +} + +static void +goodixts_childdet(device_t self, device_t child) +{ + struct goodixts_softc *sc = device_private(self); + + KASSERT(sc->sc_wsmousedev == child); + atomic_store_relaxed(&sc->sc_wsmousedev, NULL); +} + +static int +goodixts_read(struct goodixts_softc *sc, uint16_t reg, void *buf, size_t len) +{ + uint8_t cmd[2]; + int error; + + be16enc(cmd, reg); + + error = iic_acquire_bus(sc->sc_i2c, 0); + if (error) + return error; + + error = iic_exec(sc->sc_i2c, I2C_OP_READ_WITH_STOP, sc->sc_addr, cmd, + sizeof(cmd), buf, len, 0); + iic_release_bus(sc->sc_i2c, 0); + return error; +} + +static int __unused +goodixts_write_8(struct goodixts_softc *sc, uint16_t reg, uint8_t val) +{ + uint8_t cmd[3]; + int error; + + be16enc(cmd, reg); + cmd[2] = val; + + error = iic_acquire_bus(sc->sc_i2c, 0); + if (error) + return error; + + error = iic_exec(sc->sc_i2c, I2C_OP_WRITE_WITH_STOP, sc->sc_addr, cmd, + sizeof(cmd), NULL, 0, 0); + iic_release_bus(sc->sc_i2c, 0); + return error; +} + +static int +goodixts_reset(struct goodixts_softc *sc) +{ + device_t self = sc->sc_dev; + struct fdtbus_gpio_pin *irq_gpios; + struct fdtbus_gpio_pin *reset_gpios; + int error; + + reset_gpios = fdtbus_gpio_acquire(sc->sc_phandle, "reset-gpios", + GPIO_PIN_INPUT|GPIO_PIN_OUTPUT); + if (reset_gpios == NULL) { + aprint_error_dev(self, "reset-gpios not found\n"); + error = ENXIO; + goto out0; + } + + irq_gpios = fdtbus_gpio_acquire(sc->sc_phandle, "irq-gpios", + GPIO_PIN_INPUT|GPIO_PIN_OUTPUT); + if (irq_gpios == NULL) { + aprint_error_dev(self, "irq-gpios not found\n"); + error = ENXIO; + goto out1; + } + + fdtbus_gpio_write(reset_gpios, 0); + DELAY(10*1000); /* T5 >= 10ms */ + fdtbus_gpio_write(reset_gpios, 1); + + fdtbus_gpio_write(irq_gpios, 0); + DELAY(5*1000); /* 5ms <= T4 <= 20ms */ + + (void)fdtbus_gpio_read(irq_gpios); + (void)fdtbus_gpio_read(reset_gpios); + + /* Success! */ + error = 0; + +out1: fdtbus_gpio_release(irq_gpios); + fdtbus_gpio_release(reset_gpios); +out0: return error; +} + +/* sysctl hw.goodixtsN subtree */ + +static void +goodixts_sysctl_attach(struct goodixts_softc *sc) +{ + device_t self = sc->sc_dev; + int error; + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + NULL, &sc->sc_sysctl.root, + CTLFLAG_PERMANENT, CTLTYPE_NODE, device_xname(self), + SYSCTL_DESCR("Goodix touchscreen"), + NULL, 0, NULL, 0, + CTL_HW, CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s: %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_STRUCT, + "addr", SYSCTL_DESCR("I2C address"), + NULL, 0, &sc->sc_addr, sizeof(sc->sc_addr), + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.addr" + ": %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_STRING, + "id", SYSCTL_DESCR("Model identifier"), + NULL, 0, sc->sc_id, sizeof(sc->sc_id), + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.id" + ": %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_STRUCT, + "version", SYSCTL_DESCR("Model version"), + NULL, 0, &sc->sc_version, sizeof(sc->sc_version), + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.version" + ": %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_STRUCT, + "config", SYSCTL_DESCR("Configuration"), + NULL, 0, sc->sc_config.buf, sc->sc_model->gm_config_len, + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.config" + ": %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_INT, + "x", SYSCTL_DESCR("Horizontal width"), + NULL, 0, &sc->sc_x, sizeof(sc->sc_x), + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.x: %d\n", + device_xname(self), error); + return; + } + + error = sysctl_createv(&sc->sc_sysctl.log, 0, + &sc->sc_sysctl.root, NULL, + CTLFLAG_PERMANENT|CTLFLAG_READONLY|CTLFLAG_PRIVATE, CTLTYPE_INT, + "y", SYSCTL_DESCR("Vertical height"), + NULL, 0, &sc->sc_y, sizeof(sc->sc_y), + CTL_CREATE, CTL_EOL); + if (error) { + aprint_error_dev(self, "failed to set up sysctl hw.%s.y: %d\n", + device_xname(self), error); + return; + } +} + +static void +goodixts_sysctl_detach(struct goodixts_softc *sc) +{ + + sysctl_teardown(&sc->sc_sysctl.log); +} + +/* wsmouse(9) operations */ + +static int +goodixts_enable(void *cookie) +{ + struct goodixts_softc *sc = cookie; + + __USE(sc); + + return 0; +} + +static void +goodixts_disable(void *cookie) +{ + struct goodixts_softc *sc = cookie; + + __USE(sc); +} + +static int +goodixts_ioctl(void *cookie, unsigned long cmd, void *data, int flags, + struct lwp *l) +{ + struct goodixts_softc *sc = cookie; + + __USE(sc); + + return EPASSTHROUGH; +} + +/* Interrupt reports */ + +struct goodixts_report { + uint8_t hdr; +#define REPORT_NTOUCH __BITS(0,3) +#define REPORT_HAVE_KEY __BIT(4) +#define REPORT_LARGE_DETECT __BIT(6) +#define REPORT_BUFFER_STATUS __BIT(7) + struct { + uint8_t track_id; + uint8_t x_le[2]; + uint8_t y_le[2]; + uint8_t sz_le[2]; + uint8_t reserved; + } touch[10]; +}; + +static uint8_t +goodixts_keycode(const uint8_t *buf, unsigned ntouch) +{ + size_t off = offsetof(struct goodixts_report, touch[ntouch]); + + return ((const uint8_t *)buf)[off]; +} + +static int +goodixts_intr(void *cookie) +{ + struct goodixts_softc *sc = cookie; + + aprint_debug_dev(sc->sc_dev, "%s\n", __func__); + + if (atomic_swap_uint(&sc->sc_work_pending, 1) == 0) + workqueue_enqueue(sc->sc_wq, &sc->sc_work, NULL); + + return 1; /* handled */ +} + +static void +goodixts_work(struct work *work, void *cookie) +{ + struct goodixts_softc *sc = cookie; + union { + uint8_t buf[sizeof(struct goodixts_report) + 1]; + struct goodixts_report report; + } u; + unsigned ntouch, timo, keycode, i; + device_t wsmousedev; + int error; + + aprint_debug_dev(sc->sc_dev, "%s\n", __func__); + + for (timo = 20; error = ETIMEDOUT, timo --> 0; DELAY(1000)) { + error = goodixts_read(sc, GOODIXTS_REPORT, u.buf, + sizeof(u.buf)); + if (error) + break; + aprint_debug_dev(sc->sc_dev, "report header %hhx\n", + u.report.hdr); + if (u.report.hdr & REPORT_BUFFER_STATUS) { + ntouch = __SHIFTOUT(u.report.hdr, REPORT_NTOUCH); + aprint_debug_dev(sc->sc_dev, "ntouch %hhu\n", ntouch); + if (ntouch > 10) /* XXX magic constant */ + error = EIO; + break; + } + } + if (error) { + aprint_error_dev(sc->sc_dev, "failed to read report: %d", + error); + goto out; + } + + keycode = goodixts_keycode(u.buf, ntouch); + aprint_debug_dev(sc->sc_dev, "keycode %hhx\n", keycode); + __USE(keycode); /* XXX `keyboard' input? */ + + for (i = 0; i < ntouch; i++) { + aprint_debug_dev(sc->sc_dev, + "touch %u: id=%hhd x=%hd y=%hd sz=%hd", i, + u.report.touch[i].track_id, + le32dec(u.report.touch[i].x_le), + le32dec(u.report.touch[i].y_le), + le32dec(u.report.touch[i].sz_le)); + } + + wsmousedev = atomic_load_relaxed(&sc->sc_wsmousedev); + if (wsmousedev == NULL) + goto out; + for (i = 0; i < ntouch; i++) { + int x = le16dec(u.report.touch[i].x_le); + int y = le16dec(u.report.touch[i].y_le); + /* XXX size? */ + + wsmouse_input(wsmousedev, 0, x, y, 0, 0, + WSMOUSE_INPUT_ABSOLUTE_X|WSMOUSE_INPUT_ABSOLUTE_Y); + break; /* XXX multitouch */ + } + +out: atomic_store_relaxed(&sc->sc_work_pending, 0); + error = goodixts_write_8(sc, GOODIXTS_REPORT, 0); + if (error) + aprint_error_dev(sc->sc_dev, "failed to acknowledge report" + ": %d\n", error); +} diff -r 0184f7e8d2fa sys/dev/wscons/tpcalibvar.h --- a/sys/dev/wscons/tpcalibvar.h Tue Dec 01 06:55:48 2020 +0000 +++ b/sys/dev/wscons/tpcalibvar.h Fri Dec 04 21:50:36 2020 +0000 @@ -27,6 +27,15 @@ * */ +#ifndef _DEV_WSCONS_TPCALIBVAR_H_ +#define _DEV_WSCONS_TPCALIBVAR_H_ + +#include + +#include + +struct lwp; + struct tpcalib_softc { int sc_minx, sc_miny; int sc_maxx, sc_maxy; @@ -41,3 +50,5 @@ void tpcalib_reset(struct tpcalib_softc void tpcalib_trans(struct tpcalib_softc *, int, int, int *, int *); int tpcalib_ioctl(struct tpcalib_softc *, u_long, void *, int, struct lwp *); + +#endif /* _DEV_WSCONS_TPCALIBVAR_H_ */