#include "app_config.h" #include "app_action.h" #include "system/includes.h" #include "spp_user.h" #include "string.h" #include "circular_buf.h" #include "spp_trans_data.h" #include "bt_common.h" #include "btstack/avctp_user.h" #include "circular_buf.h" #include "hid_user.h" #include "standard_hid.h" #if (USER_SUPPORT_PROFILE_HID ==1) #define log_info printf #define HID_CHANGE_DESCRIPTOR 0 #define TEST_USER_HID_EN 0 #define HID_DATA 0xA0 /*Device&host*/ #define HID_DATC 0xB0 /*Device&host DEPRECATED*/ /*DATA*/ #define DATA_OTHER 0x00 #define DATA_INPUT 0x01 #define DATA_OUTPUT 0x02 #define DATA_FEATURE 0x03 static u8 *report_map; static u16 report_map_size; #define HID_REPORT_MAP_DATA report_map #define HID_REPORT_MAP_SIZE report_map_size /* #define MOUSE_REPORT_TYPE 0xA1 */ /* #define MOUSE_REPORT_ID 1 */ static void (*user_hid_send_wakeup)(void) = NULL; static u16 hid_channel; static u8 hid_run = 0; static volatile u8 is_hid_active = 0; static volatile u8 hid_s_step = 0; /* static int hid_timer_id = 0; */ int hid_timer_id = 0; int edr_hid_timer_handle = 0; #define HID_SEND_MAX_SIZE (16) //描述符数据包的长度 /* static u8 edr_hid_one_packet[HID_SEND_MAX_SIZE]; */ /* static volatile u16 edr_send_packet_len = 0; */ static volatile u8 bt_send_busy = 0; void (*user_led_status_callback)(u8 *buffer, u16 size) = NULL; #define HID_TMP_BUFSIZE (64*2) #define cbuf_get_space(a) (a)->total_len static cbuffer_t user_send_cbuf; static u8 hid_tmp_buffer[HID_TMP_BUFSIZE]; extern void hid_diy_regiest_callback(void *cb, void *interrupt_cb); extern void hid_sdp_init(const u8 *hid_descriptor, u16 size); extern uint16_t little_endian_read_16(const uint8_t *buffer, int pos); extern void put_buf(const u8 *buf, int len); void edr_hid_data_send(u8 report_id, u8 *data, u16 len); /* typedef struct { */ /* u8 report_type; */ /* u8 report_id; */ /* u8 button; */ /* s8 x; */ /* s8 y; */ /* } hid_ctl_info_t; */ typedef struct { u8 report_type; u8 report_id; u8 data[HID_SEND_MAX_SIZE - 2]; } hid_data_info_t; //----------------------------------------------------- static void edr_bt_evnet_post(u32 arg_type, u8 priv_event, u8 *args, u32 value) { struct sys_event e; e.type = SYS_BT_EVENT; e.arg = (void *)arg_type; e.u.bt.event = priv_event; if (args) { memcpy(e.u.bt.args, args, 7); } e.u.bt.value = value; sys_event_notify(&e); } /* enum { */ /* REMOTE_DEV_UNKNOWN = 0, */ /* REMOTE_DEV_ANDROID , */ /* REMOTE_DEV_IOS , */ /* }; */ //参考识别手机系统 void sdp_callback_remote_type(u8 remote_type) { log_info("edr_hid:remote_type= %d\n", remote_type); edr_bt_evnet_post(SYS_BT_EVENT_FORM_COMMON, COMMON_EVENT_EDR_REMOTE_TYPE, NULL, remote_type); //to do } //----------------------------------------------------- static u16 user_data_read_sub(u8 *buf, u16 buf_size) { u16 ret_len; if (0 == cbuf_get_data_size(&user_send_cbuf)) { return 0; } OS_ENTER_CRITICAL(); cbuf_read(&user_send_cbuf, &ret_len, 2); if (ret_len && ret_len <= buf_size) { cbuf_read(&user_send_cbuf, buf, ret_len); } OS_EXIT_CRITICAL(); return ret_len; } static void user_data_try_send(void) { u16 send_len; if (bt_send_busy) { return; } bt_send_busy = 1;//hold u8 tmp_send_buf[HID_SEND_MAX_SIZE]; send_len = user_data_read_sub(tmp_send_buf, HID_SEND_MAX_SIZE); if (send_len) { if (user_hid_send_data(tmp_send_buf, send_len)) { bt_send_busy = 0; } } else { //not send bt_send_busy = 0; } } static u32 user_data_write_sub(u8 *data, u16 len) { u16 wlen = 0; u16 buf_space = cbuf_get_space(&user_send_cbuf) - cbuf_get_data_size(&user_send_cbuf); if (len + 2 > buf_space) { return 0; } OS_ENTER_CRITICAL(); wlen = cbuf_write(&user_send_cbuf, &len, 2); wlen += cbuf_write(&user_send_cbuf, data, len); OS_EXIT_CRITICAL(); user_data_try_send(); return wlen; } //----------------------------------------------------- void user_hid_set_icon(u32 class_type) { __change_hci_class_type(class_type);// } void user_hid_set_ReportMap(u8 *map, u16 size) { report_map = map; report_map_size = size; if (hid_run) { #if !HID_CHANGE_DESCRIPTOR hid_sdp_init(HID_REPORT_MAP_DATA, HID_REPORT_MAP_SIZE); #endif } } /* const hid_ctl_info_t test_key[5] = { */ /* {0xA1, 1, 0, 0, 0}, */ /* {0xA1, 1, 0, 100, 0}, */ /* {0xA1, 1, 0, -100, 0}, */ /* }; */ /* static void test_hid_send_step(void) */ /* { */ /* static u8 xy_step = 0; */ /* u8 send_len = 5; */ /* if (!hid_s_step) { */ /* return; */ /* } */ /* xy_step = !xy_step; */ /* if (xy_step) { */ /* hid_s_step = 1; */ /* } else { */ /* hid_s_step = 2; */ /* } */ /* if (0 == user_hid_send_data((u8 *)&test_key[hid_s_step], send_len)) { */ /* hid_s_step = 0; */ /* } */ /* } */ static void user_hid_timer_handler(void *para) { /* static u8 count = 0; */ if (!hid_channel) { return; } /* if (++count > 5) { */ /* count = 0; */ /* hid_s_step = 1; */ /* test_hid_send_step(); */ /* } */ /* hid_key_send(CONSUMER_PLAY_PAUSE); */ } static u8 user_hid_idle_query(void) { return !is_hid_active; } REGISTER_LP_TARGET(hid_user_target) = { .name = "hid_user", .is_idle = user_hid_idle_query, }; static void user_hid_send_ok_callback(void) { /* putchar('K'); */ bt_send_busy = 0; if (user_hid_send_wakeup) { user_hid_send_wakeup(); } user_data_try_send(); /* test_hid_send_step(); */ } void user_hid_regiser_wakeup_send(void *cbk) { user_hid_send_wakeup = cbk; } void user_hid_disconnect(void) { if (hid_channel) { user_send_cmd_prepare(USER_CTRL_HID_DISCONNECT, 0, NULL); } } int user_hid_send_data(u8 *buf, u32 len) { int ret; hid_s_param_t s_par; if (!hid_channel) { return -1; } s_par.chl_id = hid_channel; s_par.data_len = len; s_par.data_ptr = buf; ret = user_send_cmd_prepare(USER_CTRL_HID_SEND_DATA, sizeof(hid_s_param_t), (u8 *)&s_par); if (ret) { log_info("hid send fail!!! %d\n", ret); } return ret; } static void user_hid_msg_handler(u32 msg, u8 *packet, u32 packet_size) { u16 tmp_chl; switch (msg) { case 1: hid_channel = little_endian_read_16(packet, 2); //inter_channel bt_send_busy = 0; log_info("hid connect ########################,%d\n", hid_channel); cbuf_init(&user_send_cbuf, hid_tmp_buffer, HID_TMP_BUFSIZE); break; case 2: log_info("hid disconnect ########################\n"); hid_channel = 0; cbuf_init(&user_send_cbuf, hid_tmp_buffer, HID_TMP_BUFSIZE); break; case 3: if (hid_channel == little_endian_read_16(packet, 0)) { user_hid_send_ok_callback(); } break; default: break; } } void user_hid_init(void (*user_hid_interrupt_handler)(u8 *packet, u16 size)) { log_info("%s\n", __FUNCTION__); hid_channel = 0; hid_diy_regiest_callback(user_hid_msg_handler, user_hid_interrupt_handler); #if !HID_CHANGE_DESCRIPTOR hid_sdp_init(HID_REPORT_MAP_DATA, HID_REPORT_MAP_SIZE); #endif #if TEST_USER_HID_EN hid_timer_id = sys_s_hi_timer_add((void *)0, user_hid_timer_handler, 5000); #endif cbuf_init(&user_send_cbuf, hid_tmp_buffer, HID_TMP_BUFSIZE); hid_run = 1; } void user_hid_exit(void) { log_info("%s\n", __FUNCTION__); if (hid_timer_id) { user_hid_disconnect(); sys_s_hi_timer_del(hid_timer_id); hid_timer_id = 0; } if (edr_hid_timer_handle) { sys_s_hi_timer_del(edr_hid_timer_handle); edr_hid_timer_handle = 0; } hid_run = 0; } void user_hid_enable(u8 en) { log_info("%s:%d\n", __FUNCTION__, en); if (en) { ; } else { user_hid_disconnect(); } } void edr_hid_data_send(u8 report_id, u8 *data, u16 len) { if (!hid_channel) { return; } if (len > HID_SEND_MAX_SIZE - 2) { log_info("buffer limitp!!!\n"); return; } putchar('@'); hid_data_info_t data_info; data_info.report_type = HID_DATA | DATA_INPUT; data_info.report_id = report_id; memcpy(data_info.data, data, len); if (!user_data_write_sub((u8 *)&data_info, 2 + len)) { log_info("hid buffer full!!!\n"); return; } } void edr_hid_key_deal_test(u16 key_msg) { edr_hid_data_send(1, (u8 *)&key_msg, 2); if (key_msg) { key_msg = 0; edr_hid_data_send(1, (u8 *)&key_msg, 2); } } int edr_hid_is_connected(void) { return hid_channel; } extern u32 lmp_private_get_tx_remain_buffer(); int edr_hid_tx_buff_is_ok(void) { if (lmp_private_get_tx_remain_buffer() < 0x300) { putchar('b'); return 0; } return 1; } #if HID_CHANGE_DESCRIPTOR const u8 use_hid_descriptor[] = { 0x05, 0x01, 0x09, 0x02, 0xA1, 0x01, 0x85, 0x01, 0x05, 0x09, 0x19, 0x01, 0x29, 0x03, 0x15, 0x00, 0x25, 0x01, 0x95, 0x03, 0x75, 0x01, 0x81, 0x02, 0x75, 0x05, 0x95, 0x01, 0x81, 0x03, 0x05, 0x01, 0x09, 0x01, 0xA1, 0x00, 0x09, 0x30, 0x09, 0x31, 0x16, 0x00, 0xD8, 0x26, 0x00, 0x28, 0x75, 0x10, 0x95, 0x02, 0x81, 0x06, 0xC0, 0xC0, 0x05, 0x0C, 0x09, 0x01, 0xA1, 0x01, 0x85, 0x02, 0x15, 0x00, 0x25, 0x01, 0x09, 0x34, 0x09, 0x40, 0x0A, 0x23, 0x02, 0x0A, 0x24, 0x02, 0x09, 0xE9, 0x09, 0xEA, 0x09, 0xB0, 0x09, 0xB1, 0x09, 0xB3, 0x09, 0xB4, 0x09, 0xB5, 0x09, 0xB6, 0x09, 0xB7, 0x09, 0xCD, 0x75, 0x01, 0x95, 0x0E, 0x81, 0x22, 0x75, 0x01, 0x95, 0x02, 0x81, 0x02, 0xC0 }; struct user_hid_consumer_cmd { //Bluetooth HID Protocol Message Header Octet u8 HIDP_Hdr; //Bluetooth HID Boot Reports u8 report_id; u8 button; } _GNU_PACKED_; static struct user_hid_consumer_cmd u_consumer = { .HIDP_Hdr = 0xA1, .report_id = 0x02, .button = 0, }; // consumer key #define CONSUMER_MENU 0x01 #define CONSUMER_MENU_ESCAPE 0x02 #define CONSUMER_AC_HOME 0x04 void hid_consumer_send_test(u8 menu) { if (menu == 1) { u_consumer.button = CONSUMER_MENU; } if (menu == 2) { u_consumer.button = CONSUMER_MENU_ESCAPE; } if (menu == 3) { u_consumer.button = CONSUMER_AC_HOME; } put_buf((u8 *)&u_consumer, sizeof(u_consumer)); user_data_write_sub((u8 *)&u_consumer, sizeof(u_consumer)); u_consumer.button = 0x00; user_data_write_sub((u8 *)&u_consumer, sizeof(u_consumer)); } u8 sdp_make_hid_service_data[0x200]; static void user_hid_sdp_init(const u8 *hid_descriptor, u16 size) { int real_size; real_size = sdp_create_diy_hid_service(sdp_make_hid_service_data, sizeof(sdp_make_hid_service_data), hid_descriptor, size); printf("dy_hid_service(%d):", real_size); } void user_hid_descriptor_init(void) { //__change_hci_class_type(0x240418); /*需要更换手机显示图标的可以自己改成以前的值*/ user_hid_init(); user_hid_sdp_init(use_hid_descriptor, sizeof(use_hid_descriptor)); } //用户修改成自定义的描述符说明 //1、在void bredr_handle_register();中调用user_hid_descriptor_init; //2、user_hid_sdp_init换成自己的表 //3、文件上方HID_CHANGE_DESCRIPTOR定义为1 //4、在bt_profile_config.c文件中extern sdp_make_hid_service_data数组, // 把sdp_hid_service_data替换为sdp_make_hid_service_data. //5、把const u8 hid_conn_depend_on_dev_company的值置0; #endif #endif