/** * Copyright (c) 2016 - 2017, Nordic Semiconductor ASA * * 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, except as embedded into a Nordic * Semiconductor ASA integrated circuit in a product or a software update for * such product, 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 Nordic Semiconductor ASA nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * 4. This software, with or without modification, must only be used with a * Nordic Semiconductor ASA integrated circuit. * * 5. Any software provided in binary form under this license must not be reverse * engineered, decompiled, modified and/or disassembled. * * THIS SOFTWARE IS PROVIDED BY NORDIC SEMICONDUCTOR ASA "AS IS" AND ANY EXPRESS * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY, NONINFRINGEMENT, AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL NORDIC SEMICONDUCTOR ASA OR CONTRIBUTORS 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. * */ #include "sdk_config.h" #if APP_USBD_CLASS_HID_ENABLED #include "app_usbd.h" #include "app_usbd_core.h" #include "app_usbd_hid.h" /** * @ingroup app_usbd_hid_internals USBD HID internals * @{ * @ingroup app_usbd_hid * @internal */ /** * @brief Test whether SOF HID transfer is required. * * This function handles idle period IN transfer. * * @param[in,out] p_hid_ctx Internal HID context. * @param framecnt SOF event frame counter. * * @retval true Idle transfer is required. * @retval false Idle transfer is not required. */ static bool hid_sof_required(app_usbd_hid_ctx_t * p_hid_ctx, uint16_t framecnt) { if (p_hid_ctx->idle_rate == 0) { /* Infinite idle rate */ return false; } /*Idle rate has 4ms units. Every SOF event is generated with 1ms period.*/ uint16_t rate_ms = p_hid_ctx->idle_rate * 4; if ((framecnt % rate_ms) != 0) { /* Idle transfer not required yet*/ return false; } if (p_hid_ctx->access_lock) { /* Access to internal data locked. Buffer is BUSY. * Don't send anything. Clear transfer flag. Next transfer will be triggered * from API context.*/ app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_TRANS_IN_PROGRESS); return false; } if (app_usbd_hid_trans_required(p_hid_ctx)) { /*New transfer need to be triggered*/ return true; } return false; } /** * @brief User event handler. * * @param[in] p_inst Class instance. * @param[in] p_hinst HID class instance. * @param[in] event user Event type. */ static inline void user_event_handler(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_user_event_t event) { if (p_hinst->user_event_handler != NULL) { p_hinst->user_event_handler(p_inst, event); } } /** * @brief Internal SETUP standard IN request handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t setup_req_std_in(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_setup_evt_t const * p_setup_ev) { /*Only Get Descriptor standard IN request is supported by HID class*/ if (p_setup_ev->setup.bmRequest != APP_USBD_SETUP_STDREQ_GET_DESCRIPTOR) { return NRF_ERROR_NOT_SUPPORTED; } size_t dsc_len = 0; /* Try to find descriptor in class internals*/ void const * p_dsc = app_usbd_class_descriptor_find(p_inst, p_setup_ev->setup.wValue.hb, p_setup_ev->setup.wValue.lb, &dsc_len); if (p_dsc != NULL) { return app_usbd_core_setup_rsp(&(p_setup_ev->setup), p_dsc, dsc_len); } /* HID specific descriptors*/ switch (p_setup_ev->setup.wValue.hb) { case APP_USBD_HID_DESCRIPTOR_REPORT: { return app_usbd_core_setup_rsp(&p_setup_ev->setup, p_hinst->p_report_dsc, p_hinst->report_dsc_size); } case APP_USBD_HID_DESCRIPTOR_PHYSICAL: /*Not supported*/ break; default: /*Not supported*/ break; } return NRF_ERROR_NOT_SUPPORTED; } /** * @brief Internal SETUP standard OUT request handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t setup_req_std_out(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_setup_evt_t const * p_setup_ev) { /*Only Set Descriptor standard OUT request is supported by HID class. However, it is optional * and useless in HID cases.*/ return NRF_ERROR_NOT_SUPPORTED; } /** * @brief Internal SETUP class IN request handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t setup_req_class_in(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_setup_evt_t const * p_setup_ev) { switch (p_setup_ev->setup.bmRequest) { case APP_USBD_HID_REQ_GET_REPORT: { /*Only input report is supported. Same format as over IN pipe.*/ if (p_setup_ev->setup.wValue.hb != APP_USBD_HID_REPORT_TYPE_INPUT) { break; } return p_hinst->p_hid_methods->on_get_report(p_inst, p_setup_ev); } case APP_USBD_HID_REQ_GET_IDLE: { return app_usbd_core_setup_rsp(&p_setup_ev->setup, &p_hid_ctx->idle_rate, sizeof(p_hid_ctx->idle_rate)); } case APP_USBD_HID_REQ_GET_PROTOCOL: { return app_usbd_core_setup_rsp(&p_setup_ev->setup, &p_hid_ctx->protocol, sizeof(p_hid_ctx->protocol)); } default: break; } return NRF_ERROR_NOT_SUPPORTED; } /** * @brief Internal SETUP class OUT request handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t setup_req_class_out(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_setup_evt_t const * p_setup_ev) { switch (p_setup_ev->setup.bmRequest) { case APP_USBD_HID_REQ_SET_REPORT: if (p_setup_ev->setup.wValue.hb != APP_USBD_HID_REPORT_TYPE_OUTPUT) { break; } if (p_hinst->p_hid_methods->on_get_report == NULL) { break; } return p_hinst->p_hid_methods->on_set_report(p_inst, p_setup_ev); case APP_USBD_HID_REQ_SET_IDLE: p_hid_ctx->idle_rate = p_setup_ev->setup.wValue.hb; return NRF_SUCCESS; case APP_USBD_HID_REQ_SET_PROTOCOL: p_hid_ctx->protocol = p_setup_ev->setup.wValue.w; { app_usbd_hid_user_event_t ev = (p_hid_ctx->protocol == 0) ? APP_USBD_HID_USER_EVT_SET_BOOT_PROTO : APP_USBD_HID_USER_EVT_SET_REPORT_PROTO; user_event_handler(p_inst, p_hinst, ev); } return NRF_SUCCESS; default: break; } return NRF_ERROR_NOT_SUPPORTED; } /** * @brief Internal SETUP event handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t setup_event_handler(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_setup_evt_t const * p_setup_ev) { ASSERT(p_hinst != NULL); ASSERT(p_hid_ctx != NULL); ASSERT(p_setup_ev != NULL); if (app_usbd_setup_req_dir(p_setup_ev->setup.bmRequestType) == APP_USBD_SETUP_REQDIR_IN) { switch (app_usbd_setup_req_typ(p_setup_ev->setup.bmRequestType)) { case APP_USBD_SETUP_REQTYPE_STD: return setup_req_std_in(p_inst, p_hinst, p_hid_ctx, p_setup_ev); case APP_USBD_SETUP_REQTYPE_CLASS: return setup_req_class_in(p_inst, p_hinst, p_hid_ctx, p_setup_ev); default: break; } } else /*APP_USBD_SETUP_REQDIR_OUT*/ { switch (app_usbd_setup_req_typ(p_setup_ev->setup.bmRequestType)) { case APP_USBD_SETUP_REQTYPE_STD: return setup_req_std_out(p_inst, p_hinst, p_hid_ctx, p_setup_ev); case APP_USBD_SETUP_REQTYPE_CLASS: return setup_req_class_out(p_inst, p_hinst, p_hid_ctx, p_setup_ev); default: break; } } return NRF_ERROR_NOT_SUPPORTED; } /** * @brief Endpoint IN event handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t endpoint_in_event_handler(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_complex_evt_t const * p_event) { if (p_event->drv_evt.data.eptransfer.status == NRF_USBD_EP_OK) { /* Notify user about last successful transfer. */ user_event_handler(p_inst, p_hinst, APP_USBD_HID_USER_EVT_IN_REPORT_DONE); } if (app_usbd_hid_access_lock_test(p_hid_ctx)) { /* Access to internal data locked. Buffer is BUSY. * Don't send anything. Clear transfer flag. Next transfer will be triggered * from main loop context.*/ app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_TRANS_IN_PROGRESS); return NRF_SUCCESS; } return p_hinst->p_hid_methods->ep_transfer_in(p_inst); } /** * @brief Endpoint OUT event handler. * * @param[in] p_inst Generic class instance. * @param[in] p_hinst HID class instance. * @param[in,out] p_hid_ctx HID context. * @param[in] p_setup_ev Setup event. * * @return Standard error code. */ static ret_code_t endpoint_out_event_handler(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_complex_evt_t const * p_event) { if (p_hinst->p_hid_methods->ep_transfer_out == NULL) { return NRF_ERROR_NOT_SUPPORTED; } if (p_event->drv_evt.data.eptransfer.status == NRF_USBD_EP_OK) { /* Notify user about last successful transfer. */ user_event_handler(p_inst, p_hinst, APP_USBD_HID_USER_EVT_OUT_REPORT_READY); } return p_hinst->p_hid_methods->ep_transfer_out(p_inst); } /** @} */ ret_code_t app_usbd_hid_event_handler(app_usbd_class_inst_t const * p_inst, app_usbd_hid_inst_t const * p_hinst, app_usbd_hid_ctx_t * p_hid_ctx, app_usbd_complex_evt_t const * p_event) { ret_code_t ret = NRF_SUCCESS; switch (p_event->app_evt.type) { case APP_USBD_EVT_DRV_SOF: if (!hid_sof_required(p_hid_ctx, p_event->drv_evt.data.sof.framecnt)) { break; } ret = p_hinst->p_hid_methods->ep_transfer_in(p_inst); break; case APP_USBD_EVT_DRV_RESET: app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_TRANS_IN_PROGRESS); break; case APP_USBD_EVT_DRV_SETUP: ret = setup_event_handler(p_inst, p_hinst, p_hid_ctx, &p_event->setup_evt); break; case APP_USBD_EVT_DRV_EPTRANSFER: if (NRF_USBD_EPIN_CHECK(p_event->drv_evt.data.eptransfer.ep)) { ret = endpoint_in_event_handler(p_inst, p_hinst, p_hid_ctx, p_event); } else { ret = endpoint_out_event_handler(p_inst, p_hinst, p_hid_ctx, p_event); } break; case APP_USBD_EVT_DRV_SUSPEND: app_usbd_hid_state_flag_set(p_hid_ctx, APP_USBD_HID_STATE_FLAG_SUSPENDED); break; case APP_USBD_EVT_DRV_RESUME: app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_SUSPENDED); /* Always try to trigger transfer on resume event*/ ret = p_hinst->p_hid_methods->ep_transfer_in(p_inst); break; case APP_USBD_EVT_INST_APPEND: /*SOF register: GetIdle/SetIdle support*/ ret = app_usbd_class_sof_register(p_inst); if (ret != NRF_SUCCESS) { break; } app_usbd_core_class_rwu_register(p_inst); app_usbd_hid_state_flag_set(p_hid_ctx, APP_USBD_HID_STATE_FLAG_APPENDED); break; case APP_USBD_EVT_INST_REMOVE: /*SOF unregister: GetIdle/SetIdle support*/ ret = app_usbd_class_sof_unregister(p_inst); if (ret != NRF_SUCCESS) { break; } app_usbd_core_class_rwu_unregister(p_inst); app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_APPENDED); break; case APP_USBD_EVT_START: app_usbd_hid_state_flag_set(p_hid_ctx, APP_USBD_HID_STATE_FLAG_STARTED); break; case APP_USBD_EVT_STOP: app_usbd_hid_state_flag_clr(p_hid_ctx, APP_USBD_HID_STATE_FLAG_STARTED); break; default: ret = NRF_ERROR_NOT_SUPPORTED; break; } return ret; } app_usbd_hid_report_buffer_t * app_usbd_hid_rep_buff_in_get(app_usbd_hid_inst_t const * p_hinst) { ASSERT(p_hinst); return p_hinst->p_rep_buffer_in; } #endif // APP_USBD_CLASS_HID_ENABLED