From 3fefc3f71685835ac54b26dec9df765eea0059b1 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Mon, 17 Mar 2025 15:07:29 +0100 Subject: [PATCH 1/5] touch: prepare touch-screen driver --- include/gint/defs/util.h | 6 ++ include/gint/touch.h | 40 +++++++++ src/kernel/exch.c | 6 ++ src/touch/adconv.c | 114 ++++++++++++++++++++++++ src/touch/adconv.h | 77 ++++++++++++++++ src/touch/i2c.c | 186 +++++++++++++++++++++++++++++++++++++++ src/touch/i2c.h | 124 ++++++++++++++++++++++++++ src/touch/i2c_inth.c | 166 ++++++++++++++++++++++++++++++++++ src/touch/touch.c | 53 +++++++++++ 9 files changed, 772 insertions(+) create mode 100644 include/gint/touch.h create mode 100644 src/touch/adconv.c create mode 100644 src/touch/adconv.h create mode 100644 src/touch/i2c.c create mode 100644 src/touch/i2c.h create mode 100644 src/touch/i2c_inth.c create mode 100644 src/touch/touch.c diff --git a/include/gint/defs/util.h b/include/gint/defs/util.h index 680a265..6debcc1 100644 --- a/include/gint/defs/util.h +++ b/include/gint/defs/util.h @@ -47,4 +47,10 @@ using std::max; (b) = _tmp; \ }) +/* abs() - absolute value of a variable (be careful of neg overflow) */ +#define abs(a) ({ \ + GAUTOTYPE _a = (a); \ + (_a < 0) ? -(_a) : _a; \ +}) + #endif /* GINT_DEFS_UTIL */ diff --git a/include/gint/touch.h b/include/gint/touch.h new file mode 100644 index 0000000..eecfbb5 --- /dev/null +++ b/include/gint/touch.h @@ -0,0 +1,40 @@ +//--- +// gint:touch - touch-screen driver API +//--- +#ifndef GINT_TOUCH_H +#define GINT_TOUCH_H 1 + +#ifdef __cplusplus +extern "C" { +#endif + +/* touch_calib - tounch-screen calibration information */ +typedef struct { + int scan_freq_us; + int x_mul; + int x_div; + int y_mul; + int y_div; + int dual_debounce_frame; + int dual_sensi_entry; + int dual_sensi_leave; +} touch_calib; + +/* touch_calib_get() - get calibration information */ +extern int touch_calib_get(touch_calib *calib); + +/* touch_calib_set() - set calibration information */ +extern int touch_calib_set(touch_calib *calib); + +// low-level API + +#include + +/* touch_next_event() - get the next touchscreen event */ +extern key_event_t touch_next_event(void); + +#ifdef __cplusplus +} +#endif + +#endif /* GINT_TOUCH_H */ diff --git a/src/kernel/exch.c b/src/kernel/exch.c index f04bfb0..5f79567 100644 --- a/src/kernel/exch.c +++ b/src/kernel/exch.c @@ -123,6 +123,7 @@ GNORETURN static void gint_default_panic(GUNUSED uint32_t code) if(code == 0x1080) name = "Stack overflow during world switch"; if(code == 0x10a0) name = "UBC break in register bank 1 code"; if(code == 0x10c0) name = "Missing syscall for this OS version"; + if(code == 0x10e0) name = "I2C (touch-screen) error"; dprint(6, 25, "%03x %s", code, name); @@ -173,6 +174,11 @@ GNORETURN static void gint_default_panic(GUNUSED uint32_t code) dprint(6, 160, "Opcodes: %04x %04x [%04x] %04x", opcodes[-2], opcodes[-1], opcodes[0], opcodes[1]); } + /* I2C exception error */ + if (code == 0x10e0) + { + //todo + } #endif _WEAK_dupdate(); diff --git a/src/touch/adconv.c b/src/touch/adconv.c new file mode 100644 index 0000000..6a9b283 --- /dev/null +++ b/src/touch/adconv.c @@ -0,0 +1,114 @@ +//--- +// gint:touchscreen:adconv - 0x84 register data conversion +//--- +#include + +#include "./adconv.h" +#include "./i2c.h" + +//--- +// Internals +//--- + +/* _adconv_check_dual() - check if it is dual or single */ +static int _adconv_check_dual( + struct _touch_adconv *adconv, + struct _touch_adraw *adraw +) { + static int _prev_type = 0; + int x2; + int y2; + int z2; + int val; + + x2 = ((int)((uint)(adraw->x2) >> 6)) + ((adraw->x2 & 1) * -0x400); + y2 = ((int)((uint)(adraw->y2) >> 6)) + ((adraw->y2 & 1) * -0x400); + z2 = ((int)((uint)(adraw->z2) >> 4)) + ((adraw->z2 & 1) * -0x1000); + adconv->x2 = x2; + adconv->y2 = y2; + adconv->z2 = z2; + val = (_prev_type == 0) ? 0x18 : 0x12; + _prev_type = ((abs(z2) >= val) || (max(abs(x2),abs(y2)) >= val)); + return _prev_type; +} + +//--- +// Public +//--- + +/* touchscreen_adconv_get_raw() - read 0x84 register using I2C */ +int touchscreen_adconv_get_raw(struct _touch_adraw *adraw) +{ + volatile uint8_t *IO_PRDR = (void*)0xa405013c; + + if (((*IO_PRDR) & 0x20) != 0) + { + adraw->x1 = 0; + adraw->y1 = 0; + adraw->z1 = 0; + adraw->x2 = 0; + adraw->y2 = 0; + adraw->z2 = 0; + adraw->gh = 0; + return 0; + } + i2c_reg_read(0x84, adraw, 16); + return 1; +} + +/* touchscreen_adconv_convert() - perform the raw conversion */ +int touchscreen_adconv_get_conv( + struct _touch_adconv *adconv, + struct _touch_adraw *adraw, + int type +) { + if (type == 0) + return 0; + adconv->x1 = adraw->x1 >> 4; + adconv->y1 = adraw->y1 >> 4; + adconv->z1 = adraw->z1 >> 4; + adconv->gh = adraw->gh >> 4; + adconv->dm = adraw->dm >> 6; + if (_adconv_check_dual(adconv, adraw) == 0) + return 1; + return 2; +} + +/* touchscreen_adconv_get_dots() - generate dots information */ +int touchscreen_adconv_get_dots( + struct _touch_addots *dots, + struct _touch_adconv *adconv, + int type +) { + switch (type) + { + case 0: + dots->type = TS_DOTS_TYPE_OFF; + dots->x1 = 0; + dots->y1 = 0; + dots->z1 = 0; + dots->x2 = 0; + dots->y2 = 0; + dots->z2 = 0; + break; + case 1: + dots->type = TS_DOTS_TYPE_SINGLE; + dots->x1 = ((adconv->x1 - 0x1b8) * 0x100) / 0x9e5; + dots->y1 = ((adconv->y1 - 0x104) * 0x100) / 0x660; + dots->z1 = adconv->z1; + dots->x2 = 0; + dots->y2 = 0; + dots->z2 = 0; + break; + case 2: + dots->type = TS_DOTS_TYPE_DUAL; + dots->x1 = ((adconv->x1 - 0x1b8) * 0x100) / 0x9e5; + dots->y1 = ((adconv->y1 - 0x104) * 0x100) / 0x660; + dots->z1 = adconv->z1; + dots->x2 = ((adconv->x2 - 0x1b8) * 0x100) / 0x9e5; + dots->y2 = ((adconv->y2 - 0x104) * 0x100) / 0x660; + dots->z2 = adconv->z2; + break; + } + return type; +} diff --git a/src/touch/adconv.h b/src/touch/adconv.h new file mode 100644 index 0000000..50e04a2 --- /dev/null +++ b/src/touch/adconv.h @@ -0,0 +1,77 @@ +#ifndef GINT_TOUCH_ADCONV_H +#define GINT_TOUCH_ADCONV_H 1 + +#include + +//--- +// Internals +//--- + +/* _touch_adraw - raw 0x84 register information */ +struct _touch_adraw +{ + uint16_t x1; + uint16_t y1; + uint16_t z1; + uint16_t gh; + uint16_t x2; + uint16_t y2; + uint16_t z2; + uint16_t dm; +}; + +/* _touch_adconv - post-conversion raw 0x84 register information */ +struct _touch_adconv +{ + int x1; + int y1; + int x2; + int y2; + int z1; + int z2; + uint16_t gh; + uint16_t dm; +}; + +/* _touch_addots_type - type of dots */ +enum _touch_addots_type +{ + TS_DOTS_TYPE_OFF = 0, + TS_DOTS_TYPE_SINGLE = 1, + TS_DOTS_TYPE_DUAL = 2, +}; + +/* _touch_addots - touchscreen dots information */ +struct _touch_addots +{ + enum _touch_addots_type type; + int x1; + int y1; + int z1; + int x2; + int y2; + int z2; +}; + +//--- +// Public +//--- + +/* touchscreen_adconv_get_raw() - read 0x84 register using I2C */ +extern int touchscreen_adconv_get_raw(struct _touch_adraw *adraw); + +/* touchscreen_adconv_get_conv() - perform the raw conversion */ +extern int touchscreen_adconv_get_conv( + struct _touch_adconv *adconv, + struct _touch_adraw *adraw, + int type +); + +/* touchscreen_adconv_get_dots() - generate dots information */ +extern int touchscreen_adconv_get_dots( + struct _touch_addots *dots, + struct _touch_adconv *adconv, + int type +); + +#endif /* GINT_TOUCH_ADCONV_H */ diff --git a/src/touch/i2c.c b/src/touch/i2c.c new file mode 100644 index 0000000..8d3b9a6 --- /dev/null +++ b/src/touch/i2c.c @@ -0,0 +1,186 @@ +//--- +// gint:touchscreen:i2c - I2C driver +//--- +#include +#include +#include +#include +#include + +#include "./i2c.h" + +//--- +// Internals +//--- + +/* __i2c_request - internal I2C request information */ +volatile struct i2c_request_info __i2c_request; + +/* i2c_hw_enable() - enable the I2C peripheral and configure the clock + * + * notes + * Clock configuration are the same used by Casio. Investigation must be + * made to ensure overclock/underclock validity */ +static void _i2c_hw_enable(void) +{ + SH7305_I2C.ICCR.ICE = 1; + while(true) { + if (SH7305_I2C.ICSR.BUSY == 0) + break; + } + SH7305_I2C.ICCL = 0x29; + SH7305_I2C.ICCH = 0x22; +} + +/* i2c_hw_start_operation() - start I2C operation + * + * notes + * Enable only data transmit and arbitration lost interrupt then perform + * a start condition (0x94) */ +static void _i2c_hw_start_operation(void) +{ + SH7305_I2C.ICIC.ALE = 1; + SH7305_I2C.ICIC.TACKE = 1; + SH7305_I2C.ICIC.WAITE = 0; + SH7305_I2C.ICIC.DTEE = 1; + + __auto_type iccr = SH7305_I2C.ICCR; + iccr.ICE = 1; + iccr.RACK = 0; + iccr.TRS = 1; + iccr.BBSY = 1; + iccr.SCP = 0; + SH7305_I2C.ICCR = iccr; +} + +/* i2c_request_await() - await async operation */ +static int _i2c_request_await(void) +{ + int status; + + while (true) + { + cpu_atomic_start(); + status = (__i2c_request.status == I2C_REQ_STATUS_FINISHED); + cpu_atomic_end(); + if (status) + break; + __asm__ volatile ("sleep"); + } + return 0; +} + +//--- +// Public +//--- + +/* i2c_power_on() - power on the I2C peripheral and install handlers */ +void i2c_power_on(void) +{ + volatile uint16_t *INTC_IPRH = (void*)0xa408001c; + volatile uint8_t *INTC_IMCR7 = (void*)0xa40800dc; + volatile uint8_t *INTC_IMR7 = (void*)0xa408009c; + volatile uint16_t *PFC_PJCR = (void*)0xa4050110; + volatile uint32_t *MSTPCR2 = (void*)0xa4150038; + extern void i2c_inth_tack(void); + extern void i2c_inth_wait(void); + extern void i2c_inth_trans(void); + extern void i2c_inth_al(void); + + // force-mask I2C interrupts + *INTC_IPRH &= 0xfff0; + *INTC_IMR7 |= 0xf0; + + // install I2C handler + intc_handler_function(0xe00, GINT_CALL(i2c_inth_al)); + intc_handler_function(0xe20, GINT_CALL(i2c_inth_tack)); + intc_handler_function(0xe40, GINT_CALL(i2c_inth_wait)); + intc_handler_function(0xe60, GINT_CALL(i2c_inth_trans)); + + // power-on I2C module + *MSTPCR2 &= 0xfffff7ff; + + // configure I2C PIN + *PFC_PJCR &= 0xf0ff; + + // configure I2C module + SH7305_I2C.ICCR.byte &= 0x7f; + SH7305_I2C.ICSR.byte = 0x00; + SH7305_I2C.ICIC.byte = 0x00; + + // Enable interrupt + *INTC_IPRH |= 1; + *INTC_IMCR7 |= 0xf0; +} + +// user API + +/* i2c_reg_select() - select a register */ +int i2c_reg_select(int reg) +{ + cpu_atomic_start(); + _i2c_hw_enable(); + + __i2c_request.mode = I2C_REQ_MODE_REG_SELECT; + __i2c_request.target_register = reg; + __i2c_request.buffer = NULL; + __i2c_request.buffer_size = 0; + __i2c_request.buffer_size_remaning = 0; + __i2c_request.buffer_cursor = 0; + __i2c_request.status = I2C_REQ_STATUS_START; + __i2c_request.state = I2C_REQ_STATE_START; + + cpu_atomic_end(); + _i2c_hw_start_operation(); + return _i2c_request_await(); +} + +/* i2c_reg_read() - register read operation */ +int i2c_reg_read(int reg, void *buffer, size_t size) +{ + if (buffer == NULL) + return -1; + if (size == 0) + return -2; + + cpu_atomic_start(); + _i2c_hw_enable(); + + __i2c_request.mode = I2C_REQ_MODE_REG_READ; + __i2c_request.target_register = reg; + __i2c_request.buffer = buffer; + __i2c_request.buffer_size = size; + __i2c_request.buffer_size_remaning = size; + __i2c_request.buffer_cursor = 0; + __i2c_request.status = I2C_REQ_STATUS_START; + __i2c_request.state = I2C_REQ_STATE_START; + + cpu_atomic_end(); + _i2c_hw_start_operation(); + return _i2c_request_await(); +} + +/* i2c_read_stream() - "stream" read operation (skip reg selection) */ +int i2c_read_stream(void *buffer, size_t size) +{ + if (buffer == NULL) + return -1; + if (size == 0) + return -2; + + cpu_atomic_start(); + _i2c_hw_enable(); + + __i2c_request.mode = I2C_REQ_MODE_READ_STREAM; + __i2c_request.target_register = 0x00; + __i2c_request.buffer = buffer; + __i2c_request.buffer_size = size; + __i2c_request.buffer_size_remaning = size; + __i2c_request.buffer_cursor = 0; + __i2c_request.status = I2C_REQ_STATUS_START; + __i2c_request.state = I2C_REQ_STATE_START; + + cpu_atomic_end(); + _i2c_hw_start_operation(); + return _i2c_request_await(); +} diff --git a/src/touch/i2c.h b/src/touch/i2c.h new file mode 100644 index 0000000..a4d3e4a --- /dev/null +++ b/src/touch/i2c.h @@ -0,0 +1,124 @@ +#ifndef GINT_TOUCH_I2C_H +#define GINT_TOUCH_I2C_H 1 + +#include +#include + +//--- +// User API +//--- + +/* i2c_power_on() - power on the I2C peripheral and install handlers */ +extern void i2c_power_on(void); + +/* i2c_reg_select() - select a register */ +extern int i2c_reg_select(int reg); + +/* i2c_reg_read() - register read operation */ +extern int i2c_reg_read(int reg, void *buffer, size_t size); + +/* i2c_read_stream() - "stream" read operation (skip reg selection) */ +extern int i2c_read_stream(void *buffer, size_t size); + +//--- +// Internals +//--- + +/* i2c_request_mode - enumerate request mode */ +enum i2c_request_mode { + I2C_REQ_MODE_REG_READ = 0, + I2C_REQ_MODE_REG_WRITE = 1, + I2C_REQ_MODE_REG_SELECT = 2, + I2C_REQ_MODE_READ_STREAM = 3, +}; + +/* i2c_request_status - request status */ +enum i2c_request_status { + I2C_REQ_STATUS_START = 0, + I2C_REQ_STATUS_FINISHED = 1, +}; + +enum i2c_request_state { + I2C_REQ_STATE_START = 0, + I2C_REQ_STATE_REG_SELECT = 10, + I2C_REQ_STATE_REG_SELECT_WAIT = 11, + I2C_REQ_STATE_SWITCH_TO_RECEIVE = 20, + I2C_REQ_STATE_SWITCH_TO_RECEIVE_WAIT = 21, + I2C_REQ_STATE_READ = 30, + I2C_REQ_STATE_READ_LAST = 31, + I2C_REQ_STATE_ZOMBIE = 667, + I2C_REQ_STATE_DEAD = 2617, +}; + +/* struct i2c_request_info - internal I2C request information */ +struct i2c_request_info { + enum i2c_request_mode mode; + int target_register; + uint8_t *buffer; + size_t buffer_size; + size_t buffer_size_remaning; + size_t buffer_cursor; + enum i2c_request_status status; + int state; +}; + +/* __i2c_request - internal current I2C request */ +extern volatile struct i2c_request_info __i2c_request; + +//--- +// Hardware information +//--- + +typedef struct { + // read/write register + uint8_t ICDR; + pad(3); + + // control register + byte_union(ICCR, + uint8_t ICE :1; + uint8_t RACK :1; + uint8_t :1; + uint8_t TRS :1; + uint8_t :1; + uint8_t BBSY :1; + uint8_t :1; + uint8_t SCP :1; + ); + pad(3); + + // status register + byte_union(ICSR, + uint8_t SCLM :1; + uint8_t SDAM :1; + uint8_t :1; + uint8_t BUSY :1; + uint8_t AL :1; + uint8_t TACK :1; + uint8_t WAIT :1; + uint8_t DTE :1; + ); + pad(3); + + // interrupt control register + byte_union(ICIC, + uint8_t :1; + uint8_t :1; + uint8_t :1; + uint8_t :1; + uint8_t ALE :1; + uint8_t TACKE :1; + uint8_t WAITE :1; + uint8_t DTEE :1; + ); + pad(3); + + // clock control registers + uint8_t ICCL; + pad(3); + uint8_t ICCH; +} GPACKED(1) sh7305_i2c_t; + +#define SH7305_I2C (*((volatile sh7305_i2c_t *)0xa4470000)) + +#endif /* CLOVE_TOUCH_I2C_H */ diff --git a/src/touch/i2c_inth.c b/src/touch/i2c_inth.c new file mode 100644 index 0000000..844b291 --- /dev/null +++ b/src/touch/i2c_inth.c @@ -0,0 +1,166 @@ +//--- +// gint:touchscreen:i2c_inth - I2C interrupt handlers +//--- +#include + +#include "./i2c.h" + +//--- +// Private +//--- + +/* _i2c_inth_read() - generic read byte operation */ +static int _i2c_inth_io_operation(bool read) +{ + if (read) { + __i2c_request.buffer[ + __i2c_request.buffer_cursor + ] = SH7305_I2C.ICDR; + } else { + SH7305_I2C.ICDR = __i2c_request.buffer[ + __i2c_request.buffer_cursor + ]; + } + __i2c_request.buffer_cursor += 1; + __i2c_request.buffer_size_remaning -= 1; + return __i2c_request.buffer_size_remaning; +} + +/* _i2c_inth_stop() - stop I2C module (with force wait) */ +static void _i2c_inth_stop(void) +{ + while(true) { + if (SH7305_I2C.ICSR.BUSY == 0) + break; + SH7305_I2C.ICDR = SH7305_I2C.ICDR; + } + SH7305_I2C.ICIC.byte = 0x00; + SH7305_I2C.ICSR.byte = 0x00; + SH7305_I2C.ICCR.byte = 0x00; + __i2c_request.state = I2C_REQ_STATE_DEAD; + __i2c_request.status = I2C_REQ_STATUS_FINISHED; +} + +//--- +// Public +//--- + +/* i2c_inth_trans() - TRANS interrupt handler */ +void i2c_inth_trans(void) +{ + switch (__i2c_request.state) + { + // start byte (writing or reading) and select the proper mode + // note that the "read stream" just change the internal state + // because the starting byte (0x81) is force-sended over there + case I2C_REQ_STATE_START: + SH7305_I2C.ICIC.DTEE = 0; + SH7305_I2C.ICIC.WAITE = 1; + if (__i2c_request.mode == I2C_REQ_MODE_READ_STREAM) { + SH7305_I2C.ICDR = 0x81; + __i2c_request.state = I2C_REQ_STATE_SWITCH_TO_RECEIVE_WAIT; + } else { + SH7305_I2C.ICDR = 0x80; + __i2c_request.state = I2C_REQ_STATE_REG_SELECT; + } + break; + + // switch the I2C module to receive mode and setup the next state + // use WAIT interrupt to really switch the I2C to receive + case I2C_REQ_STATE_SWITCH_TO_RECEIVE: + SH7305_I2C.ICIC.DTEE = 0; + SH7305_I2C.ICDR = 0x81; + __i2c_request.state = I2C_REQ_STATE_SWITCH_TO_RECEIVE_WAIT; + break; + + // for the read operation, the only things to do is the ACK at the + // end of the operation + case I2C_REQ_STATE_READ: + SH7305_I2C.ICIC.DTEE = 0; + if (_i2c_inth_io_operation(true) == 0) + _i2c_inth_stop(); + break; + + // error, unsupported sequences display tests/debug information + default: + gint_panic(0x10e0); + } +} + +/* i2c_inth_wait() - WAIT interrupt handler */ +void i2c_inth_wait(void) +{ + switch (__i2c_request.state) + { + // indicate which register to perform the operation + // we will wait the next WAIT interrupt to ensure that the data has + // been sent + case I2C_REQ_STATE_REG_SELECT: + SH7305_I2C.ICDR = __i2c_request.target_register; + __i2c_request.state = I2C_REQ_STATE_REG_SELECT_WAIT; + SH7305_I2C.ICSR.WAIT = 0; + break; + + // the selected register is confirmed, stop or restart + // note that if the "select register" request is performed (or if + // no buffer is provided) then we send the stop condition (0x90) + // and force-stop the module. + // Otherwise, we trigger a new request (0x94) manually to (after + // the restart interruption) switch from "reading" or "writing" + // mode + case I2C_REQ_STATE_REG_SELECT_WAIT: + if ( + __i2c_request.mode == I2C_REQ_MODE_REG_SELECT + || __i2c_request.buffer_size_remaning == 0 + || __i2c_request.buffer == NULL + ) { + SH7305_I2C.ICCR.byte = 0x90; + SH7305_I2C.ICSR.WAIT = 0; + _i2c_inth_stop(); + } else { + SH7305_I2C.ICCR.byte = 0x94; + __i2c_request.state = I2C_REQ_STATE_SWITCH_TO_RECEIVE; + SH7305_I2C.ICIC.DTEE = 1; + SH7305_I2C.ICSR.WAIT = 0; + } + break; + + // switch to receive mode + // waiting the next WAIT interrupt to start reading + case I2C_REQ_STATE_SWITCH_TO_RECEIVE_WAIT: + SH7305_I2C.ICCR.byte = 0x81; + __i2c_request.state = I2C_REQ_STATE_READ; + SH7305_I2C.ICSR.WAIT = 0; + break; + + // read operation + // can either be WAIT or DTE interruption, a special case is + // performed to avoid DTE interruption + case I2C_REQ_STATE_READ: + if (__i2c_request.buffer_size_remaning > 1) { + if (SH7305_I2C.ICSR.DTE != 0) + _i2c_inth_io_operation(true); + } + if (__i2c_request.buffer_size_remaning == 1) + SH7305_I2C.ICCR.byte = 0xc0; + SH7305_I2C.ICIC.DTEE = 1; + SH7305_I2C.ICSR.WAIT = 0; + break; + + // error cases, unable to handle the WAIT interrupt + default: + gint_panic(0x10e0); + } +} + +/* i2c_inth_tack() - TACK interrupt handler */ +void i2c_inth_tack(void) +{ + gint_panic(0x10e0); +} + +/* i2c_inth_al() - AL interrupt handler */ +void i2c_inth_al(void) +{ + gint_panic(0x10e0); +} diff --git a/src/touch/touch.c b/src/touch/touch.c new file mode 100644 index 0000000..f0bce86 --- /dev/null +++ b/src/touch/touch.c @@ -0,0 +1,53 @@ +//--- +// gint:touch - touchscreen driver +//---- +#include + +#include + +//--- +// Internals +//--- + +touch_calib __ts_calib; + +//--- +// Public +//--- + +// user-API + +/* touch_calib_get() - get calibration information */ +int touch_calib_get(touch_calib *calib) +{ + if (calib == NULL) + return -1; + memcpy(calib, &__ts_calib, sizeof(touch_calib)); + return 0; +} + +/* touch_calib_set() - set calibration information */ +int touch_calib_set(touch_calib *calib) +{ + if (calib == NULL) + return -1; + memcpy(&__ts_calib, calib, sizeof(touch_calib)); + return 0; +} + +// low-level API + +/* touch_next_event() - get the next touchscreen event */ +key_event_t touch_next_event(void) +{ +#if 0 + struct _ts_adraw adraw; + struct _ts_adconv adconv; + int type; + + type = touchscreen_adconv_get_raw(&adraw); + type = touchscreen_adconv_get_conv(&adconv, &adraw, type); + type = touchscreen_adconv_get_dots(dots, &adconv, type); + return type; +#endif +} From df1cf45e98857e8627e9e049e58cc60679465e0d Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Tue, 18 Mar 2025 11:42:22 +0100 Subject: [PATCH 2/5] touch: add build support + proper driver/world-switch support --- CMakeLists.txt | 5 ++ include/gint/defs/util.h | 6 -- include/gint/drivers/states.h | 10 +++ include/gint/mpu/intc.h | 2 +- include/gint/touch.h | 1 - src/touch/adconv.c | 3 + src/touch/i2c.c | 123 ++++++++++++++++++++++------------ src/touch/i2c.h | 27 ++++++-- src/touch/touch.c | 89 ++++++++++++++++++++++++ 9 files changed, 213 insertions(+), 53 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index fd0176a..8a1f9b9 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -253,6 +253,11 @@ set(SOURCES src/usb/write4.S # Video driver interface src/video/video.c + # Touch-screen driver + src/touch/i2c.c + src/touch/i2c_inth.c + src/touch/adconv.c + src/touch/touch.c ) set(ASSETS_FX src/font5x7.png src/gdb/icons-i1msb.png) diff --git a/include/gint/defs/util.h b/include/gint/defs/util.h index 6debcc1..680a265 100644 --- a/include/gint/defs/util.h +++ b/include/gint/defs/util.h @@ -47,10 +47,4 @@ using std::max; (b) = _tmp; \ }) -/* abs() - absolute value of a variable (be careful of neg overflow) */ -#define abs(a) ({ \ - GAUTOTYPE _a = (a); \ - (_a < 0) ? -(_a) : _a; \ -}) - #endif /* GINT_DEFS_UTIL */ diff --git a/include/gint/drivers/states.h b/include/gint/drivers/states.h index d18e36a..84ec3fd 100644 --- a/include/gint/drivers/states.h +++ b/include/gint/drivers/states.h @@ -109,6 +109,16 @@ typedef struct { } usb_state_t; +/* Touch-screen modules */ +typedef struct { + uint16_t PRCR; + uint16_t PJCR; + uint8_t ICCR; + uint8_t ICIC; + uint8_t ICCL; + uint8_t ICCH; +} touch_state_t; + #ifdef __cplusplus } #endif diff --git a/include/gint/mpu/intc.h b/include/gint/mpu/intc.h index 7ccb06a..be4e0ca 100644 --- a/include/gint/mpu/intc.h +++ b/include/gint/mpu/intc.h @@ -192,7 +192,7 @@ typedef volatile struct uint16_t _MSIOF0:4; /* SH7724: Sync SCIF channel 0 */ uint16_t _MSIOF1:4; /* SH7724: Sync SCIF channel 1 */ uint16_t _1 :4; /* Unknown (TODO) */ - uint16_t _2 :4; /* Unknown (TODO) */ + uint16_t I2C:4; /* I2C */ ); pad(2); diff --git a/include/gint/touch.h b/include/gint/touch.h index eecfbb5..1ed5d21 100644 --- a/include/gint/touch.h +++ b/include/gint/touch.h @@ -10,7 +10,6 @@ extern "C" { /* touch_calib - tounch-screen calibration information */ typedef struct { - int scan_freq_us; int x_mul; int x_div; int y_mul; diff --git a/src/touch/adconv.c b/src/touch/adconv.c index 6a9b283..08585a8 100644 --- a/src/touch/adconv.c +++ b/src/touch/adconv.c @@ -1,6 +1,8 @@ //--- // gint:touchscreen:adconv - 0x84 register data conversion //--- +#include + #include #include "./adconv.h" @@ -11,6 +13,7 @@ //--- /* _adconv_check_dual() - check if it is dual or single */ +// fixme : atomic static int _adconv_check_dual( struct _touch_adconv *adconv, struct _touch_adraw *adraw diff --git a/src/touch/i2c.c b/src/touch/i2c.c index 8d3b9a6..5728e77 100644 --- a/src/touch/i2c.c +++ b/src/touch/i2c.c @@ -1,6 +1,8 @@ //--- // gint:touchscreen:i2c - I2C driver //--- +#include + #include #include #include @@ -74,47 +76,6 @@ static int _i2c_request_await(void) // Public //--- -/* i2c_power_on() - power on the I2C peripheral and install handlers */ -void i2c_power_on(void) -{ - volatile uint16_t *INTC_IPRH = (void*)0xa408001c; - volatile uint8_t *INTC_IMCR7 = (void*)0xa40800dc; - volatile uint8_t *INTC_IMR7 = (void*)0xa408009c; - volatile uint16_t *PFC_PJCR = (void*)0xa4050110; - volatile uint32_t *MSTPCR2 = (void*)0xa4150038; - extern void i2c_inth_tack(void); - extern void i2c_inth_wait(void); - extern void i2c_inth_trans(void); - extern void i2c_inth_al(void); - - // force-mask I2C interrupts - *INTC_IPRH &= 0xfff0; - *INTC_IMR7 |= 0xf0; - - // install I2C handler - intc_handler_function(0xe00, GINT_CALL(i2c_inth_al)); - intc_handler_function(0xe20, GINT_CALL(i2c_inth_tack)); - intc_handler_function(0xe40, GINT_CALL(i2c_inth_wait)); - intc_handler_function(0xe60, GINT_CALL(i2c_inth_trans)); - - // power-on I2C module - *MSTPCR2 &= 0xfffff7ff; - - // configure I2C PIN - *PFC_PJCR &= 0xf0ff; - - // configure I2C module - SH7305_I2C.ICCR.byte &= 0x7f; - SH7305_I2C.ICSR.byte = 0x00; - SH7305_I2C.ICIC.byte = 0x00; - - // Enable interrupt - *INTC_IPRH |= 1; - *INTC_IMCR7 |= 0xf0; -} - -// user API - /* i2c_reg_select() - select a register */ int i2c_reg_select(int reg) { @@ -184,3 +145,83 @@ int i2c_read_stream(void *buffer, size_t size) _i2c_hw_start_operation(); return _i2c_request_await(); } + +//--- +// Driver and state management +//--- + +#include +#include +#include + +/* i2c_hpowered() - check if the module is powered */ +bool i2c_hpowered(void) +{ + return (SH7305_POWER.MSTPCR2.I2C == 0); +} + +/* i2c_hpoweron() - power on the module */ +void i2c_hpoweron(void) +{ + SH7305_POWER.MSTPCR2.I2C = 0; + SH7305_I2C.ICCR.ICE = 0; + SH7305_I2C.ICDR = 0; +} + +/* i2c_hpoweroff() - power on the module */ +void i2c_hpoweroff(void) +{ + SH7305_POWER.MSTPCR2.I2C = 1; +} + +/* i2c_hsave() - save hardware information */ +void i2c_hsave(touch_state_t *state) +{ + state->PJCR = SH7305_PFC.PJCR.word; + state->ICCR = SH7305_I2C.ICCR.byte; + state->ICIC = SH7305_I2C.ICIC.byte; + state->ICCL = SH7305_I2C.ICCL; + state->ICCH = SH7305_I2C.ICCH; +} + +/* i2c_hrestore() - restore hardware information */ +void i2c_hrestore(touch_state_t *state) +{ + SH7305_I2C.ICCH = state->ICCH; + SH7305_I2C.ICCL = state->ICCL; + SH7305_I2C.ICIC.byte = state->ICIC; + SH7305_I2C.ICCR.byte = state->ICCR; + SH7305_PFC.PJCR.word = state->PJCR; +} + +/* i2c_configure() - configure and install interrupt handlers */ +void i2c_configure(void) +{ + extern void i2c_inth_tack(void); + extern void i2c_inth_wait(void); + extern void i2c_inth_trans(void); + extern void i2c_inth_al(void); + + // install I2C handler + intc_handler_function(0xe00, GINT_CALL(i2c_inth_al)); + intc_handler_function(0xe20, GINT_CALL(i2c_inth_tack)); + intc_handler_function(0xe40, GINT_CALL(i2c_inth_wait)); + intc_handler_function(0xe60, GINT_CALL(i2c_inth_trans)); + + // configure I2C PIN + SH7305_PFC.PJCR.P5MD = 0b00; + SH7305_PFC.PJCR.P4MD = 0b00; + + // configure I2C module + SH7305_I2C.ICCR.byte &= 0x7f; + SH7305_I2C.ICSR.byte = 0x00; + SH7305_I2C.ICIC.byte = 0x00; + + // init high-level driver information + memset((void*)&__i2c_request, 0x00, sizeof(struct i2c_request_info)); + __i2c_request.status = I2C_REQ_STATUS_FINISHED; + + // Enable interrupt + SH7305_INTC._->IPRH.I2C = 1; + SH7305_INTC.MSKCLR->IMR7 |= 0xf0; +} diff --git a/src/touch/i2c.h b/src/touch/i2c.h index a4d3e4a..8966ca8 100644 --- a/src/touch/i2c.h +++ b/src/touch/i2c.h @@ -8,9 +8,6 @@ // User API //--- -/* i2c_power_on() - power on the I2C peripheral and install handlers */ -extern void i2c_power_on(void); - /* i2c_reg_select() - select a register */ extern int i2c_reg_select(int reg); @@ -121,4 +118,26 @@ typedef struct { #define SH7305_I2C (*((volatile sh7305_i2c_t *)0xa4470000)) -#endif /* CLOVE_TOUCH_I2C_H */ +// state and world-switch + +#include + +/* i2c_configure() - driver/hardware configuration */ +extern void i2c_configure(void); + +/* i2c_hsave() - save hardware information */ +extern void i2c_hsave(touch_state_t *states); + +/* i2c_hrestore() - restore hardware information */ +extern void i2c_hrestore(touch_state_t *states); + +/* i2c_hpowered() - check if the module is powered */ +extern bool i2c_hpowered(void); + +/* i2c_hpoweron() - power on the module */ +extern void i2c_hpoweron(void); + +/* i2c_hpoweroff() - power off the module */ +extern void i2c_hpoweroff(void); + +#endif /* GINT_TOUCH_I2C_H */ diff --git a/src/touch/touch.c b/src/touch/touch.c index f0bce86..3b7d8b1 100644 --- a/src/touch/touch.c +++ b/src/touch/touch.c @@ -3,8 +3,13 @@ //---- #include +#include +#include #include +#include "./i2c.h" +#include "./adconv.h" + //--- // Internals //--- @@ -35,11 +40,27 @@ int touch_calib_set(touch_calib *calib) return 0; } +/* touch_get_dots() - get dots information */ +int touch_get_dots(struct _touch_addots *dots) +{ + struct _touch_adraw adraw; + struct _touch_adconv adconv; + int type; + + type = touchscreen_adconv_get_raw(&adraw); + type = touchscreen_adconv_get_conv(&adconv, &adraw, type); + type = touchscreen_adconv_get_dots(dots, &adconv, type); + return type; +} + // low-level API /* touch_next_event() - get the next touchscreen event */ key_event_t touch_next_event(void) { + key_event_t evt = { 0 }; + evt.type = KEYEV_NONE; + return evt; #if 0 struct _ts_adraw adraw; struct _ts_adconv adconv; @@ -51,3 +72,71 @@ key_event_t touch_next_event(void) return type; #endif } + +//--- +// Driver and state +//--- + +/* _touch_configure() - configure touch-screen */ +static void _touch_configure(void) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + i2c_configure(); + *(IO_PRCR) = (*(IO_PRCR) & 0xf3ff) | 0xc00; + __ts_calib.x_mul = 0x20b; + __ts_calib.x_div = 0x9b6; + __ts_calib.y_mul = 0x0f4; + __ts_calib.y_div = 0x66f; + __ts_calib.dual_debounce_frame = 0; + __ts_calib.dual_sensi_entry = 0x18; + __ts_calib.dual_sensi_leave = 0x24; +} + +/* _touch_hsave() - save hardware state */ +static void _touch_hsave(touch_state_t *state) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + i2c_hsave(state); + state->PRCR = *(IO_PRCR); +} + +/* _touch_hrestore() - restore hardware state */ +static void _touch_hrestore(touch_state_t *state) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + *(IO_PRCR) = state->PRCR; + i2c_hrestore(state); +} + +/* _touch_hpowered() - check if the module is powered */ +static bool _touch_hpowered(void) +{ + return i2c_hpowered(); +} + +/* _touch_hpoweron() - power on the module */ +static void _touch_hpoweron(void) +{ + i2c_hpoweron(); +} + +/* _touch_hpoweroff() - power off the module */ +static void _touch_hpoweroff(void) +{ + i2c_hpoweroff(); +} + +gint_driver_t drv_touch = { + .name = "TOUCH", + .configure = _touch_configure, + .hsave = (void *)_touch_hsave, + .hrestore = (void *)_touch_hrestore, + .hpowered = _touch_hpowered, + .hpoweron = _touch_hpoweron, + .hpoweroff = _touch_hpoweroff, + .state_size = sizeof(touch_state_t), +}; +GINT_DECLARE_DRIVER(16, drv_touch); From 24d932906b66d866cbf905d3308bc980b3af7e76 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 10:29:03 +0100 Subject: [PATCH 3/5] touch: add missing unbind/funbind driver primitive --- src/touch/i2c.c | 29 ++++++++++++++++++++++++++++- src/touch/i2c.h | 6 ++++++ src/touch/touch.c | 15 +++++++++++++++ 3 files changed, 49 insertions(+), 1 deletion(-) diff --git a/src/touch/i2c.c b/src/touch/i2c.c index 5728e77..f4f7d07 100644 --- a/src/touch/i2c.c +++ b/src/touch/i2c.c @@ -154,6 +154,7 @@ int i2c_read_stream(void *buffer, size_t size) #include #include + /* i2c_hpowered() - check if the module is powered */ bool i2c_hpowered(void) { @@ -166,6 +167,8 @@ void i2c_hpoweron(void) SH7305_POWER.MSTPCR2.I2C = 0; SH7305_I2C.ICCR.ICE = 0; SH7305_I2C.ICDR = 0; + SH7305_I2C.ICCL = 0x00; + SH7305_I2C.ICCH = 0x00; } /* i2c_hpoweroff() - power on the module */ @@ -213,7 +216,10 @@ void i2c_configure(void) SH7305_PFC.PJCR.P4MD = 0b00; // configure I2C module - SH7305_I2C.ICCR.byte &= 0x7f; + SH7305_I2C.ICCR.ICE = 0; + SH7305_I2C.ICDR = 0; + SH7305_I2C.ICCL = 0x00; + SH7305_I2C.ICCH = 0x00; SH7305_I2C.ICSR.byte = 0x00; SH7305_I2C.ICIC.byte = 0x00; @@ -225,3 +231,24 @@ void i2c_configure(void) SH7305_INTC._->IPRH.I2C = 1; SH7305_INTC.MSKCLR->IMR7 |= 0xf0; } + +/* i2c_unbin() - unbind from gint to casio */ +void i2c_unbin(void) +{ + _i2c_request_await(); +} + +/* i2c_funbin() - funbind from casio to gint */ +void i2c_funbin(void) +{ + if (i2c_hpowered() == false) + return; + // fixme : avoid force terminate + // We cannot easily know the state of Casio's driver since they use + // an state machine to work, and finding the current state require a + // lot of hardcoded OS-specific offsets information. So, for now, + // force terminate the transaction and disable the module to avoid + // any error + SH7305_I2C.ICCR.ICE = 0; + SH7305_I2C.ICDR = 0; +} diff --git a/src/touch/i2c.h b/src/touch/i2c.h index 8966ca8..19abdee 100644 --- a/src/touch/i2c.h +++ b/src/touch/i2c.h @@ -140,4 +140,10 @@ extern void i2c_hpoweron(void); /* i2c_hpoweroff() - power off the module */ extern void i2c_hpoweroff(void); +/* i2c_funbin() - funbind from casio to gint */ +extern void i2c_funbin(void); + +/* i2c_unbin() - unbind from gint to casio */ +extern void i2c_unbin(void); + #endif /* GINT_TOUCH_I2C_H */ diff --git a/src/touch/touch.c b/src/touch/touch.c index 3b7d8b1..22bf7ce 100644 --- a/src/touch/touch.c +++ b/src/touch/touch.c @@ -129,6 +129,19 @@ static void _touch_hpoweroff(void) i2c_hpoweroff(); } +/* _touch_unbin() - unbind from gint to casio */ +static void _touch_unbin(void) +{ + i2c_unbin(); +} + +/* _touch_funbin() - funbind from casio to gint */ +static void _touch_funbin(void) +{ + i2c_funbin(); +} + +/* drv_touch - touch-screen driver declaration */ gint_driver_t drv_touch = { .name = "TOUCH", .configure = _touch_configure, @@ -137,6 +150,8 @@ gint_driver_t drv_touch = { .hpowered = _touch_hpowered, .hpoweron = _touch_hpoweron, .hpoweroff = _touch_hpoweroff, + .unbind = _touch_unbin, + .funbind = _touch_funbin, .state_size = sizeof(touch_state_t), }; GINT_DECLARE_DRIVER(16, drv_touch); From 0ec45f92d36a1cbb665468c53df729835e792041 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 14:39:05 +0100 Subject: [PATCH 4/5] touch: support event generation + support calibration information --- CMakeLists.txt | 1 + include/gint/keyboard.h | 30 +++++--- src/touch/adconv.c | 46 +++++++++--- src/touch/driver.c | 107 +++++++++++++++++++++++++++ src/touch/driver.h | 17 +++++ src/touch/touch.c | 157 ++++++++++++---------------------------- 6 files changed, 225 insertions(+), 133 deletions(-) create mode 100644 src/touch/driver.c create mode 100644 src/touch/driver.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 8a1f9b9..b72d3d6 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -258,6 +258,7 @@ set(SOURCES src/touch/i2c_inth.c src/touch/adconv.c src/touch/touch.c + src/touch/driver.c ) set(ASSETS_FX src/font5x7.png src/gdb/icons-i1msb.png) diff --git a/include/gint/keyboard.h b/include/gint/keyboard.h index 06170d8..47a68a8 100644 --- a/include/gint/keyboard.h +++ b/include/gint/keyboard.h @@ -128,22 +128,32 @@ typedef struct // The following attributes will be union'd with touch info on the CP. - /* Matrix code: physical location of the key being it. */ - u8 row, col; - - /* Reserved for future use. */ - uint :16; + union { + struct { + /* Matrix code: physical location of the key being it. */ + u8 row; + u8 col; + }; + struct { + /* X/Y touch-screen coordinate */ + u16 x; + u16 y; + }; + }; } GPACKED(4) key_event_t; /* Keyboard event types, as in the [type] field of key_event_t */ enum { - KEYEV_NONE = 0, /* No event available (poll() only) */ - KEYEV_DOWN = 1, /* Key was pressed */ - KEYEV_UP = 2, /* Key was released */ - KEYEV_HOLD = 3, /* A key that was pressed has been held down */ - KEYEV_OSMENU = 4, /* We went to the main menu and back */ + KEYEV_NONE = 0, /* No event available (poll() only) */ + KEYEV_DOWN = 1, /* Key was pressed */ + KEYEV_UP = 2, /* Key was released */ + KEYEV_HOLD = 3, /* A key that was pressed has been held down */ + KEYEV_OSMENU = 4, /* We went to the main menu and back */ + KEYEV_TOUCH_PRESSED = 5, /* Touch was detected */ + KEYEV_TOUCH_DRAG = 6, /* A touch that was detected has been held down */ + KEYEV_TOUCH_RELEASE = 7, /* Touch was released */ }; /* Keyboard frequency analysis is a runtime setting since gint 2.4. This macro diff --git a/src/touch/adconv.c b/src/touch/adconv.c index 08585a8..44c1ed8 100644 --- a/src/touch/adconv.c +++ b/src/touch/adconv.c @@ -4,35 +4,45 @@ #include #include +#include #include "./adconv.h" #include "./i2c.h" +#include "./driver.h" //--- // Internals //--- +/* __touch_drv_info - global driver information */ +extern struct _touch_drv_info __touch_drv_info; + /* _adconv_check_dual() - check if it is dual or single */ -// fixme : atomic static int _adconv_check_dual( struct _touch_adconv *adconv, struct _touch_adraw *adraw ) { - static int _prev_type = 0; + extern struct _touch_drv_info __touch_drv_info; + bool is_dual; int x2; int y2; int z2; int val; + cpu_atomic_start(); x2 = ((int)((uint)(adraw->x2) >> 6)) + ((adraw->x2 & 1) * -0x400); y2 = ((int)((uint)(adraw->y2) >> 6)) + ((adraw->y2 & 1) * -0x400); z2 = ((int)((uint)(adraw->z2) >> 4)) + ((adraw->z2 & 1) * -0x1000); adconv->x2 = x2; adconv->y2 = y2; adconv->z2 = z2; - val = (_prev_type == 0) ? 0x18 : 0x12; - _prev_type = ((abs(z2) >= val) || (max(abs(x2),abs(y2)) >= val)); - return _prev_type; + val = __touch_drv_info.calibration.dual_sensi_entry; + if (__touch_drv_info.adinfo.prev_is_dual) + val = __touch_drv_info.calibration.dual_sensi_leave; + is_dual = ((abs(z2) >= val) || (max(abs(x2),abs(y2)) >= val)); + __touch_drv_info.adinfo.prev_is_dual = is_dual; + cpu_atomic_end(); + return is_dual; } //--- @@ -53,6 +63,9 @@ int touchscreen_adconv_get_raw(struct _touch_adraw *adraw) adraw->y2 = 0; adraw->z2 = 0; adraw->gh = 0; + cpu_atomic_start(); + __touch_drv_info.adinfo.prev_is_dual = false; + cpu_atomic_end(); return 0; } i2c_reg_read(0x84, adraw, 16); @@ -83,6 +96,16 @@ int touchscreen_adconv_get_dots( struct _touch_adconv *adconv, int type ) { + int x_div; + int x_mul; + int y_div; + int y_mul; + + cpu_atomic_start(); + x_div = __touch_drv_info.calibration.x_div; + x_mul = __touch_drv_info.calibration.x_mul; + y_div = __touch_drv_info.calibration.y_div; + y_mul = __touch_drv_info.calibration.y_mul; switch (type) { case 0: @@ -96,8 +119,8 @@ int touchscreen_adconv_get_dots( break; case 1: dots->type = TS_DOTS_TYPE_SINGLE; - dots->x1 = ((adconv->x1 - 0x1b8) * 0x100) / 0x9e5; - dots->y1 = ((adconv->y1 - 0x104) * 0x100) / 0x660; + dots->x1 = ((adconv->x1 - x_mul) * 0x100) / x_div; + dots->y1 = ((adconv->y1 - y_mul) * 0x100) / y_div; dots->z1 = adconv->z1; dots->x2 = 0; dots->y2 = 0; @@ -105,13 +128,14 @@ int touchscreen_adconv_get_dots( break; case 2: dots->type = TS_DOTS_TYPE_DUAL; - dots->x1 = ((adconv->x1 - 0x1b8) * 0x100) / 0x9e5; - dots->y1 = ((adconv->y1 - 0x104) * 0x100) / 0x660; + dots->x1 = ((adconv->x1 - x_mul) * 0x100) / x_div; + dots->y1 = ((adconv->y1 - y_mul) * 0x100) / y_div; dots->z1 = adconv->z1; - dots->x2 = ((adconv->x2 - 0x1b8) * 0x100) / 0x9e5; - dots->y2 = ((adconv->y2 - 0x104) * 0x100) / 0x660; + dots->x2 = ((adconv->x2 - x_mul) * 0x100) / x_div; + dots->y2 = ((adconv->y2 - y_mul) * 0x100) / y_div; dots->z2 = adconv->z2; break; } + cpu_atomic_end(); return type; } diff --git a/src/touch/driver.c b/src/touch/driver.c new file mode 100644 index 0000000..1ea21a8 --- /dev/null +++ b/src/touch/driver.c @@ -0,0 +1,107 @@ +//--- +// gint:touch:driver - touch-screen driver declaration +//--- +#include + +#include +#include + +#include "./driver.h" +#include "./i2c.h" + +//--- +// Internals +//--- + +/* __touch_drv_info - internal driver information */ +extern struct _touch_drv_info __touch_drv_info; + +/* _touch_configure() - configure touch-screen */ +static void _touch_configure(void) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + i2c_configure(); + *(IO_PRCR) = (*(IO_PRCR) & 0xf3ff) | 0xc00; + memset(&__touch_drv_info, 0x00, sizeof(struct _touch_drv_info)); + __touch_drv_info.prev_evt.type = KEYEV_NONE; + __touch_drv_info.prev_evt.x = 0xffff; + __touch_drv_info.prev_evt.y = 0xffff; + __touch_drv_info.calibration.x_mul = 0x20b; + __touch_drv_info.calibration.x_div = 0x9b6; + __touch_drv_info.calibration.y_mul = 0x0f4; + __touch_drv_info.calibration.y_div = 0x66f; + __touch_drv_info.calibration.dual_debounce_frame = 0; + __touch_drv_info.calibration.dual_sensi_entry = 0x18; + __touch_drv_info.calibration.dual_sensi_leave = 0x24; +} + +/* _touch_hsave() - save hardware state */ +static void _touch_hsave(touch_state_t *state) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + i2c_hsave(state); + state->PRCR = *(IO_PRCR); +} + +/* _touch_hrestore() - restore hardware state */ +static void _touch_hrestore(touch_state_t *state) +{ + volatile uint16_t *IO_PRCR = (void*)0xa405011c; + + *(IO_PRCR) = state->PRCR; + i2c_hrestore(state); +} + +/* _touch_hpowered() - check if the module is powered */ +static bool _touch_hpowered(void) +{ + return i2c_hpowered(); +} + +/* _touch_hpoweron() - power on the module */ +static void _touch_hpoweron(void) +{ + i2c_hpoweron(); +} + +/* _touch_hpoweroff() - power off the module */ +static void _touch_hpoweroff(void) +{ + i2c_hpoweroff(); +} + +/* _touch_unbin() - unbind from gint to casio */ +static void _touch_unbin(void) +{ + i2c_unbin(); +} + +/* _touch_funbin() - funbind from casio to gint */ +static void _touch_funbin(void) +{ + i2c_funbin(); +} + +//--- +// Public +//--- + +/* __touch_drv_info - internal driver information */ +struct _touch_drv_info __touch_drv_info; + +/* drv_touch - touch-screen driver declaration */ +gint_driver_t drv_touch = { + .name = "TOUCH", + .configure = _touch_configure, + .hsave = (void *)_touch_hsave, + .hrestore = (void *)_touch_hrestore, + .hpowered = _touch_hpowered, + .hpoweron = _touch_hpoweron, + .hpoweroff = _touch_hpoweroff, + .unbind = _touch_unbin, + .funbind = _touch_funbin, + .state_size = sizeof(touch_state_t), +}; +GINT_DECLARE_DRIVER(16, drv_touch); diff --git a/src/touch/driver.h b/src/touch/driver.h new file mode 100644 index 0000000..d6caa95 --- /dev/null +++ b/src/touch/driver.h @@ -0,0 +1,17 @@ +#ifndef GINT_TOUCH_DRIVER_H +#define GINT_TOUCH_DRIVER_H 1 + +#include +#include + +/* _touch_drv_info() - internal driver information */ +struct _touch_drv_info +{ + touch_calib calibration; + key_event_t prev_evt; + struct { + bool prev_is_dual; + } adinfo; +}; + +#endif /* GINT_TOUCH_DRIVER_H */ diff --git a/src/touch/touch.c b/src/touch/touch.c index 22bf7ce..bd2fa28 100644 --- a/src/touch/touch.c +++ b/src/touch/touch.c @@ -1,20 +1,21 @@ //--- -// gint:touch - touchscreen driver +// gint:touch - touchscreen driver (high-level) //---- #include -#include -#include #include +#include #include "./i2c.h" #include "./adconv.h" +#include "./driver.h" //--- // Internals //--- -touch_calib __ts_calib; +/* __touch_drv_info - internal driver information */ +extern struct _touch_drv_info __touch_drv_info; //--- // Public @@ -27,7 +28,7 @@ int touch_calib_get(touch_calib *calib) { if (calib == NULL) return -1; - memcpy(calib, &__ts_calib, sizeof(touch_calib)); + memcpy(calib, &__touch_drv_info.calibration, sizeof(touch_calib)); return 0; } @@ -36,122 +37,54 @@ int touch_calib_set(touch_calib *calib) { if (calib == NULL) return -1; - memcpy(&__ts_calib, calib, sizeof(touch_calib)); + memcpy(&__touch_drv_info.calibration, calib, sizeof(touch_calib)); return 0; } -/* touch_get_dots() - get dots information */ -int touch_get_dots(struct _touch_addots *dots) -{ - struct _touch_adraw adraw; - struct _touch_adconv adconv; - int type; - - type = touchscreen_adconv_get_raw(&adraw); - type = touchscreen_adconv_get_conv(&adconv, &adraw, type); - type = touchscreen_adconv_get_dots(dots, &adconv, type); - return type; -} - // low-level API /* touch_next_event() - get the next touchscreen event */ key_event_t touch_next_event(void) { - key_event_t evt = { 0 }; - evt.type = KEYEV_NONE; - return evt; -#if 0 - struct _ts_adraw adraw; - struct _ts_adconv adconv; + struct _touch_adconv adconv; + struct _touch_addots addots; + struct _touch_adraw adraw; + key_event_t evt; int type; + evt.type = KEYEV_TOUCH_RELEASE; type = touchscreen_adconv_get_raw(&adraw); - type = touchscreen_adconv_get_conv(&adconv, &adraw, type); - type = touchscreen_adconv_get_dots(dots, &adconv, type); - return type; -#endif + if (type != 0) + { + type = touchscreen_adconv_get_conv(&adconv, &adraw, type); + type = touchscreen_adconv_get_dots(&addots, &adconv, type); + if (type == 1) + { + evt.type = KEYEV_TOUCH_DRAG; + evt.x = addots.x1; + evt.y = addots.y1; + } + } + cpu_atomic_start(); + if (evt.type == KEYEV_TOUCH_DRAG) { + if ( + (__touch_drv_info.prev_evt.type == KEYEV_NONE) || + (__touch_drv_info.prev_evt.type == KEYEV_TOUCH_RELEASE) + ) { + evt.type = KEYEV_TOUCH_PRESSED; + } + } + if (evt.type == KEYEV_TOUCH_RELEASE) { + if ( + (__touch_drv_info.prev_evt.type == KEYEV_NONE) || + (__touch_drv_info.prev_evt.type == KEYEV_TOUCH_RELEASE) + ) { + evt.type = KEYEV_NONE; + } + } + __touch_drv_info.prev_evt.type = evt.type; + __touch_drv_info.prev_evt.x = evt.x; + __touch_drv_info.prev_evt.y = evt.y; + cpu_atomic_end(); + return evt; } - -//--- -// Driver and state -//--- - -/* _touch_configure() - configure touch-screen */ -static void _touch_configure(void) -{ - volatile uint16_t *IO_PRCR = (void*)0xa405011c; - - i2c_configure(); - *(IO_PRCR) = (*(IO_PRCR) & 0xf3ff) | 0xc00; - __ts_calib.x_mul = 0x20b; - __ts_calib.x_div = 0x9b6; - __ts_calib.y_mul = 0x0f4; - __ts_calib.y_div = 0x66f; - __ts_calib.dual_debounce_frame = 0; - __ts_calib.dual_sensi_entry = 0x18; - __ts_calib.dual_sensi_leave = 0x24; -} - -/* _touch_hsave() - save hardware state */ -static void _touch_hsave(touch_state_t *state) -{ - volatile uint16_t *IO_PRCR = (void*)0xa405011c; - - i2c_hsave(state); - state->PRCR = *(IO_PRCR); -} - -/* _touch_hrestore() - restore hardware state */ -static void _touch_hrestore(touch_state_t *state) -{ - volatile uint16_t *IO_PRCR = (void*)0xa405011c; - - *(IO_PRCR) = state->PRCR; - i2c_hrestore(state); -} - -/* _touch_hpowered() - check if the module is powered */ -static bool _touch_hpowered(void) -{ - return i2c_hpowered(); -} - -/* _touch_hpoweron() - power on the module */ -static void _touch_hpoweron(void) -{ - i2c_hpoweron(); -} - -/* _touch_hpoweroff() - power off the module */ -static void _touch_hpoweroff(void) -{ - i2c_hpoweroff(); -} - -/* _touch_unbin() - unbind from gint to casio */ -static void _touch_unbin(void) -{ - i2c_unbin(); -} - -/* _touch_funbin() - funbind from casio to gint */ -static void _touch_funbin(void) -{ - i2c_funbin(); -} - -/* drv_touch - touch-screen driver declaration */ -gint_driver_t drv_touch = { - .name = "TOUCH", - .configure = _touch_configure, - .hsave = (void *)_touch_hsave, - .hrestore = (void *)_touch_hrestore, - .hpowered = _touch_hpowered, - .hpoweron = _touch_hpoweron, - .hpoweroff = _touch_hpoweroff, - .unbind = _touch_unbin, - .funbind = _touch_funbin, - .state_size = sizeof(touch_state_t), -}; -GINT_DECLARE_DRIVER(16, drv_touch); From 8515c2aecb54005281d566dc1c443c9b1bb3719b Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 15:56:46 +0100 Subject: [PATCH 5/5] touch: fix typo --- src/touch/i2c_inth.c | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/touch/i2c_inth.c b/src/touch/i2c_inth.c index 844b291..6b12ce0 100644 --- a/src/touch/i2c_inth.c +++ b/src/touch/i2c_inth.c @@ -51,8 +51,8 @@ void i2c_inth_trans(void) switch (__i2c_request.state) { // start byte (writing or reading) and select the proper mode - // note that the "read stream" just change the internal state - // because the starting byte (0x81) is force-sended over there + // note that the "read stream" just changes the internal state + // because the starting byte (0x81) is force-sent over there case I2C_REQ_STATE_START: SH7305_I2C.ICIC.DTEE = 0; SH7305_I2C.ICIC.WAITE = 1; @@ -73,8 +73,8 @@ void i2c_inth_trans(void) __i2c_request.state = I2C_REQ_STATE_SWITCH_TO_RECEIVE_WAIT; break; - // for the read operation, the only things to do is the ACK at the - // end of the operation + // for the read operation, the only thing to do is to send the + // ACK at the end of the operation case I2C_REQ_STATE_READ: SH7305_I2C.ICIC.DTEE = 0; if (_i2c_inth_io_operation(true) == 0) @@ -105,9 +105,9 @@ void i2c_inth_wait(void) // note that if the "select register" request is performed (or if // no buffer is provided) then we send the stop condition (0x90) // and force-stop the module. - // Otherwise, we trigger a new request (0x94) manually to (after - // the restart interruption) switch from "reading" or "writing" - // mode + // Otherwise, we trigger a new request (0x94) to (after + // the restart interruption) switch manually from "reading" or + // "writing" mode case I2C_REQ_STATE_REG_SELECT_WAIT: if ( __i2c_request.mode == I2C_REQ_MODE_REG_SELECT @@ -134,8 +134,8 @@ void i2c_inth_wait(void) break; // read operation - // can either be WAIT or DTE interruption, a special case is - // performed to avoid DTE interruption + // can either be WAIT or DTE interrupt, a special case is + // performed to avoid DTE interrupt case I2C_REQ_STATE_READ: if (__i2c_request.buffer_size_remaning > 1) { if (SH7305_I2C.ICSR.DTE != 0)