2013-10-16 12:35:55 +07:00
|
|
|
/**************************************************************************/
|
|
|
|
/*!
|
2014-03-24 16:07:17 +07:00
|
|
|
@file keyboard_host_app.c
|
2013-10-16 12:35:55 +07:00
|
|
|
@author hathach (tinyusb.org)
|
|
|
|
|
|
|
|
@section LICENSE
|
|
|
|
|
|
|
|
Software License Agreement (BSD License)
|
|
|
|
|
|
|
|
Copyright (c) 2013, hathach (tinyusb.org)
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
3. Neither the name of the copyright holders nor the
|
|
|
|
names of its contributors may be used to endorse or promote products
|
|
|
|
derived from this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
|
|
|
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
|
|
|
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION HOWEVER CAUSED AND
|
|
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
INCLUDING NEGLIGENCE OR OTHERWISE ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
This file is part of the tinyusb stack.
|
|
|
|
*/
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INCLUDE
|
|
|
|
//--------------------------------------------------------------------+
|
2014-03-24 16:07:17 +07:00
|
|
|
#include "keyboard_host_app.h"
|
2013-10-16 12:35:55 +07:00
|
|
|
#include "app_os_prio.h"
|
|
|
|
|
|
|
|
#if TUSB_CFG_HOST_HID_KEYBOARD
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// MACRO CONSTANT TYPEDEF
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
#define QUEUE_KEYBOARD_REPORT_DEPTH 4
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// INTERNAL OBJECT & FUNCTION DECLARATION
|
|
|
|
//--------------------------------------------------------------------+
|
2018-03-01 11:17:11 +07:00
|
|
|
static osal_queue_t queue_kbd_hdl;
|
2014-03-10 13:13:13 +07:00
|
|
|
TUSB_CFG_ATTR_USBRAM static hid_keyboard_report_t usb_keyboard_report;
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
static inline uint8_t keycode_to_ascii(uint8_t modifier, uint8_t keycode) ATTR_CONST ATTR_ALWAYS_INLINE;
|
2013-10-25 17:28:31 +07:00
|
|
|
static inline void process_kbd_report(hid_keyboard_report_t const * report);
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
2014-03-31 12:06:13 +07:00
|
|
|
// tinyusb callbacks
|
2013-10-16 12:35:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2015-05-01 19:11:25 +07:00
|
|
|
void tuh_hid_keyboard_mounted_cb(uint8_t dev_addr)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
|
|
|
// application set-up
|
2014-03-14 11:59:39 +07:00
|
|
|
printf("\na Keyboard device (address %d) is mounted\n", dev_addr);
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
osal_queue_flush(queue_kbd_hdl);
|
2015-05-01 19:11:25 +07:00
|
|
|
tuh_hid_keyboard_get_report(dev_addr, (uint8_t*) &usb_keyboard_report); // first report
|
2013-10-16 12:35:55 +07:00
|
|
|
}
|
|
|
|
|
2015-05-01 19:11:25 +07:00
|
|
|
void tuh_hid_keyboard_unmounted_cb(uint8_t dev_addr)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
|
|
|
// application tear-down
|
2014-03-14 11:59:39 +07:00
|
|
|
printf("\na Keyboard device (address %d) is unmounted\n", dev_addr);
|
2013-10-16 12:35:55 +07:00
|
|
|
}
|
|
|
|
|
2014-03-31 12:06:13 +07:00
|
|
|
// invoked ISR context
|
2015-05-01 19:11:25 +07:00
|
|
|
void tuh_hid_keyboard_isr(uint8_t dev_addr, tusb_event_t event)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
|
|
|
switch(event)
|
|
|
|
{
|
|
|
|
case TUSB_EVENT_XFER_COMPLETE:
|
2014-04-24 23:57:21 +07:00
|
|
|
(void) osal_queue_send(queue_kbd_hdl, &usb_keyboard_report);
|
2015-05-01 19:11:25 +07:00
|
|
|
tuh_hid_keyboard_get_report(dev_addr, (uint8_t*) &usb_keyboard_report);
|
2013-10-16 12:35:55 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_EVENT_XFER_ERROR:
|
2015-05-01 19:11:25 +07:00
|
|
|
tuh_hid_keyboard_get_report(dev_addr, (uint8_t*) &usb_keyboard_report); // ignore & continue
|
2013-10-16 12:35:55 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default :
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// APPLICATION
|
|
|
|
//--------------------------------------------------------------------+
|
2014-03-24 16:07:17 +07:00
|
|
|
void keyboard_host_app_init(void)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
2013-10-25 17:28:31 +07:00
|
|
|
memclr_(&usb_keyboard_report, sizeof(hid_keyboard_report_t));
|
2013-10-16 12:35:55 +07:00
|
|
|
|
2018-03-01 11:17:11 +07:00
|
|
|
queue_kbd_hdl = osal_queue_create( QUEUE_KEYBOARD_REPORT_DEPTH, sizeof(hid_keyboard_report_t) );
|
2013-10-16 12:35:55 +07:00
|
|
|
ASSERT_PTR( queue_kbd_hdl, VOID_RETURN );
|
|
|
|
|
2018-03-01 11:17:11 +07:00
|
|
|
VERIFY( osal_task_create(keyboard_host_app_task, "kbd", 128, NULL, KEYBOARD_APP_TASK_PRIO, NULL), );
|
2013-10-16 12:35:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
//------------- main task -------------//
|
2018-03-01 11:17:11 +07:00
|
|
|
void keyboard_host_app_task(void* param)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
2018-03-01 11:17:11 +07:00
|
|
|
(void) param;
|
2014-03-14 11:59:39 +07:00
|
|
|
|
2014-04-25 15:16:52 +07:00
|
|
|
OSAL_TASK_LOOP_BEGIN
|
|
|
|
|
2013-10-25 17:28:31 +07:00
|
|
|
hid_keyboard_report_t kbd_report;
|
2014-03-13 16:17:38 +07:00
|
|
|
tusb_error_t error;
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
osal_queue_receive(queue_kbd_hdl, &kbd_report, OSAL_TIMEOUT_WAIT_FOREVER, &error);
|
2014-03-13 16:17:38 +07:00
|
|
|
(void) error; // suppress compiler warning
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
process_kbd_report(&kbd_report);
|
|
|
|
|
|
|
|
OSAL_TASK_LOOP_END
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// HELPER
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// look up new key in previous keys
|
2014-03-31 12:06:13 +07:00
|
|
|
static inline bool find_key_in_report(hid_keyboard_report_t const *p_report, uint8_t keycode)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
|
|
|
for(uint8_t i=0; i<6; i++)
|
|
|
|
{
|
2014-03-13 18:11:59 +07:00
|
|
|
if (p_report->keycode[i] == keycode) return true;
|
2013-10-16 12:35:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:28:31 +07:00
|
|
|
static inline void process_kbd_report(hid_keyboard_report_t const *p_new_report)
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
2014-03-13 18:11:59 +07:00
|
|
|
static hid_keyboard_report_t prev_report = { 0, 0, {0} }; // previous report to check key released
|
2013-10-16 12:35:55 +07:00
|
|
|
|
|
|
|
//------------- example code ignore control (non-printable) key affects -------------//
|
|
|
|
for(uint8_t i=0; i<6; i++)
|
|
|
|
{
|
|
|
|
if ( p_new_report->keycode[i] )
|
|
|
|
{
|
2014-03-31 12:06:13 +07:00
|
|
|
if ( find_key_in_report(&prev_report, p_new_report->keycode[i]) )
|
2013-10-16 12:35:55 +07:00
|
|
|
{
|
2014-03-13 18:11:59 +07:00
|
|
|
// exist in previous report means the current key is holding
|
2013-10-16 12:35:55 +07:00
|
|
|
}else
|
|
|
|
{
|
2014-03-31 12:06:13 +07:00
|
|
|
// not existed in previous report means the current key is pressed
|
2014-03-02 21:39:18 +07:00
|
|
|
uint8_t ch = keycode_to_ascii(p_new_report->modifier, p_new_report->keycode[i]);
|
|
|
|
putchar(ch);
|
|
|
|
if ( ch == '\r' ) putchar('\n'); // added new line for enter key
|
2013-10-16 12:35:55 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// TODO example skips key released
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_report = *p_new_report;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint8_t keycode_to_ascii(uint8_t modifier, uint8_t keycode)
|
|
|
|
{
|
|
|
|
// TODO max of keycode_ascii_tbl
|
|
|
|
return keycode > 128 ? 0 :
|
|
|
|
hid_keycode_to_ascii_tbl [modifier & (KEYBOARD_MODIFIER_LEFTSHIFT | KEYBOARD_MODIFIER_RIGHTSHIFT) ? 1 : 0] [keycode];
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|