gadget-kspconsole/unicore-mx/lib/usbh/usbh_urb.c

167 lines
4.8 KiB
C
Raw Normal View History

2023-11-27 15:13:55 +01:00
/*
* This file is part of the unicore-mx project.
*
* Copyright (C) 2016 Kuldeep Singh Dhaka <kuldeepdhaka9@gmail.com>
*
* This library is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "usbh-private.h"
#if defined(USBH_DEBUG)
static const char *stringify_transfer_status(usbh_transfer_status status) {
switch (status) {
case USBH_ONE_PACKET_DATA: return "USBH_ONE_PACKET_DATA";
case USBH_SUCCESS: return "USBH_SUCCESS";
case USBH_ERR_TIMEOUT: return "USBH_ERR_TIMEOUT";
case USBH_ERR_IO: return "USBH_ERR_IO";
case USBH_ERR_SIZE: return "USBH_ERR_SIZE";
case USBH_ERR_STALL: return "USBH_ERR_STALL";
case USBH_ERR_RES_UNAVAIL: return "USBH_ERR_RES_UNAVAIL";
case USBH_ERR_NO_DEVICE: return "USBH_ERR_NO_DEVICE";
case USBH_ERR_BABBLE: return "USBH_ERR_BABBLE";
case USBH_ERR_DTOG: return "USBH_ERR_DTOG";
case USBH_ERR_SHORT_PACKET: return "USBH_ERR_SHORT_PACKET";
case USBH_ERR_INVALID: return "USBH_ERR_INVALID";
case USBH_ERR_CANCEL: return "USBH_ERR_CANCEL";
default: return "**UNKNOWN**";
}
}
#endif
/**
* Invalidate the URB.
* It will mark this as free for other use
* @param urb USB Request Block
*/
void usbh_urb_invalidate(usbh_urb *urb)
{
urb->id = USBH_INVALID_URB_ID;
}
/**
* Free the URB as it purpose has been served.
* @param urb USB Request Block
* @param status Transfer status to provide to callback
*/
void usbh_urb_free(usbh_urb *urb, usbh_transfer_status status)
{
LOG_CALL
usbh_urb_id cached_urb_id = urb->id;
urb->id = USBH_INVALID_URB_ID;
if (urb->backend_tag != INVALID_BACKEND_TAG) {
usbh_host *host = urb->transfer.device->host;
host->backend->transfer_cancel(host, urb);
}
LOGF_LN("URB %"PRIu64" transfer status = %s", cached_urb_id,
stringify_transfer_status(status));
if (urb->transfer.callback != NULL) {
if (urb->transfer.flags & USBH_FLAG_NO_SUCCESS_CALLBACK) {
if (status == USBH_SUCCESS) {
return;
}
}
const usbh_transfer transfer = urb->transfer; /* incase got overwritten */
transfer.callback(&transfer, status, cached_urb_id);
}
}
/**
* Get the transfer direction
* @param transfer USB Transfer
* @return true if out, false on in
*/
static bool transfer_dir(const usbh_transfer *transfer)
{
if (transfer->ep_type == USBH_EP_CONTROL) {
return !(transfer->setup.bmRequestType & USB_REQ_TYPE_DIRECTION);
} else {
return IS_OUT_ENDPOINT(transfer->ep_addr);
}
}
/**
* Called by backend to get a pointer to receive/transmit 1 packet
* @param urb USB Request Block
* @param len The number of bytes it should be able to hold/provide
* @return pointer to data
*/
void *usbh_urb_get_data_pointer(usbh_urb *urb, uint16_t len)
{
usbh_transfer *transfer = &urb->transfer;
bool out = transfer_dir(transfer);
#if defined(USBH_DEBUG)
if ((transfer->transferred + len) > transfer->length) {
LOGF_LN("URB %"PRIu64" buffer overflow detected! "
"(backend want to %s %"PRIu16" bytes to buffer)", urb->id,
out ? "read" : "write", len);
LOGF_LN("transfer->length: %"PRIu16, transfer->length);
LOGF_LN("transfer->transferred: %"PRIu16, transfer->transferred);
}
#else
(void) len;
#endif
if (transfer->flags & USBH_FLAG_PER_PACKET_CALLBACK) {
if (out) {
/* OUT endpoint, get data from user */
URB_CALLBACK(urb, USBH_ONE_PACKET_DATA);
}
}
if (transfer->flags & USBH_FLAG_NO_MEMORY_INCREMENT) {
/* User said to reuse same buffer location everytime. */
return transfer->data;
}
return transfer->data + transfer->transferred;
}
/**
* Called by backend when it get data
* @param urb USB Request Block
* @param len
*/
void usbh_urb_inc_data_pointer(usbh_urb *urb, uint16_t len)
{
usbh_transfer *transfer = &urb->transfer;
bool out = transfer_dir(transfer);
#if defined(USBH_DEBUG)
if ((transfer->transferred + len) > transfer->length) {
LOGF_LN("URB %"PRIu64" buffer overflow detected! "
"(backend is reporting that it has %s %"PRIu16" bytes)", urb->id,
out ? "readed" : "written", len);
LOGF_LN("transfer->length: %"PRIu16, transfer->length);
LOGF_LN("transfer->transferred: %"PRIu16, transfer->transferred);
}
#endif
transfer->transferred += len;
if (transfer->flags & USBH_FLAG_PER_PACKET_CALLBACK) {
if (!out) {
/* IN endpoint, give data to user */
URB_CALLBACK(urb, USBH_ONE_PACKET_DATA);
}
}
}