2018-08-23 14:42:55 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Koen Zandberg <koen@bergzand.net>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for
|
|
|
|
* more details.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @defgroup drivers_periph_usbdev usbdev - USB Device Driver API
|
|
|
|
* @ingroup drivers_periph
|
|
|
|
* @brief This is a generic low-level USB driver interface
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* # About
|
|
|
|
*
|
|
|
|
* usbdev specifies a common USB device API for low level USB interfaces. The
|
|
|
|
* interface is split in a definition for the USB interface hardware and for
|
|
|
|
* individual USB endpoints.
|
|
|
|
*
|
|
|
|
* # Design goals
|
|
|
|
*
|
|
|
|
* 1. Support for multiple instances on a single board
|
|
|
|
* 2. Interface optimized for MCU peripheral interfaces.
|
|
|
|
*
|
|
|
|
* # Details
|
|
|
|
*
|
|
|
|
* The driver interface is split in two separate parts. One part is a global USB
|
|
|
|
* device interface, the other is an endpoint control API.
|
|
|
|
*
|
|
|
|
* The usb device driver can manage parts of the USB interface itself such as
|
|
|
|
* the pull up resistor state or the USB speed.
|
|
|
|
*
|
|
|
|
* The endpoint control API manages a single endpoint. This allows for a modular
|
|
|
|
* approach where multiple USB functions/interfaces can be multiplexed over a
|
|
|
|
* single USB interface. Each interface can be implemented as a separate module.
|
|
|
|
* The interface handler does not have to care about the usb device itself or
|
|
|
|
* it's endpoint number. It can simply request an available endpoint from the
|
|
|
|
* usb device with the @ref usbdev_new_ep function.
|
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* Each interface handler can request multiple endpoints from the usbdev device.
|
|
|
|
* It must supply the expected maximum endpoint transfer size and other
|
|
|
|
* properties of the endpoint. A pointer to an usbdev endpoint is returned if
|
|
|
|
* an endpoint fitting the requirements is available.
|
|
|
|
*
|
|
|
|
* Data transmission is done via the @ref usbdev_ep_xmit function. A buffer and
|
|
|
|
* length must be supplied. Received data from the host ends up at this buffer
|
|
|
|
* automatically by the low level drivers after reception. For reception the
|
|
|
|
* maximum expected transfer length must be supplied, equal to the maximum
|
|
|
|
* endpoint transfer size. The data received can be less than this.
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* For transmitting data back to the host, a similar approach is used. The data
|
2021-10-26 19:39:12 +02:00
|
|
|
* to be transmitted is supplied as buffer and the @ref usbdev_ep_xmit function
|
|
|
|
* is called with the buffer and the size of the data.
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* To ensure that the data buffers adhere to the restrictions of the low level
|
|
|
|
* USB peripheral memory interface, the specific @ref usbdev_ep_buf_t data type
|
|
|
|
* must be used. It behaves as a regular `uint8_t` byte buffer, but is
|
|
|
|
* instantiated with the attributes to ensure that the low level DMA interface
|
|
|
|
* can use it.
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* A callback function is required for signalling events from the driver. The
|
|
|
|
* @ref USBDEV_EVENT_ESR is special in that it indicates that the USB peripheral
|
|
|
|
* had an interrupt that needs to be serviced in a non-interrupt context. This
|
|
|
|
* requires the usbdev_esr function to be called. In the case that the event is
|
|
|
|
* signalled via the @ref usbdev_t::epcb callback, the @ref usbdev_ep_esr
|
|
|
|
* should be called to service the endpoint specific events from the
|
|
|
|
* peripheral.
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Definitions low-level USB driver interface
|
|
|
|
*
|
|
|
|
* @author Koen Zandberg <koen@bergzand.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef PERIPH_USBDEV_H
|
|
|
|
#define PERIPH_USBDEV_H
|
|
|
|
|
2021-10-30 12:04:36 +02:00
|
|
|
#include <stdbool.h>
|
2018-08-23 14:42:55 +02:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
#include "assert.h"
|
2023-04-15 12:12:09 +02:00
|
|
|
#include "periph_conf.h"
|
2018-08-23 14:42:55 +02:00
|
|
|
#include "usb.h"
|
|
|
|
#include "usb/usbopt.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief usbdev_t forward declaration
|
|
|
|
*/
|
|
|
|
typedef struct usbdev usbdev_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief usbdev_ep_t forward declaration
|
|
|
|
*/
|
|
|
|
typedef struct usbdev_ep usbdev_ep_t;
|
|
|
|
|
|
|
|
/**
|
2021-10-26 19:39:12 +02:00
|
|
|
* @brief USBDEV endpoint buffer CPU-specific requirements
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* Can be overridden by periph_cpu if needed by the USB peripheral DMA.
|
2018-08-23 14:42:55 +02:00
|
|
|
*/
|
2021-10-26 19:39:12 +02:00
|
|
|
#ifndef USBDEV_CPU_DMA_REQUIREMENTS
|
|
|
|
#define USBDEV_CPU_DMA_REQUIREMENTS
|
2018-08-23 14:42:55 +02:00
|
|
|
#endif
|
|
|
|
|
2021-10-26 19:39:12 +02:00
|
|
|
/**
|
|
|
|
* @brief Instantiation type for usbdev endpoint buffers
|
|
|
|
*
|
|
|
|
* Functions passing around pointers to these buffers can still use `uint8_t`
|
|
|
|
* for the argument type.
|
|
|
|
*
|
|
|
|
* Example usage:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* usbdev_ep_buf_t buffer[64];
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @note This is a define and not a typedef so that the above works. With a
|
2021-10-26 22:05:23 +02:00
|
|
|
* typedef it would instantiate an array of aligned uint8_t and not an aligned
|
|
|
|
* array of uint8_t (assuming the requirement is alignment).
|
2021-10-26 19:39:12 +02:00
|
|
|
*/
|
|
|
|
#define usbdev_ep_buf_t USBDEV_CPU_DMA_REQUIREMENTS uint8_t
|
|
|
|
|
2023-04-15 12:12:09 +02:00
|
|
|
/**
|
|
|
|
* @brief USBDEV specific requirement for setting the device address
|
|
|
|
*
|
|
|
|
* The address in the USB device can be set either directly after the SETUP
|
|
|
|
* stage on receipt of the `SET ADDRESS Request` or after the associated status
|
|
|
|
* stage. When the USB device address has to be set depends on the hardware.
|
|
|
|
* If `USBDEV_CPU_SET_ADDR_AFTER_STATUS` has the value 1 (default), the address
|
|
|
|
* is only set in the USB device after the status stage. Overwrite it with 0
|
|
|
|
* in `periph_cpu.h` to set the address already directly after the SETUP stage.
|
|
|
|
*/
|
|
|
|
#ifndef USBDEV_CPU_SET_ADDR_AFTER_STATUS
|
|
|
|
#define USBDEV_CPU_SET_ADDR_AFTER_STATUS 1
|
|
|
|
#endif
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Number of USB IN and OUT endpoints allocated
|
|
|
|
*
|
|
|
|
* Configures the number of endpoints allocated. An equal number of IN and OUT
|
|
|
|
* endpoints are allocated
|
|
|
|
*/
|
|
|
|
#ifndef USBDEV_NUM_ENDPOINTS
|
|
|
|
#define USBDEV_NUM_ENDPOINTS 8
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief List of event types that can be send from the device driver to the
|
|
|
|
* upper layer
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
/**
|
|
|
|
* @brief Driver needs it's ESR (event service routine) handled
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_ESR = 1,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Host connection detected
|
|
|
|
*
|
|
|
|
* A host has connected to the device. Detection should happen by detecting
|
|
|
|
* the USB power, but other detection mechanisms might be used.
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_HOST_CONNECT,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Host disconnected from the device
|
|
|
|
*
|
|
|
|
* Similar to @ref USBDEV_EVENT_HOST_CONNECT, except that the host
|
|
|
|
* disconnected instead.
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_HOST_DISCONNECT,
|
|
|
|
|
|
|
|
/**
|
2019-10-23 10:36:49 +02:00
|
|
|
* @brief Line reset occurred
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* A line reset is a host initiated USB reset to the peripheral
|
|
|
|
*
|
|
|
|
* The peripheral driver must clears the following settings before
|
|
|
|
* emitting this event:
|
|
|
|
* - Device address
|
|
|
|
* - Endpoint configuration, including stall settings
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_RESET,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Start of Frame received
|
|
|
|
*
|
|
|
|
* A Start of Frame (SoF) packet is received from the host.
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_SOF,
|
|
|
|
|
2019-03-19 20:04:30 +01:00
|
|
|
/**
|
|
|
|
* @brief USB suspend condition active
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_SUSPEND,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief USB suspend condition no longer active
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_RESUME,
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Transaction completed event.
|
|
|
|
*
|
|
|
|
* An endpoint must emit this event after a transaction with the host
|
2019-10-23 10:36:49 +02:00
|
|
|
* occurred to indicate that the data in the buffer is used or new
|
2018-08-23 14:42:55 +02:00
|
|
|
* depending on the endpoint direction
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_TR_COMPLETE,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Transaction stall event.
|
|
|
|
*
|
|
|
|
* An endpoint should emit this event after a stall reply has been
|
|
|
|
* transmitted to the host
|
|
|
|
*/
|
|
|
|
USBDEV_EVENT_TR_STALL,
|
|
|
|
|
|
|
|
/* expand list if required */
|
|
|
|
} usbdev_event_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Event callback for signaling usbdev event to upper layers
|
|
|
|
*
|
|
|
|
* @param[in] usbdev usbdev context
|
|
|
|
* @param[in] event type of the event
|
|
|
|
*/
|
|
|
|
typedef void (*usbdev_event_cb_t)(usbdev_t *usbdev, usbdev_event_t event);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Event callback for signaling endpoint events to upper layers
|
|
|
|
*
|
|
|
|
* @param[in] ep usbdev endpoint context
|
|
|
|
* @param[in] event type of the event
|
|
|
|
*/
|
|
|
|
typedef void (*usbdev_ep_event_cb_t)(usbdev_ep_t *ep,
|
|
|
|
usbdev_event_t event);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief usbdev device descriptor
|
|
|
|
*/
|
|
|
|
struct usbdev {
|
|
|
|
const struct usbdev_driver *driver; /**< usbdev driver struct */
|
|
|
|
usbdev_event_cb_t cb; /**< Event callback supplied by
|
|
|
|
* upper layer */
|
|
|
|
usbdev_ep_event_cb_t epcb; /**< Endpoint event callback for
|
|
|
|
* upper layer */
|
|
|
|
void *context; /**< Ptr to the thread context */
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief usbdev endpoint descriptor
|
|
|
|
*/
|
|
|
|
struct usbdev_ep {
|
|
|
|
usbdev_t *dev; /**< USB device this endpoint belongs to */
|
2021-10-26 19:39:12 +02:00
|
|
|
size_t len; /**< Endpoint configured max transfer size in bytes */
|
2018-08-23 14:42:55 +02:00
|
|
|
usb_ep_dir_t dir; /**< Endpoint direction */
|
|
|
|
usb_ep_type_t type; /**< Endpoint type */
|
|
|
|
uint8_t num; /**< Endpoint number */
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief usbdev driver functions
|
|
|
|
*
|
|
|
|
* Helpers (such as @ref usbdev_init) are provided and should be used instead.
|
|
|
|
* Directly calling these functions is not recommended.
|
|
|
|
*/
|
|
|
|
typedef struct usbdev_driver {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the USB peripheral device
|
|
|
|
*
|
|
|
|
* This initializes the USB device but must not enable the USB pull up.
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
*/
|
|
|
|
void (*init)(usbdev_t *usbdev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Retrieve an USB endpoint of the specified type
|
|
|
|
*
|
|
|
|
* requesting an endpoint of @ref USB_EP_TYPE_CONTROL must always return
|
|
|
|
* endpoint 0 of the specified direction
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] type USB endpoint type
|
|
|
|
* @param[in] dir USB endpoint direction
|
2021-10-26 19:39:12 +02:00
|
|
|
* @param[in] size USB endpoint buffer size
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* @return ptr to the new USB endpoint descriptor
|
|
|
|
* @return NULL on error
|
|
|
|
*/
|
2021-10-26 19:39:12 +02:00
|
|
|
usbdev_ep_t *(*new_ep)(usbdev_t *dev, usb_ep_type_t type, usb_ep_dir_t dir, size_t size);
|
2018-08-23 14:42:55 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get an option value from a given usb device
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[out] value pointer to store the option's value in
|
|
|
|
* @param[in] max_len maximal amount of byte that fit into @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes written to @p value
|
|
|
|
* @return `< 0` on error 0
|
|
|
|
*/
|
|
|
|
int (*get)(usbdev_t *usbdev, usbopt_t opt,
|
|
|
|
void *value, size_t max_len);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an option value for a given usb device
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[in] value value to set
|
|
|
|
* @param[in] value_len the length of @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes used from @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
|
|
|
int (*set)(usbdev_t *usbdev, usbopt_t opt,
|
|
|
|
const void *value, size_t value_len);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief a driver's user-space event service handler
|
|
|
|
*
|
|
|
|
* This function will be called from a USB stack's loop when being
|
|
|
|
* notified by usbdev_event_isr.
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
*/
|
|
|
|
void (*esr)(usbdev_t *dev);
|
|
|
|
|
2021-10-30 12:04:36 +02:00
|
|
|
/**
|
|
|
|
* @brief Stall both OUT and IN packets on endpoint 0 until a setup packet
|
|
|
|
* is received.
|
|
|
|
*
|
|
|
|
* @note The stall condition should be cleared automatically either by
|
|
|
|
* hardware or by the usbdev implementation after receiving a setup packet.
|
|
|
|
*/
|
|
|
|
void (*ep0_stall)(usbdev_t *usbdev);
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Initialize the USB endpoint
|
|
|
|
*
|
|
|
|
* This initializes the USB endpoint with the settings from the
|
|
|
|
* @ref usbdev_ep.
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
*/
|
|
|
|
void (*ep_init)(usbdev_ep_t *ep);
|
|
|
|
|
2021-10-30 12:04:36 +02:00
|
|
|
/**
|
|
|
|
* @brief Enable or disable the stall condition on the USB endpoint
|
|
|
|
*
|
|
|
|
* After clearing the stall condition on the endpoint, the usb peripheral
|
|
|
|
* must reinitialize the data toggle to DATA0.
|
|
|
|
*
|
|
|
|
* @note For enabling stall on endpoint 0 @ref usbdev_driver_t::ep0_stall
|
|
|
|
* must be used.
|
|
|
|
*
|
|
|
|
* @pre (ep->num != 0)
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] enable True to set stall, false to disable stall
|
|
|
|
*/
|
|
|
|
void (*ep_stall)(usbdev_ep_t *ep, bool enable);
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Get an option value from a given usb device endpoint
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[out] value pointer to store the option's value in
|
|
|
|
* @param[in] max_len maximum number of byte that fit into @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes written to @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
|
|
|
int (*ep_get)(usbdev_ep_t *ep, usbopt_ep_t opt,
|
|
|
|
void *value, size_t max_len);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an option value for a given usb device endpoint
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[in] value pointer to the value
|
|
|
|
* @param[in] value_len the length of @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes used from @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
|
|
|
int (*ep_set)(usbdev_ep_t *ep, usbopt_ep_t opt,
|
|
|
|
const void *value, size_t value_len);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief an endpoint's user-space event handler
|
|
|
|
*
|
|
|
|
* Must be called in response to an @ref USBDEV_EVENT_ESR event in
|
|
|
|
* userspace context.
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor to service
|
|
|
|
*/
|
|
|
|
void (*ep_esr)(usbdev_ep_t *ep);
|
|
|
|
|
|
|
|
/**
|
2021-10-26 19:39:12 +02:00
|
|
|
* @brief Transmit a data buffer
|
|
|
|
*
|
|
|
|
* Ownership of the @p buf is transferred to the usbdev device after calling
|
|
|
|
* this. Do not modify (or unallocate) the buffer between calling this and
|
|
|
|
* when it is released via the @ref USBDEV_EVENT_TR_COMPLETE event.
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* This clears the stall setting in the endpoint if that is enabled.
|
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* @note The @p buf passed here must have been declared as
|
|
|
|
* @ref usbdev_ep_buf_t before so that DMA restrictions are applied to it
|
|
|
|
*
|
2018-08-23 14:42:55 +02:00
|
|
|
* @param[in] ep USB endpoint descriptor
|
2021-10-26 19:39:12 +02:00
|
|
|
* @param[inout] buf Buffer with the data to transfer
|
|
|
|
* @param[in] len (Max) Length of the data to transfer
|
2018-08-23 14:42:55 +02:00
|
|
|
*/
|
2021-10-26 19:39:12 +02:00
|
|
|
int (*xmit)(usbdev_ep_t *ep, uint8_t *buf, size_t len);
|
2018-08-23 14:42:55 +02:00
|
|
|
} usbdev_driver_t;
|
|
|
|
|
2019-03-19 20:14:31 +01:00
|
|
|
/**
|
|
|
|
* @brief Low level USB peripheral driver initialization
|
|
|
|
*
|
|
|
|
* This function prepares all usbdev peripherals available for initialization
|
|
|
|
*/
|
|
|
|
void usbdev_init_lowlevel(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Retrieve usbdev context from the peripheral
|
|
|
|
*
|
|
|
|
* @param num usbdev peripheral number to retrieve
|
|
|
|
*
|
|
|
|
* @returns the usbdev context at index @p num
|
|
|
|
*/
|
|
|
|
usbdev_t *usbdev_get_ctx(unsigned num);
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Initialize the USB peripheral device
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::init
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param dev usb device to initialize
|
|
|
|
*/
|
|
|
|
static inline void usbdev_init(usbdev_t *dev)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
dev->driver->init(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Retrieve an USB endpoint of the specified type
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::new_ep
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] type USB endpoint type
|
|
|
|
* @param[in] dir USB endpoint direction
|
2021-10-26 19:39:12 +02:00
|
|
|
* @param[in] size Maximum USB endpoint buffer size used
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* @return ptr to the new USB endpoint descriptor
|
|
|
|
* @return NULL on error
|
|
|
|
*/
|
|
|
|
static inline usbdev_ep_t * usbdev_new_ep(usbdev_t *dev, usb_ep_type_t type,
|
2021-10-26 19:39:12 +02:00
|
|
|
usb_ep_dir_t dir, size_t size)
|
2018-08-23 14:42:55 +02:00
|
|
|
{
|
|
|
|
assert(dev);
|
2021-10-26 19:39:12 +02:00
|
|
|
return dev->driver->new_ep(dev, type, dir, size);
|
2018-08-23 14:42:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get an option value from a given usb device
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::get
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[out] value pointer to store the option's value in
|
|
|
|
* @param[in] max_len maximal amount of byte that fit into @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes written to @p value
|
|
|
|
* @return `< 0` on error 0
|
|
|
|
*/
|
2019-06-05 12:05:12 +02:00
|
|
|
static inline int usbdev_get(usbdev_t *dev, usbopt_t opt,
|
2018-08-23 14:42:55 +02:00
|
|
|
void *value, size_t max_len)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
return dev->driver->get(dev, opt, value, max_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an option value for a given usb device
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::set
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[in] value value to set
|
|
|
|
* @param[in] value_len the length of @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes used from @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
2019-06-05 12:05:12 +02:00
|
|
|
static inline int usbdev_set(usbdev_t *dev, usbopt_t opt,
|
2018-08-23 14:42:55 +02:00
|
|
|
const void *value, size_t value_len)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
return dev->driver->set(dev, opt, value, value_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief a driver's user-space event service handler
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::esr
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
*/
|
|
|
|
static inline void usbdev_esr(usbdev_t *dev)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
dev->driver->esr(dev);
|
|
|
|
}
|
|
|
|
|
2021-10-30 12:04:36 +02:00
|
|
|
/**
|
|
|
|
* @brief Stall both OUT and IN packets on endpoint 0 until a setup packet
|
|
|
|
* is received.
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep0_stall
|
|
|
|
*
|
|
|
|
* @note The stall condition is automatically cleared after receiving a
|
|
|
|
* setup packet.
|
|
|
|
*
|
|
|
|
* @pre `(dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] dev USB device descriptor
|
|
|
|
*/
|
|
|
|
static inline void usbdev_ep0_stall(usbdev_t *dev)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
dev->driver->ep0_stall(dev);
|
|
|
|
}
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Initialize the USB endpoint
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep_init
|
|
|
|
*
|
|
|
|
* @pre `(ep != NULL)`
|
|
|
|
* @pre `(ep->dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
*/
|
|
|
|
static inline void usbdev_ep_init(usbdev_ep_t *ep)
|
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
|
|
|
ep->dev->driver->ep_init(ep);
|
|
|
|
}
|
|
|
|
|
2021-10-30 12:04:36 +02:00
|
|
|
/**
|
|
|
|
* @brief Enable or disable the stall condition on the USB endpoint
|
|
|
|
*
|
|
|
|
* @note For enabling stall on endpoint 0 @ref usbdev_driver_t::ep0_stall
|
|
|
|
* must be used.
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep_stall
|
|
|
|
*
|
|
|
|
* @pre (ep->num != 0)
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] enable True to set stall, false to disable stall
|
|
|
|
*/
|
|
|
|
static inline void usbdev_ep_stall(usbdev_ep_t *ep, bool enable)
|
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
|
|
|
ep->dev->driver->ep_stall(ep, enable);
|
|
|
|
}
|
|
|
|
|
2018-08-23 14:42:55 +02:00
|
|
|
/**
|
|
|
|
* @brief Get an option value from a given usb device endpoint
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep_get
|
|
|
|
*
|
|
|
|
* @pre `(ep != NULL)`
|
|
|
|
* @pre `(ep->dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[out] value pointer to store the option's value in
|
|
|
|
* @param[in] max_len maximum number of byte that fit into @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes written to @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
|
|
|
static inline int usbdev_ep_get(usbdev_ep_t *ep, usbopt_ep_t opt,
|
|
|
|
void *value, size_t max_len)
|
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
|
|
|
return ep->dev->driver->ep_get(ep, opt, value, max_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an option value for a given usb device endpoint
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep_set
|
|
|
|
*
|
|
|
|
* @pre `(ep != NULL)`
|
|
|
|
* @pre `(ep->dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[in] opt option type
|
|
|
|
* @param[in] value pointer to the value
|
|
|
|
* @param[in] value_len the length of @p value
|
|
|
|
*
|
|
|
|
* @return number of bytes used from @p value
|
|
|
|
* @return `< 0` on error
|
|
|
|
*/
|
|
|
|
static inline int usbdev_ep_set(usbdev_ep_t *ep, usbopt_ep_t opt,
|
|
|
|
const void *value, size_t value_len)
|
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
|
|
|
return ep->dev->driver->ep_set(ep, opt, value, value_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief an endpoint's user-space event handler
|
|
|
|
*
|
|
|
|
* @see @ref usbdev_driver_t::ep_esr
|
|
|
|
*
|
|
|
|
* @pre `(ep != NULL)`
|
|
|
|
* @pre `(ep->dev != NULL)`
|
|
|
|
*
|
|
|
|
* @param[in] ep USB endpoint descriptor to service
|
|
|
|
*/
|
|
|
|
static inline void usbdev_ep_esr(usbdev_ep_t *ep)
|
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
|
|
|
ep->dev->driver->ep_esr(ep);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-10-26 19:39:12 +02:00
|
|
|
* @brief Submit a buffer for a USB endpoint transmission
|
|
|
|
*
|
|
|
|
* When dealing with an OUT type endpoint, @p len must be the maximum allowed
|
|
|
|
* transfer size for the endpoint. The host is allowed to transfer fewer bytes
|
|
|
|
* than @p len.
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* @see @ref usbdev_driver_t::xmit
|
2018-08-23 14:42:55 +02:00
|
|
|
*
|
|
|
|
* @pre `(ep != NULL)`
|
|
|
|
* @pre `(ep->dev != NULL)`
|
|
|
|
*
|
2021-10-26 19:39:12 +02:00
|
|
|
* @param[in] ep USB endpoint descriptor
|
|
|
|
* @param[inout] buf Buffer to submit for transmission
|
|
|
|
* @param[in] len length of the buffer in bytes to be transmitted or received
|
2018-08-23 14:42:55 +02:00
|
|
|
*/
|
2021-10-26 19:39:12 +02:00
|
|
|
static inline int usbdev_ep_xmit(usbdev_ep_t *ep, uint8_t *buf, size_t len)
|
2018-08-23 14:42:55 +02:00
|
|
|
{
|
|
|
|
assert(ep);
|
|
|
|
assert(ep->dev);
|
2021-10-26 19:39:12 +02:00
|
|
|
return ep->dev->driver->xmit(ep, buf, len);
|
2018-08-23 14:42:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* PERIPH_USBDEV_H */
|
|
|
|
/** @} */
|