mirror of
https://github.com/grymoire/i2c_puppet-Linux.git
synced 2025-01-15 17:28:22 +01:00
196 lines
5.0 KiB
C
196 lines
5.0 KiB
C
#include "usb.h"
|
|
|
|
#include "backlight.h"
|
|
#include "keyboard.h"
|
|
#include "touchpad.h"
|
|
#include "reg.h"
|
|
|
|
#include <hardware/irq.h>
|
|
#include <pico/mutex.h>
|
|
#include <tusb.h>
|
|
|
|
#define USB_LOW_PRIORITY_IRQ 31
|
|
#define USB_TASK_INTERVAL_US 1000
|
|
|
|
static struct
|
|
{
|
|
mutex_t mutex;
|
|
bool mouse_moved;
|
|
uint8_t mouse_btn;
|
|
|
|
uint8_t write_buffer[2];
|
|
uint8_t write_len;
|
|
} self;
|
|
|
|
// TODO: Should mods always be sent?
|
|
// TODO: What about Ctrl?
|
|
// TODO: What should L1, L2, R1, R2 do
|
|
// TODO: Should touch send arrow keys as an option?
|
|
|
|
static void low_priority_worker_irq(void)
|
|
{
|
|
if (mutex_try_enter(&self.mutex, NULL)) {
|
|
tud_task();
|
|
|
|
mutex_exit(&self.mutex);
|
|
}
|
|
}
|
|
|
|
static int64_t timer_task(alarm_id_t id, void *user_data)
|
|
{
|
|
(void)id;
|
|
(void)user_data;
|
|
|
|
irq_set_pending(USB_LOW_PRIORITY_IRQ);
|
|
|
|
return USB_TASK_INTERVAL_US;
|
|
}
|
|
|
|
static void key_cb(char key, enum key_state state)
|
|
{
|
|
if (tud_hid_n_ready(USB_ITF_KEYBOARD) && reg_is_bit_set(REG_ID_CF2, CF2_USB_KEYB_ON)) {
|
|
uint8_t conv_table[128][2] = { HID_ASCII_TO_KEYCODE };
|
|
conv_table['\n'][1] = HID_KEY_ENTER; // Fixup: Enter instead of Return
|
|
conv_table[KEY_JOY_UP][1] = HID_KEY_ARROW_UP;
|
|
conv_table[KEY_JOY_DOWN][1] = HID_KEY_ARROW_DOWN;
|
|
conv_table[KEY_JOY_LEFT][1] = HID_KEY_ARROW_LEFT;
|
|
conv_table[KEY_JOY_RIGHT][1] = HID_KEY_ARROW_RIGHT;
|
|
|
|
uint8_t keycode[6] = { 0 };
|
|
uint8_t modifier = 0;
|
|
|
|
if (state == KEY_STATE_PRESSED) {
|
|
if (conv_table[(int)key][0])
|
|
modifier = KEYBOARD_MODIFIER_LEFTSHIFT;
|
|
|
|
keycode[0] = conv_table[(int)key][1];
|
|
}
|
|
|
|
if (state != KEY_STATE_HOLD)
|
|
tud_hid_n_keyboard_report(USB_ITF_KEYBOARD, 0, modifier, keycode);
|
|
}
|
|
|
|
if (tud_hid_n_ready(USB_ITF_MOUSE) && reg_is_bit_set(REG_ID_CF2, CF2_USB_MOUSE_ON)) {
|
|
if (key == KEY_JOY_CENTER) {
|
|
if (state == KEY_STATE_PRESSED) {
|
|
self.mouse_btn = MOUSE_BUTTON_LEFT;
|
|
self.mouse_moved = false;
|
|
tud_hid_n_mouse_report(USB_ITF_MOUSE, 0, MOUSE_BUTTON_LEFT, 0, 0, 0, 0);
|
|
} else if ((state == KEY_STATE_HOLD) && !self.mouse_moved) {
|
|
self.mouse_btn = MOUSE_BUTTON_RIGHT;
|
|
tud_hid_n_mouse_report(USB_ITF_MOUSE, 0, MOUSE_BUTTON_RIGHT, 0, 0, 0, 0);
|
|
} else if (state == KEY_STATE_RELEASED) {
|
|
self.mouse_btn = 0x00;
|
|
tud_hid_n_mouse_report(USB_ITF_MOUSE, 0, 0x00, 0, 0, 0, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
static struct key_callback key_callback =
|
|
{
|
|
.func = key_cb
|
|
};
|
|
|
|
static void touch_cb(int8_t x, int8_t y)
|
|
{
|
|
if (!tud_hid_n_ready(USB_ITF_MOUSE) || !reg_is_bit_set(REG_ID_CF2, CF2_USB_MOUSE_ON))
|
|
return;
|
|
|
|
self.mouse_moved = true;
|
|
|
|
tud_hid_n_mouse_report(USB_ITF_MOUSE, 0, self.mouse_btn, x, y, 0, 0);
|
|
}
|
|
static struct touch_callback touch_callback =
|
|
{
|
|
.func = touch_cb
|
|
};
|
|
|
|
uint16_t tud_hid_get_report_cb(uint8_t itf, uint8_t report_id, hid_report_type_t report_type, uint8_t *buffer, uint16_t reqlen)
|
|
{
|
|
// TODO not Implemented
|
|
(void)itf;
|
|
(void)report_id;
|
|
(void)report_type;
|
|
(void)buffer;
|
|
(void)reqlen;
|
|
|
|
return 0;
|
|
|
|
// if (itf != USB_ITF_HID_GENERIC)
|
|
// return 0;
|
|
|
|
// printf("%s: itf: %d, report id: %d, type: %d, len: %d\r\n", __func__, itf, report_id, report_type, reqlen);
|
|
|
|
// memcpy(buffer, self.write_buffer, self.write_len);
|
|
|
|
// return self.write_len;
|
|
}
|
|
|
|
void tud_hid_set_report_cb(uint8_t itf, uint8_t report_id, hid_report_type_t report_type, uint8_t const *buffer, uint16_t len)
|
|
{
|
|
// TODO set LED based on CAPLOCK, NUMLOCK etc...
|
|
(void)itf;
|
|
(void)report_id;
|
|
(void)report_type;
|
|
(void)buffer;
|
|
(void)len;
|
|
|
|
// if (itf != USB_ITF_HID_GENERIC)
|
|
// return;
|
|
|
|
// printf("%s: itf: %d, report id: %d, type: %d, buff: %02X %02X %02X len: %d\r\n", __func__, itf, report_id, report_type, buffer[0], buffer[1], buffer[2], len);
|
|
|
|
// if (len < 1)
|
|
// return;
|
|
|
|
// const bool is_write = (buffer[0] & PACKET_WRITE_MASK);
|
|
// const uint8_t reg = (buffer[0] & ~PACKET_WRITE_MASK);
|
|
|
|
// printf("%s: read complete, is_write: %d, reg: 0x%02X\r\n", __func__, is_write, reg);
|
|
|
|
// if (is_write && (len < 2))
|
|
// return;
|
|
|
|
// printf("%s: data: 0x%02X\r\n", __func__, buffer[1]);
|
|
|
|
// reg_process_packet(buffer[0], buffer[1], (uint8_t*)&self.write_buffer, &self.write_len);
|
|
|
|
// printf("%s: write_buff: %02X %02X, len: %d\r\n", __func__, self.write_buffer[0], self.write_buffer[1], self.write_len);
|
|
|
|
// tud_hid_n_report(itf, report_id, self.write_buffer, self.write_len);
|
|
}
|
|
|
|
void tud_vendor_rx_cb(uint8_t itf)
|
|
{
|
|
printf("%s: itf: %d, avail: %d\r\n", __func__, itf, tud_vendor_n_available(itf));
|
|
|
|
uint8_t buff[64] = { 0 };
|
|
tud_vendor_n_read(itf, buff, 64);
|
|
// printf("%s: %02X %02X %02X\r\n", __func__, buff[0], buff[1], buff[2]);
|
|
|
|
reg_process_packet(buff[0], buff[1], self.write_buffer, &self.write_len);
|
|
|
|
tud_vendor_n_write(itf, self.write_buffer, self.write_len);
|
|
}
|
|
|
|
void usb_init(void)
|
|
{
|
|
tusb_init();
|
|
|
|
keyboard_add_key_callback(&key_callback);
|
|
|
|
touchpad_add_touch_callback(&touch_callback);
|
|
|
|
// create a new interrupt that calls tud_task, and trigger that interrupt from a timer
|
|
irq_set_exclusive_handler(USB_LOW_PRIORITY_IRQ, low_priority_worker_irq);
|
|
irq_set_enabled(USB_LOW_PRIORITY_IRQ, true);
|
|
|
|
mutex_init(&self.mutex);
|
|
add_alarm_in_us(USB_TASK_INTERVAL_US, timer_task, NULL, true);
|
|
}
|
|
|
|
mutex_t *usb_get_mutex(void)
|
|
{
|
|
return &self.mutex;
|
|
}
|