mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
424e1e1886
This commit adds the USB HCD (Host Controller Driver) and accompanying unit tests.
1908 lines
74 KiB
C
1908 lines
74 KiB
C
// Copyright 2015-2020 Espressif Systems (Shanghai) PTE LTD
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include <string.h>
|
|
#include "sys/queue.h"
|
|
#include "esp_heap_caps.h"
|
|
#include "esp_intr_alloc.h"
|
|
#include "esp_timer.h"
|
|
#include "esp_err.h"
|
|
#include "esp_rom_gpio.h"
|
|
#include "hal/usbh_hal.h"
|
|
#include "soc/gpio_pins.h"
|
|
#include "soc/gpio_sig_map.h"
|
|
#include "driver/periph_ctrl.h"
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/task.h"
|
|
#include "freertos/semphr.h"
|
|
#include "hcd.h"
|
|
|
|
// ----------------------------------------------------- Macros --------------------------------------------------------
|
|
|
|
// --------------------- Constants -------------------------
|
|
|
|
/**
|
|
* @brief Number of transfer descriptors per transfer for various transfer types
|
|
*
|
|
* Control: Requires 3 transfer descriptors for a single transfer
|
|
* corresponding to each stage of a control transfer
|
|
* Bulk: Requires 1 transfer descriptor for each transfer
|
|
*/
|
|
#define NUM_DESC_PER_XFER_CTRL 3
|
|
#define NUM_DESC_PER_XFER_BULK 1
|
|
#define XFER_LIST_LEN_CTRL 1
|
|
#define XFER_LIST_LEN_BULK 1
|
|
|
|
#define INIT_DELAY_MS 30 //A delay of at least 25ms to enter Host mode. Make it 30ms to be safe
|
|
#define DEBOUNCE_DELAY_MS 250 //A debounce delay of 250ms
|
|
#define RESET_HOLD_MS 30 //Spec requires at least 10ms. Make it 30ms to be safe
|
|
#define RESET_RECOVERY_MS 30 //Reset recovery delay of 10ms (make it 30 ms to be safe) to allow for connected device to recover (and for port enabled interrupt to occur)
|
|
#define RESUME_HOLD_MS 30 //Spec requires at least 20ms, Make it 30ms to be safe
|
|
#define RESUME_RECOVERY_MS 20 //Resume recovery of at least 10ms. Make it 20 ms to be safe. This will include the 3 LS bit times of the EOP
|
|
|
|
#define CTRL_EP_MAX_MPS_LS 8 //Largest Maximum Packet Size for Low Speed control endpoints
|
|
#define CTRL_EP_MAX_MPS_FS 64 //Largest Maximum Packet Size for Full Speed control endpoints
|
|
|
|
#define NUM_PORTS 1 //The controller only has one port.
|
|
|
|
typedef enum {
|
|
XFER_REQ_STATE_IDLE, //The transfer request is not enqueued
|
|
XFER_REQ_STATE_PENDING, //The transfer request is enqueued and pending execution
|
|
XFER_REQ_STATE_INFLIGHT, //The transfer request is currently being executed
|
|
XFER_REQ_STATE_DONE, //The transfer request has completed executed or is retired, and is waiting to be dequeued
|
|
} xfer_req_state_t;
|
|
|
|
// -------------------- Convenience ------------------------
|
|
|
|
#define HCD_ENTER_CRITICAL_ISR() portENTER_CRITICAL_ISR(&hcd_lock)
|
|
#define HCD_EXIT_CRITICAL_ISR() portEXIT_CRITICAL_ISR(&hcd_lock)
|
|
#define HCD_ENTER_CRITICAL() portENTER_CRITICAL(&hcd_lock)
|
|
#define HCD_EXIT_CRITICAL() portEXIT_CRITICAL(&hcd_lock)
|
|
|
|
#define HCD_CHECK(cond, ret_val) ({ \
|
|
if (!(cond)) { \
|
|
return (ret_val); \
|
|
} \
|
|
})
|
|
#define HCD_CHECK_FROM_CRIT(cond, ret_val) ({ \
|
|
if (!(cond)) { \
|
|
HCD_EXIT_CRITICAL(); \
|
|
return ret_val; \
|
|
} \
|
|
})
|
|
|
|
// ------------------------------------------------------ Types --------------------------------------------------------
|
|
|
|
typedef struct xfer_req_obj xfer_req_t;
|
|
typedef struct pipe_obj pipe_t;
|
|
typedef struct port_obj port_t;
|
|
|
|
/**
|
|
* @brief Object representing an HCD transfer request
|
|
*/
|
|
struct xfer_req_obj {
|
|
TAILQ_ENTRY(xfer_req_obj) tailq_entry; //TailQ entry for pending or done tailq in pipe object
|
|
pipe_t *pipe; //Target pipe of transfer request
|
|
usb_irp_t *irp; //Target IRP
|
|
void *context; //Context variable of transfer request
|
|
xfer_req_state_t state; //Current state of the transfer request
|
|
};
|
|
|
|
/**
|
|
* @brief Object representing a pipe in the HCD layer
|
|
*/
|
|
struct pipe_obj {
|
|
//Transfer requests related
|
|
TAILQ_HEAD(tailhead_xfer_req_pend, xfer_req_obj) pend_xfer_req_tailq;
|
|
TAILQ_HEAD(tailhead_xfer_req_done, xfer_req_obj) done_xfer_req_tailq;
|
|
int num_xfer_req_pending;
|
|
int num_xfer_req_done;
|
|
xfer_req_t *inflight_xfer_req; //Pointer to the current transfer request being executed by the pipe. NULL if none.
|
|
//Port related
|
|
port_t *port; //The port to which this pipe is routed through
|
|
TAILQ_ENTRY(pipe_obj) tailq_entry; //TailQ entry for port's list of pipes
|
|
//HAl channel related
|
|
void *xfer_desc_list;
|
|
usbh_hal_chan_t *chan_obj;
|
|
usbh_hal_ep_char_t ep_char;
|
|
//Pipe status, state, and events
|
|
hcd_pipe_state_t state;
|
|
hcd_pipe_event_t last_event;
|
|
TaskHandle_t task_waiting_pipe_notif; //Task handle used for internal pipe events
|
|
union {
|
|
struct {
|
|
uint32_t waiting_xfer_done: 1;
|
|
uint32_t paused: 1;
|
|
uint32_t pipe_cmd_processing: 1;
|
|
//Flags only used by control transfers
|
|
uint32_t ctrl_data_stg_in: 1;
|
|
uint32_t ctrl_data_stg_skip: 1;
|
|
uint32_t reserved3: 3;
|
|
uint32_t xfer_desc_list_len: 8;
|
|
uint32_t reserved16: 16;
|
|
};
|
|
uint32_t val;
|
|
} flags;
|
|
//Pipe callback and context
|
|
hcd_pipe_isr_callback_t callback;
|
|
void *callback_arg;
|
|
void *context;
|
|
};
|
|
|
|
/**
|
|
* @brief Object representing a port in the HCD layer
|
|
*/
|
|
struct port_obj {
|
|
usbh_hal_context_t *hal;
|
|
//Pipes routed through this port
|
|
TAILQ_HEAD(tailhead_pipes_idle, pipe_obj) pipes_idle_tailq;
|
|
TAILQ_HEAD(tailhead_pipes_queued, pipe_obj) pipes_queued_tailq;
|
|
int num_pipes_idle;
|
|
int num_pipes_queued;
|
|
//Port status, state, and events
|
|
hcd_port_state_t state;
|
|
usb_speed_t speed;
|
|
hcd_port_event_t last_event;
|
|
TaskHandle_t task_waiting_port_notif; //Task handle used for internal port events
|
|
union {
|
|
struct {
|
|
uint32_t event_pending: 1; //The port has an event that needs to be handled
|
|
uint32_t event_processing: 1; //The port is current processing (handling) an event
|
|
uint32_t cmd_processing: 1; //Used to indicate command handling is ongoing
|
|
uint32_t waiting_all_pipes_pause: 1; //Waiting for all pipes routed through this port to be paused
|
|
uint32_t disable_requested: 1;
|
|
uint32_t conn_devc_ena: 1; //Used to indicate the port is connected to a device that has been reset
|
|
uint32_t reserved10: 10;
|
|
uint32_t num_pipes_waiting_pause: 16;
|
|
};
|
|
uint32_t val;
|
|
} flags;
|
|
bool initialized;
|
|
//Port callback and context
|
|
hcd_port_isr_callback_t callback;
|
|
void *callback_arg;
|
|
SemaphoreHandle_t port_mux;
|
|
void *context;
|
|
};
|
|
|
|
/**
|
|
* @brief Object representing the HCD
|
|
*/
|
|
typedef struct {
|
|
//Ports (Hardware only has one)
|
|
port_t *port_obj;
|
|
intr_handle_t isr_hdl;
|
|
} hcd_obj_t;
|
|
|
|
static portMUX_TYPE hcd_lock = portMUX_INITIALIZER_UNLOCKED;
|
|
static hcd_obj_t *s_hcd_obj = NULL; //Note: "s_" is for the static pointer
|
|
|
|
// ------------------------------------------------- Forward Declare ---------------------------------------------------
|
|
|
|
// ----------------------- Events --------------------------
|
|
|
|
/**
|
|
* @brief Wait for an internal event from a port
|
|
*
|
|
* @note For each port, there can only be one thread/task waiting for an internal port event
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param port Port object
|
|
*/
|
|
static void _internal_port_event_wait(port_t *port);
|
|
|
|
/**
|
|
* @brief Notify (from an ISR context) the thread/task waiting for the internal port event
|
|
*
|
|
* @param port Port object
|
|
* @return true A yield is required
|
|
* @return false Whether a yield is required or not
|
|
*/
|
|
static bool _internal_port_event_notify_from_isr(port_t *port);
|
|
|
|
/**
|
|
* @brief Wait for an internal event from a particular pipe
|
|
*
|
|
* @note For each pipe, there can only be one thread/task waiting for an internal port event
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param pipe Pipe object
|
|
*/
|
|
static void _internal_pipe_event_wait(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Notify (from an ISR context) the thread/task waiting for an internal pipe event
|
|
*
|
|
* @param pipe Pipe object
|
|
* @param from_isr Whether this is called from an ISR or not
|
|
* @return true A yield is required
|
|
* @return false Whether a yield is required or not. Always false when from_isr is also false
|
|
*/
|
|
static bool _internal_pipe_event_notify(pipe_t *pipe, bool from_isr);
|
|
|
|
// ------------------------ Port ---------------------------
|
|
|
|
/**
|
|
* @brief Invalidates all the pipes routed through a port
|
|
*
|
|
* This should be called when port or its connected device is no longer valid (e.g., the port is suddenly reset/disabled
|
|
* or the device suddenly disconnects)
|
|
*
|
|
* @note This function may run one or more callbacks, and will exit and enter the critical section to do so
|
|
*
|
|
* Entry:
|
|
* - The port or its connected device is no longer valid. This guarantees that none of the pipes will be transferring
|
|
* Exit:
|
|
* - Each pipe will have any pending transfer request moved to their respective done tailq
|
|
* - Each pipe will be put into the invalid state
|
|
* - Generate a HCD_PIPE_EVENT_INVALID event on each pipe and run their respective callbacks
|
|
*
|
|
* @param port Port object
|
|
*/
|
|
static void _port_invalidate_all_pipes(port_t *port);
|
|
|
|
/**
|
|
* @brief Pause all pipes routed through a port
|
|
*
|
|
* Call this before attempting to reset or suspend a port
|
|
*
|
|
* Entry:
|
|
* - The port is in the HCD_PORT_STATE_ENABLED state (i.e., there is a connected device which has been reset)
|
|
* Exit:
|
|
* - All pipes of the port have either paused, or are waiting to complete their inflight transfer request to pause
|
|
* - If waiting for one or more pipes, _internal_port_event_wait() must be called after this function returns
|
|
*
|
|
* @param port Port object
|
|
* @return true All pipes have been paused
|
|
* @return false Need to wait for one or more pipes to pause. Call _internal_port_event_wait() afterwards
|
|
*/
|
|
static bool _port_pause_all_pipes(port_t *port);
|
|
|
|
/**
|
|
* @brief Un-pause all pipes routed through a port
|
|
*
|
|
* Call this before after coming out of a port reset or resume.
|
|
*
|
|
* Entry:
|
|
* - The port is in the HCD_PORT_STATE_ENABLED state
|
|
* - All pipes are paused
|
|
* Exit:
|
|
* - All pipes un-paused. If those pipes have pending transfer requests, they will be started.
|
|
*
|
|
* @param port Port object
|
|
*/
|
|
static void _port_unpause_all_pipes(port_t *port);
|
|
|
|
/**
|
|
* @brief Send a reset condition on a port's bus
|
|
*
|
|
* Entry:
|
|
* - The port must be in the HCD_PORT_STATE_ENABLED or HCD_PORT_STATE_DISABLED state
|
|
* Exit:
|
|
* - Reset condition sent on the port's bus
|
|
*
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param port Port object
|
|
* @return true Reset condition successfully sent
|
|
* @return false Failed to send reset condition due to unexpected port state
|
|
*/
|
|
static bool _port_bus_reset(port_t *port);
|
|
|
|
/**
|
|
* @brief Send a suspend condition on a port's bus
|
|
*
|
|
* This function will first pause pipes routed through a port, and then send a suspend condition.
|
|
*
|
|
* Entry:
|
|
* - The port must be in the HCD_PORT_STATE_ENABLED state
|
|
* Exit:
|
|
* - All pipes paused and the port is put into the suspended state
|
|
*
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param port Port object
|
|
* @return true Suspend condition successfully sent. Port is now in the HCD_PORT_STATE_SUSPENDED state
|
|
* @return false Failed to send a suspend condition due to unexpected port state
|
|
*/
|
|
static bool _port_bus_suspend(port_t *port);
|
|
|
|
/**
|
|
* @brief Send a resume condition on a port's bus
|
|
*
|
|
* This function will send a resume condition, and then un-pause all the pipes routed through a port
|
|
*
|
|
* Entry:
|
|
* - The port must be in the HCD_PORT_STATE_SUSPENDED state
|
|
* Exit:
|
|
* - The port is put into the enabled state and all pipes un-paused
|
|
*
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param port Port object
|
|
* @return true Resume condition successfully sent. Port is now in the HCD_PORT_STATE_ENABLED state
|
|
* @return false Failed to send a resume condition due to unexpected port state.
|
|
*/
|
|
static bool _port_bus_resume(port_t *port);
|
|
|
|
/**
|
|
* @brief Disable a port
|
|
*
|
|
* Entry:
|
|
* - The port must be in the HCD_PORT_STATE_ENABLED or HCD_PORT_STATE_SUSPENDED state
|
|
* Exit:
|
|
* - All pipes paused (should already be paused if port was suspended), and the port is put into the disabled state.
|
|
*
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param port Port object
|
|
* @return true Port successfully disabled
|
|
* @return false Port to disable port due to unexpected port state
|
|
*/
|
|
static bool _port_disable(port_t *port);
|
|
|
|
/**
|
|
* @brief Debounce port after a connection or disconnection event
|
|
*
|
|
* This function should be called after a port connection or disconnect event. This function will execute a debounce
|
|
* delay then check the actual connection/disconnections state.
|
|
*
|
|
* @param port Port object
|
|
* @return true A device is connected
|
|
* @return false No device connected
|
|
*/
|
|
static bool _port_debounce(port_t *port);
|
|
|
|
// ------------------------ Pipe ---------------------------
|
|
|
|
/**
|
|
* @brief Get the next pending transfer request from the pending tailq
|
|
*
|
|
* Entry:
|
|
* - The inflight transfer request must be set to NULL (indicating the pipe currently has no inflight transfer request)
|
|
* Exit:
|
|
* - If (num_xfer_req_pending > 0), the first transfer request is removed from pend_xfer_req_tailq and and
|
|
* inflight_xfer_req is set to that transfer request.
|
|
* - If there are no more queued transfer requests, inflight_xfer_req is left as NULL
|
|
*
|
|
* @param pipe Pipe object
|
|
* @return true A pending transfer request is now set as the inflight transfer request
|
|
* @return false No more pending transfer requests
|
|
*/
|
|
static bool _pipe_get_next_xfer_req(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Return the inflight transfer request to the done tailq
|
|
*
|
|
* Entry:
|
|
* - The inflight transfer request must already have been parsed (i.e., results have been checked)
|
|
* Exit:
|
|
* - The inflight transfer request is returned to the done tailq and inflight_xfer_req is set to NULL
|
|
*
|
|
* @param pipe Pipe object
|
|
*/
|
|
static void _pipe_ret_cur_xfer_req(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Wait until a pipe's inflight transfer request is done
|
|
*
|
|
* If the pipe has an inflight transfer request, this function will block until it is done (via a internal pipe event).
|
|
* If the pipe has no inflight transfer request, this function do nothing and return immediately.
|
|
* If the pipe's state changes unexpectedely, this function will return false.
|
|
*
|
|
* @note This function is blocking (will exit and re-enter the critical section to do so)
|
|
*
|
|
* @param pipe Pipe object
|
|
* @return true Pipes inflight transfer request is done
|
|
* @return false Pipes state unexpectedly changed
|
|
*/
|
|
static bool _pipe_wait_done(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Retires all transfer requests (those that were previously inflight or pending)
|
|
*
|
|
* Retiring all transfer requests will result in any pending transfer request being moved to the done tailq. This
|
|
* function will update the IPR status of each transfer request.
|
|
* - If the retiring is self-initiated (i.e., due to a pipe command), the IRP status will be set to USB_TRANSFER_STATUS_CANCELLED.
|
|
* - If the retiring is NOT self-initiated (i.e., the pipe is no longer valid), the IRP status will be set to USB_TRANSFER_STATUS_NO_DEVICE
|
|
*
|
|
* Entry:
|
|
* - There can be no inflight transfer request (must already be parsed and returned to done queue)
|
|
* Exit:
|
|
* - If there was an inflight transfer request, it is parsed and returned to the done queue
|
|
* - If there are any pending transfer requests:
|
|
* - They are moved to the done tailq
|
|
*
|
|
* @param pipe Pipe object
|
|
* @param cancelled Are we actively Pipe retire is initialized by the user due to a command, thus transfer request are actively
|
|
* cancelled
|
|
*/
|
|
static void _pipe_retire(pipe_t *pipe, bool self_initiated);
|
|
|
|
/**
|
|
* @brief Decode a HAL channel error to the corresponding pipe event
|
|
*
|
|
* @param chan_error The HAL channel error
|
|
* @return hcd_pipe_event_t The corresponding pipe error event
|
|
*/
|
|
static inline hcd_pipe_event_t pipe_decode_error_event(usbh_hal_chan_error_t chan_error);
|
|
|
|
// ------------------ Transfer Requests --------------------
|
|
|
|
/**
|
|
* @brief Fill a transfer request into the pipe's transfer descriptor list
|
|
*
|
|
* Entry:
|
|
* - The pipe's inflight_xfer_req must be set to the next transfer request
|
|
* Exit:
|
|
* - inflight_xfer_req filled into the pipe's transfer descriptor list
|
|
* - Starting PIDs and directions set
|
|
* - Channel slot acquired. Will need to call usbh_hal_chan_activate() to actually start execution
|
|
*
|
|
* @param pipe Pipe where inflight_xfer_req is already set to the next transfer request
|
|
*/
|
|
static void _xfer_req_fill(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Continue a transfer request
|
|
*
|
|
* @note This is currently only used for control transfers
|
|
*
|
|
* @param pipe Pipe where inflight_xfer_req contains the transfer request to continue
|
|
*/
|
|
static void _xfer_req_continue(pipe_t *pipe);
|
|
|
|
/**
|
|
* @brief Parse the results of a pipe's transfer descriptor list into a transfer request
|
|
*
|
|
* Entry:
|
|
* - The pipe must have stop transferring either due a channel event or a port disconnection.
|
|
* - The pipe's state and last_event must be updated before parsing the transfer request as
|
|
* they will used to determine the resuult of the transfer request
|
|
* Exit:
|
|
* - The pipe's inflight_xfer_req is filled with result of the transfer request (i.e., the underlying IRP has its status set)
|
|
*
|
|
* @param pipe Pipe where inflight_xfer_req contains the completed transfer request
|
|
* @param error_occurred Are we parsing after the pipe had an error (or has become invalid)
|
|
*/
|
|
static void _xfer_req_parse(pipe_t *pipe, bool error_occurred);
|
|
|
|
// ----------------------------------------------- Interrupt Handling --------------------------------------------------
|
|
|
|
// ------------------- Internal Event ----------------------
|
|
|
|
static void _internal_port_event_wait(port_t *port)
|
|
{
|
|
//There must NOT be another thread/task already waiting for an internal event
|
|
assert(port->task_waiting_port_notif == NULL);
|
|
port->task_waiting_port_notif = xTaskGetCurrentTaskHandle();
|
|
HCD_EXIT_CRITICAL();
|
|
//Wait to be notified from ISR
|
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
HCD_ENTER_CRITICAL();
|
|
port->task_waiting_port_notif = NULL;
|
|
}
|
|
|
|
static bool _internal_port_event_notify_from_isr(port_t *port)
|
|
{
|
|
//There must be a thread/task waiting for an internal event
|
|
assert(port->task_waiting_port_notif != NULL);
|
|
BaseType_t xTaskWoken = pdFALSE;
|
|
//Unblock the thread/task waiting for the notification
|
|
HCD_EXIT_CRITICAL_ISR();
|
|
vTaskNotifyGiveFromISR(port->task_waiting_port_notif, &xTaskWoken);
|
|
HCD_ENTER_CRITICAL_ISR();
|
|
return (xTaskWoken == pdTRUE);
|
|
}
|
|
|
|
static void _internal_pipe_event_wait(pipe_t *pipe)
|
|
{
|
|
//There must NOT be another thread/task already waiting for an internal event
|
|
assert(pipe->task_waiting_pipe_notif == NULL);
|
|
pipe->task_waiting_pipe_notif = xTaskGetCurrentTaskHandle();
|
|
HCD_EXIT_CRITICAL();
|
|
//Wait to be notified from ISR
|
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
HCD_ENTER_CRITICAL();
|
|
pipe->task_waiting_pipe_notif = NULL;
|
|
}
|
|
|
|
static bool _internal_pipe_event_notify(pipe_t *pipe, bool from_isr)
|
|
{
|
|
//There must be a thread/task waiting for an internal event
|
|
assert(pipe->task_waiting_pipe_notif != NULL);
|
|
bool ret;
|
|
if (from_isr) {
|
|
BaseType_t xTaskWoken = pdFALSE;
|
|
HCD_EXIT_CRITICAL_ISR();
|
|
//Unblock the thread/task waiting for the pipe notification
|
|
vTaskNotifyGiveFromISR(pipe->task_waiting_pipe_notif, &xTaskWoken);
|
|
HCD_ENTER_CRITICAL_ISR();
|
|
ret = (xTaskWoken == pdTRUE);
|
|
} else {
|
|
HCD_EXIT_CRITICAL();
|
|
xTaskNotifyGive(pipe->task_waiting_pipe_notif);
|
|
HCD_ENTER_CRITICAL();
|
|
ret = false;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// ----------------- Interrupt Handlers --------------------
|
|
|
|
/**
|
|
* @brief Handle a HAL port interrupt and obtain the corresponding port event
|
|
*
|
|
* @param[in] port Port object
|
|
* @param[in] hal_port_event The HAL port event
|
|
* @param[out] yield Set to true if a yield is required as a result of handling the interrupt
|
|
* @return hcd_port_event_t Returns a port event, or HCD_PORT_EVENT_NONE if no port event occurred
|
|
*/
|
|
static hcd_port_event_t _intr_hdlr_hprt(port_t *port, usbh_hal_port_event_t hal_port_event, bool *yield)
|
|
{
|
|
hcd_port_event_t port_event = HCD_PORT_EVENT_NONE;
|
|
switch (hal_port_event) {
|
|
case USBH_HAL_PORT_EVENT_CONN: {
|
|
//Don't update state immediately, we still need to debounce.
|
|
port_event = HCD_PORT_EVENT_CONNECTION;
|
|
break;
|
|
}
|
|
case USBH_HAL_PORT_EVENT_DISCONN: {
|
|
if (port->flags.conn_devc_ena) {
|
|
//The port was previously enabled, so this is a sudden disconenction
|
|
port->state = HCD_PORT_STATE_RECOVERY;
|
|
port_event = HCD_PORT_EVENT_SUDDEN_DISCONN;
|
|
} else {
|
|
//For normal disconnections, don't update state immediately as we still need to debounce.
|
|
port_event = HCD_PORT_EVENT_DISCONNECTION;
|
|
}
|
|
port->flags.conn_devc_ena = 0;
|
|
break;
|
|
}
|
|
case USBH_HAL_PORT_EVENT_ENABLED: {
|
|
usbh_hal_port_enable(port->hal); //Initialize remaining host port registers
|
|
port->speed = usbh_hal_port_get_conn_speed(port->hal);
|
|
port->state = HCD_PORT_STATE_ENABLED;
|
|
port->flags.conn_devc_ena = 1;
|
|
//This was triggered by a command, so no event needs to be propagated.
|
|
break;
|
|
}
|
|
case USBH_HAL_PORT_EVENT_DISABLED: {
|
|
port->flags.conn_devc_ena = 0;
|
|
//Disabled could be due to a disable request or reset request, or due to a port error
|
|
if (port->state != HCD_PORT_STATE_RESETTING) { //Ignore the disable event if it's due to a reset request
|
|
port->state = HCD_PORT_STATE_DISABLED;
|
|
if (port->flags.disable_requested) {
|
|
//Disabled by request (i.e. by port command). Generate an internal event
|
|
port->flags.disable_requested = 0;
|
|
*yield |= _internal_port_event_notify_from_isr(port);
|
|
} else {
|
|
//Disabled due to a port error
|
|
port_event = HCD_PORT_EVENT_ERROR;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case USBH_HAL_PORT_EVENT_OVRCUR:
|
|
case USBH_HAL_PORT_EVENT_OVRCUR_CLR: { //Could occur if a quick overcurrent then clear happens
|
|
if (port->state != HCD_PORT_STATE_NOT_POWERED) {
|
|
//We need to power OFF the port to protect it
|
|
usbh_hal_port_toggle_power(port->hal, false);
|
|
port->state = HCD_PORT_STATE_NOT_POWERED;
|
|
port_event = HCD_PORT_EVENT_OVERCURRENT;
|
|
}
|
|
port->flags.conn_devc_ena = 0;
|
|
break;
|
|
}
|
|
default: {
|
|
abort();
|
|
break;
|
|
}
|
|
}
|
|
return port_event;
|
|
}
|
|
|
|
/**
|
|
* @brief Handles a HAL channel interrupt
|
|
*
|
|
* This function should be called on a HAL channel when it has an interrupt. Most HAL channel events will correspond to
|
|
* to a pipe event, but not always. This function will store the pipe event and return a pipe object pointer if a pipe
|
|
* event occurred, or return NULL otherwise.
|
|
*
|
|
* @param[in] chan_obj Pointer to HAL channel object with interrupt
|
|
* @param[out] yield Set to true if a yield is required as a result of handling the interrupt
|
|
* @return hcd_pipe_event_t The pipe event
|
|
*/
|
|
static hcd_pipe_event_t _intr_hdlr_chan(pipe_t *pipe, usbh_hal_chan_t *chan_obj, bool *yield)
|
|
{
|
|
usbh_hal_chan_event_t chan_event = usbh_hal_chan_decode_intr(chan_obj);
|
|
hcd_pipe_event_t event = HCD_PIPE_EVENT_NONE;
|
|
//Check the the pipe's port still has a connected and enabled device before processing the interrupt
|
|
if (!pipe->port->flags.conn_devc_ena) {
|
|
return event; //Treat as a no event.
|
|
}
|
|
|
|
switch (chan_event) {
|
|
case USBH_HAL_CHAN_EVENT_SLOT_DONE: {
|
|
//An entire transfer descriptor list has completed execution
|
|
pipe->last_event = HCD_PIPE_EVENT_XFER_REQ_DONE;
|
|
event = HCD_PIPE_EVENT_XFER_REQ_DONE;
|
|
_xfer_req_parse(pipe, false); //Parse results of transfer request
|
|
_pipe_ret_cur_xfer_req(pipe); //Return the transfer request to the pipe's done tailq
|
|
if (pipe->flags.waiting_xfer_done) {
|
|
//A port/pipe command is waiting for this pipe to complete its transfer. So don't load the next transfer
|
|
pipe->flags.waiting_xfer_done = 0;
|
|
if (pipe->port->flags.waiting_all_pipes_pause) {
|
|
//Port command is waiting for all pipes to be paused
|
|
pipe->flags.paused = 1;
|
|
pipe->port->flags.num_pipes_waiting_pause--;
|
|
if (pipe->port->flags.num_pipes_waiting_pause == 0) {
|
|
//All pipes have finished pausing, Notify the blocked port command
|
|
pipe->port->flags.waiting_all_pipes_pause = 0;
|
|
*yield |= _internal_port_event_notify_from_isr(pipe->port);
|
|
}
|
|
} else {
|
|
//Pipe command is waiting for transfer to complete
|
|
*yield |= _internal_pipe_event_notify(pipe, true);
|
|
}
|
|
} else if (_pipe_get_next_xfer_req(pipe)) {
|
|
//Fill the descriptor list with the transfer request and start the transfer
|
|
_xfer_req_fill(pipe);
|
|
usbh_hal_chan_activate(chan_obj, 0); //Start with the first descriptor
|
|
}
|
|
break;
|
|
}
|
|
case USBH_HAL_CHAN_EVENT_SLOT_HALT: {
|
|
//A transfer descriptor list has partially completed. This currently only happens on control pipes
|
|
assert(pipe->ep_char.type == USB_XFER_TYPE_CTRL);
|
|
_xfer_req_continue(pipe); //Continue the transfer request.
|
|
//We are continuing a transfer, so no event has occurred
|
|
break;
|
|
}
|
|
case USBH_HAL_CHAN_EVENT_ERROR: {
|
|
//Get and store the pipe error event
|
|
usbh_hal_chan_error_t chan_error = usbh_hal_chan_get_error(chan_obj);
|
|
usbh_hal_chan_clear_error(chan_obj);
|
|
pipe->last_event = pipe_decode_error_event(chan_error);
|
|
event = pipe->last_event;
|
|
pipe->state = HCD_PIPE_STATE_HALTED;
|
|
//Parse the failed transfer request and update it's IRP status
|
|
_xfer_req_parse(pipe, true);
|
|
_pipe_ret_cur_xfer_req(pipe); //Return the transfer request to the pipe's done tailq
|
|
break;
|
|
}
|
|
case USBH_HAL_CHAN_EVENT_HALT_REQ: //We currently don't halt request so this event should never occur
|
|
default:
|
|
abort();
|
|
break;
|
|
}
|
|
|
|
return event;
|
|
}
|
|
|
|
/**
|
|
* @brief Main interrupt handler
|
|
*
|
|
* - Handle all HPRT (Host Port) related interrupts first as they may change the
|
|
* state of the driver (e.g., a disconnect event)
|
|
* - If any channels (pipes) have pending interrupts, handle them one by one
|
|
* - The HCD has not blocking functions, so the user's ISR callback is run to
|
|
* allow the users to send whatever OS primitives they need.
|
|
* @param arg
|
|
*/
|
|
static void intr_hdlr_main(void *arg)
|
|
{
|
|
port_t *port = (port_t *)arg;
|
|
bool yield = false;
|
|
|
|
HCD_ENTER_CRITICAL_ISR();
|
|
usbh_hal_port_event_t hal_port_evt = usbh_hal_decode_intr(port->hal);
|
|
if (hal_port_evt == USBH_HAL_PORT_EVENT_CHAN) {
|
|
//Channel event. Cycle through each pending channel
|
|
usbh_hal_chan_t *chan_obj = usbh_hal_get_chan_pending_intr(port->hal);
|
|
while (chan_obj != NULL) {
|
|
pipe_t *pipe = (pipe_t *)usbh_hal_chan_get_context(chan_obj);
|
|
hcd_pipe_event_t event = _intr_hdlr_chan(pipe, chan_obj, &yield);
|
|
//Run callback if a pipe event has occurred and the pipe also has a callback
|
|
if (event != HCD_PIPE_EVENT_NONE && pipe->callback != NULL) {
|
|
HCD_EXIT_CRITICAL_ISR();
|
|
yield |= pipe->callback((hcd_pipe_handle_t)pipe, event, pipe->callback_arg, true);
|
|
HCD_ENTER_CRITICAL_ISR();
|
|
}
|
|
//Check for more channels with pending interrupts. Returns NULL if there are no more
|
|
chan_obj = usbh_hal_get_chan_pending_intr(port->hal);
|
|
}
|
|
} else if (hal_port_evt != USBH_HAL_PORT_EVENT_NONE) { //Port event
|
|
hcd_port_event_t port_event = _intr_hdlr_hprt(port, hal_port_evt, &yield);
|
|
if (port_event != HCD_PORT_EVENT_NONE) {
|
|
port->last_event = port_event;
|
|
port->flags.event_pending = 1;
|
|
if (port->callback != NULL) {
|
|
HCD_EXIT_CRITICAL_ISR();
|
|
yield |= port->callback((hcd_port_handle_t)port, port_event, port->callback_arg, true);
|
|
HCD_ENTER_CRITICAL_ISR();
|
|
}
|
|
}
|
|
}
|
|
HCD_EXIT_CRITICAL_ISR();
|
|
|
|
if (yield) {
|
|
portYIELD_FROM_ISR();
|
|
}
|
|
}
|
|
|
|
// --------------------------------------------- Host Controller Driver ------------------------------------------------
|
|
|
|
static port_t *port_obj_alloc(void)
|
|
{
|
|
port_t *port = calloc(1, sizeof(port_t));
|
|
usbh_hal_context_t *hal = malloc(sizeof(usbh_hal_context_t));
|
|
SemaphoreHandle_t port_mux = xSemaphoreCreateMutex();
|
|
if (port == NULL || hal == NULL || port_mux == NULL) {
|
|
free(port);
|
|
free(hal);
|
|
if (port_mux != NULL) {
|
|
vSemaphoreDelete(port_mux);
|
|
}
|
|
return NULL;
|
|
}
|
|
port->hal = hal;
|
|
port->port_mux = port_mux;
|
|
return port;
|
|
}
|
|
|
|
static void port_obj_free(port_t *port)
|
|
{
|
|
if (port == NULL) {
|
|
return;
|
|
}
|
|
vSemaphoreDelete(port->port_mux);
|
|
free(port->hal);
|
|
free(port);
|
|
}
|
|
|
|
// ----------------------- Public --------------------------
|
|
|
|
esp_err_t hcd_install(const hcd_config_t *config)
|
|
{
|
|
HCD_ENTER_CRITICAL();
|
|
HCD_CHECK_FROM_CRIT(s_hcd_obj == NULL, ESP_ERR_INVALID_STATE);
|
|
HCD_EXIT_CRITICAL();
|
|
|
|
esp_err_t err_ret;
|
|
//Allocate memory and resources for driver object and all port objects
|
|
hcd_obj_t *p_hcd_obj_dmy = calloc(1, sizeof(hcd_obj_t));
|
|
if (p_hcd_obj_dmy == NULL) {
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
//Allocate resources for each port (there's only one)
|
|
p_hcd_obj_dmy->port_obj = port_obj_alloc();
|
|
esp_err_t intr_alloc_ret = esp_intr_alloc(ETS_USB_INTR_SOURCE,
|
|
config->intr_flags | ESP_INTR_FLAG_INTRDISABLED, //The interruupt must be disabled until the port is initialized
|
|
intr_hdlr_main,
|
|
(void *)p_hcd_obj_dmy->port_obj,
|
|
&p_hcd_obj_dmy->isr_hdl);
|
|
if (p_hcd_obj_dmy->port_obj == NULL) {
|
|
err_ret = ESP_ERR_NO_MEM;
|
|
}
|
|
if (intr_alloc_ret != ESP_OK) {
|
|
err_ret = intr_alloc_ret;
|
|
goto err;
|
|
}
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
if (s_hcd_obj != NULL) {
|
|
HCD_EXIT_CRITICAL();
|
|
err_ret = ESP_ERR_INVALID_STATE;
|
|
goto err;
|
|
}
|
|
s_hcd_obj = p_hcd_obj_dmy;
|
|
//Set HW prereqs for each port (there's only one)
|
|
periph_module_enable(PERIPH_USB_MODULE);
|
|
periph_module_reset(PERIPH_USB_MODULE);
|
|
/*
|
|
Configure GPIOS for Host mode operation using internal PHY
|
|
- Forces ID to GND for A side
|
|
- Forces B Valid to GND as we are A side host
|
|
- Forces VBUS Valid to HIGH
|
|
- Froces A Valid to HIGH
|
|
*/
|
|
esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ZERO_INPUT, USB_OTG_IDDIG_IN_IDX, false);
|
|
esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ZERO_INPUT, USB_SRP_BVALID_IN_IDX, false);
|
|
esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ONE_INPUT, USB_OTG_VBUSVALID_IN_IDX, false);
|
|
esp_rom_gpio_connect_in_signal(GPIO_MATRIX_CONST_ONE_INPUT, USB_OTG_AVALID_IN_IDX, false);
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_OK;
|
|
err:
|
|
if (intr_alloc_ret == ESP_OK) {
|
|
esp_intr_free(p_hcd_obj_dmy->isr_hdl);
|
|
}
|
|
port_obj_free(p_hcd_obj_dmy->port_obj);
|
|
free(p_hcd_obj_dmy);
|
|
return err_ret;
|
|
}
|
|
|
|
esp_err_t hcd_uninstall(void)
|
|
{
|
|
HCD_ENTER_CRITICAL();
|
|
//Check that all ports have been disabled (theres only one)
|
|
if (s_hcd_obj == NULL || s_hcd_obj->port_obj->initialized) {
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
periph_module_disable(PERIPH_USB_MODULE);
|
|
hcd_obj_t *p_hcd_obj_dmy = s_hcd_obj;
|
|
s_hcd_obj = NULL;
|
|
HCD_EXIT_CRITICAL();
|
|
|
|
//Free resources
|
|
port_obj_free(p_hcd_obj_dmy->port_obj);
|
|
esp_intr_free(p_hcd_obj_dmy->isr_hdl);
|
|
free(p_hcd_obj_dmy);
|
|
return ESP_OK;
|
|
}
|
|
|
|
// ------------------------------------------------------ Port ---------------------------------------------------------
|
|
|
|
// ----------------------- Private -------------------------
|
|
|
|
static void _port_invalidate_all_pipes(port_t *port)
|
|
{
|
|
//This function should only be called when the port is invalid
|
|
assert(!port->flags.conn_devc_ena);
|
|
pipe_t *pipe;
|
|
//Process all pipes that have queued transfer requests
|
|
TAILQ_FOREACH(pipe, &port->pipes_queued_tailq, tailq_entry) {
|
|
//Mark the pipe as invalid and set an invalid event
|
|
pipe->state = HCD_PIPE_STATE_INVALID;
|
|
pipe->last_event = HCD_PIPE_EVENT_INVALID;
|
|
//If the pipe had an inflight transfer, parse and return it
|
|
if (pipe->inflight_xfer_req != NULL) {
|
|
_xfer_req_parse(pipe, true);
|
|
_pipe_ret_cur_xfer_req(pipe);
|
|
}
|
|
//Retire any remaining transfer requests
|
|
_pipe_retire(pipe, false);
|
|
if (pipe->task_waiting_pipe_notif != NULL) {
|
|
//Unblock the thread/task waiting for a notification from the pipe as the pipe is no longer valid.
|
|
_internal_pipe_event_notify(pipe, false);
|
|
}
|
|
if (pipe->callback != NULL) {
|
|
HCD_EXIT_CRITICAL();
|
|
(void) pipe->callback((hcd_pipe_handle_t)pipe, HCD_PIPE_EVENT_INVALID, pipe->callback_arg, false);
|
|
HCD_ENTER_CRITICAL();
|
|
}
|
|
}
|
|
//Process all idle pipes
|
|
TAILQ_FOREACH(pipe, &port->pipes_idle_tailq, tailq_entry) {
|
|
//Mark pipe as invalid and call its callback
|
|
pipe->state = HCD_PIPE_STATE_INVALID;
|
|
pipe->last_event = HCD_PIPE_EVENT_INVALID;
|
|
if (pipe->callback != NULL) {
|
|
HCD_EXIT_CRITICAL();
|
|
(void) pipe->callback((hcd_pipe_handle_t)pipe, HCD_PIPE_EVENT_INVALID, pipe->callback_arg, false);
|
|
HCD_ENTER_CRITICAL();
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool _port_pause_all_pipes(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_ENABLED);
|
|
pipe_t *pipe;
|
|
int num_pipes_waiting_done = 0;
|
|
//Process all pipes that have queued transfer requests
|
|
TAILQ_FOREACH(pipe, &port->pipes_queued_tailq, tailq_entry) {
|
|
if (pipe->inflight_xfer_req != NULL) {
|
|
//Pipe has an inflight transfer. Indicate to the pipe we are waiting the transfer to complete
|
|
pipe->flags.waiting_xfer_done = 1;
|
|
num_pipes_waiting_done++;
|
|
} else {
|
|
//No inflight transfer so no need to wait
|
|
pipe->flags.paused = 1;
|
|
}
|
|
}
|
|
//Process all idle pipes. They don't have queue transfer so just mark them as paused
|
|
TAILQ_FOREACH(pipe, &port->pipes_idle_tailq, tailq_entry) {
|
|
pipe->flags.paused = 1;
|
|
}
|
|
if (num_pipes_waiting_done > 0) {
|
|
//Indicate we need to wait for one or more pipes to complete their transfers
|
|
port->flags.num_pipes_waiting_pause = num_pipes_waiting_done;
|
|
port->flags.waiting_all_pipes_pause = 1;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void _port_unpause_all_pipes(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_ENABLED);
|
|
pipe_t *pipe;
|
|
//Process all idle pipes. They don't have queue transfer so just mark them as un-paused
|
|
TAILQ_FOREACH(pipe, &port->pipes_idle_tailq, tailq_entry) {
|
|
pipe->flags.paused = 0;
|
|
}
|
|
//Process all pipes that have queued transfer requests
|
|
TAILQ_FOREACH(pipe, &port->pipes_queued_tailq, tailq_entry) {
|
|
pipe->flags.paused = 0;
|
|
//If the pipe has more pending transfer request, start them.
|
|
if (_pipe_get_next_xfer_req(pipe)) {
|
|
_xfer_req_fill(pipe);
|
|
usbh_hal_chan_activate(pipe->chan_obj, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool _port_bus_reset(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_ENABLED || port->state == HCD_PORT_STATE_DISABLED);
|
|
//Put and hold the bus in the reset state. If the port was previously enabled, a disabled event will occur after this
|
|
port->state = HCD_PORT_STATE_RESETTING;
|
|
usbh_hal_port_toggle_reset(port->hal, true);
|
|
HCD_EXIT_CRITICAL();
|
|
vTaskDelay(pdMS_TO_TICKS(RESET_HOLD_MS));
|
|
HCD_ENTER_CRITICAL();
|
|
if (port->state != HCD_PORT_STATE_RESETTING) {
|
|
//The port state has unexpectedly changed
|
|
goto bailout;
|
|
}
|
|
//Return the bus to the idle state and hold it for the required reset recovery time. Port enabled event should occur
|
|
usbh_hal_port_toggle_reset(port->hal, false);
|
|
HCD_EXIT_CRITICAL();
|
|
vTaskDelay(pdMS_TO_TICKS(RESET_RECOVERY_MS));
|
|
HCD_ENTER_CRITICAL();
|
|
if (port->state != HCD_PORT_STATE_ENABLED || !port->flags.conn_devc_ena) {
|
|
//The port state has unexpectedly changed
|
|
goto bailout;
|
|
}
|
|
return true;
|
|
bailout:
|
|
return false;
|
|
}
|
|
|
|
static bool _port_bus_suspend(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_ENABLED);
|
|
//Pause all pipes before suspending the bus
|
|
if (!_port_pause_all_pipes(port)) {
|
|
//Need to wait for some pipes to pause. Wait for notification from ISR
|
|
_internal_port_event_wait(port);
|
|
if (port->state != HCD_PORT_STATE_ENABLED || !port->flags.conn_devc_ena) {
|
|
//Port state unexpectedley changed
|
|
goto bailout;
|
|
}
|
|
}
|
|
//All pipes are guaranteed paused at this point. Proceed to suspend the port
|
|
usbh_hal_port_suspend(port->hal);
|
|
port->state = HCD_PORT_STATE_SUSPENDED;
|
|
return true;
|
|
bailout:
|
|
return false;
|
|
}
|
|
|
|
static bool _port_bus_resume(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_SUSPENDED);
|
|
//Put and hold the bus in the K state.
|
|
usbh_hal_port_toggle_resume(port->hal, true);
|
|
port->state = HCD_PORT_STATE_RESUMING;
|
|
HCD_EXIT_CRITICAL();
|
|
vTaskDelay(pdMS_TO_TICKS(RESUME_HOLD_MS));
|
|
HCD_ENTER_CRITICAL();
|
|
//Return and hold the bus to the J state (as port of the LS EOP)
|
|
usbh_hal_port_toggle_resume(port->hal, false);
|
|
if (port->state != HCD_PORT_STATE_RESUMING || !port->flags.conn_devc_ena) {
|
|
//Port state unexpectedley changed
|
|
goto bailout;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
vTaskDelay(pdMS_TO_TICKS(RESUME_RECOVERY_MS));
|
|
HCD_ENTER_CRITICAL();
|
|
if (port->state != HCD_PORT_STATE_RESUMING || !port->flags.conn_devc_ena) {
|
|
//Port state unexpectedley changed
|
|
goto bailout;
|
|
}
|
|
port->state = HCD_PORT_STATE_ENABLED;
|
|
_port_unpause_all_pipes(port);
|
|
return true;
|
|
bailout:
|
|
return false;
|
|
}
|
|
|
|
static bool _port_disable(port_t *port)
|
|
{
|
|
assert(port->state == HCD_PORT_STATE_ENABLED || port->state == HCD_PORT_STATE_SUSPENDED);
|
|
if (port->state == HCD_PORT_STATE_ENABLED) {
|
|
//There may be pipes that are still transferring, so pause them.
|
|
if (!_port_pause_all_pipes(port)) {
|
|
//Need to wait for some pipes to pause. Wait for notification from ISR
|
|
_internal_port_event_wait(port);
|
|
if (port->state != HCD_PORT_STATE_ENABLED || !port->flags.conn_devc_ena) {
|
|
//Port state unexpectedley changed
|
|
goto bailout;
|
|
}
|
|
}
|
|
}
|
|
//All pipes are guaranteed paused at this point. Proceed to suspend the port. This should trigger an internal event
|
|
port->flags.disable_requested = 1;
|
|
usbh_hal_port_disable(port->hal);
|
|
_internal_port_event_wait(port);
|
|
if (port->state != HCD_PORT_STATE_DISABLED) {
|
|
//Port state unexpectedley changed
|
|
goto bailout;
|
|
}
|
|
_port_invalidate_all_pipes(port);
|
|
return true;
|
|
bailout:
|
|
return false;
|
|
}
|
|
|
|
static bool _port_debounce(port_t *port)
|
|
{
|
|
if (port->state == HCD_PORT_STATE_NOT_POWERED) {
|
|
//Disconnect event due to power off, no need to debounce or update port state.
|
|
return false;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
vTaskDelay(pdMS_TO_TICKS(DEBOUNCE_DELAY_MS));
|
|
HCD_ENTER_CRITICAL();
|
|
//Check the post-debounce state of the bus (i.e., whether it's actually connected/disconnected)
|
|
bool is_connected = usbh_hal_port_check_if_connected(port->hal);
|
|
if (is_connected) {
|
|
port->state = HCD_PORT_STATE_DISABLED;
|
|
} else {
|
|
port->state = HCD_PORT_STATE_DISCONNECTED;
|
|
}
|
|
//Disable debounce lock
|
|
usbh_hal_disable_debounce_lock(port->hal);
|
|
return is_connected;
|
|
}
|
|
|
|
// ----------------------- Public --------------------------
|
|
|
|
esp_err_t hcd_port_init(int port_number, hcd_port_config_t *port_config, hcd_port_handle_t *port_hdl)
|
|
{
|
|
HCD_CHECK(port_number > 0 && port_config != NULL && port_hdl != NULL, ESP_ERR_INVALID_ARG);
|
|
HCD_CHECK(port_number <= NUM_PORTS, ESP_ERR_NOT_FOUND);
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
HCD_CHECK_FROM_CRIT(s_hcd_obj != NULL && !s_hcd_obj->port_obj->initialized, ESP_ERR_INVALID_STATE);
|
|
//Port object memory and resources (such as mutex) already be allocated. Just need to initialize necessary fields only
|
|
port_t *port_obj = s_hcd_obj->port_obj;
|
|
TAILQ_INIT(&port_obj->pipes_idle_tailq);
|
|
TAILQ_INIT(&port_obj->pipes_queued_tailq);
|
|
port_obj->state = HCD_PORT_STATE_NOT_POWERED;
|
|
port_obj->last_event = HCD_PORT_EVENT_NONE;
|
|
port_obj->callback = port_config->callback;
|
|
port_obj->callback_arg = port_config->callback_arg;
|
|
port_obj->context = port_config->context;
|
|
usbh_hal_init(port_obj->hal);
|
|
port_obj->initialized = true;
|
|
esp_intr_enable(s_hcd_obj->isr_hdl);
|
|
*port_hdl = (hcd_port_handle_t)port_obj;
|
|
HCD_EXIT_CRITICAL();
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(INIT_DELAY_MS)); //Need a short delay before host mode takes effect
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t hcd_port_deinit(hcd_port_handle_t port_hdl)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
HCD_CHECK_FROM_CRIT(s_hcd_obj != NULL && port->initialized
|
|
&& port->num_pipes_idle == 0 && port->num_pipes_queued == 0
|
|
&& (port->state == HCD_PORT_STATE_NOT_POWERED || port->state == HCD_PORT_STATE_RECOVERY)
|
|
&& port->flags.val == 0 && port->task_waiting_port_notif == NULL,
|
|
ESP_ERR_INVALID_STATE);
|
|
port->initialized = false;
|
|
esp_intr_disable(s_hcd_obj->isr_hdl);
|
|
usbh_hal_deinit(port->hal);
|
|
HCD_EXIT_CRITICAL();
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t hcd_port_command(hcd_port_handle_t port_hdl, hcd_port_cmd_t command)
|
|
{
|
|
esp_err_t ret = ESP_ERR_INVALID_STATE;
|
|
port_t *port = (port_t *)port_hdl;
|
|
xSemaphoreTake(port->port_mux, portMAX_DELAY);
|
|
HCD_ENTER_CRITICAL();
|
|
if (port->initialized && !port->flags.event_pending) { //Port events need to be handled first before issuing a command
|
|
port->flags.cmd_processing = 1;
|
|
switch (command) {
|
|
case HCD_PORT_CMD_POWER_ON: {
|
|
//Port can only be powered on if currently unpowered
|
|
if (port->state == HCD_PORT_STATE_NOT_POWERED) {
|
|
port->state = HCD_PORT_STATE_DISCONNECTED;
|
|
usbh_hal_port_start(port->hal);
|
|
usbh_hal_port_toggle_power(port->hal, true);
|
|
ret = ESP_OK;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_CMD_POWER_OFF: {
|
|
//Port can only be unpowered if already powered
|
|
if (port->state != HCD_PORT_STATE_NOT_POWERED) {
|
|
port->state = HCD_PORT_STATE_NOT_POWERED;
|
|
usbh_hal_port_stop(port->hal);
|
|
usbh_hal_port_toggle_power(port->hal, false);
|
|
//If a device is currently connected, this should trigger a disconnect event
|
|
ret = ESP_OK;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_CMD_RESET: {
|
|
//Port can only a reset when it is in the enabled or disabled states (in case of new connection)
|
|
if (port->state == HCD_PORT_STATE_ENABLED || port->state == HCD_PORT_STATE_DISABLED) {
|
|
ret = (_port_bus_reset(port)) ? ESP_OK : ESP_ERR_INVALID_RESPONSE;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_CMD_SUSPEND: {
|
|
//Port can only be suspended if already in the enabled state
|
|
if (port->state == HCD_PORT_STATE_ENABLED) {
|
|
ret = (_port_bus_suspend(port)) ? ESP_OK : ESP_ERR_INVALID_RESPONSE;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_CMD_RESUME: {
|
|
//Port can only be resumed if already suspended
|
|
if (port->state == HCD_PORT_STATE_SUSPENDED) {
|
|
ret = (_port_bus_resume(port)) ? ESP_OK : ESP_ERR_INVALID_RESPONSE;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_CMD_DISABLE: {
|
|
//Can only disable the port when already enabled or suspended
|
|
if (port->state == HCD_PORT_STATE_ENABLED || port->state == HCD_PORT_STATE_SUSPENDED) {
|
|
ret = (_port_disable(port)) ? ESP_OK : ESP_ERR_INVALID_RESPONSE;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
port->flags.cmd_processing = 0;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
xSemaphoreGive(port->port_mux);
|
|
return ret;
|
|
}
|
|
|
|
hcd_port_state_t hcd_port_get_state(hcd_port_handle_t port_hdl)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
hcd_port_state_t ret;
|
|
HCD_ENTER_CRITICAL();
|
|
ret = port->state;
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
esp_err_t hcd_port_get_speed(hcd_port_handle_t port_hdl, usb_speed_t *speed)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
HCD_CHECK(speed != NULL, ESP_ERR_INVALID_ARG);
|
|
HCD_ENTER_CRITICAL();
|
|
//Device speed is only valid if there is a resetted device connected to the port
|
|
HCD_CHECK_FROM_CRIT(port->flags.conn_devc_ena, ESP_ERR_INVALID_STATE);
|
|
*speed = usbh_hal_port_get_conn_speed(port->hal);
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_OK;
|
|
}
|
|
|
|
hcd_port_event_t hcd_port_handle_event(hcd_port_handle_t port_hdl)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
hcd_port_event_t ret = HCD_PORT_EVENT_NONE;
|
|
xSemaphoreTake(port->port_mux, portMAX_DELAY);
|
|
HCD_ENTER_CRITICAL();
|
|
if (port->initialized && port->flags.event_pending) {
|
|
port->flags.event_pending = 0;
|
|
port->flags.event_processing = 1;
|
|
ret = port->last_event;
|
|
switch (ret) {
|
|
case HCD_PORT_EVENT_CONNECTION: {
|
|
if (_port_debounce(port)) {
|
|
ret = HCD_PORT_EVENT_CONNECTION;
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PORT_EVENT_DISCONNECTION:
|
|
if (_port_debounce(port)) {
|
|
//A device is still connected, so it was just a debounce
|
|
port->state = HCD_PORT_STATE_DISABLED;
|
|
ret = HCD_PORT_EVENT_NONE;
|
|
} else {
|
|
//No device conencted after debounce delay. This is an actual disconenction
|
|
port->state = HCD_PORT_STATE_DISCONNECTED;
|
|
ret = HCD_PORT_EVENT_DISCONNECTION;
|
|
}
|
|
break;
|
|
case HCD_PORT_EVENT_ERROR:
|
|
case HCD_PORT_EVENT_OVERCURRENT:
|
|
case HCD_PORT_EVENT_SUDDEN_DISCONN: {
|
|
_port_invalidate_all_pipes(port);
|
|
break;
|
|
}
|
|
default: {
|
|
break;
|
|
}
|
|
}
|
|
port->flags.event_processing = 0;
|
|
} else {
|
|
ret = HCD_PORT_EVENT_NONE;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
xSemaphoreGive(port->port_mux);
|
|
return ret;
|
|
}
|
|
|
|
esp_err_t hcd_port_recover(hcd_port_handle_t port_hdl)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
HCD_ENTER_CRITICAL();
|
|
HCD_CHECK_FROM_CRIT(s_hcd_obj != NULL && port->initialized && port->state == HCD_PORT_STATE_RECOVERY
|
|
&& port->num_pipes_idle == 0 && port->num_pipes_queued == 0
|
|
&& port->flags.val == 0 && port->task_waiting_port_notif == NULL,
|
|
ESP_ERR_INVALID_STATE);
|
|
//We are about to do a soft reset on the peripheral. Disable the peripheral throughout
|
|
esp_intr_disable(s_hcd_obj->isr_hdl);
|
|
usbh_hal_core_soft_reset(port->hal);
|
|
port->state = HCD_PORT_STATE_NOT_POWERED;
|
|
port->last_event = HCD_PORT_EVENT_NONE;
|
|
port->flags.val = 0;
|
|
esp_intr_enable(s_hcd_obj->isr_hdl);
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_OK;
|
|
}
|
|
|
|
void *hcd_port_get_ctx(hcd_port_handle_t port_hdl)
|
|
{
|
|
port_t *port = (port_t *)port_hdl;
|
|
void *ret;
|
|
HCD_ENTER_CRITICAL();
|
|
ret = port->context;
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
// --------------------------------------------------- HCD Pipes -------------------------------------------------------
|
|
|
|
// ----------------------- Private -------------------------
|
|
|
|
static bool _pipe_get_next_xfer_req(pipe_t *pipe)
|
|
{
|
|
assert(pipe->inflight_xfer_req == NULL);
|
|
bool ret;
|
|
//This function assigns the next pending transfer request to the inflight_xfer_req
|
|
if (pipe->num_xfer_req_pending > 0) {
|
|
//Set inflight_xfer_req to the next pending transfer request
|
|
pipe->inflight_xfer_req = TAILQ_FIRST(&pipe->pend_xfer_req_tailq);
|
|
TAILQ_REMOVE(&pipe->pend_xfer_req_tailq, pipe->inflight_xfer_req, tailq_entry);
|
|
pipe->inflight_xfer_req->state = XFER_REQ_STATE_INFLIGHT;
|
|
pipe->num_xfer_req_pending--;
|
|
ret = true;
|
|
} else {
|
|
ret = false;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static void _pipe_ret_cur_xfer_req(pipe_t *pipe)
|
|
{
|
|
assert(pipe->inflight_xfer_req != NULL);
|
|
//Add the transfer request to the pipe's done tailq
|
|
TAILQ_INSERT_TAIL(&pipe->done_xfer_req_tailq, pipe->inflight_xfer_req, tailq_entry);
|
|
pipe->inflight_xfer_req->state = XFER_REQ_STATE_DONE;
|
|
pipe->inflight_xfer_req = NULL;
|
|
pipe->num_xfer_req_done++;
|
|
}
|
|
|
|
static bool _pipe_wait_done(pipe_t *pipe)
|
|
{
|
|
//Check if there is a currently inflight transfer request
|
|
if (pipe->inflight_xfer_req != NULL) {
|
|
//Wait for pipe to complete its transfer
|
|
pipe->flags.waiting_xfer_done = 1;
|
|
_internal_pipe_event_wait(pipe);
|
|
if (pipe->state == HCD_PIPE_STATE_INVALID) {
|
|
//The pipe become invalid whilst waiting for its internal event
|
|
pipe->flags.waiting_xfer_done = 0; //Need to manually reset this bit in this case
|
|
return false;
|
|
}
|
|
bool chan_halted = usbh_hal_chan_slot_request_halt(pipe->chan_obj);
|
|
assert(chan_halted);
|
|
(void) chan_halted;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void _pipe_retire(pipe_t *pipe, bool self_initiated)
|
|
{
|
|
//Cannot have any inflight transfer request
|
|
assert(pipe->inflight_xfer_req == NULL);
|
|
if (pipe->num_xfer_req_pending > 0) {
|
|
//Process all remaining pending transfer requests
|
|
xfer_req_t *xfer_req;
|
|
TAILQ_FOREACH(xfer_req, &pipe->pend_xfer_req_tailq, tailq_entry) {
|
|
xfer_req->state = XFER_REQ_STATE_DONE;
|
|
//If we are initiating the retire, mark the transfer request as cancelled
|
|
xfer_req->irp->status = (self_initiated) ? USB_TRANSFER_STATUS_CANCELLED : USB_TRANSFER_STATUS_NO_DEVICE;
|
|
}
|
|
//Concatenated pending tailq to the done tailq
|
|
TAILQ_CONCAT(&pipe->done_xfer_req_tailq, &pipe->pend_xfer_req_tailq, tailq_entry);
|
|
pipe->num_xfer_req_done += pipe->num_xfer_req_pending;
|
|
pipe->num_xfer_req_pending = 0;
|
|
}
|
|
}
|
|
|
|
static inline hcd_pipe_event_t pipe_decode_error_event(usbh_hal_chan_error_t chan_error)
|
|
{
|
|
hcd_pipe_event_t event = HCD_PIPE_EVENT_NONE;
|
|
switch (chan_error) {
|
|
case USBH_HAL_CHAN_ERROR_XCS_XACT:
|
|
event = HCD_PIPE_EVENT_ERROR_XFER;
|
|
break;
|
|
case USBH_HAL_CHAN_ERROR_BNA:
|
|
event = HCD_PIPE_EVENT_ERROR_XFER_NOT_AVAIL;
|
|
break;
|
|
case USBH_HAL_CHAN_ERROR_PKT_BBL:
|
|
event = HCD_PIPE_EVENT_ERROR_OVERFLOW;
|
|
break;
|
|
case USBH_HAL_CHAN_ERROR_STALL:
|
|
event = HCD_PIPE_EVENT_ERROR_STALL;
|
|
break;
|
|
}
|
|
return event;
|
|
}
|
|
|
|
// ----------------------- Public --------------------------
|
|
|
|
esp_err_t hcd_pipe_alloc(hcd_port_handle_t port_hdl, const hcd_pipe_config_t *pipe_config, hcd_pipe_handle_t *pipe_hdl)
|
|
{
|
|
HCD_CHECK(port_hdl != NULL && pipe_config != NULL && pipe_hdl != NULL, ESP_ERR_INVALID_ARG);
|
|
port_t *port = (port_t *)port_hdl;
|
|
HCD_ENTER_CRITICAL();
|
|
//Can only allocate a pipe if the targetted port is initialized and conencted to an enabled device
|
|
HCD_CHECK_FROM_CRIT(port->initialized && port->flags.conn_devc_ena, ESP_ERR_INVALID_STATE);
|
|
usb_speed_t port_speed = port->speed;
|
|
HCD_EXIT_CRITICAL();
|
|
//Cannot connect to a FS device if the port is LS
|
|
HCD_CHECK(port_speed == USB_SPEED_FULL || (port_speed == USB_SPEED_LOW && pipe_config->dev_speed == USB_SPEED_LOW), ESP_ERR_NOT_SUPPORTED);
|
|
|
|
esp_err_t ret = ESP_OK;
|
|
//Get the type of pipe to allocate
|
|
usb_xfer_type_t type;
|
|
bool is_default_pipe;
|
|
if (pipe_config->ep_desc == NULL) { //A NULL ep_desc indicates we are allocating a default pipe
|
|
type = USB_XFER_TYPE_CTRL;
|
|
is_default_pipe = true;
|
|
} else {
|
|
type = USB_DESC_EP_GET_XFERTYPE(pipe_config->ep_desc);
|
|
is_default_pipe = false;
|
|
}
|
|
size_t num_xfer_desc = 0;
|
|
switch (type) {
|
|
case USB_XFER_TYPE_CTRL: {
|
|
num_xfer_desc = XFER_LIST_LEN_CTRL * NUM_DESC_PER_XFER_CTRL;
|
|
break;
|
|
}
|
|
case USB_XFER_TYPE_BULK: {
|
|
if (pipe_config->dev_speed == USB_SPEED_LOW) {
|
|
return ESP_ERR_NOT_SUPPORTED; //Low speed devices do not support bulk transfers
|
|
}
|
|
num_xfer_desc = XFER_LIST_LEN_BULK * NUM_DESC_PER_XFER_BULK;
|
|
break;
|
|
}
|
|
default: {
|
|
//Isochronous and Interrupt pipes currently not supported
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
|
|
//Allocate the pipe resources
|
|
pipe_t *pipe = calloc(1, sizeof(pipe_t));
|
|
usbh_hal_chan_t *chan_obj = malloc(sizeof(usbh_hal_chan_t));
|
|
void *xfer_desc_list = heap_caps_aligned_calloc(USBH_HAL_DMA_MEM_ALIGN, num_xfer_desc, USBH_HAL_XFER_DESC_SIZE, MALLOC_CAP_DMA);
|
|
if (pipe == NULL|| chan_obj == NULL || xfer_desc_list == NULL) {
|
|
ret = ESP_ERR_NO_MEM;
|
|
goto err;
|
|
}
|
|
|
|
//Initialize pipe object
|
|
TAILQ_INIT(&pipe->pend_xfer_req_tailq);
|
|
TAILQ_INIT(&pipe->done_xfer_req_tailq);
|
|
pipe->port = port;
|
|
pipe->xfer_desc_list = xfer_desc_list;
|
|
pipe->flags.xfer_desc_list_len = num_xfer_desc;
|
|
pipe->chan_obj = chan_obj;
|
|
pipe->ep_char.type = type;
|
|
if (is_default_pipe) {
|
|
pipe->ep_char.bEndpointAddress = 0;
|
|
//Set the default pipe's MPS to the worst case MPS for the device's speed
|
|
pipe->ep_char.mps = (pipe_config->dev_speed == USB_SPEED_FULL) ? CTRL_EP_MAX_MPS_FS : CTRL_EP_MAX_MPS_LS;
|
|
} else {
|
|
pipe->ep_char.bEndpointAddress = pipe_config->ep_desc->bEndpointAddress;
|
|
pipe->ep_char.mps = pipe_config->ep_desc->wMaxPacketSize;
|
|
}
|
|
pipe->ep_char.dev_addr = pipe_config->dev_addr;
|
|
pipe->ep_char.ls_via_fs_hub = (port_speed == USB_SPEED_FULL && pipe_config->dev_speed == USB_SPEED_LOW);
|
|
pipe->state = HCD_PIPE_STATE_ACTIVE;
|
|
pipe->callback = pipe_config->callback;
|
|
pipe->callback_arg = pipe_config->callback_arg;
|
|
pipe->context = pipe_config->context;
|
|
|
|
//Allocate channel
|
|
HCD_ENTER_CRITICAL();
|
|
if (!port->initialized || !port->flags.conn_devc_ena) {
|
|
HCD_EXIT_CRITICAL();
|
|
ret = ESP_ERR_INVALID_STATE;
|
|
goto err;
|
|
}
|
|
bool chan_allocated = usbh_hal_chan_alloc(port->hal, pipe->chan_obj, (void *) pipe);
|
|
if (!chan_allocated) {
|
|
HCD_EXIT_CRITICAL();
|
|
ret = ESP_ERR_NOT_SUPPORTED;
|
|
goto err;
|
|
}
|
|
usbh_hal_chan_set_ep_char(pipe->chan_obj, &pipe->ep_char);
|
|
|
|
//Add the pipe to the list of idle pipes in the port object
|
|
TAILQ_INSERT_TAIL(&port->pipes_idle_tailq, pipe, tailq_entry);
|
|
port->num_pipes_idle++;
|
|
HCD_EXIT_CRITICAL();
|
|
*pipe_hdl = (hcd_pipe_handle_t)pipe;
|
|
return ret;
|
|
|
|
err:
|
|
free(xfer_desc_list);
|
|
free(chan_obj);
|
|
free(pipe);
|
|
return ret;
|
|
}
|
|
|
|
esp_err_t hcd_pipe_free(hcd_pipe_handle_t pipe_hdl)
|
|
{
|
|
pipe_t *pipe = (pipe_t *)pipe_hdl;
|
|
HCD_ENTER_CRITICAL();
|
|
//Check that all transfer requests have been removed and pipe has no pending events
|
|
HCD_CHECK_FROM_CRIT(pipe->inflight_xfer_req == NULL
|
|
&& pipe->num_xfer_req_pending == 0
|
|
&& pipe->num_xfer_req_done == 0,
|
|
ESP_ERR_INVALID_STATE);
|
|
//Remove pipe from the list of idle pipes (it must be in the idle list because it should have no queued transfer requests)
|
|
TAILQ_REMOVE(&pipe->port->pipes_idle_tailq, pipe, tailq_entry);
|
|
pipe->port->num_pipes_idle--;
|
|
usbh_hal_chan_free(pipe->port->hal, pipe->chan_obj);
|
|
HCD_EXIT_CRITICAL();
|
|
|
|
//Free pipe resources
|
|
free(pipe->xfer_desc_list);
|
|
free(pipe->chan_obj);
|
|
free(pipe);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t hcd_pipe_update(hcd_pipe_handle_t pipe_hdl, uint8_t dev_addr, int mps)
|
|
{
|
|
pipe_t *pipe = (pipe_t *)pipe_hdl;
|
|
HCD_ENTER_CRITICAL();
|
|
//Check if pipe is in the correct state to be updated
|
|
HCD_CHECK_FROM_CRIT(pipe->state != HCD_PIPE_STATE_INVALID
|
|
&& !pipe->flags.pipe_cmd_processing
|
|
&& pipe->num_xfer_req_pending == 0
|
|
&& pipe->num_xfer_req_done == 0,
|
|
ESP_ERR_INVALID_STATE);
|
|
//Check that all transfer requests have been removed and pipe has no pending events
|
|
pipe->ep_char.dev_addr = dev_addr;
|
|
pipe->ep_char.mps = mps;
|
|
usbh_hal_chan_set_ep_char(pipe->chan_obj, &pipe->ep_char);
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_OK;
|
|
}
|
|
|
|
void *hcd_pipe_get_ctx(hcd_pipe_handle_t pipe_hdl)
|
|
{
|
|
pipe_t *pipe = (pipe_t *) pipe_hdl;
|
|
void *ret;
|
|
HCD_ENTER_CRITICAL();
|
|
ret = pipe->context;
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
hcd_pipe_state_t hcd_pipe_get_state(hcd_pipe_handle_t pipe_hdl)
|
|
{
|
|
hcd_pipe_state_t ret;
|
|
pipe_t *pipe = (pipe_t *) pipe_hdl;
|
|
HCD_ENTER_CRITICAL();
|
|
//If there is no enabled device, all existing pipes are invalid.
|
|
if (pipe->port->state != HCD_PORT_STATE_ENABLED
|
|
&& pipe->port->state != HCD_PORT_STATE_SUSPENDED
|
|
&& pipe->port->state != HCD_PORT_STATE_RESUMING) {
|
|
ret = HCD_PIPE_STATE_INVALID;
|
|
} else {
|
|
ret = pipe->state;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
esp_err_t hcd_pipe_command(hcd_pipe_handle_t pipe_hdl, hcd_pipe_cmd_t command)
|
|
{
|
|
pipe_t *pipe = (pipe_t *) pipe_hdl;
|
|
bool ret = ESP_OK;
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
//Cannot execute pipe commands the pipe is already executing a command, or if the pipe or its port are no longer valid
|
|
if (pipe->flags.pipe_cmd_processing || !pipe->port->flags.conn_devc_ena || pipe->state == HCD_PIPE_STATE_INVALID) {
|
|
ret = ESP_ERR_INVALID_STATE;
|
|
} else {
|
|
pipe->flags.pipe_cmd_processing = 1;
|
|
switch (command) {
|
|
case HCD_PIPE_CMD_ABORT: {
|
|
//Retire all scheduled transfer requests. Pipe's state remains unchanged
|
|
if (!_pipe_wait_done(pipe)) { //Stop any on going transfers
|
|
ret = ESP_ERR_INVALID_RESPONSE;
|
|
break;
|
|
}
|
|
_pipe_retire(pipe, true); //Retire any pending transfers
|
|
break;
|
|
}
|
|
case HCD_PIPE_CMD_RESET: {
|
|
//Retire all scheduled transfer requests. Pipe's state moves to active
|
|
if (!_pipe_wait_done(pipe)) { //Stop any on going transfers
|
|
ret = ESP_ERR_INVALID_RESPONSE;
|
|
break;
|
|
}
|
|
_pipe_retire(pipe, true); //Retire any pending transfers
|
|
pipe->state = HCD_PIPE_STATE_ACTIVE;
|
|
break;
|
|
}
|
|
case HCD_PIPE_CMD_CLEAR: { //Can only do this if port is still active
|
|
//Pipe's state moves from halted to active
|
|
if (pipe->state == HCD_PIPE_STATE_HALTED) {
|
|
pipe->state = HCD_PIPE_STATE_ACTIVE;
|
|
//Start the next pending transfer if it exists
|
|
if (_pipe_get_next_xfer_req(pipe)) {
|
|
//Fill the descriptor list with the transfer request and start the transfer
|
|
_xfer_req_fill(pipe);
|
|
usbh_hal_chan_activate(pipe->chan_obj, 0); //Start with the first descriptor
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case HCD_PIPE_CMD_HALT: {
|
|
//Pipe's state moves to halted
|
|
if (!_pipe_wait_done(pipe)) { //Stop any on going transfers
|
|
ret = ESP_ERR_INVALID_RESPONSE;
|
|
break;
|
|
}
|
|
pipe->state = HCD_PIPE_STATE_HALTED;
|
|
break;
|
|
}
|
|
}
|
|
pipe->flags.pipe_cmd_processing = 0;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
hcd_pipe_event_t hcd_pipe_get_event(hcd_pipe_handle_t pipe_hdl)
|
|
{
|
|
pipe_t *pipe = (pipe_t *) pipe_hdl;
|
|
hcd_pipe_event_t ret;
|
|
HCD_ENTER_CRITICAL();
|
|
ret = pipe->last_event;
|
|
pipe->last_event = HCD_PIPE_EVENT_NONE;
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
// ----------------------------------------------- HCD Transfer Requests -----------------------------------------------
|
|
|
|
// ----------------------- Private -------------------------
|
|
|
|
static void _xfer_req_fill(pipe_t *pipe)
|
|
{
|
|
//inflight_xfer_req of the pipe must already set to the target transfer request
|
|
assert(pipe->inflight_xfer_req != NULL);
|
|
//Fill transfer descriptor list with a single transfer request
|
|
usb_irp_t *usb_irp = pipe->inflight_xfer_req->irp;
|
|
switch (pipe->ep_char.type) {
|
|
case USB_XFER_TYPE_CTRL: {
|
|
//Get information about the contorl transfer by analyzing the setup packet (the first 8 bytes)
|
|
usb_ctrl_req_t *ctrl_req = (usb_ctrl_req_t *)usb_irp->data_buffer;
|
|
pipe->flags.ctrl_data_stg_in = ((ctrl_req->bRequestType & USB_B_REQUEST_TYPE_DIR_IN) != 0);
|
|
pipe->flags.ctrl_data_stg_skip = (usb_irp->num_bytes == 0);
|
|
|
|
//Fill setup stage
|
|
usbh_hal_xfer_desc_fill(pipe->xfer_desc_list, 0, usb_irp->data_buffer, sizeof(usb_ctrl_req_t),
|
|
USBH_HAL_XFER_DESC_FLAG_SETUP | USBH_HAL_XFER_DESC_FLAG_HALT);
|
|
if (pipe->flags.ctrl_data_stg_skip) {
|
|
//Fill a NULL packet if there is no data stage
|
|
usbh_hal_xfer_desc_fill(pipe->xfer_desc_list, 1, NULL, 0, USBH_HAL_XFER_DESC_FLAG_NULL);
|
|
} else {
|
|
//Fill data stage
|
|
usbh_hal_xfer_desc_fill(pipe->xfer_desc_list, 1, usb_irp->data_buffer + sizeof(usb_ctrl_req_t), usb_irp->num_bytes,
|
|
((pipe->flags.ctrl_data_stg_in) ? USBH_HAL_XFER_DESC_FLAG_IN : 0) | USBH_HAL_XFER_DESC_FLAG_HALT);
|
|
}
|
|
//Fill status stage (i.e., a zero length packet). If data stage is skipped, the status stage is always IN.
|
|
usbh_hal_xfer_desc_fill(pipe->xfer_desc_list, 2, NULL, 0,
|
|
((pipe->flags.ctrl_data_stg_in && !pipe->flags.ctrl_data_stg_skip) ? 0 : USBH_HAL_XFER_DESC_FLAG_IN) | USBH_HAL_XFER_DESC_FLAG_HALT);
|
|
//Set the channel's direction to OUT and PID to 0 respectively for the the setup stage
|
|
usbh_hal_chan_set_dir(pipe->chan_obj, false); //Setup stage is always OUT
|
|
usbh_hal_chan_set_pid(pipe->chan_obj, 0); //Setup stage always has a PID of DATA0
|
|
break;
|
|
}
|
|
case USB_XFER_TYPE_BULK: {
|
|
bool is_in = pipe->ep_char.bEndpointAddress & USB_B_ENDPOINT_ADDRESS_EP_DIR_MASK;
|
|
usbh_hal_xfer_desc_fill(pipe->xfer_desc_list, 0, usb_irp->data_buffer, usb_irp->num_bytes,
|
|
((is_in) ? USBH_HAL_XFER_DESC_FLAG_IN : 0) | USBH_HAL_XFER_DESC_FLAG_HALT);
|
|
break;
|
|
}
|
|
default: {
|
|
break; //Isoc and Interrupt transfers not supported yet
|
|
}
|
|
}
|
|
//Claim slot
|
|
usbh_hal_chan_slot_acquire(pipe->chan_obj, pipe->xfer_desc_list, pipe->flags.xfer_desc_list_len, (void *)pipe);
|
|
}
|
|
|
|
static void _xfer_req_continue(pipe_t *pipe)
|
|
{
|
|
int next_idx = usbh_hal_chan_get_next_desc_index(pipe->chan_obj);
|
|
bool next_dir_is_in; //Next descriptor direction is IN
|
|
int next_pid; //Next PID (DATA0 or DATA 1)
|
|
int num_to_skip; //Number of descriptors to skip
|
|
if (next_idx == 1) {
|
|
//Just finished setup stage
|
|
if (pipe->flags.ctrl_data_stg_skip) {
|
|
//Skipping data stage. Go straight to status stage
|
|
next_dir_is_in = true; //With no data stage, status stage must be IN
|
|
next_pid = 1; //Status stage always has a PID of DATA1
|
|
num_to_skip = 1; //Skip over the null descriptor representing the skipped data stage
|
|
} else {
|
|
//Go to data stage
|
|
next_dir_is_in = pipe->flags.ctrl_data_stg_in;
|
|
next_pid = 1; //Data stage always starts with a PID of DATA1
|
|
num_to_skip = 0;
|
|
}
|
|
} else { //next_idx == 2
|
|
//Going to status stage from data stage
|
|
next_dir_is_in = !pipe->flags.ctrl_data_stg_in; //Status stage is opposite direction of data stage
|
|
next_pid = 1; //Status stage always has a PID of DATA1
|
|
num_to_skip = 0;
|
|
}
|
|
|
|
usbh_hal_chan_set_dir(pipe->chan_obj, next_dir_is_in);
|
|
usbh_hal_chan_set_pid(pipe->chan_obj, next_pid);
|
|
usbh_hal_chan_activate(pipe->chan_obj, num_to_skip); //Start the next stage
|
|
}
|
|
|
|
static void _xfer_req_parse(pipe_t *pipe, bool error_occurred)
|
|
{
|
|
assert(pipe->inflight_xfer_req != NULL);
|
|
//Release the slot
|
|
void *xfer_desc_list;
|
|
int xfer_desc_len;
|
|
usbh_hal_chan_slot_release(pipe->chan_obj, &xfer_desc_list, &xfer_desc_len);
|
|
assert(xfer_desc_list == pipe->xfer_desc_list);
|
|
(void) xfer_desc_len;
|
|
|
|
//Parse the transfer descriptor list for the result of the transfer
|
|
usb_irp_t *usb_irp = pipe->inflight_xfer_req->irp;
|
|
usb_transfer_status_t xfer_status;
|
|
int xfer_rem_len;
|
|
if (error_occurred) {
|
|
//Either a pipe error has occurred or the pipe is no longer valid
|
|
if (pipe->state == HCD_PIPE_STATE_INVALID) {
|
|
xfer_status = USB_TRANSFER_STATUS_NO_DEVICE;
|
|
} else {
|
|
//Must have been a pipe error event
|
|
switch (pipe->last_event) {
|
|
case HCD_PIPE_EVENT_ERROR_XFER: //Excessive transaction error
|
|
xfer_status = USB_TRANSFER_STATUS_ERROR;
|
|
break;
|
|
case HCD_PIPE_EVENT_ERROR_OVERFLOW:
|
|
xfer_status = USB_TRANSFER_STATUS_OVERFLOW;
|
|
break;
|
|
case HCD_PIPE_EVENT_ERROR_STALL:
|
|
xfer_status = USB_TRANSFER_STATUS_STALL;
|
|
break;
|
|
default:
|
|
//HCD_PIPE_EVENT_ERROR_XFER_NOT_AVAIL should never occur
|
|
abort();
|
|
break;
|
|
}
|
|
}
|
|
//We assume no bytes transmitted because of an error.
|
|
xfer_rem_len = usb_irp->num_bytes;
|
|
} else {
|
|
int desc_status;
|
|
switch (pipe->ep_char.type) {
|
|
case USB_XFER_TYPE_CTRL: {
|
|
if (pipe->flags.ctrl_data_stg_skip) {
|
|
//There was no data stage. Just set it as successful
|
|
desc_status = USBH_HAL_XFER_DESC_STS_SUCCESS;
|
|
xfer_rem_len = 0;
|
|
} else {
|
|
//Check the data stage (index 1)
|
|
usbh_hal_xfer_desc_parse(pipe->xfer_desc_list, 1, &xfer_rem_len, &desc_status);
|
|
}
|
|
break;
|
|
}
|
|
case USB_XFER_TYPE_BULK: {
|
|
usbh_hal_xfer_desc_parse(pipe->xfer_desc_list, 0, &xfer_rem_len, &desc_status);
|
|
break;
|
|
}
|
|
default: {
|
|
//We don't supportISOC and INTR pipes yet
|
|
desc_status = USBH_HAL_XFER_DESC_STS_NOT_EXECUTED;
|
|
xfer_rem_len = 0;
|
|
xfer_status = USB_TRANSFER_STATUS_ERROR;
|
|
abort();
|
|
break;
|
|
}
|
|
}
|
|
xfer_status = USB_TRANSFER_STATUS_COMPLETED;
|
|
assert(desc_status == USBH_HAL_XFER_DESC_STS_SUCCESS);
|
|
}
|
|
//Write back results to IRP
|
|
usb_irp->actual_num_bytes = usb_irp->num_bytes - xfer_rem_len;
|
|
usb_irp->status = xfer_status;
|
|
}
|
|
|
|
// ----------------------- Public --------------------------
|
|
|
|
hcd_xfer_req_handle_t hcd_xfer_req_alloc()
|
|
{
|
|
xfer_req_t *xfer_req = calloc(1, sizeof(xfer_req_t));
|
|
xfer_req->state = XFER_REQ_STATE_IDLE;
|
|
return (hcd_xfer_req_handle_t) xfer_req;
|
|
}
|
|
|
|
void hcd_xfer_req_free(hcd_xfer_req_handle_t req_hdl)
|
|
{
|
|
if (req_hdl == NULL) {
|
|
return;
|
|
}
|
|
xfer_req_t *xfer_req = (xfer_req_t *) req_hdl;
|
|
//Cannot free a transfer request that is still being used
|
|
assert(xfer_req->state == XFER_REQ_STATE_IDLE);
|
|
free(xfer_req);
|
|
}
|
|
|
|
void hcd_xfer_req_set_target(hcd_xfer_req_handle_t req_hdl, hcd_pipe_handle_t pipe_hdl, usb_irp_t *irp, void *context)
|
|
{
|
|
xfer_req_t *xfer_req = (xfer_req_t *) req_hdl;
|
|
//Can only set an transfer request's target when the transfer request is idl
|
|
assert(xfer_req->state == XFER_REQ_STATE_IDLE);
|
|
xfer_req->pipe = (pipe_t *) pipe_hdl;
|
|
xfer_req->irp = irp;
|
|
xfer_req->context = context;
|
|
}
|
|
|
|
void hcd_xfer_req_get_target(hcd_xfer_req_handle_t req_hdl, hcd_pipe_handle_t *pipe_hdl, usb_irp_t **irp, void **context)
|
|
{
|
|
xfer_req_t *xfer_req = (xfer_req_t *) req_hdl;
|
|
*pipe_hdl = (hcd_pipe_handle_t) xfer_req->pipe;
|
|
*irp = xfer_req->irp;
|
|
*context = xfer_req->context;
|
|
}
|
|
|
|
esp_err_t hcd_xfer_req_enqueue(hcd_xfer_req_handle_t req_hdl)
|
|
{
|
|
xfer_req_t *xfer_req = (xfer_req_t *) req_hdl;
|
|
HCD_CHECK(xfer_req->pipe != NULL && xfer_req->irp != NULL //The transfer request's target must be set
|
|
&& xfer_req->state == XFER_REQ_STATE_IDLE, //The transfer request cannot be already enqueued
|
|
ESP_ERR_INVALID_STATE);
|
|
pipe_t *pipe = xfer_req->pipe;
|
|
HCD_ENTER_CRITICAL();
|
|
HCD_CHECK_FROM_CRIT(pipe->port->state == HCD_PORT_STATE_ENABLED //The pipe's port must be in the correct state
|
|
&& pipe->state == HCD_PIPE_STATE_ACTIVE //The pipe must be in the correct state
|
|
&& !pipe->flags.pipe_cmd_processing, //Pipe cannot currently be processing a pipe command
|
|
ESP_ERR_INVALID_STATE);
|
|
//Check if we can start execution on the pipe immediately
|
|
if (!pipe->flags.paused && pipe->num_xfer_req_pending == 0 && pipe->inflight_xfer_req == NULL) {
|
|
//Pipe isn't executing any transfers. Start immediately
|
|
pipe->inflight_xfer_req = xfer_req;
|
|
_xfer_req_fill(pipe);
|
|
usbh_hal_chan_activate(pipe->chan_obj, 0); //Start with the first descriptor
|
|
xfer_req->state = XFER_REQ_STATE_INFLIGHT;
|
|
if (pipe->num_xfer_req_done == 0) {
|
|
//This is the first transfer request to be enqueued into the pipe. Move the pipe to the list of queued pipes
|
|
TAILQ_REMOVE(&pipe->port->pipes_idle_tailq, pipe, tailq_entry);
|
|
TAILQ_INSERT_TAIL(&pipe->port->pipes_queued_tailq, pipe, tailq_entry);
|
|
pipe->port->num_pipes_idle--;
|
|
pipe->port->num_pipes_queued++;
|
|
}
|
|
} else {
|
|
//Add the transfer request to the pipe's pending tailq
|
|
TAILQ_INSERT_TAIL(&pipe->pend_xfer_req_tailq, xfer_req, tailq_entry);
|
|
pipe->num_xfer_req_pending++;
|
|
xfer_req->state = XFER_REQ_STATE_PENDING;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
return ESP_OK;
|
|
}
|
|
|
|
hcd_xfer_req_handle_t hcd_xfer_req_dequeue(hcd_pipe_handle_t pipe_hdl)
|
|
{
|
|
pipe_t *pipe = (pipe_t *)pipe_hdl;
|
|
hcd_xfer_req_handle_t ret;
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
if (pipe->num_xfer_req_done > 0) {
|
|
xfer_req_t *xfer_req = TAILQ_FIRST(&pipe->done_xfer_req_tailq);
|
|
TAILQ_REMOVE(&pipe->done_xfer_req_tailq, xfer_req, tailq_entry);
|
|
pipe->num_xfer_req_done--;
|
|
assert(xfer_req->state == XFER_REQ_STATE_DONE);
|
|
xfer_req->state = XFER_REQ_STATE_IDLE;
|
|
ret = (hcd_xfer_req_handle_t) xfer_req;
|
|
if (pipe->num_xfer_req_done == 0 && pipe->num_xfer_req_pending == 0) {
|
|
//This pipe has no more enqueued transfers. Move the pipe to the list of idle pipes
|
|
TAILQ_REMOVE(&pipe->port->pipes_queued_tailq, pipe, tailq_entry);
|
|
TAILQ_INSERT_TAIL(&pipe->port->pipes_idle_tailq, pipe, tailq_entry);
|
|
pipe->port->num_pipes_idle++;
|
|
pipe->port->num_pipes_queued--;
|
|
}
|
|
} else {
|
|
ret = NULL;
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|
|
|
|
esp_err_t hcd_xfer_req_abort(hcd_xfer_req_handle_t req_hdl)
|
|
{
|
|
xfer_req_t *xfer_req = (xfer_req_t *) req_hdl;
|
|
esp_err_t ret;
|
|
|
|
HCD_ENTER_CRITICAL();
|
|
switch (xfer_req->state) {
|
|
case XFER_REQ_STATE_PENDING: {
|
|
//Transfer request has not been executed so it can be aborted
|
|
pipe_t *pipe = xfer_req->pipe;
|
|
//Remove it form the pending queue
|
|
TAILQ_REMOVE(&pipe->pend_xfer_req_tailq, xfer_req, tailq_entry);
|
|
pipe->num_xfer_req_pending--;
|
|
//Add it to the done queue
|
|
TAILQ_INSERT_TAIL(&pipe->done_xfer_req_tailq, xfer_req, tailq_entry);
|
|
pipe->num_xfer_req_done++;
|
|
//Update the transfer request and associated IRP's status
|
|
xfer_req->state = XFER_REQ_STATE_DONE;
|
|
xfer_req->irp->status = USB_TRANSFER_STATUS_CANCELLED;
|
|
ret = ESP_OK;
|
|
break;
|
|
}
|
|
case XFER_REQ_STATE_IDLE: {
|
|
//Cannot abort a transfer request that was never enqueued
|
|
ret = ESP_ERR_INVALID_STATE;
|
|
break;
|
|
}
|
|
default :{
|
|
//Transfer request is currently or has already been executed. Nothing to do.
|
|
ret = ESP_OK;
|
|
break;
|
|
}
|
|
}
|
|
HCD_EXIT_CRITICAL();
|
|
return ret;
|
|
}
|