From 4cd81571bccb284580767ce37c487ec8ec536759 Mon Sep 17 00:00:00 2001 From: Lephe Date: Sun, 6 Oct 2024 08:51:22 +0200 Subject: [PATCH 01/29] gdb: add missing stdbool.h include in gdb.h Fixes sh-elf-gdb#1 --- include/gint/gdb.h | 1 + 1 file changed, 1 insertion(+) diff --git a/include/gint/gdb.h b/include/gint/gdb.h index 93bdab6..4f89c91 100644 --- a/include/gint/gdb.h +++ b/include/gint/gdb.h @@ -10,6 +10,7 @@ extern "C" { #endif #include +#include /* gdb_cpu_state_t: State of the CPU when breaking This struct keep the same register indices as those declared by GDB to allow From 3e7e8e1fa0f5ce67675afab1751a1a391360d675 Mon Sep 17 00:00:00 2001 From: Lephenixnoir Date: Sat, 19 Oct 2024 09:43:49 +0200 Subject: [PATCH 02/29] kernel: fix 512-kB RAM check on mono being duped by cache This was noticed by Sentaro21 and tested by CalcLoverHK on a Slim. --- src/kernel/hardware.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/kernel/hardware.c b/src/kernel/hardware.c index 793c39e..56e9dcb 100644 --- a/src/kernel/hardware.c +++ b/src/kernel/hardware.c @@ -90,10 +90,10 @@ void hw_detect(void) gint[HWFS] = HWFS_CASIOWIN; } - /* Detect RAM by checking if 8804'0000 is the same as 8800'0000. */ + /* Detect RAM by checking if a804'0000 is the same as a800'0000. */ - volatile uint8_t *R4 = (void *)0x88040000; - volatile uint8_t *R0 = (void *)0x88000000; + volatile uint8_t *R4 = (void *)0xa8040000; + volatile uint8_t *R0 = (void *)0xa8000000; /* Make backups */ uint8_t b0 = *R0; From 444ab2dd8b22c45d5c1cef1e5cd0b6c8cf158cb3 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Fri, 13 Dec 2024 13:38:59 +0100 Subject: [PATCH 03/29] kernel: simplify PIE support --- fx9860g.ld.c | 4 ++-- fxcg50.ld.c | 4 ++-- fxcp_hh2.ld.c | 4 ++-- src/kernel/kernel.c | 10 ++++++++-- src/kernel/vbr.h | 4 ---- 5 files changed, 14 insertions(+), 12 deletions(-) diff --git a/fx9860g.ld.c b/fx9860g.ld.c index a4ac9dc..f06c72c 100644 --- a/fx9860g.ld.c +++ b/fx9860g.ld.c @@ -70,11 +70,11 @@ SECTIONS _gint_exch_start = . ; KEEP(*(.gint.exch)) - _gint_exch_size = ABSOLUTE(. - _gint_exch_start); + _gint_exch_end = . ; _gint_tlbh_start = . ; KEEP(*(.gint.tlbh)) - _gint_tlbh_size = ABSOLUTE(. - _gint_tlbh_start); + _gint_tlbh_end = . ; *(.text .text.*) *(C P) diff --git a/fxcg50.ld.c b/fxcg50.ld.c index 8ad5e3e..c4ba63b 100644 --- a/fxcg50.ld.c +++ b/fxcg50.ld.c @@ -59,11 +59,11 @@ SECTIONS _gint_exch_start = . ; KEEP(*(.gint.exch)) - _gint_exch_size = ABSOLUTE(. - _gint_exch_start); + _gint_exch_end = . ; _gint_tlbh_start = . ; KEEP(*(.gint.tlbh)) - _gint_tlbh_size = ABSOLUTE(. - _gint_tlbh_start); + _gint_tlbh_end = . ; *(.text .text.*) } > rom diff --git a/fxcp_hh2.ld.c b/fxcp_hh2.ld.c index 5036a1b..897d8da 100644 --- a/fxcp_hh2.ld.c +++ b/fxcp_hh2.ld.c @@ -68,12 +68,12 @@ SECTIONS . = ALIGN(0x10); _gint_exch_start = . ; *(.gint.exch) - _gint_exch_size = ABSOLUTE(. - _gint_exch_start); + _gint_exch_end = . ; . = ALIGN(0x10); _gint_tlbh_start = . ; *(.gint.tlbh) - _gint_tlbh_size = ABSOLUTE(. - _gint_tlbh_start); + _gint_tlbh_end = . ; *(.text .text.*) } > ram AT> bin diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index 602a9db..5f45b2e 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -76,9 +76,15 @@ void kinit(void) #endif /* Event handler entry points */ + extern uint32_t gint_exch_start; + extern uint32_t gint_exch_end; + extern uint32_t gint_tlbh_start; + extern uint32_t gint_tlbh_end; void *inth_entry = isSH3() ? gint_inth_7705 : gint_inth_7305; - uint32_t exch_size = (uint32_t)&gint_exch_size; - uint32_t tlbh_size = (uint32_t)&gint_tlbh_size; + uint32_t exch_size = \ + (uint32_t)&gint_exch_end - (uint32_t)&gint_exch_start; + uint32_t tlbh_size = \ + (uint32_t)&gint_tlbh_end - (uint32_t)&gint_tlbh_start; /* Load the event handler entry points into memory */ memcpy((void *)VBR + 0x100, gint_exch, exch_size); diff --git a/src/kernel/vbr.h b/src/kernel/vbr.h index dc0c1a8..e549874 100644 --- a/src/kernel/vbr.h +++ b/src/kernel/vbr.h @@ -11,8 +11,4 @@ void gint_tlbh(void); void gint_inth_7705(void); void gint_inth_7305(void); -/* Size of exception and TLB handlers */ -extern char gint_exch_size; -extern char gint_tlbh_size; - #endif /* GINT_CORE_VBR */ From f42762010348534e7c9cc018a24a64e5b3478225 Mon Sep 17 00:00:00 2001 From: Lephenixnoir Date: Mon, 16 Dec 2024 13:15:07 +0100 Subject: [PATCH 04/29] minor: add missing include in defs/call.h --- include/gint/defs/call.h | 1 + 1 file changed, 1 insertion(+) diff --git a/include/gint/defs/call.h b/include/gint/defs/call.h index 5a2f2bf..1dd3c7f 100644 --- a/include/gint/defs/call.h +++ b/include/gint/defs/call.h @@ -12,6 +12,7 @@ #define GINT_DEFS_CALL #include +#include #ifdef __cplusplus extern "C" { From b687af4e943e3794b59932949700049b507c40da Mon Sep 17 00:00:00 2001 From: Lephe Date: Sun, 24 Nov 2024 23:25:21 +0100 Subject: [PATCH 05/29] kernel: allows syscalls to be called from fixed addresses --- src/kernel/kernel.c | 3 + src/kernel/kernel.h | 3 + src/kernel/osmenu.c | 5 -- src/kernel/syscalls.S | 157 ++++++++++++++++++++++++------------------ 4 files changed, 97 insertions(+), 71 deletions(-) diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index 5f45b2e..d629735 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -40,6 +40,9 @@ void *gint_stack_top = NULL; /* kinit(): Install and start gint */ void kinit(void) { + /* Figure out which CASIOWIN API to use based on the OS type. */ + gint_set_CASIOWIN_API(0); + #if GINT_HW_FX /* On fx-9860G, VBR is loaded at the end of the user RAM. On SH4, the end of the user RAM hosts the stack, for which we leave 12 kB diff --git a/src/kernel/kernel.h b/src/kernel/kernel.h index 3efc8cb..a5b5952 100644 --- a/src/kernel/kernel.h +++ b/src/kernel/kernel.h @@ -18,4 +18,7 @@ void kinit(void); /* kquit(): Quit gint and give back control to the system */ void kquit(void); +/* Select the CASIOWIN API for syscalls. */ +void gint_set_CASIOWIN_API(int API); + #endif /* GINT_CORE_KERNEL */ diff --git a/src/kernel/osmenu.c b/src/kernel/osmenu.c index ea503cd..22afa1a 100644 --- a/src/kernel/osmenu.c +++ b/src/kernel/osmenu.c @@ -122,11 +122,6 @@ void gint_osmenu_native(void) } #endif - /* Mysteriously crashes when coming back; might be useful another time - instead of GetKeyWait() - int C=0x04, R=0x09; - __SpecialMatrixCodeProcessing(&C, &R); */ - __osmenu_id = __Timer_Install(0, __osmenu_handler, 0 /* ms */); if(__osmenu_id <= 0) return; __Timer_Start(__osmenu_id); diff --git a/src/kernel/syscalls.S b/src/kernel/syscalls.S index 387e18f..4c819c7 100644 --- a/src/kernel/syscalls.S +++ b/src/kernel/syscalls.S @@ -144,82 +144,107 @@ syscall_table: #endif /* GINT_OS_FX */ #if GINT_OS_CG +#define CASIOWIN_API_VERSIONS 2 +.global _gint_set_CASIOWIN_API -/* Dynamic allocation */ +/* System for dynamically selecting between the syscall and fixed version of + each function based on the OS version. + @r0: Internal function ID (from table below) */ +_CASIOWIN_call: + mov #(4*CASIOWIN_API_VERSIONS), r1 + mulu.w r0, r1 + mov.l 3f, r0 + mov.l @r0, r2 + sts macl, r1 -___malloc: - syscall(0x1f44) -___free: - syscall(0x1f42) -___realloc: - syscall(0x1f46) + /* API version 0 is the normal syscall table */ + tst r2, r2 + mova .CASIOWIN_TABLE, r0 + bt.s 1f + mov.l @(r0, r1), r0 -/* BFile driver */ + /* Other API versions are the direct calls */ + jmp @r0 + nop -_BFile_Remove: - syscall(0x1db4) -_BFile_Rename: - syscall(0x1db3) -_BFile_Create: - syscall(0x1dae) -_BFile_Open: - mov #0, r6 - syscall(0x1da3) -_BFile_Close: - syscall(0x1da4) -_BFile_Size: - syscall(0x1da6) -_BFile_Seek: - syscall(0x1da9) -_BFile_GetPos: - syscall(0x1dab) -_BFile_Write: - syscall(0x1daf) -_BFile_Read: - syscall(0x1dac) -_BFile_FindFirst: - syscall(0x1db6) -_BFile_FindNext: - syscall(0x1db8) -_BFile_FindClose: - syscall(0x1dba) +1: mov.l 2f, r1 + jmp @r1 + nop -/* Return to menu */ +2: .long 0x80020070 -___Timer_Install: - syscall(0x8d9) -___Timer_Start: - syscall(0x8db) -___Timer_Stop: - syscall(0x8dc) -___Timer_Deinstall: - syscall(0x8da) -___PutKeyCode: - syscall(0x12c6) -___GetKeyWait: - syscall(0x12bf) -___ClearKeyBuffer: - syscall(0x12c7) -___GetVRAMAddress: - syscall(0x1e6) -___ConfigureStatusArea: - syscall(0x2b7) -___SetQuitHandler: - syscall(0x1e6e) +_gint_set_CASIOWIN_API: + mov.l 3f, r0 + rts + mov.l r4, @r0 -.global ___SpecialMatrixCodeProcessing -___SpecialMatrixCodeProcessing: - syscall(0x1e60) +.balign 4 +3: .long .CASIOWIN_API -/* Reset */ +.CASIOWIN_TABLE: + .long 0x1f44, 0x00000000 /* malloc */ + .long 0x1f42, 0x00000000 /* free */ + .long 0x1f46, 0x00000000 /* realloc */ + .long 0x1db4, 0x00000000 /* BFile_Remove */ + .long 0x1db3, 0x00000000 /* BFile_Rename */ + .long 0x1dae, 0x00000000 /* BFile_Create */ + .long 0x1da3, 0x00000000 /* BFile_Open */ + .long 0x1da4, 0x00000000 /* BFile_Close */ + .long 0x1da6, 0x00000000 /* BFile_Size */ + .long 0x1da9, 0x00000000 /* BFile_Seek */ + .long 0x1dab, 0x00000000 /* BFile_GetPos */ + .long 0x1daf, 0x00000000 /* BFile_Write */ + .long 0x1dac, 0x00000000 /* BFile_Read */ + .long 0x1db6, 0x00000000 /* BFile_FindFirst */ + .long 0x1db8, 0x00000000 /* BFile_FindNext */ + .long 0x1dba, 0x00000000 /* BFile_FindClose */ + .long 0x08d9, 0x00000000 /* Timer_Install */ + .long 0x08db, 0x00000000 /* Timer_Start */ + .long 0x08dc, 0x00000000 /* Timer_Stop */ + .long 0x08da, 0x00000000 /* Timer_Deinstall */ + .long 0x12c6, 0x00000000 /* PutKeyCode */ + .long 0x12bf, 0x00000000 /* GetKeyWait */ + .long 0x12c7, 0x00000000 /* ClearKeyBuffer */ + .long 0x01e6, 0x00000000 /* GetVRAMAddress */ + .long 0x02b7, 0x00000000 /* ConfigureStatusArea */ + .long 0x1e6e, 0x00000000 /* SetQuitHandler */ + .long 0x1839, 0x00000000 /* PowerOff */ + .long 0x1187, 0x00000000 /* Reset */ -___PowerOff: - syscall(0x1839) -___Reset: - syscall(0x1187) +#define casiowin_call(id) bra _CASIOWIN_call; mov id, r0 -syscall_table: - .long 0x80020070 +___malloc: casiowin_call(#0) +___free: casiowin_call(#1) +___realloc: casiowin_call(#2) +_BFile_Remove: casiowin_call(#3) +_BFile_Rename: casiowin_call(#4) +_BFile_Create: casiowin_call(#5) +_BFile_Open: mov #0, r6; casiowin_call(#6) +_BFile_Close: casiowin_call(#7) +_BFile_Size: casiowin_call(#8) +_BFile_Seek: casiowin_call(#9) +_BFile_GetPos: casiowin_call(#10) +_BFile_Write: casiowin_call(#11) +_BFile_Read: casiowin_call(#12) +_BFile_FindFirst: casiowin_call(#13) +_BFile_FindNext: casiowin_call(#14) +_BFile_FindClose: casiowin_call(#15) +___Timer_Install: casiowin_call(#16) +___Timer_Start: casiowin_call(#17) +___Timer_Stop: casiowin_call(#18) +___Timer_Deinstall: casiowin_call(#19) +___PutKeyCode: casiowin_call(#20) +___GetKeyWait: casiowin_call(#21) +___ClearKeyBuffer: casiowin_call(#22) +___GetVRAMAddress: casiowin_call(#23) +___ConfigureStatusArea: casiowin_call(#24) +___SetQuitHandler: casiowin_call(#25) +___PowerOff: casiowin_call(#26) +___Reset: casiowin_call(#27) + +.data +.CASIOWIN_API: + .long 0 #endif /* GINT_OS_CG */ From a2c85b757788bec2aff14c60d507ab7cbd996643 Mon Sep 17 00:00:00 2001 From: Lephe Date: Mon, 25 Nov 2024 17:29:48 +0100 Subject: [PATCH 06/29] kernel: partial support for Math+ OS 1.00 --- src/kernel/exch.c | 2 + src/kernel/hardware.c | 1 + src/kernel/kernel.c | 11 +++++- src/kernel/osmenu.c | 4 -- src/kernel/syscalls.S | 85 +++++++++++++++++++++++++++---------------- 5 files changed, 65 insertions(+), 38 deletions(-) diff --git a/src/kernel/exch.c b/src/kernel/exch.c index b1c11e9..f04bfb0 100644 --- a/src/kernel/exch.c +++ b/src/kernel/exch.c @@ -80,6 +80,7 @@ GNORETURN static void gint_default_panic(GUNUSED uint32_t code) if(code == 0x1060) name = "Memory init failed"; if(code == 0x1080) name = "Stack overflow"; if(code == 0x10a0) name = "UBC in bank 1 code"; + // if(code == 0x10c0) name = "Missing syscall"; // not on FX if(name[0]) dtext(1, 9, name); else dprint(1, 9, "%03x", code); @@ -121,6 +122,7 @@ GNORETURN static void gint_default_panic(GUNUSED uint32_t code) if(code == 0x1060) name = "Memory initialization failed (heap)"; 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"; dprint(6, 25, "%03x %s", code, name); diff --git a/src/kernel/hardware.c b/src/kernel/hardware.c index 56e9dcb..8a02815 100644 --- a/src/kernel/hardware.c +++ b/src/kernel/hardware.c @@ -11,6 +11,7 @@ #include #include +#include "kernel.h" /* Holds information about the current platform */ GBSS uint32_t gint[HW_KEYS]; diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index d629735..c797ef5 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -40,8 +40,15 @@ void *gint_stack_top = NULL; /* kinit(): Install and start gint */ void kinit(void) { - /* Figure out which CASIOWIN API to use based on the OS type. */ - gint_set_CASIOWIN_API(0); + /* Figure out which CASIOWIN API to use based on the OS type. */ + #if GINT_OS_CG + char *version = (void *)0x80020020; + if(!memcmp(version, "01.00", 5)) + gint_set_CASIOWIN_API(1); + else + gint_set_CASIOWIN_API(0); + #endif + #if GINT_HW_FX /* On fx-9860G, VBR is loaded at the end of the user RAM. On SH4, the diff --git a/src/kernel/osmenu.c b/src/kernel/osmenu.c index 22afa1a..c5173c0 100644 --- a/src/kernel/osmenu.c +++ b/src/kernel/osmenu.c @@ -13,7 +13,6 @@ int __Timer_Deinstall(int id); int __PutKeyCode(int row, int column, int keycode); int __GetKeyWait(int *col,int *row,int type,int time,int menu,uint16_t *key); void __ClearKeyBuffer(void); /* ? */ -void __ConfigureStatusArea(int mode); void __SetQuitHandler(void (*callback)(void)); #if !GINT_OS_CP @@ -102,9 +101,6 @@ void gint_osmenu_native(void) gint_copy_vram(); #if GINT_OS_CG - /* Unfortunately ineffective (main menu probably reenables it) - __ConfigureStatusArea(3); */ - /* Try to use the internal function directly if we could figure out its address by dynamically disassembling */ os_menu_function_t *fun = find_os_menu_function(); diff --git a/src/kernel/syscalls.S b/src/kernel/syscalls.S index 4c819c7..a13f2b6 100644 --- a/src/kernel/syscalls.S +++ b/src/kernel/syscalls.S @@ -146,70 +146,91 @@ syscall_table: #if GINT_OS_CG #define CASIOWIN_API_VERSIONS 2 .global _gint_set_CASIOWIN_API +.global _gint_get_CASIOWIN_API /* System for dynamically selecting between the syscall and fixed version of each function based on the OS version. @r0: Internal function ID (from table below) */ _CASIOWIN_call: - mov #(4*CASIOWIN_API_VERSIONS), r1 + mov #CASIOWIN_API_VERSIONS, r1 mulu.w r0, r1 mov.l 3f, r0 mov.l @r0, r2 sts macl, r1 + add r2, r1 + shll2 r1 /* API version 0 is the normal syscall table */ tst r2, r2 mova .CASIOWIN_TABLE, r0 - bt.s 1f + bt.s .syscall mov.l @(r0, r1), r0 /* Other API versions are the direct calls */ + tst r0, r0 + bt .missingCall + jmp @r0 nop -1: mov.l 2f, r1 +.missingCall: + mov.l .gint_panic, r0 + mov.w 2f, r4 + jmp @r0 + nop + +.syscall: + mov.l 1f, r1 jmp @r1 nop -2: .long 0x80020070 +2: .word 0x10c0 +.balign 4 +1: .long 0x80020070 +.gint_panic: + .long _gint_panic _gint_set_CASIOWIN_API: mov.l 3f, r0 rts mov.l r4, @r0 +_gint_get_CASIOWIN_API: + mov.l 3f, r0 + rts + mov.l @r0, r0 + .balign 4 3: .long .CASIOWIN_API .CASIOWIN_TABLE: - .long 0x1f44, 0x00000000 /* malloc */ - .long 0x1f42, 0x00000000 /* free */ - .long 0x1f46, 0x00000000 /* realloc */ - .long 0x1db4, 0x00000000 /* BFile_Remove */ - .long 0x1db3, 0x00000000 /* BFile_Rename */ - .long 0x1dae, 0x00000000 /* BFile_Create */ - .long 0x1da3, 0x00000000 /* BFile_Open */ - .long 0x1da4, 0x00000000 /* BFile_Close */ - .long 0x1da6, 0x00000000 /* BFile_Size */ - .long 0x1da9, 0x00000000 /* BFile_Seek */ - .long 0x1dab, 0x00000000 /* BFile_GetPos */ - .long 0x1daf, 0x00000000 /* BFile_Write */ - .long 0x1dac, 0x00000000 /* BFile_Read */ - .long 0x1db6, 0x00000000 /* BFile_FindFirst */ - .long 0x1db8, 0x00000000 /* BFile_FindNext */ - .long 0x1dba, 0x00000000 /* BFile_FindClose */ - .long 0x08d9, 0x00000000 /* Timer_Install */ - .long 0x08db, 0x00000000 /* Timer_Start */ - .long 0x08dc, 0x00000000 /* Timer_Stop */ - .long 0x08da, 0x00000000 /* Timer_Deinstall */ - .long 0x12c6, 0x00000000 /* PutKeyCode */ - .long 0x12bf, 0x00000000 /* GetKeyWait */ - .long 0x12c7, 0x00000000 /* ClearKeyBuffer */ - .long 0x01e6, 0x00000000 /* GetVRAMAddress */ - .long 0x02b7, 0x00000000 /* ConfigureStatusArea */ - .long 0x1e6e, 0x00000000 /* SetQuitHandler */ - .long 0x1839, 0x00000000 /* PowerOff */ - .long 0x1187, 0x00000000 /* Reset */ + .long 0x1f44, 0x8025e0fc /* malloc */ + .long 0x1f42, 0x8025dec8 /* free */ + .long 0x1f46, 0x8025ec3c /* realloc */ + .long 0x1db4, 0x802404d2 /* BFile_Remove */ + .long 0x1db3, 0x80240482 /* BFile_Rename */ + .long 0x1dae, 0x802401b0 /* BFile_Create */ + .long 0x1da3, 0x8023fb90 /* BFile_Open */ + .long 0x1da4, 0x8023fd0e /* BFile_Close */ + .long 0x1da6, 0x8023fdc4 /* BFile_Size */ + .long 0x1da9, 0x8023ff2c /* BFile_Seek */ + .long 0x1dab, 0x8024003c /* BFile_GetPos */ + .long 0x1daf, 0x8024025e /* BFile_Write */ + .long 0x1dac, 0x80240082 /* BFile_Read */ + .long 0x1db6, 0 /* BFile_FindFirst */ + .long 0x1db8, 0x80240b06 /* BFile_FindNext */ + .long 0x1dba, 0x80240c10 /* BFile_FindClose */ + .long 0x08d9, 0x800b130c /* Timer_Install */ + .long 0x08db, 0x800b1456 /* Timer_Start */ + .long 0x08dc, 0x800b14b2 /* Timer_Stop */ + .long 0x08da, 0x800b13d4 /* Timer_Deinstall */ + .long 0x12c6, 0 /* PutKeyCode */ + .long 0x12bf, 0x8017be56 /* GetKeyWait */ + .long 0x12c7, 0 /* ClearKeyBuffer */ + .long 0x01e6, 0 /* GetVRAMAddress */ + .long 0x1e6e, 0 /* SetQuitHandler */ + .long 0x1839, 0 /* PowerOff */ + .long 0x1187, 0 /* Reset */ #define casiowin_call(id) bra _CASIOWIN_call; mov id, r0 From 1efe73728202803215b8219285464768ef162840 Mon Sep 17 00:00:00 2001 From: Lephe Date: Fri, 27 Dec 2024 15:29:09 +0100 Subject: [PATCH 07/29] kernel: further support for the Math+ in general * Add a new HWCALC value HWCALC_FXCG100, detected based on being on an Area-3 RAM model and having OS version that's either less than 3 or 3 and built after January 2025. * Disable the _ostk heap arena, as the region might simply not exist, and improve the VRAM allocation code to account for this better than the hardcoded macro previously in place for the fx-CP 400. * Disable gint_osmenu() which can't work with MPM right now. * Add BFile_FindFirst() and GetVRAMAddress() syscalls. --- include/gint/hardware.h | 2 ++ src/kernel/hardware.c | 22 ++++++++++++++++++++-- src/kernel/kernel.c | 22 ++++++++++++---------- src/kernel/osmenu.c | 5 +++++ src/kernel/start.c | 5 +++-- src/kernel/syscalls.S | 4 ++-- src/render-cg/dvram.c | 13 ++++++------- 7 files changed, 50 insertions(+), 23 deletions(-) diff --git a/include/gint/hardware.h b/include/gint/hardware.h index 6044ec8..6a252a2 100644 --- a/include/gint/hardware.h +++ b/include/gint/hardware.h @@ -111,6 +111,8 @@ void hw_detect(void); #define HWCALC_FX9860G_SLIM 7 /* fx-CP 400 */ #define HWCALC_FXCP400 8 +/* fx-CG 100, successor to the CG-50 in the Prizm family. Also Graph Math+ */ +#define HWCALC_FXCG100 9 /* ** Keyboard diff --git a/src/kernel/hardware.c b/src/kernel/hardware.c index 8a02815..3340513 100644 --- a/src/kernel/hardware.c +++ b/src/kernel/hardware.c @@ -127,10 +127,28 @@ void hw_detect(void) gint[HWCPUVR] = PVR; gint[HWCPUPR] = PRR; - /* Tell Prizms apart from fx-CG 50 by checking the stack address*/ uint32_t stack; __asm__("mov r15, %0" : "=r"(stack)); - gint[HWCALC] = (stack < 0x8c000000) ? HWCALC_PRIZM : HWCALC_FXCG50; + char const *version = (void *)0x80020020; + char const *osdate = (void *)0x80b5ffe0; + + /* Tell Prizms apart from fx-CG 50 by checking the stack address*/ + if(stack < 0x8c000000) { + gint[HWCALC] = HWCALC_PRIZM; + } + /* Tell Math+/fx-CG 100 apart from CG-50 by checking OS version + date. + CG-50 OS versions use OS 3. Math+, for some reason, rewinds back to OS 1 + and got updated to OS 2 in late 2024. We decide that we are on a CG-50 OS + if the version is 3 and the date is 201x-2024. */ + else { + /* All known CG-50 versions have date at this address due to the footer + location. WARNING: not a future-proof address! */ + char d0 = osdate[0], d1 = osdate[1], d2 = osdate[2], d3 = osdate[3]; + bool cg50 = version[1] == '3' && d0 == '2' && d1 == '0' && + (d2 == '1' || (d2 == '2' && d3 <= '4')); + gint[HWCALC] = cg50 ? HWCALC_FXCG50 : HWCALC_FXCG100; + } + gint[HWFS] = HWFS_FUGUE; /* Tell the fx-CG emulator apart using the product ID */ diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index c797ef5..3541e70 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -118,16 +118,18 @@ void kinit(void) /* Create an arena in the OS stack as well, for VRAM and more data */ #if GINT_HW_CG && !defined(GINT_NO_OS_STACK) - static kmalloc_arena_t os_stack = { 0 }; - os_stack.name = "_ostk"; - os_stack.is_default = true; - if(gint[HWCALC] == HWCALC_PRIZM || gint[HWCALC] == HWCALC_FXCG_MANAGER) - os_stack.start = (void *)0x880f0000; - else - os_stack.start = (void *)0x8c0f0000; - os_stack.end = os_stack.start + (350 * 1024); - kmalloc_init_arena(&os_stack, true); - kmalloc_add_arena(&os_stack); + if(gint[HWCALC] != HWCALC_FXCG100) { + static kmalloc_arena_t os_stack = { 0 }; + os_stack.name = "_ostk"; + os_stack.is_default = true; + if(gint[HWCALC] == HWCALC_PRIZM || gint[HWCALC] == HWCALC_FXCG_MANAGER) + os_stack.start = (void *)0x880f0000; + else + os_stack.start = (void *)0x8c0f0000; + os_stack.end = os_stack.start + (350 * 1024); + kmalloc_init_arena(&os_stack, true); + kmalloc_add_arena(&os_stack); + } #endif /* Allocate world buffers for the OS and for gint */ diff --git a/src/kernel/osmenu.c b/src/kernel/osmenu.c index c5173c0..5124b67 100644 --- a/src/kernel/osmenu.c +++ b/src/kernel/osmenu.c @@ -95,6 +95,11 @@ static os_menu_function_t *find_os_menu_function(void) void gint_osmenu_native(void) { +#if GINT_OS_CG + if(gint[HWCALC] == HWCALC_FXCG100) + return; +#endif + // TODO: OS menu on fx-CP #if !GINT_OS_CP __ClearKeyBuffer(); diff --git a/src/kernel/start.c b/src/kernel/start.c index 26e4257..5159a92 100644 --- a/src/kernel/start.c +++ b/src/kernel/start.c @@ -42,8 +42,9 @@ int8_t gint_restart = 0; /* gint_setrestart(): Set whether to restart the add-in after exiting */ void gint_setrestart(int restart) { - /* There is now return-to-menu so no restart on CP */ - gint_restart = restart && !GINT_OS_CP; + /* No restart on the machines for which there is no return-to-menu, i.e. on + fx-CP and on the fx-CG 100. */ + gint_restart = restart && !GINT_OS_CP && gint[HWCALC] != HWCALC_FXCG100; } /* Return value of main() */ diff --git a/src/kernel/syscalls.S b/src/kernel/syscalls.S index a13f2b6..94bc19f 100644 --- a/src/kernel/syscalls.S +++ b/src/kernel/syscalls.S @@ -217,7 +217,7 @@ _gint_get_CASIOWIN_API: .long 0x1dab, 0x8024003c /* BFile_GetPos */ .long 0x1daf, 0x8024025e /* BFile_Write */ .long 0x1dac, 0x80240082 /* BFile_Read */ - .long 0x1db6, 0 /* BFile_FindFirst */ + .long 0x1db6, 0x80240888 /* BFile_FindFirst */ .long 0x1db8, 0x80240b06 /* BFile_FindNext */ .long 0x1dba, 0x80240c10 /* BFile_FindClose */ .long 0x08d9, 0x800b130c /* Timer_Install */ @@ -227,7 +227,7 @@ _gint_get_CASIOWIN_API: .long 0x12c6, 0 /* PutKeyCode */ .long 0x12bf, 0x8017be56 /* GetKeyWait */ .long 0x12c7, 0 /* ClearKeyBuffer */ - .long 0x01e6, 0 /* GetVRAMAddress */ + .long 0x01e6, 0x8004579a /* GetVRAMAddress */ .long 0x1e6e, 0 /* SetQuitHandler */ .long 0x1839, 0 /* PowerOff */ .long 0x1187, 0 /* Reset */ diff --git a/src/render-cg/dvram.c b/src/render-cg/dvram.c index aac777a..7f9f287 100644 --- a/src/render-cg/dvram.c +++ b/src/render-cg/dvram.c @@ -97,16 +97,15 @@ bool dvram_init(void) { int const MARGIN = 32; + char const *arena = NULL; + if(kmalloc_get_arena("_ostk")) + arena = "_ostk"; + /* Leave MARGIN bytes on each side of the region; this enables some important optimizations in the image renderer. We also add another 32 bytes so we can manually 32-align the region */ - uint32_t region = (uint32_t)kmalloc(DWIDTH*DHEIGHT*2 + MARGIN*2 + 32, -#if !defined(GINT_NO_OS_STACK) - "_ostk" -#else - NULL -#endif - ); + uint32_t region = + (uint32_t)kmalloc(DWIDTH*DHEIGHT*2 + MARGIN*2 + 32, arena); if(region == 0) return false; From 58c8e5f8fe16e5983e60196519249373048e1d78 Mon Sep 17 00:00:00 2001 From: Lephe Date: Tue, 7 Jan 2025 21:14:51 +0100 Subject: [PATCH 08/29] keydev: support for the Math+ layout and track row/col key_event_t is now 8 bytes instead of 4, a change that was doomed to happen anyway to deal with touch input (where it's not clear either whether 8 bytes will be enough for double touch). --- include/gint/drivers/keydev.h | 2 ++ include/gint/keyboard.h | 17 ++++++++--- include/gint/keycodes.h | 19 ++++++++++-- src/keysc/keydev.c | 56 +++++++++++++++++++++++++++-------- 4 files changed, 76 insertions(+), 18 deletions(-) diff --git a/include/gint/drivers/keydev.h b/include/gint/drivers/keydev.h index 4d9b956..0e15645 100644 --- a/include/gint/drivers/keydev.h +++ b/include/gint/drivers/keydev.h @@ -184,6 +184,8 @@ typedef struct { /* Candidate key for repeats (or 0 if no key is candidate yet) */ int16_t rep_key; + /* Matrix code for rep_key, if rep_key is not 0. */ + uint8_t rep_row, rep_col; /* Number of repeats already sent */ int16_t rep_count; /* Time since key was first pressed (us) */ diff --git a/include/gint/keyboard.h b/include/gint/keyboard.h index 2777479..7d4cfe7 100644 --- a/include/gint/keyboard.h +++ b/include/gint/keyboard.h @@ -114,16 +114,25 @@ extern "C" { * 0xffff is "just before" 0x0000, not "long after". */ typedef struct { - uint time :16; /* Time of event, unique over short periods */ + /* Time of event, unique over short periods */ + u16 time; uint :2; /* Reserved for future use */ - uint mod :1; /* Whether modifiers are used */ uint shift :1; /* If mod=1, whether SHIFT was pressed */ uint alpha :1; /* If mod=1, whether ALPHA was pressed */ - uint type :3; /* Type of key event */ - uint key :8; /* Hit key */ + + /* Key that was pressed or released. */ + u8 key; + + // 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; } GPACKED(4) key_event_t; diff --git a/include/gint/keycodes.h b/include/gint/keycodes.h index 5e904fa..dc2dd3f 100644 --- a/include/gint/keycodes.h +++ b/include/gint/keycodes.h @@ -9,7 +9,6 @@ extern "C" { #endif -/* Raw matrix codes */ enum { KEY_F1 = 0x91, KEY_F2 = 0x92, @@ -87,7 +86,23 @@ enum { KEY_EQUALS = 0xa5, KEY_CLEAR = KEY_EXIT, - /* Key aliases (handle with care =D) */ + /* Key codes for the CG-100 */ + KEY_ON = 0xa6, + KEY_HOME = KEY_MENU, + KEY_PREVTAB = 0xa7, + KEY_NEXTTAB = 0xa8, + KEY_PAGEUP = 0xa9, + KEY_PAGEDOWN = 0xaa, + KEY_SETTINGS = 0xab, + KEY_BACK = KEY_EXIT, + KEY_OK = 0xac, + KEY_CATALOG = 0xad, + KEY_TOOLS = KEY_OPTN, + KEY_FORMAT = KEY_FD, + KEY_SQRT = 0xae, + KEY_EXPFUN = 0xaf, + + /* Key aliases (deprecated--no more will be added) */ KEY_X2 = KEY_SQUARE, KEY_CARET = KEY_POWER, KEY_SWITCH = KEY_FD, diff --git a/src/keysc/keydev.c b/src/keysc/keydev.c index 4edee84..1071458 100644 --- a/src/keysc/keydev.c +++ b/src/keysc/keydev.c @@ -8,6 +8,7 @@ #include #include #include +#include #include #include @@ -45,12 +46,39 @@ static void keycode_to_keymatrix(int keycode, int *row, int *col) *col = 0; } #else -static GINLINE int keymatrix_to_keycode(int row, int col) +static uint8_t const CG100_keymap[] = { + KEY_ON, KEY_HOME, KEY_PREVTAB, KEY_UP, KEY_NEXTTAB, KEY_PAGEUP, + KEY_SETTINGS, KEY_BACK, KEY_LEFT, KEY_OK, KEY_RIGHT, KEY_PAGEDOWN, + KEY_SHIFT, KEY_ALPHA, KEY_VARS, KEY_DOWN, KEY_CATALOG, KEY_TOOLS, + KEY_XOT, KEY_FRAC, KEY_SQRT, KEY_POWER, KEY_SQUARE, KEY_EXPFUN, + KEY_COMMA, KEY_SIN, KEY_COS, KEY_TAN, KEY_LEFTP, KEY_RIGHTP, +}; +static int keymatrix_to_keycode(int row, int col) { + if(gint[HWCALC] == HWCALC_FXCG100) { + if(row >= 7 && row <= 9) + return CG100_keymap[6 * (9-row) + (6-col)]; + if(row == 1 && col == 3) + return KEY_FORMAT; + } return (row << 4) + (7 - col); } -static GINLINE void keycode_to_keymatrix(int keycode, int *row, int *col) +static void keycode_to_keymatrix(int keycode, int *row, int *col) { + if(gint[HWCALC] == HWCALC_FXCG100) { + if(keycode == KEY_FORMAT) { + *row = 1; + *col = 3; + return; + } + for(int i = 0; i < (int)sizeof(CG100_keymap); i++) { + if(CG100_keymap[i] == keycode) { + *row = 9 - i / 6; + *col = 6 - (i % 6); + return; + } + } + } *row = keycode >> 4; *col = 7 - (keycode & 7); } @@ -125,6 +153,8 @@ void keydev_process_state(keydev_t *d, uint8_t scan[12]) for(int mask = 0x80, col = 7; mask; mask >>= 1, col--) { + ev.row = row; + ev.col = col; ev.key = keymatrix_to_keycode(row, col); /* Update state only if the push succeeds */ if((diff & mask) && keydev_queue_push(d, ev)) @@ -173,6 +203,8 @@ key_event_t keydev_repeat_event(keydev_t *d) ev.type = KEYEV_HOLD; ev.key = d->rep_key; + ev.row = d->rep_row; + ev.col = d->rep_col; return ev; } @@ -188,11 +220,11 @@ void keydev_tick(keydev_t *d, uint us) /* Disable repeat if the repeating key was released */ if(d->rep_key != 0) { - int row, col; - keycode_to_keymatrix(d->rep_key, &row, &col); - if(!(d->state_now[row] & (1 << col))) + if(!(d->state_now[d->rep_row] & (1 << d->rep_col))) { d->rep_key = 0; + d->rep_row = 0; + d->rep_col = 0; d->rep_count = -1; d->rep_time = -1; d->rep_delay = -1; @@ -232,18 +264,18 @@ key_event_t keydev_unqueue_event(keydev_t *d) return ev; /* Update the event state accordingly */ - int row, col; - keycode_to_keymatrix(ev.key, &row, &col); - int mask = 1 << col; + int mask = 1 << ev.col; if(ev.type == KEYEV_DOWN) { - d->state_queue[row] |= mask; - d->state_flips[row] ^= mask; + d->state_queue[ev.row] |= mask; + d->state_flips[ev.row] ^= mask; /* Mark this key as the currently repeating one */ if(d->rep_key == 0 && can_repeat(d, ev.key)) { d->rep_key = ev.key; + d->rep_row = ev.row; + d->rep_col = ev.col; d->rep_count = -1; d->rep_time = 0; d->rep_delay = 0; @@ -251,8 +283,8 @@ key_event_t keydev_unqueue_event(keydev_t *d) } else if(ev.type == KEYEV_UP) { - d->state_queue[row] &= ~mask; - d->state_flips[row] ^= mask; + d->state_queue[ev.row] &= ~mask; + d->state_flips[ev.row] ^= mask; } return ev; From d04bb51f7ab9cf1b454f17a9bffd259cd32bb241 Mon Sep 17 00:00:00 2001 From: Lephe Date: Tue, 7 Jan 2025 21:32:49 +0100 Subject: [PATCH 09/29] keydev: add Catalog+4th row combo for F1...F6 on Math+ --- src/keysc/keydev.c | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/keysc/keydev.c b/src/keysc/keydev.c index 1071458..9e05003 100644 --- a/src/keysc/keydev.c +++ b/src/keysc/keydev.c @@ -263,6 +263,14 @@ key_event_t keydev_unqueue_event(keydev_t *d) if(!queue_poll(d, &ev)) return ev; + /* Compatibility combinations can transform the .key attribute */ +#if GINT_HW_CG + if(gint[HWCALC] == HWCALC_FXCG100) { + if(keydev_keydown(d, KEY_CATALOG) && ev.row == 6) + ev.key = KEY_F1 + (6 - ev.col); + } +#endif + /* Update the event state accordingly */ int mask = 1 << ev.col; From 918d96c081d642c5e31d65a63aded83828ee6ace Mon Sep 17 00:00:00 2001 From: Lephe Date: Fri, 10 Jan 2025 12:23:21 +0100 Subject: [PATCH 10/29] cpg: don't overclock beyond known limits on Math+ --- src/cpg/overclock.c | 62 +++++++++++++++++++++++++++++++++++++++++++++ src/kernel/kernel.c | 4 +-- 2 files changed, 64 insertions(+), 2 deletions(-) diff --git a/src/cpg/overclock.c b/src/cpg/overclock.c index 125615a..396a5cb 100644 --- a/src/cpg/overclock.c +++ b/src/cpg/overclock.c @@ -433,6 +433,66 @@ static struct cpg_overclock_setting const settings_fxcg50[5] = { .CS5aWCR = 0x000203C1 }, }; +// TODO: These structures are big and many settings overlap. Make it smaller. +// This is fxcg50[0,1,2,3,3]. +static struct cpg_overclock_setting const settings_fxcg100[5] = { + /* CLOCK_SPEED_F1 */ + { .FLLFRQ = 0x00004000 + 900, + .FRQCR = 0x0F011112, + .CS0BCR = 0x36DA0400, + .CS2BCR = 0x36DA3400, + .CS3BCR = 0x36DB4400, + .CS5aBCR = 0x17DF0400, + .CS0WCR = 0x000003C0, + .CS2WCR = 0x000003C0, + .CS3WCR = 0x000024D1, + .CS5aWCR = 0x000203C1 }, + /* CLOCK_SPEED_F2 */ + { .FLLFRQ = 0x00004000 + 900, + .FRQCR = (SH4_PLL_16x<<24)+(SH4_DIV_4<<20)+(SH4_DIV_8<<12)+(SH4_DIV_8<<8)+SH4_DIV_8, + .CS0BCR = 0x24920400, + .CS2BCR = 0x24923400, + .CS3BCR = 0x24924400, + .CS5aBCR = 0x17DF0400, + .CS0WCR = 0x00000340, + .CS2WCR = 0x000003C0, + .CS3WCR = 0x000024D1, + .CS5aWCR = 0x000203C1 }, + /* CLOCK_SPEED_F3 */ + { .FLLFRQ = 0x00004000 + 900, + .FRQCR = (SH4_PLL_26x<<24)+(SH4_DIV_4<<20)+(SH4_DIV_8<<12)+(SH4_DIV_8<<8)+SH4_DIV_8, + .CS0BCR = 0x24920400, + .CS2BCR = 0x24923400, + .CS3BCR = 0x24924400, + .CS5aBCR = 0x17DF0400, + .CS0WCR = 0x00000240, + .CS2WCR = 0x000003C0, + .CS3WCR = 0x000024D1, + .CS5aWCR = 0x000203C1 }, + /* CLOCK_SPEED_F4 */ + { .FLLFRQ = 0x00004000 + 900, + .FRQCR = (SH4_PLL_32x<<24)+(SH4_DIV_2<<20)+(SH4_DIV_4<<12)+(SH4_DIV_8<<8)+SH4_DIV_16, + .CS0BCR = 0x24920400, + .CS2BCR = 0x24923400, + .CS3BCR = 0x24924400, + .CS5aBCR = 0x17DF0400, + .CS0WCR = 0x000002C0, + .CS2WCR = 0x000003C0, + .CS3WCR = 0x000024D1, + .CS5aWCR = 0x000203C1 }, + /* CLOCK_SPEED_F5 is made identical to CLOCK_SPEED_F4 because clearly the + Graph Math+ cannot handle the higher bus speed. */ + { .FLLFRQ = 0x00004000 + 900, + .FRQCR = (SH4_PLL_32x<<24)+(SH4_DIV_2<<20)+(SH4_DIV_4<<12)+(SH4_DIV_8<<8)+SH4_DIV_16, + .CS0BCR = 0x24920400, + .CS2BCR = 0x24923400, + .CS3BCR = 0x24924400, + .CS5aBCR = 0x17DF0400, + .CS0WCR = 0x000002C0, + .CS2WCR = 0x000003C0, + .CS3WCR = 0x000024D1, + .CS5aWCR = 0x000203C1 }, +}; #endif static struct cpg_overclock_setting const *get_settings(void) @@ -451,6 +511,8 @@ static struct cpg_overclock_setting const *get_settings(void) return settings_prizm; if(gint[HWCALC] == HWCALC_FXCG50) return settings_fxcg50; + if(gint[HWCALC] == HWCALC_FXCG100) + return settings_fxcg100; #endif return NULL; diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index 3541e70..0a08e71 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -40,8 +40,8 @@ void *gint_stack_top = NULL; /* kinit(): Install and start gint */ void kinit(void) { - /* Figure out which CASIOWIN API to use based on the OS type. */ - #if GINT_OS_CG + /* Figure out which CASIOWIN API to use based on the OS type. */ + #if GINT_OS_CG char *version = (void *)0x80020020; if(!memcmp(version, "01.00", 5)) gint_set_CASIOWIN_API(1); From df140ff8468a5581cd33b964aa76ea32951d0795 Mon Sep 17 00:00:00 2001 From: Lephe Date: Fri, 10 Jan 2025 12:23:56 +0100 Subject: [PATCH 11/29] keysc: fix keycodes mapping to multiple keys on Math+ --- src/keysc/keydev.c | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/keysc/keydev.c b/src/keysc/keydev.c index 9e05003..207e2a7 100644 --- a/src/keysc/keydev.c +++ b/src/keysc/keydev.c @@ -79,8 +79,16 @@ static void keycode_to_keymatrix(int keycode, int *row, int *col) } } } + *row = keycode >> 4; *col = 7 - (keycode & 7); + + if(gint[HWCALC] == HWCALC_FXCG100 && + (*row > 4 || (*row == 1 && *col == 3))) { + // key that doesn't exist + *row = 0; + *col = 1; + } } #endif From 061b371ff1fb677f866ff329f972fd546434a9ca Mon Sep 17 00:00:00 2001 From: Lephe Date: Fri, 10 Jan 2025 12:24:28 +0100 Subject: [PATCH 12/29] keysc: add MPU module description for KEYSC --- include/gint/mpu/keysc.h | 34 ++++++++++++++++++++++++++++++++++ 1 file changed, 34 insertions(+) create mode 100644 include/gint/mpu/keysc.h diff --git a/include/gint/mpu/keysc.h b/include/gint/mpu/keysc.h new file mode 100644 index 0000000..ed6d932 --- /dev/null +++ b/include/gint/mpu/keysc.h @@ -0,0 +1,34 @@ +//--- +// gint:mpu:keysc - Key Scan Controller +//--- + +#ifndef GINT_MPU_KEYSC +#define GINT_MPU_KEYSC + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include + +typedef volatile struct { + uint16_t KIUDATA[6]; + uint16_t KIUCNTREG; + uint16_t KIAUTOFIXREG; + uint16_t KIUMODEREG; + uint16_t KIUSTATEREG; + uint16_t KIUINTREG; + uint16_t KIUWSETREG; + uint16_t KIUINTERVALREG; + uint16_t KOUTPINSET; + uint16_t KINPINSET; +} GPACKED(4) sh7305_keysc_t; + +#define SH7305_KEYSC (*(sh7305_keysc_t *)0xa44b0000) + +#ifdef __cplusplus +} +#endif + +#endif /* GINT_MPU_KEYSC */ From 3c36a2e14e856e515618db0a07068310f0fc33c6 Mon Sep 17 00:00:00 2001 From: Lephe Date: Tue, 28 Jan 2025 23:15:05 +0100 Subject: [PATCH 13/29] kernel: syscall support for Math+ OS v2.00 --- src/kernel/kernel.c | 2 ++ src/kernel/syscalls.S | 56 +++++++++++++++++++++---------------------- 2 files changed, 30 insertions(+), 28 deletions(-) diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index 0a08e71..7ef15eb 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -45,6 +45,8 @@ void kinit(void) char *version = (void *)0x80020020; if(!memcmp(version, "01.00", 5)) gint_set_CASIOWIN_API(1); + else if(gint[HWCALC] == HWCALC_FXCG100 && !memcmp(version, "02.00", 5)) + gint_set_CASIOWIN_API(2); else gint_set_CASIOWIN_API(0); #endif diff --git a/src/kernel/syscalls.S b/src/kernel/syscalls.S index 94bc19f..f9f0a06 100644 --- a/src/kernel/syscalls.S +++ b/src/kernel/syscalls.S @@ -144,7 +144,7 @@ syscall_table: #endif /* GINT_OS_FX */ #if GINT_OS_CG -#define CASIOWIN_API_VERSIONS 2 +#define CASIOWIN_API_VERSIONS 3 .global _gint_set_CASIOWIN_API .global _gint_get_CASIOWIN_API @@ -204,33 +204,33 @@ _gint_get_CASIOWIN_API: 3: .long .CASIOWIN_API .CASIOWIN_TABLE: - .long 0x1f44, 0x8025e0fc /* malloc */ - .long 0x1f42, 0x8025dec8 /* free */ - .long 0x1f46, 0x8025ec3c /* realloc */ - .long 0x1db4, 0x802404d2 /* BFile_Remove */ - .long 0x1db3, 0x80240482 /* BFile_Rename */ - .long 0x1dae, 0x802401b0 /* BFile_Create */ - .long 0x1da3, 0x8023fb90 /* BFile_Open */ - .long 0x1da4, 0x8023fd0e /* BFile_Close */ - .long 0x1da6, 0x8023fdc4 /* BFile_Size */ - .long 0x1da9, 0x8023ff2c /* BFile_Seek */ - .long 0x1dab, 0x8024003c /* BFile_GetPos */ - .long 0x1daf, 0x8024025e /* BFile_Write */ - .long 0x1dac, 0x80240082 /* BFile_Read */ - .long 0x1db6, 0x80240888 /* BFile_FindFirst */ - .long 0x1db8, 0x80240b06 /* BFile_FindNext */ - .long 0x1dba, 0x80240c10 /* BFile_FindClose */ - .long 0x08d9, 0x800b130c /* Timer_Install */ - .long 0x08db, 0x800b1456 /* Timer_Start */ - .long 0x08dc, 0x800b14b2 /* Timer_Stop */ - .long 0x08da, 0x800b13d4 /* Timer_Deinstall */ - .long 0x12c6, 0 /* PutKeyCode */ - .long 0x12bf, 0x8017be56 /* GetKeyWait */ - .long 0x12c7, 0 /* ClearKeyBuffer */ - .long 0x01e6, 0x8004579a /* GetVRAMAddress */ - .long 0x1e6e, 0 /* SetQuitHandler */ - .long 0x1839, 0 /* PowerOff */ - .long 0x1187, 0 /* Reset */ + .long 0x1f44, 0x8025e0fc, 0x80366708 /* malloc */ + .long 0x1f42, 0x8025dec8, 0x803664d4 /* free */ + .long 0x1f46, 0x8025ec3c, 0x803672c8 /* realloc */ + .long 0x1db4, 0x802404d2, 0x80334212 /* BFile_Remove */ + .long 0x1db3, 0x80240482, 0x803341c2 /* BFile_Rename */ + .long 0x1dae, 0x802401b0, 0x80333ef0 /* BFile_Create */ + .long 0x1da3, 0x8023fb90, 0x803338d0 /* BFile_Open */ + .long 0x1da4, 0x8023fd0e, 0x80333a4e /* BFile_Close */ + .long 0x1da6, 0x8023fdc4, 0x80333b04 /* BFile_Size */ + .long 0x1da9, 0x8023ff2c, 0x80333c6c /* BFile_Seek */ + .long 0x1dab, 0x8024003c, 0x80333d7c /* BFile_GetPos */ + .long 0x1daf, 0x8024025e, 0x80333f9e /* BFile_Write */ + .long 0x1dac, 0x80240082, 0x80333dc2 /* BFile_Read */ + .long 0x1db6, 0x80240888, 0x803345c8 /* BFile_FindFirst */ + .long 0x1db8, 0x80240b06, 0x80334846 /* BFile_FindNext */ + .long 0x1dba, 0x80240c10, 0x80334950 /* BFile_FindClose */ + .long 0x08d9, 0x800b130c, 0x8010de28 /* Timer_Install */ + .long 0x08db, 0x800b1456, 0x8010df72 /* Timer_Start */ + .long 0x08dc, 0x800b14b2, 0x8010dfce /* Timer_Stop */ + .long 0x08da, 0x800b13d4, 0x8010def0 /* Timer_Deinstall */ + .long 0x12c6, 0, 0 /* PutKeyCode */ + .long 0x12bf, 0x8017be56, 0x802382fe /* GetKeyWait */ + .long 0x12c7, 0, 0 /* ClearKeyBuffer */ + .long 0x01e6, 0x8004579a, 0x8007569e /* GetVRAMAddress */ + .long 0x1e6e, 0, 0 /* SetQuitHandler */ + .long 0x1839, 0, 0 /* PowerOff */ + .long 0x1187, 0, 0 /* Reset */ #define casiowin_call(id) bra _CASIOWIN_call; mov id, r0 From 3178aa69871f5315d5b7f635c715a7e332f33e09 Mon Sep 17 00:00:00 2001 From: CalcLoverHK <0v0katai@gmail.com> Date: Sun, 16 Feb 2025 17:04:27 +0800 Subject: [PATCH 14/29] cpg: add superhyway clock frequency calculation --- include/gint/clock.h | 2 ++ src/cpg/cpg.c | 3 +++ 2 files changed, 5 insertions(+) diff --git a/include/gint/clock.h b/include/gint/clock.h index acb5479..3236b93 100644 --- a/include/gint/clock.h +++ b/include/gint/clock.h @@ -35,6 +35,7 @@ typedef struct int Bphi_div; int Iphi_div; int Pphi_div; + int Sphi_div; union { int CKIO_f; @@ -44,6 +45,7 @@ typedef struct int Bphi_f; int Iphi_f; int Pphi_f; + int Sphi_f; } clock_frequency_t; diff --git a/src/cpg/cpg.c b/src/cpg/cpg.c index ae230b4..0d6ed07 100644 --- a/src/cpg/cpg.c +++ b/src/cpg/cpg.c @@ -84,10 +84,12 @@ static void sh7305_probe(void) int divb = CPG.FRQCR.BFC; int divi = CPG.FRQCR.IFC; int divp = CPG.FRQCR.P1FC; + int divs = CPG.FRQCR.SFC; freq.Bphi_div = 1 << (divb + 1); freq.Iphi_div = 1 << (divi + 1); freq.Pphi_div = 1 << (divp + 1); + freq.Sphi_div = 1 << (divs + 1); /* Deduce the input frequency of divider 1 */ int base = 32768; @@ -99,6 +101,7 @@ static void sh7305_probe(void) freq.Bphi_f = base >> (divb + 1); freq.Iphi_f = base >> (divi + 1); freq.Pphi_f = base >> (divp + 1); + freq.Sphi_f = base >> (divs + 1); } #undef CPG From 02b5f19cfc023b907feacdfca6f641cbcd74ead8 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Feb 2025 16:11:44 +0100 Subject: [PATCH 15/29] fix missing C_INVERT support in dline() for render-cg --- src/render-cg/gint_dline.c | 27 ++++++++++++++++++++------- src/render/render.h | 4 ++-- 2 files changed, 22 insertions(+), 9 deletions(-) diff --git a/src/render-cg/gint_dline.c b/src/render-cg/gint_dline.c index 18941ea..ff8bf53 100644 --- a/src/render-cg/gint_dline.c +++ b/src/render-cg/gint_dline.c @@ -4,7 +4,7 @@ #if GINT_RENDER_RGB /* gint_dhline(): Optimized horizontal line */ -void gint_dhline(int x1, int x2, int y, uint16_t color) +void gint_dhline(int x1, int x2, int y, int color) { if(y < dwindow.top || y >= dwindow.bottom) return; if(x1 > x2) swap(x1, x2); @@ -16,8 +16,13 @@ void gint_dhline(int x1, int x2, int y, uint16_t color) /* Use longwords to do the copy, but first paint the endpoints to heed for odd x1 and x2. Checking the parity may be a waste of time. */ - gint_vram[offset + x1] = color; - gint_vram[offset + x2] = color; + if (color != C_INVERT) { + gint_vram[offset + x1] = color; + gint_vram[offset + x2] = color; + } else { + gint_vram[offset + x1] ^= 0xffff; + gint_vram[offset + x2] ^= 0xffff; + } /* Now round to longword boundaries and copy everything in-between with longwords */ @@ -26,13 +31,17 @@ void gint_dhline(int x1, int x2, int y, uint16_t color) uint32_t *start = (void *)(gint_vram + offset + x1); uint32_t *end = (void *)(gint_vram + offset + x2); - uint32_t op = (color << 16) | color; - while(end > start) *--end = op; + if (color != C_INVERT) { + uint32_t op = (color << 16) | color; + while(end > start) *--end = op; + } else { + while(end > start) *--end ^= 0xffffffff; + } } /* gint_dvline(): Optimized vertical line */ -void gint_dvline(int y1, int y2, int x, uint16_t color) +void gint_dvline(int y1, int y2, int x, int color) { if(x < dwindow.left || x >= dwindow.right) return; if(y1 > y2) swap(y1, y2); @@ -42,7 +51,11 @@ void gint_dvline(int y1, int y2, int x, uint16_t color) uint16_t *v = gint_vram + DWIDTH * y1 + x; int height = y2 - y1 + 1; - while(height-- > 0) *v = color, v += DWIDTH; + if (color != C_INVERT) { + while(height-- > 0) *v = color, v += DWIDTH; + } else { + while(height-- > 0) *v ^= 0xffff, v += DWIDTH; + } } #endif diff --git a/src/render/render.h b/src/render/render.h index 888e439..a292801 100644 --- a/src/render/render.h +++ b/src/render/render.h @@ -11,12 +11,12 @@ /* gint_dhline(): Optimized horizontal line @x1 @x2 @y Coordinates of endpoints of line (both included) @color Any color suitable for dline() */ -void gint_dhline(int x1, int x2, int y, color_t color); +void gint_dhline(int x1, int x2, int y, int color); /* gint_dvline(): Optimized vertical line @y1 @y2 @x Coordinates of endpoints of line (both included) @color Any color suitable for dline() */ -void gint_dvline(int y1, int y2, int x, color_t color); +void gint_dvline(int y1, int y2, int x, int color); //--- // Font rendering (topti) From 10a2f2ecdec97b33ada1e498042bd67246720e2e Mon Sep 17 00:00:00 2001 From: Lephe Date: Wed, 5 Feb 2025 01:22:01 +0100 Subject: [PATCH 16/29] usb: add WCID support to have WinUSB driver automatically This should work on Windows Vista onwards. By specifying Windows OS 1.0 descriptors announcing compatibility with WinUSB, we get it as a driver plug-and-play style with no manual intervention (e.g. no Zadig). From there libusb can enumerate the device, which is awesome. --- src/usb/setup.c | 112 +++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 110 insertions(+), 2 deletions(-) diff --git a/src/usb/setup.c b/src/usb/setup.c index 82ac80e..8b800ea 100644 --- a/src/usb/setup.c +++ b/src/usb/setup.c @@ -33,7 +33,9 @@ static usb_dc_device_t dc_device = { .bMaxPacketSize0 = 64, .idVendor = htole16(0x07cf), /* Casio Computer Co., Ltd. */ .idProduct = htole16(ID_PRODUCT), - .bcdDevice = htole16(0x0100), + /* CASIO sets bcdDevice to 0x0100. Use a different value so that Windows + won't rely on cached registry entries and check for WCID support. */ + .bcdDevice = htole16(0x0177), .iManufacturer = 0, .iProduct = 0, .iSerialNumber = 0, @@ -55,6 +57,65 @@ static usb_dc_string_t dc_string0 = { .data = { htole16(0x0409) }, /* English (US) */ }; +struct MOS1_String_0xEE_Descriptor { + usb_dc_string_t dc; + u16 signature[7]; + u8 bMS_VendorCode; + u8 bPad; +}; +static struct MOS1_String_0xEE_Descriptor dc_string_ee = { + .dc = { .bLength = 18, .bDescriptorType = USB_DC_STRING }, + .signature = { 0x4D00, 0x5300, 0x4600, 0x5400, 0x3100, 0x3000, 0x3000 }, + .bMS_VendorCode = 0x7b, /* Arbitrarily-chosen value */ + .bPad = 0, +}; + +struct MOS1_Extended_Compat_ID_Function_Section { + u8 bFirstInterfaceNumber; + u8 bReserved1; + u8 compatibleID[8]; + u8 subcompatibleID[8]; + u8 bReserved2[6]; +}; +struct MOS1_Extended_Compat_ID_Descriptor { + // 16 byte header + u32 dwLength; + u16 bcdVersion; + u16 wIndex; + u8 bCount; + u8 bReserved[7]; + + // hardcoded to size 1 for convenience here + struct MOS1_Extended_Compat_ID_Function_Section f; +}; +static struct MOS1_Extended_Compat_ID_Descriptor dc_compatid = { + .dwLength = htole32(0x10 + 24 * 1), + .bcdVersion = htole16(0x0100), + .wIndex = htole16(0x0004), + .bCount = 1, + .f = { + .bFirstInterfaceNumber = 0, + .bReserved1 = 0x01, + .compatibleID = { 0x57, 0x49, 0x4e, 0x55, 0x53, 0x42, 0x00, 0x00 }, + .subcompatibleID = { 0 }, + .bReserved2 = { 0 }, + }, +}; + +struct MOS1_Extended_Properties_Descriptor { + // 10 byte header + u32 dwLength; + u16 bcdVersion; + u16 wIndex; + u16 wCount; +}; +static struct MOS1_Extended_Properties_Descriptor dc_extprops = { + .dwLength = htole32(10), + .bcdVersion = htole16(0x0100), + .wIndex = htole16(0x0005), + .wCount = htole16(0), +}; + GCONSTRUCTOR static void set_strings(void) { char const *serial_base = @@ -139,12 +200,40 @@ static void req_get_descriptor(int wValue, int wLength) else if(type == USB_DC_STRING) { - usb_dc_string_t *dc = usb_dc_string_get(num); + usb_dc_string_t *dc; + if(num == 0xee) { + dc = &dc_string_ee.dc; + USB_LOG("Selecting MOS1 string descriptor\n"); + } + else + dc = usb_dc_string_get(num); if(dc) dcp_write(dc, dc->bLength); else USB.DCPCTR.PID = 2; } } +static void req_get_MOS_feature_descriptor(int wValue, int wIndex, int wLength) +{ + int intf = (wValue >> 8) & 0xff; + int page = (wValue & 0xff); + + USB_LOG("GET_MS_DESCRIPTOR: i%d p%d #%d len:%d\n", intf, page, wIndex, wLength); + + if(wIndex == 0x0004) { + /* Extended Compat ID descriptor */ + dcp_write(&dc_compatid, wLength); + USB_LOG("Compat ID descriptor given\n"); + } + else if(wIndex == 0x0005) { + /* Extended Properties descriptor */ + dcp_write(&dc_extprops, wLength); + USB_LOG("Extended properties descriptor given\n"); + } + else { + USB_LOG("Unknown MS descriptor!\n"); + } +} + static void req_get_configuration(void) { USB_LOG("GET_CONFIGURATION -> %d\n", 1); @@ -158,6 +247,12 @@ static void req_set_configuration(int wValue) USB.DCPCTR.PID = (wValue == 1) ? 1 : 2; } +static void req_get_device_status(void) +{ + USB_LOG("GET_STATUS device -> 0x0001\n"); + dcp_write("\x01\x00", 2); +} + void usb_req_setup(void) { /* Respond to setup requests */ @@ -181,6 +276,19 @@ void usb_req_setup(void) else if(bmRequestType == 0x00 && bRequest == SET_CONFIGURATION) req_set_configuration(wValue); + else if(bmRequestType == 0x80 && bRequest == GET_STATUS) + req_get_device_status(); + // 0x81 / GET_STATUS : get intf status + // 0x82 / GET_STATUS : get endpoint status + + // CESG502 initial 0x01 comm + // else if(bmRequestType == 0x41 && bRequest == 0x01) + // USB.DCPCTR.PID = 1; + + else if((bmRequestType == 0xc0 || bmRequestType == 0xc1) + && bRequest == 0x7b) + req_get_MOS_feature_descriptor(wValue, wIndex, wLength); + /* TODO: Other standard SETUP requests */ else USB_LOG("SETUP: bRequest=%02x bmRequestType=%02x wValue=%04x\n" " wIndex=%04x wLength=%d -> ???\n", From 814c7bc3e22b16102acdbe8381be5f090f470011 Mon Sep 17 00:00:00 2001 From: Lephe Date: Thu, 20 Feb 2025 19:16:33 +0100 Subject: [PATCH 17/29] cpg: allow overclock settings to be made permanent --- include/gint/clock.h | 5 +++++ src/cpg/overclock.c | 12 ++++++++++++ 2 files changed, 17 insertions(+) diff --git a/include/gint/clock.h b/include/gint/clock.h index 3236b93..7a8c4f5 100644 --- a/include/gint/clock.h +++ b/include/gint/clock.h @@ -157,6 +157,11 @@ void cpg_get_overclock_setting(struct cpg_overclock_setting *s); /* Applies the specified overclock setting. */ void cpg_set_overclock_setting(struct cpg_overclock_setting const *s); +/* Sets whether or node CPG parameters are retained when world switching, i.e. + mostly whether overclock setting will remain while in the OS world and when + leaving the add-in. */ +void cpg_set_overclock_permanent(bool permanent); + //--- // Sleep functions //--- diff --git a/src/cpg/overclock.c b/src/cpg/overclock.c index 396a5cb..bf9b416 100644 --- a/src/cpg/overclock.c +++ b/src/cpg/overclock.c @@ -12,6 +12,7 @@ #include #include #include +#include #include #include #include @@ -142,6 +143,17 @@ void cpg_set_overclock_setting(struct cpg_overclock_setting const *s) cpu_atomic_end(); } +void cpg_set_overclock_permanent(bool permanent) +{ + extern gint_driver_t drv_cpg; + int i = &drv_cpg - gint_drivers; + + if(permanent) + gint_driver_flags[i] |= GINT_DRV_SHARED; + else + gint_driver_flags[i] &= ~GINT_DRV_SHARED; +} + #if GINT_HW_FX static struct cpg_overclock_setting const settings_fx9860g_sh3[5] = { From 5a74fbac4d92159f3f2c9be8dd3287d47a57e62e Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Sat, 22 Feb 2025 15:27:56 +0100 Subject: [PATCH 18/29] dline: fix odd x1/x2 handling with C_INVERT color (render-cg) --- src/render-cg/gint_dline.c | 43 +++++++++++++++++++++----------------- 1 file changed, 24 insertions(+), 19 deletions(-) diff --git a/src/render-cg/gint_dline.c b/src/render-cg/gint_dline.c index ff8bf53..8bdafa3 100644 --- a/src/render-cg/gint_dline.c +++ b/src/render-cg/gint_dline.c @@ -15,14 +15,19 @@ void gint_dhline(int x1, int x2, int y, int color) int offset = DWIDTH * y; /* Use longwords to do the copy, but first paint the endpoints to heed - for odd x1 and x2. Checking the parity may be a waste of time. */ - if (color != C_INVERT) { - gint_vram[offset + x1] = color; - gint_vram[offset + x2] = color; - } else { - gint_vram[offset + x1] ^= 0xffff; - gint_vram[offset + x2] ^= 0xffff; - } + for odd x1 and x2. Checking the parity may be a waste of time for + "real" color, but must be checked when C_INVERT in involved to + avoid "cancelling" invert effect with potential overdraw of the + next operation. */ + if (color != C_INVERT) { + gint_vram[offset + x1] = color; + gint_vram[offset + x2] = color; + } else { + if (x1 & 1) + gint_vram[offset + x1] ^= 0xffff; + if (x2 & 1) + gint_vram[offset + x2] ^= 0xffff; + } /* Now round to longword boundaries and copy everything in-between with longwords */ @@ -32,12 +37,12 @@ void gint_dhline(int x1, int x2, int y, int color) uint32_t *start = (void *)(gint_vram + offset + x1); uint32_t *end = (void *)(gint_vram + offset + x2); - if (color != C_INVERT) { - uint32_t op = (color << 16) | color; - while(end > start) *--end = op; - } else { - while(end > start) *--end ^= 0xffffffff; - } + if (color != C_INVERT) { + uint32_t op = (color << 16) | color; + while(end > start) *--end = op; + } else { + while(end > start) *--end ^= 0xffffffff; + } } /* gint_dvline(): Optimized vertical line */ @@ -51,11 +56,11 @@ void gint_dvline(int y1, int y2, int x, int color) uint16_t *v = gint_vram + DWIDTH * y1 + x; int height = y2 - y1 + 1; - if (color != C_INVERT) { - while(height-- > 0) *v = color, v += DWIDTH; - } else { - while(height-- > 0) *v ^= 0xffff, v += DWIDTH; - } + if (color != C_INVERT) { + while(height-- > 0) *v = color, v += DWIDTH; + } else { + while(height-- > 0) *v ^= 0xffff, v += DWIDTH; + } } #endif From 66f173bd1117e6f5cd675f834ade204364f951d1 Mon Sep 17 00:00:00 2001 From: Lephe Date: Mon, 24 Feb 2025 19:41:28 +0100 Subject: [PATCH 19/29] kernel: get arenas from MPM load info --- src/kernel/kernel.c | 33 ++++++++++++++++++++++++++++++++- src/kernel/kernel.h | 4 +++- src/kernel/start.c | 16 ++++++++-------- 3 files changed, 43 insertions(+), 10 deletions(-) diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c index 7ef15eb..dffd482 100644 --- a/src/kernel/kernel.c +++ b/src/kernel/kernel.c @@ -33,13 +33,17 @@ uint8_t *gint_driver_flags = NULL; /* Top of the stack */ void *gint_stack_top = NULL; +u32 *gint_load_info = NULL; + //--- // Initialization and unloading //--- /* kinit(): Install and start gint */ -void kinit(void) +void kinit(u32 *load_info) { + gint_load_info = load_info; + /* Figure out which CASIOWIN API to use based on the OS type. */ #if GINT_OS_CG char *version = (void *)0x80020020; @@ -134,6 +138,33 @@ void kinit(void) } #endif + /* Create arenas in locations provided by loader */ + #if GINT_OS_CG + u32 load_arena_start = 0; + char const *names[4] = { "_ld1", "_ld2", "_ld3", "_ld4" }; + int count = 0; + for(int i = 0; load_info && load_info[i] && count < 4; i += 2) + { + if(load_info[i] == 0x00000010) + load_arena_start = load_info[i+1]; + else if(load_info[i] == 0x00000011 && load_arena_start) + { + u32 load_arena_end = load_info[i+1]; + kmalloc_arena_t *arena = kmalloc(sizeof *arena, NULL); + if(arena) + { + arena->name = names[count]; + arena->is_default = true; + arena->start = (void *)load_arena_start; + arena->end = (void *)load_arena_end; + kmalloc_init_arena(arena, true); + kmalloc_add_arena(arena); + } + load_arena_start = 0; + } + } + #endif + /* Allocate world buffers for the OS and for gint */ gint_world_os = gint_world_alloc(); gint_world_addin = gint_world_alloc(); diff --git a/src/kernel/kernel.h b/src/kernel/kernel.h index a5b5952..717345b 100644 --- a/src/kernel/kernel.h +++ b/src/kernel/kernel.h @@ -5,6 +5,8 @@ #ifndef GINT_CORE_KERNEL #define GINT_CORE_KERNEL +#include + /* gint_load_onchip_sections(): Initialize on-chip memory sections */ void gint_load_onchip_sections(void); @@ -13,7 +15,7 @@ void gint_load_onchip_sections(void); void gint_copy_vram(void); /* kinit(): Install and start gint */ -void kinit(void); +void kinit(u32 *load_info); /* kquit(): Quit gint and give back control to the system */ void kquit(void); diff --git a/src/kernel/start.c b/src/kernel/start.c index 5159a92..bd0fc9e 100644 --- a/src/kernel/start.c +++ b/src/kernel/start.c @@ -109,7 +109,7 @@ void gint_load_onchip_sections(void) } } -static int start2(int isappli, int optnum) +static int start2(int load_type, u32 *load_info) { /* We are currently in a dynamic userspace mapping of an add-in run from the storage memory. We are running in privileged mode with one @@ -174,8 +174,11 @@ static int start2(int isappli, int optnum) } #endif - /* Install gint, switch VBR and initialize drivers */ - kinit(); + /* Install gint, switch VBR and initialize drivers. If loading information + was provided, pass it on. */ + if(load_type == 0x4d504d30 /* 'MPM0' */ && !((u32)load_info & 3)) {} + else load_info = NULL; + kinit(load_info); /* We are now running on our own in kernel mode. Since we have taken control of interrupts, pretty much any interaction with the system @@ -190,9 +193,6 @@ static int start2(int isappli, int optnum) what it wants in exit() after main() finishes executing */ if(!setjmp(gint_exitbuf)) { callarray(&bctors, &ectors); - // TODO: record isappli and optnum in globals - (void)isappli; - (void)optnum; exit(main()); } else { @@ -211,11 +211,11 @@ static int start2(int isappli, int optnum) } GSECTION(".text.entry") -int start(int isappli, int optnum) +int start(int load_type, u32 *load_info) { int rc; while(1) { - rc = start2(isappli, optnum); + rc = start2(load_type, load_info); if(!gint_restart) break; gint_osmenu_native(); } From 302aeb5cdf9d4f3bc9d8ce030c99d2f298029765 Mon Sep 17 00:00:00 2001 From: Lephenixnoir Date: Tue, 11 Mar 2025 15:32:37 +0100 Subject: [PATCH 20/29] cpg: fix incorrect access size to SDMR3_CL2 and SDMR3_CL3 --- src/cpg/overclock.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/cpg/overclock.c b/src/cpg/overclock.c index bf9b416..ac65ff2 100644 --- a/src/cpg/overclock.c +++ b/src/cpg/overclock.c @@ -21,8 +21,8 @@ // Low-level clock speed access //--- -#define SH7305_SDMR3_CL2 ((volatile uint8_t *)0xFEC15040) -#define SH7305_SDMR3_CL3 ((volatile uint8_t *)0xFEC15060) +#define SH7305_SDMR3_CL2 ((volatile uint16_t *)0xFEC15040) +#define SH7305_SDMR3_CL3 ((volatile uint16_t *)0xFEC15060) //--- // Predefined clock speeds From ceaf8b27f8c06b6e10b48120cbb26f153bf99aa9 Mon Sep 17 00:00:00 2001 From: Lephe Date: Sat, 15 Mar 2025 17:08:29 +0100 Subject: [PATCH 21/29] r61523: add partial update function --- include/gint/drivers/r61523.h | 4 ++++ src/r61523/r61523.c | 18 +++++++++++++++++- 2 files changed, 21 insertions(+), 1 deletion(-) diff --git a/include/gint/drivers/r61523.h b/include/gint/drivers/r61523.h index c4b5b3c..70d9fa1 100644 --- a/include/gint/drivers/r61523.h +++ b/include/gint/drivers/r61523.h @@ -16,6 +16,10 @@ extern "C" { /* r61523_display(): Update the entire display (320x528) */ void r61523_display(uint16_t *vram); +/* r61523_display_rect(): Update a rectangle (both bounds included). */ +void r61523_display_rect( + uint16_t *vram, int xmin, int xmax, int ymin, int ymax); + /* r61523_win_set(): Set the display window */ void r61523_win_set(int x1, int x2, int y1, int y2); diff --git a/src/r61523/r61523.c b/src/r61523/r61523.c index 598d236..41ce85d 100644 --- a/src/r61523/r61523.c +++ b/src/r61523/r61523.c @@ -109,7 +109,6 @@ void r61523_win_set(int x1, int x2, int y1, int y2) void r61523_display(uint16_t *vram) { r61523_win_set(0, 319, 0, 527); - select(44); int row_offset = 0; @@ -125,6 +124,23 @@ void r61523_display(uint16_t *vram) } } +void r61523_display_rect( + uint16_t *vram, int xmin, int xmax, int ymin, int ymax) +{ + // dma_transfer_wait(0); + r61523_win_set(xmin, xmax, ymin, ymax); + select(44); + + vram += 320 * ymin + xmin; + uint16_t volatile *DISPLAY = (void *)0xb4000000; + + for(int y = 0; y < ymax - ymin + 1; y++) { + for(int x = 0; x < xmax - xmin + 1; x++) + *DISPLAY = vram[x]; + vram += 320; + } +} + static bool r61523_update(int x, int y, image_t const *fb, int flags) { if(fb->format != IMAGE_RGB565) From 1c97d613f8fc90654adb01767b117cea9ab6a75d Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Fri, 21 Mar 2025 09:30:54 +0100 Subject: [PATCH 22/29] r61523: fix newer screens variant --- src/r61523/r61523.c | 64 ++++++++++++++++++++++++++++++++++++--------- 1 file changed, 51 insertions(+), 13 deletions(-) diff --git a/src/r61523/r61523.c b/src/r61523/r61523.c index 41ce85d..827e63c 100644 --- a/src/r61523/r61523.c +++ b/src/r61523/r61523.c @@ -17,12 +17,18 @@ #define REG_VRANGE 0x2b #define REG_DATA 0x2c #define REG_DEVICE_CODE_READ 0xbf +#define REG_DEVICE_CODE_VARIANT 0xda /* Interface with the controller */ static volatile uint16_t *DISPLAY = (void *)0xb4000000; /* Bit 4 of Port R controls the RS bit of the display driver */ static volatile uint8_t *PRDR = (void *)0xa405013c; +/* Screen variant information + * This number is 0x00 for the old R61523 that everyone knows, and 0x16 or + * 0x52 for the newer ones found in recent FXCP400 devices. */ +static uint16_t r61523_variant = 0; + /* Select a register */ GINLINE static void select(uint16_t reg) { @@ -56,16 +62,38 @@ static void read_Nu16(uint16_t *array, int N) // Generic functions //--- -void r61523_identify(uint32_t *manufacturerCode, uint16_t *deviceCode) -{ - select(REG_DEVICE_CODE_READ); +/* r61523_identify() - identify screen hardware information + * + * notes + * Since recent versions of the FXCP400, new screens are used that break + * the current driver. These screens are particular since they have the + * same manufacturer / device code, but they use an undocumented register + * (0xda) to determine the variant. + * + * The register 0xda should be read twice. Casio ignores the first reading + * and only keep track of the second. So, let's do the same here. */ +void r61523_identify( + uint32_t *manufacturerCode, + uint16_t *deviceCode, + uint16_t *variant +) { uint16_t packets[5]; - read_Nu16(packets, 5); - if(manufacturerCode) - *manufacturerCode = (packets[1] << 16) | packets[2]; - if(deviceCode) - *deviceCode = (packets[3] << 16) | packets[4]; + if (manufacturerCode != NULL || deviceCode != NULL) + { + select(REG_DEVICE_CODE_READ); + read_Nu16(packets, 5); + if(manufacturerCode) + *manufacturerCode = (packets[1] << 16) | packets[2]; + if(deviceCode) + *deviceCode = (packets[3] << 16) | packets[4]; + } + if (variant) + { + select(REG_DEVICE_CODE_VARIANT); + *variant = read(); + *variant = read(); + } } //--- @@ -74,10 +102,13 @@ void r61523_identify(uint32_t *manufacturerCode, uint16_t *deviceCode) void r61523_win_set(int x1, int x2, int y1, int y2) { - /* R61523 has a 360x640 area; the CP-400 uses the top-right corner for its - 320x528 display, so skip over the first 40 columns */ - x1 += 40; - x2 += 40; + if (r61523_variant != 0x16 || r61523_variant != 0x52) + { + /* R61523 has a 360x640 area; the CP-400 uses the top-right corner + * for its 320x528 display, so skip over the first 40 columns */ + x1 += 40; + x2 += 40; + } uint16_t volatile *DISPLAY = (void *)0xb4000000; @@ -168,10 +199,17 @@ static bool r61523_update(int x, int y, image_t const *fb, int flags) // Driver metadata //--- +/* constructor() - determine which variant of screen we have */ +static void constructor(void) +{ + r61523_identify(NULL, NULL, &r61523_variant); +} + /* As far as I can tell there's no way to read the current window from the controller so this driver is completely stateless for now. */ gint_driver_t drv_r61523 = { - .name = "R61523", + .name = "R61523", + .constructor = constructor, }; GINT_DECLARE_DRIVER(26, drv_r61523); From c2bfa302a06189ca6828f8a6087f2d4fd74ca44b Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Mon, 17 Mar 2025 15:07:29 +0100 Subject: [PATCH 23/29] 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 21086bcb4c1e4d979a74044b9d6a44101e95b676 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Tue, 18 Mar 2025 11:42:22 +0100 Subject: [PATCH 24/29] 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 2e27c535ed02ca822dd2cc6ab7f9a7fd0faf6970 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 10:29:03 +0100 Subject: [PATCH 25/29] 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 718957b4bba6a1def9d839a92bf244514f7b56e2 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 14:39:05 +0100 Subject: [PATCH 26/29] 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 7d4cfe7..3734b76 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 cc802b60253a947bc0bbe823da7a66e4b8a87bf5 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Wed, 19 Mar 2025 15:56:46 +0100 Subject: [PATCH 27/29] 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) From f1b441eeef5ae78edadfd4fad2fece61f856d0f8 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Sun, 23 Mar 2025 17:16:54 +0100 Subject: [PATCH 28/29] touch: code cleanup --- include/gint/intc.h | 5 +++ include/gint/mpu/i2c.h | 74 ++++++++++++++++++++++++++++++++++++++++++ include/gint/touch.h | 10 +++--- src/intc/intc.c | 5 +++ src/touch/adconv.c | 34 +++++++++---------- src/touch/adconv.h | 12 +++---- src/touch/driver.c | 22 ++++++------- src/touch/driver.h | 2 +- src/touch/i2c.c | 25 ++++++-------- src/touch/i2c.h | 63 +++-------------------------------- src/touch/i2c_inth.c | 2 +- src/touch/touch.c | 18 +++++----- 12 files changed, 149 insertions(+), 123 deletions(-) create mode 100644 include/gint/mpu/i2c.h diff --git a/include/gint/intc.h b/include/gint/intc.h index 1f43ae2..e2db631 100644 --- a/include/gint/intc.h +++ b/include/gint/intc.h @@ -52,6 +52,11 @@ enum { INTC_SPU_DSP1, /* USB communication */ INTC_USB, + /* I2C */ + INTC_I2C_AL, + INTC_I2C_TACK, + INTC_I2C_WAIT, + INTC_I2C_DTE, }; //--- diff --git a/include/gint/mpu/i2c.h b/include/gint/mpu/i2c.h new file mode 100644 index 0000000..b759585 --- /dev/null +++ b/include/gint/mpu/i2c.h @@ -0,0 +1,74 @@ +#ifndef GINT_MPU_I2C_H +#define GINT_MPU_I2C_H 1 + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include + +//--- +// SH7305 I2C Bus Interface. Refer to: +// "Renesas SH7724 User's Manual: Hardware" +// Section 32: "I2C Bus Interface (I2C)" +//--- + +/* sh7305_i2c_t - I2C peripheral definition */ +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)) + +#ifdef __cplusplus +} +#endif + +#endif /* GINT_MPU_I2C_H */ diff --git a/include/gint/touch.h b/include/gint/touch.h index 1ed5d21..51e353f 100644 --- a/include/gint/touch.h +++ b/include/gint/touch.h @@ -10,20 +10,20 @@ extern "C" { /* touch_calib - tounch-screen calibration information */ typedef struct { - int x_mul; + int x_base; int x_div; - int y_mul; + int y_base; int y_div; int dual_debounce_frame; int dual_sensi_entry; int dual_sensi_leave; -} touch_calib; +} touch_calibration_t; /* touch_calib_get() - get calibration information */ -extern int touch_calib_get(touch_calib *calib); +extern int touch_calib_get(touch_calibration_t *calib); /* touch_calib_set() - set calibration information */ -extern int touch_calib_set(touch_calib *calib); +extern int touch_calib_set(touch_calibration_t *calib); // low-level API diff --git a/src/intc/intc.c b/src/intc/intc.c index caba6a2..b13fccb 100644 --- a/src/intc/intc.c +++ b/src/intc/intc.c @@ -71,6 +71,11 @@ static struct info { { IPRC, 0x000f, IMR4, 0x08, _ }, /* USB */ { IPRF, 0x00f0, IMR9, 0x02, _ /* Driver not SH3-compatible yet */ }, + /* I2C */ + { IPRH, 0x000f, IMR7, 0x10, _ }, + { IPRH, 0x000f, IMR7, 0x20, _ }, + { IPRH, 0x000f, IMR7, 0x40, _ }, + { IPRH, 0x000f, IMR7, 0x80, _ }, }; /* Compact SH3 VBR-space scheme diff --git a/src/touch/adconv.c b/src/touch/adconv.c index 44c1ed8..66dad73 100644 --- a/src/touch/adconv.c +++ b/src/touch/adconv.c @@ -1,5 +1,5 @@ //--- -// gint:touchscreen:adconv - 0x84 register data conversion +// gint:touch:adconv - 0x84 register data conversion //--- #include @@ -49,8 +49,8 @@ static int _adconv_check_dual( // Public //--- -/* touchscreen_adconv_get_raw() - read 0x84 register using I2C */ -int touchscreen_adconv_get_raw(struct _touch_adraw *adraw) +/* touch_adconv_get_raw() - read 0x84 register using I2C */ +int touch_adconv_get_raw(struct _touch_adraw *adraw) { volatile uint8_t *IO_PRDR = (void*)0xa405013c; @@ -72,8 +72,8 @@ int touchscreen_adconv_get_raw(struct _touch_adraw *adraw) return 1; } -/* touchscreen_adconv_convert() - perform the raw conversion */ -int touchscreen_adconv_get_conv( +/* touch_adconv_convert() - perform the raw conversion */ +int touch_adconv_get_conv( struct _touch_adconv *adconv, struct _touch_adraw *adraw, int type @@ -90,22 +90,22 @@ int touchscreen_adconv_get_conv( return 2; } -/* touchscreen_adconv_get_dots() - generate dots information */ -int touchscreen_adconv_get_dots( +/* touch_adconv_get_dots() - generate dots information */ +int touch_adconv_get_dots( struct _touch_addots *dots, struct _touch_adconv *adconv, int type ) { int x_div; - int x_mul; + int x_base; int y_div; - int y_mul; + int y_base; cpu_atomic_start(); x_div = __touch_drv_info.calibration.x_div; - x_mul = __touch_drv_info.calibration.x_mul; + x_base = __touch_drv_info.calibration.x_base; y_div = __touch_drv_info.calibration.y_div; - y_mul = __touch_drv_info.calibration.y_mul; + y_base = __touch_drv_info.calibration.y_base; switch (type) { case 0: @@ -119,8 +119,8 @@ int touchscreen_adconv_get_dots( break; case 1: dots->type = TS_DOTS_TYPE_SINGLE; - dots->x1 = ((adconv->x1 - x_mul) * 0x100) / x_div; - dots->y1 = ((adconv->y1 - y_mul) * 0x100) / y_div; + dots->x1 = ((adconv->x1 - x_base) * 0x100) / x_div; + dots->y1 = ((adconv->y1 - y_base) * 0x100) / y_div; dots->z1 = adconv->z1; dots->x2 = 0; dots->y2 = 0; @@ -128,11 +128,11 @@ int touchscreen_adconv_get_dots( break; case 2: dots->type = TS_DOTS_TYPE_DUAL; - dots->x1 = ((adconv->x1 - x_mul) * 0x100) / x_div; - dots->y1 = ((adconv->y1 - y_mul) * 0x100) / y_div; + dots->x1 = ((adconv->x1 - x_base) * 0x100) / x_div; + dots->y1 = ((adconv->y1 - y_base) * 0x100) / y_div; dots->z1 = adconv->z1; - dots->x2 = ((adconv->x2 - x_mul) * 0x100) / x_div; - dots->y2 = ((adconv->y2 - y_mul) * 0x100) / y_div; + dots->x2 = ((adconv->x2 - x_base) * 0x100) / x_div; + dots->y2 = ((adconv->y2 - y_base) * 0x100) / y_div; dots->z2 = adconv->z2; break; } diff --git a/src/touch/adconv.h b/src/touch/adconv.h index 50e04a2..c95d6a5 100644 --- a/src/touch/adconv.h +++ b/src/touch/adconv.h @@ -57,18 +57,18 @@ struct _touch_addots // Public //--- -/* touchscreen_adconv_get_raw() - read 0x84 register using I2C */ -extern int touchscreen_adconv_get_raw(struct _touch_adraw *adraw); +/* touch_adconv_get_raw() - read 0x84 register using I2C */ +extern int touch_adconv_get_raw(struct _touch_adraw *adraw); -/* touchscreen_adconv_get_conv() - perform the raw conversion */ -extern int touchscreen_adconv_get_conv( +/* touch_adconv_get_conv() - perform the raw conversion */ +extern int touch_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( +/* touch_adconv_get_dots() - generate dots information */ +extern int touch_adconv_get_dots( struct _touch_addots *dots, struct _touch_adconv *adconv, int type diff --git a/src/touch/driver.c b/src/touch/driver.c index 1ea21a8..7df11ce 100644 --- a/src/touch/driver.c +++ b/src/touch/driver.c @@ -27,9 +27,9 @@ static void _touch_configure(void) __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_base = 0x20b; __touch_drv_info.calibration.x_div = 0x9b6; - __touch_drv_info.calibration.y_mul = 0x0f4; + __touch_drv_info.calibration.y_base = 0x0f4; __touch_drv_info.calibration.y_div = 0x66f; __touch_drv_info.calibration.dual_debounce_frame = 0; __touch_drv_info.calibration.dual_sensi_entry = 0x18; @@ -72,16 +72,16 @@ static void _touch_hpoweroff(void) i2c_hpoweroff(); } -/* _touch_unbin() - unbind from gint to casio */ -static void _touch_unbin(void) +/* _touch_unbind() - unbind from gint to casio */ +static void _touch_unbind(void) { - i2c_unbin(); + i2c_unbind(); } -/* _touch_funbin() - funbind from casio to gint */ -static void _touch_funbin(void) +/* _touch_funbind() - funbind from casio to gint */ +static void _touch_funbind(void) { - i2c_funbin(); + i2c_funbind(); } //--- @@ -100,8 +100,8 @@ gint_driver_t drv_touch = { .hpowered = _touch_hpowered, .hpoweron = _touch_hpoweron, .hpoweroff = _touch_hpoweroff, - .unbind = _touch_unbin, - .funbind = _touch_funbin, + .unbind = _touch_unbind, + .funbind = _touch_funbind, .state_size = sizeof(touch_state_t), }; -GINT_DECLARE_DRIVER(16, drv_touch); +GINT_DECLARE_DRIVER(24, drv_touch); diff --git a/src/touch/driver.h b/src/touch/driver.h index d6caa95..fb3f422 100644 --- a/src/touch/driver.h +++ b/src/touch/driver.h @@ -7,7 +7,7 @@ /* _touch_drv_info() - internal driver information */ struct _touch_drv_info { - touch_calib calibration; + touch_calibration_t calibration; key_event_t prev_evt; struct { bool prev_is_dual; diff --git a/src/touch/i2c.c b/src/touch/i2c.c index f4f7d07..5894569 100644 --- a/src/touch/i2c.c +++ b/src/touch/i2c.c @@ -1,5 +1,5 @@ //--- -// gint:touchscreen:i2c - I2C driver +// gint:touch:i2c - I2C driver //--- #include @@ -45,14 +45,7 @@ static void _i2c_hw_start_operation(void) 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; + SH7305_I2C.ICCR.byte = 0x94; } /* i2c_request_await() - await async operation */ @@ -228,18 +221,20 @@ void i2c_configure(void) __i2c_request.status = I2C_REQ_STATUS_FINISHED; // Enable interrupt - SH7305_INTC._->IPRH.I2C = 1; - SH7305_INTC.MSKCLR->IMR7 |= 0xf0; + intc_priority(INTC_I2C_AL, 1); + intc_priority(INTC_I2C_TACK, 1); + intc_priority(INTC_I2C_WAIT, 1); + intc_priority(INTC_I2C_DTE, 1); } -/* i2c_unbin() - unbind from gint to casio */ -void i2c_unbin(void) +/* i2c_unbind() - unbind from gint to casio */ +void i2c_unbind(void) { _i2c_request_await(); } -/* i2c_funbin() - funbind from casio to gint */ -void i2c_funbin(void) +/* i2c_funbind() - funbind from casio to gint */ +void i2c_funbind(void) { if (i2c_hpowered() == false) return; diff --git a/src/touch/i2c.h b/src/touch/i2c.h index 19abdee..a442c48 100644 --- a/src/touch/i2c.h +++ b/src/touch/i2c.h @@ -66,61 +66,8 @@ 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)) - -// state and world-switch - #include +#include /* i2c_configure() - driver/hardware configuration */ extern void i2c_configure(void); @@ -140,10 +87,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_funbind() - funbind from casio to gint */ +extern void i2c_funbind(void); -/* i2c_unbin() - unbind from gint to casio */ -extern void i2c_unbin(void); +/* i2c_unbind() - unbind from gint to casio */ +extern void i2c_unbind(void); #endif /* GINT_TOUCH_I2C_H */ diff --git a/src/touch/i2c_inth.c b/src/touch/i2c_inth.c index 6b12ce0..1b59809 100644 --- a/src/touch/i2c_inth.c +++ b/src/touch/i2c_inth.c @@ -1,5 +1,5 @@ //--- -// gint:touchscreen:i2c_inth - I2C interrupt handlers +// gint:touch:i2c_inth - I2C interrupt handlers //--- #include diff --git a/src/touch/touch.c b/src/touch/touch.c index bd2fa28..2cfe2f8 100644 --- a/src/touch/touch.c +++ b/src/touch/touch.c @@ -1,5 +1,5 @@ //--- -// gint:touch - touchscreen driver (high-level) +// gint:touch - touch driver (high-level) //---- #include @@ -24,26 +24,26 @@ extern struct _touch_drv_info __touch_drv_info; // user-API /* touch_calib_get() - get calibration information */ -int touch_calib_get(touch_calib *calib) +int touch_calib_get(touch_calibration_t *calib) { if (calib == NULL) return -1; - memcpy(calib, &__touch_drv_info.calibration, sizeof(touch_calib)); + memcpy(calib, &__touch_drv_info.calibration, sizeof(*calib)); return 0; } /* touch_calib_set() - set calibration information */ -int touch_calib_set(touch_calib *calib) +int touch_calib_set(touch_calibration_t *calib) { if (calib == NULL) return -1; - memcpy(&__touch_drv_info.calibration, calib, sizeof(touch_calib)); + memcpy(&__touch_drv_info.calibration, calib, sizeof(*calib)); return 0; } // low-level API -/* touch_next_event() - get the next touchscreen event */ +/* touch_next_event() - get the next touch event */ key_event_t touch_next_event(void) { struct _touch_adconv adconv; @@ -53,11 +53,11 @@ key_event_t touch_next_event(void) int type; evt.type = KEYEV_TOUCH_RELEASE; - type = touchscreen_adconv_get_raw(&adraw); + type = touch_adconv_get_raw(&adraw); if (type != 0) { - type = touchscreen_adconv_get_conv(&adconv, &adraw, type); - type = touchscreen_adconv_get_dots(&addots, &adconv, type); + type = touch_adconv_get_conv(&adconv, &adraw, type); + type = touch_adconv_get_dots(&addots, &adconv, type); if (type == 1) { evt.type = KEYEV_TOUCH_DRAG; From 353c33d21cf9b06858ba9250dad57d7603944f25 Mon Sep 17 00:00:00 2001 From: Yann MAGNIN Date: Sun, 23 Mar 2025 17:27:00 +0100 Subject: [PATCH 29/29] touch: only enable touchscreen for FXCP400 + fix C++ header support --- include/gint/touch.h | 6 ++++++ src/touch/adconv.c | 5 +++++ src/touch/adconv.h | 13 +++++++++++++ src/touch/driver.c | 5 +++++ src/touch/driver.h | 13 +++++++++++++ src/touch/i2c.c | 5 +++++ src/touch/i2c.h | 13 +++++++++++++ src/touch/i2c_inth.c | 5 +++++ src/touch/touch.c | 5 +++++ 9 files changed, 70 insertions(+) diff --git a/include/gint/touch.h b/include/gint/touch.h index 51e353f..0e65638 100644 --- a/include/gint/touch.h +++ b/include/gint/touch.h @@ -8,6 +8,10 @@ extern "C" { #endif +#include + +#if GINT_HW_CP + /* touch_calib - tounch-screen calibration information */ typedef struct { int x_base; @@ -32,6 +36,8 @@ extern int touch_calib_set(touch_calibration_t *calib); /* touch_next_event() - get the next touchscreen event */ extern key_event_t touch_next_event(void); +#endif /* GINT_HW_CP */ + #ifdef __cplusplus } #endif diff --git a/src/touch/adconv.c b/src/touch/adconv.c index 66dad73..c007c02 100644 --- a/src/touch/adconv.c +++ b/src/touch/adconv.c @@ -5,6 +5,9 @@ #include #include +#include + +#if GINT_HW_CP #include "./adconv.h" #include "./i2c.h" @@ -139,3 +142,5 @@ int touch_adconv_get_dots( cpu_atomic_end(); return type; } + +#endif /* GINT_HW_CP */ diff --git a/src/touch/adconv.h b/src/touch/adconv.h index c95d6a5..cf64b1b 100644 --- a/src/touch/adconv.h +++ b/src/touch/adconv.h @@ -1,7 +1,14 @@ #ifndef GINT_TOUCH_ADCONV_H #define GINT_TOUCH_ADCONV_H 1 +#ifdef __cplusplus +extern "C" { +#endif + #include +#include + +#if GINT_HW_CP //--- // Internals @@ -74,4 +81,10 @@ extern int touch_adconv_get_dots( int type ); +#endif /* GINT_HW_CP */ + +#ifdef __cplusplus +} +#endif + #endif /* GINT_TOUCH_ADCONV_H */ diff --git a/src/touch/driver.c b/src/touch/driver.c index 7df11ce..078cacd 100644 --- a/src/touch/driver.c +++ b/src/touch/driver.c @@ -5,6 +5,9 @@ #include #include +#include + +#if GINT_HW_CP #include "./driver.h" #include "./i2c.h" @@ -105,3 +108,5 @@ gint_driver_t drv_touch = { .state_size = sizeof(touch_state_t), }; GINT_DECLARE_DRIVER(24, drv_touch); + +#endif /* GINT_HW_CP */ diff --git a/src/touch/driver.h b/src/touch/driver.h index fb3f422..f73cd1b 100644 --- a/src/touch/driver.h +++ b/src/touch/driver.h @@ -1,8 +1,15 @@ #ifndef GINT_TOUCH_DRIVER_H #define GINT_TOUCH_DRIVER_H 1 +#ifdef __cplusplus +extern "C" { +#endif + #include #include +#include + +#if GINT_HW_CP /* _touch_drv_info() - internal driver information */ struct _touch_drv_info @@ -14,4 +21,10 @@ struct _touch_drv_info } adinfo; }; +#endif /* GINT_HW_CP */ + +#ifdef __cplusplus +} +#endif + #endif /* GINT_TOUCH_DRIVER_H */ diff --git a/src/touch/i2c.c b/src/touch/i2c.c index 5894569..517f03a 100644 --- a/src/touch/i2c.c +++ b/src/touch/i2c.c @@ -8,6 +8,9 @@ #include #include #include +#include + +#if GINT_HW_CP #include "./i2c.h" @@ -247,3 +250,5 @@ void i2c_funbind(void) SH7305_I2C.ICCR.ICE = 0; SH7305_I2C.ICDR = 0; } + +#endif /* GINT_HW_CP */ diff --git a/src/touch/i2c.h b/src/touch/i2c.h index a442c48..daf4ea5 100644 --- a/src/touch/i2c.h +++ b/src/touch/i2c.h @@ -1,8 +1,15 @@ #ifndef GINT_TOUCH_I2C_H #define GINT_TOUCH_I2C_H 1 +#ifdef __cplusplus +extern "C" { +#endif + #include #include +#include + +#if GINT_HW_CP //--- // User API @@ -93,4 +100,10 @@ extern void i2c_funbind(void); /* i2c_unbind() - unbind from gint to casio */ extern void i2c_unbind(void); +#endif /* GINT_HW_CP */ + +#ifdef __cplusplus +} +#endif + #endif /* GINT_TOUCH_I2C_H */ diff --git a/src/touch/i2c_inth.c b/src/touch/i2c_inth.c index 1b59809..233bcfd 100644 --- a/src/touch/i2c_inth.c +++ b/src/touch/i2c_inth.c @@ -2,6 +2,9 @@ // gint:touch:i2c_inth - I2C interrupt handlers //--- #include +#include + +#if GINT_HW_CP #include "./i2c.h" @@ -164,3 +167,5 @@ void i2c_inth_al(void) { gint_panic(0x10e0); } + +#endif /* GINT_HW_CP */ diff --git a/src/touch/touch.c b/src/touch/touch.c index 2cfe2f8..f59efcb 100644 --- a/src/touch/touch.c +++ b/src/touch/touch.c @@ -5,6 +5,9 @@ #include #include +#include + +#if GINT_HW_CP #include "./i2c.h" #include "./adconv.h" @@ -88,3 +91,5 @@ key_event_t touch_next_event(void) cpu_atomic_end(); return evt; } + +#endif /* GINT_HW_CP */