490 lines
11 KiB
C
490 lines
11 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Cadence CDNSP DRD Driver.
|
|
*
|
|
* Copyright (C) 2020 Cadence.
|
|
*
|
|
* Author: Pawel Laszczak <pawell@cadence.com>
|
|
*
|
|
*/
|
|
|
|
#include <linux/usb/composite.h>
|
|
#include <linux/usb/gadget.h>
|
|
#include <linux/list.h>
|
|
|
|
#include "cdnsp-gadget.h"
|
|
#include "cdnsp-trace.h"
|
|
|
|
static void cdnsp_ep0_stall(struct cdnsp_device *pdev)
|
|
{
|
|
struct cdnsp_request *preq;
|
|
struct cdnsp_ep *pep;
|
|
|
|
pep = &pdev->eps[0];
|
|
preq = next_request(&pep->pending_list);
|
|
|
|
if (pdev->three_stage_setup) {
|
|
cdnsp_halt_endpoint(pdev, pep, true);
|
|
|
|
if (preq)
|
|
cdnsp_gadget_giveback(pep, preq, -ECONNRESET);
|
|
} else {
|
|
pep->ep_state |= EP0_HALTED_STATUS;
|
|
|
|
if (preq)
|
|
list_del(&preq->list);
|
|
|
|
cdnsp_status_stage(pdev);
|
|
}
|
|
}
|
|
|
|
static int cdnsp_ep0_delegate_req(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
int ret;
|
|
|
|
spin_unlock(&pdev->lock);
|
|
ret = pdev->gadget_driver->setup(&pdev->gadget, ctrl);
|
|
spin_lock(&pdev->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int cdnsp_ep0_set_config(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
enum usb_device_state state = pdev->gadget.state;
|
|
u32 cfg;
|
|
int ret;
|
|
|
|
cfg = le16_to_cpu(ctrl->wValue);
|
|
|
|
switch (state) {
|
|
case USB_STATE_ADDRESS:
|
|
trace_cdnsp_ep0_set_config("from Address state");
|
|
break;
|
|
case USB_STATE_CONFIGURED:
|
|
trace_cdnsp_ep0_set_config("from Configured state");
|
|
break;
|
|
default:
|
|
dev_err(pdev->dev, "Set Configuration - bad device state\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (!cfg)
|
|
usb_gadget_set_state(&pdev->gadget, USB_STATE_ADDRESS);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cdnsp_ep0_set_address(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
enum usb_device_state state = pdev->gadget.state;
|
|
struct cdnsp_slot_ctx *slot_ctx;
|
|
unsigned int slot_state;
|
|
int ret;
|
|
u32 addr;
|
|
|
|
addr = le16_to_cpu(ctrl->wValue);
|
|
|
|
if (addr > 127) {
|
|
dev_err(pdev->dev, "Invalid device address %d\n", addr);
|
|
return -EINVAL;
|
|
}
|
|
|
|
slot_ctx = cdnsp_get_slot_ctx(&pdev->out_ctx);
|
|
|
|
if (state == USB_STATE_CONFIGURED) {
|
|
dev_err(pdev->dev, "Can't Set Address from Configured State\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
pdev->device_address = le16_to_cpu(ctrl->wValue);
|
|
|
|
slot_ctx = cdnsp_get_slot_ctx(&pdev->out_ctx);
|
|
slot_state = GET_SLOT_STATE(le32_to_cpu(slot_ctx->dev_state));
|
|
if (slot_state == SLOT_STATE_ADDRESSED)
|
|
cdnsp_reset_device(pdev);
|
|
|
|
/*set device address*/
|
|
ret = cdnsp_setup_device(pdev, SETUP_CONTEXT_ADDRESS);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (addr)
|
|
usb_gadget_set_state(&pdev->gadget, USB_STATE_ADDRESS);
|
|
else
|
|
usb_gadget_set_state(&pdev->gadget, USB_STATE_DEFAULT);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int cdnsp_status_stage(struct cdnsp_device *pdev)
|
|
{
|
|
pdev->ep0_stage = CDNSP_STATUS_STAGE;
|
|
pdev->ep0_preq.request.length = 0;
|
|
|
|
return cdnsp_ep_enqueue(pdev->ep0_preq.pep, &pdev->ep0_preq);
|
|
}
|
|
|
|
static int cdnsp_w_index_to_ep_index(u16 wIndex)
|
|
{
|
|
if (!(wIndex & USB_ENDPOINT_NUMBER_MASK))
|
|
return 0;
|
|
|
|
return ((wIndex & USB_ENDPOINT_NUMBER_MASK) * 2) +
|
|
(wIndex & USB_ENDPOINT_DIR_MASK ? 1 : 0) - 1;
|
|
}
|
|
|
|
static int cdnsp_ep0_handle_status(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
struct cdnsp_ep *pep;
|
|
__le16 *response;
|
|
int ep_sts = 0;
|
|
u16 status = 0;
|
|
u32 recipient;
|
|
|
|
recipient = ctrl->bRequestType & USB_RECIP_MASK;
|
|
|
|
switch (recipient) {
|
|
case USB_RECIP_DEVICE:
|
|
status = pdev->gadget.is_selfpowered;
|
|
status |= pdev->may_wakeup << USB_DEVICE_REMOTE_WAKEUP;
|
|
|
|
if (pdev->gadget.speed >= USB_SPEED_SUPER) {
|
|
status |= pdev->u1_allowed << USB_DEV_STAT_U1_ENABLED;
|
|
status |= pdev->u2_allowed << USB_DEV_STAT_U2_ENABLED;
|
|
}
|
|
break;
|
|
case USB_RECIP_INTERFACE:
|
|
/*
|
|
* Function Remote Wake Capable D0
|
|
* Function Remote Wakeup D1
|
|
*/
|
|
return cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
case USB_RECIP_ENDPOINT:
|
|
ep_sts = cdnsp_w_index_to_ep_index(le16_to_cpu(ctrl->wIndex));
|
|
pep = &pdev->eps[ep_sts];
|
|
ep_sts = GET_EP_CTX_STATE(pep->out_ctx);
|
|
|
|
/* check if endpoint is stalled */
|
|
if (ep_sts == EP_STATE_HALTED)
|
|
status = BIT(USB_ENDPOINT_HALT);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
response = (__le16 *)pdev->setup_buf;
|
|
*response = cpu_to_le16(status);
|
|
|
|
pdev->ep0_preq.request.length = sizeof(*response);
|
|
pdev->ep0_preq.request.buf = pdev->setup_buf;
|
|
|
|
return cdnsp_ep_enqueue(pdev->ep0_preq.pep, &pdev->ep0_preq);
|
|
}
|
|
|
|
static void cdnsp_enter_test_mode(struct cdnsp_device *pdev)
|
|
{
|
|
u32 temp;
|
|
|
|
temp = readl(&pdev->active_port->regs->portpmsc) & ~GENMASK(31, 28);
|
|
temp |= PORT_TEST_MODE(pdev->test_mode);
|
|
writel(temp, &pdev->active_port->regs->portpmsc);
|
|
}
|
|
|
|
static int cdnsp_ep0_handle_feature_device(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl,
|
|
int set)
|
|
{
|
|
enum usb_device_state state;
|
|
enum usb_device_speed speed;
|
|
u16 tmode;
|
|
|
|
state = pdev->gadget.state;
|
|
speed = pdev->gadget.speed;
|
|
|
|
switch (le16_to_cpu(ctrl->wValue)) {
|
|
case USB_DEVICE_REMOTE_WAKEUP:
|
|
pdev->may_wakeup = !!set;
|
|
trace_cdnsp_may_wakeup(set);
|
|
break;
|
|
case USB_DEVICE_U1_ENABLE:
|
|
if (state != USB_STATE_CONFIGURED || speed < USB_SPEED_SUPER)
|
|
return -EINVAL;
|
|
|
|
pdev->u1_allowed = !!set;
|
|
trace_cdnsp_u1(set);
|
|
break;
|
|
case USB_DEVICE_U2_ENABLE:
|
|
if (state != USB_STATE_CONFIGURED || speed < USB_SPEED_SUPER)
|
|
return -EINVAL;
|
|
|
|
pdev->u2_allowed = !!set;
|
|
trace_cdnsp_u2(set);
|
|
break;
|
|
case USB_DEVICE_LTM_ENABLE:
|
|
return -EINVAL;
|
|
case USB_DEVICE_TEST_MODE:
|
|
if (state != USB_STATE_CONFIGURED || speed > USB_SPEED_HIGH)
|
|
return -EINVAL;
|
|
|
|
tmode = le16_to_cpu(ctrl->wIndex);
|
|
|
|
if (!set || (tmode & 0xff) != 0)
|
|
return -EINVAL;
|
|
|
|
tmode = tmode >> 8;
|
|
|
|
if (tmode > USB_TEST_FORCE_ENABLE || tmode < USB_TEST_J)
|
|
return -EINVAL;
|
|
|
|
pdev->test_mode = tmode;
|
|
|
|
/*
|
|
* Test mode must be set before Status Stage but controller
|
|
* will start testing sequence after Status Stage.
|
|
*/
|
|
cdnsp_enter_test_mode(pdev);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cdnsp_ep0_handle_feature_intf(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl,
|
|
int set)
|
|
{
|
|
u16 wValue, wIndex;
|
|
int ret;
|
|
|
|
wValue = le16_to_cpu(ctrl->wValue);
|
|
wIndex = le16_to_cpu(ctrl->wIndex);
|
|
|
|
switch (wValue) {
|
|
case USB_INTRF_FUNC_SUSPEND:
|
|
ret = cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/*
|
|
* Remote wakeup is enabled when any function within a device
|
|
* is enabled for function remote wakeup.
|
|
*/
|
|
if (wIndex & USB_INTRF_FUNC_SUSPEND_RW)
|
|
pdev->may_wakeup++;
|
|
else
|
|
if (pdev->may_wakeup > 0)
|
|
pdev->may_wakeup--;
|
|
|
|
return 0;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cdnsp_ep0_handle_feature_endpoint(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl,
|
|
int set)
|
|
{
|
|
struct cdnsp_ep *pep;
|
|
u16 wValue;
|
|
|
|
wValue = le16_to_cpu(ctrl->wValue);
|
|
pep = &pdev->eps[cdnsp_w_index_to_ep_index(le16_to_cpu(ctrl->wIndex))];
|
|
|
|
switch (wValue) {
|
|
case USB_ENDPOINT_HALT:
|
|
if (!set && (pep->ep_state & EP_WEDGE)) {
|
|
/* Resets Sequence Number */
|
|
cdnsp_halt_endpoint(pdev, pep, 0);
|
|
cdnsp_halt_endpoint(pdev, pep, 1);
|
|
break;
|
|
}
|
|
|
|
return cdnsp_halt_endpoint(pdev, pep, set);
|
|
default:
|
|
dev_warn(pdev->dev, "WARN Incorrect wValue %04x\n", wValue);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cdnsp_ep0_handle_feature(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl,
|
|
int set)
|
|
{
|
|
switch (ctrl->bRequestType & USB_RECIP_MASK) {
|
|
case USB_RECIP_DEVICE:
|
|
return cdnsp_ep0_handle_feature_device(pdev, ctrl, set);
|
|
case USB_RECIP_INTERFACE:
|
|
return cdnsp_ep0_handle_feature_intf(pdev, ctrl, set);
|
|
case USB_RECIP_ENDPOINT:
|
|
return cdnsp_ep0_handle_feature_endpoint(pdev, ctrl, set);
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
static int cdnsp_ep0_set_sel(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
enum usb_device_state state = pdev->gadget.state;
|
|
u16 wLength;
|
|
|
|
if (state == USB_STATE_DEFAULT)
|
|
return -EINVAL;
|
|
|
|
wLength = le16_to_cpu(ctrl->wLength);
|
|
|
|
if (wLength != 6) {
|
|
dev_err(pdev->dev, "Set SEL should be 6 bytes, got %d\n",
|
|
wLength);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/*
|
|
* To handle Set SEL we need to receive 6 bytes from Host. So let's
|
|
* queue a usb_request for 6 bytes.
|
|
*/
|
|
pdev->ep0_preq.request.length = 6;
|
|
pdev->ep0_preq.request.buf = pdev->setup_buf;
|
|
|
|
return cdnsp_ep_enqueue(pdev->ep0_preq.pep, &pdev->ep0_preq);
|
|
}
|
|
|
|
static int cdnsp_ep0_set_isoch_delay(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
if (le16_to_cpu(ctrl->wIndex) || le16_to_cpu(ctrl->wLength))
|
|
return -EINVAL;
|
|
|
|
pdev->gadget.isoch_delay = le16_to_cpu(ctrl->wValue);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cdnsp_ep0_std_request(struct cdnsp_device *pdev,
|
|
struct usb_ctrlrequest *ctrl)
|
|
{
|
|
int ret;
|
|
|
|
switch (ctrl->bRequest) {
|
|
case USB_REQ_GET_STATUS:
|
|
ret = cdnsp_ep0_handle_status(pdev, ctrl);
|
|
break;
|
|
case USB_REQ_CLEAR_FEATURE:
|
|
ret = cdnsp_ep0_handle_feature(pdev, ctrl, 0);
|
|
break;
|
|
case USB_REQ_SET_FEATURE:
|
|
ret = cdnsp_ep0_handle_feature(pdev, ctrl, 1);
|
|
break;
|
|
case USB_REQ_SET_ADDRESS:
|
|
ret = cdnsp_ep0_set_address(pdev, ctrl);
|
|
break;
|
|
case USB_REQ_SET_CONFIGURATION:
|
|
ret = cdnsp_ep0_set_config(pdev, ctrl);
|
|
break;
|
|
case USB_REQ_SET_SEL:
|
|
ret = cdnsp_ep0_set_sel(pdev, ctrl);
|
|
break;
|
|
case USB_REQ_SET_ISOCH_DELAY:
|
|
ret = cdnsp_ep0_set_isoch_delay(pdev, ctrl);
|
|
break;
|
|
case USB_REQ_SET_INTERFACE:
|
|
/*
|
|
* Add request into pending list to block sending status stage
|
|
* by libcomposite.
|
|
*/
|
|
list_add_tail(&pdev->ep0_preq.list,
|
|
&pdev->ep0_preq.pep->pending_list);
|
|
|
|
ret = cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
if (ret == -EBUSY)
|
|
ret = 0;
|
|
|
|
list_del(&pdev->ep0_preq.list);
|
|
break;
|
|
default:
|
|
ret = cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void cdnsp_setup_analyze(struct cdnsp_device *pdev)
|
|
{
|
|
struct usb_ctrlrequest *ctrl = &pdev->setup;
|
|
int ret = 0;
|
|
u16 len;
|
|
|
|
trace_cdnsp_ctrl_req(ctrl);
|
|
|
|
if (!pdev->gadget_driver)
|
|
goto out;
|
|
|
|
if (pdev->gadget.state == USB_STATE_NOTATTACHED) {
|
|
dev_err(pdev->dev, "ERR: Setup detected in unattached state\n");
|
|
ret = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
/* Restore the ep0 to Stopped/Running state. */
|
|
if (pdev->eps[0].ep_state & EP_HALTED) {
|
|
trace_cdnsp_ep0_halted("Restore to normal state");
|
|
cdnsp_halt_endpoint(pdev, &pdev->eps[0], 0);
|
|
}
|
|
|
|
/*
|
|
* Finishing previous SETUP transfer by removing request from
|
|
* list and informing upper layer
|
|
*/
|
|
if (!list_empty(&pdev->eps[0].pending_list)) {
|
|
struct cdnsp_request *req;
|
|
|
|
trace_cdnsp_ep0_request("Remove previous");
|
|
req = next_request(&pdev->eps[0].pending_list);
|
|
cdnsp_ep_dequeue(&pdev->eps[0], req);
|
|
}
|
|
|
|
len = le16_to_cpu(ctrl->wLength);
|
|
if (!len) {
|
|
pdev->three_stage_setup = false;
|
|
pdev->ep0_expect_in = false;
|
|
} else {
|
|
pdev->three_stage_setup = true;
|
|
pdev->ep0_expect_in = !!(ctrl->bRequestType & USB_DIR_IN);
|
|
}
|
|
|
|
if ((ctrl->bRequestType & USB_TYPE_MASK) == USB_TYPE_STANDARD)
|
|
ret = cdnsp_ep0_std_request(pdev, ctrl);
|
|
else
|
|
ret = cdnsp_ep0_delegate_req(pdev, ctrl);
|
|
|
|
if (!len)
|
|
pdev->ep0_stage = CDNSP_STATUS_STAGE;
|
|
|
|
if (ret == USB_GADGET_DELAYED_STATUS) {
|
|
trace_cdnsp_ep0_status_stage("delayed");
|
|
return;
|
|
}
|
|
out:
|
|
if (ret < 0)
|
|
cdnsp_ep0_stall(pdev);
|
|
else if (pdev->ep0_stage == CDNSP_STATUS_STAGE)
|
|
cdnsp_status_stage(pdev);
|
|
}
|