Hi, On Mon, Oct 06, 2014 at 06:55:05PM +0300, Laurent Pinchart wrote: > The ISP1761 is a dual-mode host and device controller backward > compatible on the host side with the ISP1760. Add support for the device > controller. > > Signed-off-by: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> finally my part :-) Unfortunately I don't have HW to verify all this, so I'd really like to see some test results. It would be nice if you had access to a windows box to run USB20CV on this UDC too, but I guess that's too much to ask :-p Anyway, a few comments below. I might come up with more as I digest this code. > diff --git a/drivers/usb/gadget/udc/Kconfig b/drivers/usb/gadget/udc/Kconfig > index 34ebaa6..7bdfdf0 100644 > --- a/drivers/usb/gadget/udc/Kconfig > +++ b/drivers/usb/gadget/udc/Kconfig > @@ -109,6 +109,13 @@ config USB_GR_UDC > Select this to support Aeroflex Gaisler GRUSBDC cores from the GRLIB > VHDL IP core library. > > +config USB_ISP1761_UDC > + boolean "NXP ISP1761 USB Device Controller" not tristate ? > diff --git a/drivers/usb/host/isp1760-core.c b/drivers/usb/host/isp1760-core.c > index 595f234..54aedc2 100644 > --- a/drivers/usb/host/isp1760-core.c > +++ b/drivers/usb/host/isp1760-core.c > @@ -24,9 +24,11 @@ > #include "isp1760-core.h" > #include "isp1760-hcd.h" > #include "isp1760-regs.h" > +#include "isp1760-udc.h" > > static void isp1760_init_core(struct isp1760_device *isp) > { > + u32 otgctrl; > u32 hwmode; > > /* Low-level chip reset */ > @@ -65,6 +67,17 @@ static void isp1760_init_core(struct isp1760_device *isp) > hwmode |= HW_INTR_EDGE_TRIG; > > /* > + * The ISP1761 has a dedicated DC IRQ line but supports sharing the HC > + * IRQ line for both the host and device controllers. Hardcode IRQ > + * sharing for now and disable the DC interrupts globally to avoid > + * spurious interrupts during HCD registration. > + */ > + if (isp->devflags & ISP1760_FLAG_ISP1761) { > + isp1760_write32(isp->regs, DC_MODE, 0); > + hwmode |= HW_COMN_IRQ; > + } > + > + /* > * We have to set this first in case we're in 16-bit mode. > * Write it twice to ensure correct upper bits if switching > * to 16-bit mode. > @@ -72,11 +85,35 @@ static void isp1760_init_core(struct isp1760_device *isp) > isp1760_write32(isp->regs, HC_HW_MODE_CTRL, hwmode); > isp1760_write32(isp->regs, HC_HW_MODE_CTRL, hwmode); > > + /* > + * PORT 1 Control register of the ISP1760 is the OTG control register > + * on ISP1761. > + * > + * TODO: Really support OTG. For now we configure port 1 in device mode > + * when OTG is requested. > + */ > + if ((isp->devflags & ISP1760_FLAG_ISP1761) && > + (isp->devflags & ISP1760_FLAG_OTG_EN)) > + otgctrl = ((HW_DM_PULLDOWN | HW_DP_PULLDOWN) << 16) > + | HW_OTG_DISABLE; > + else > + otgctrl = (HW_SW_SEL_HC_DC << 16) > + | (HW_VBUS_DRV | HW_SEL_CP_EXT); > + > + isp1760_write32(isp->regs, HC_PORT1_CTRL, otgctrl); > + usleep_range(10000, 11000); do you need a comment on this usleep_range() ? Perhaps pointing to a section of documentation which state this is necessary ? 10ms is quite a bit of wait. > + > dev_info(isp->dev, "bus width: %u, oc: %s\n", > isp->devflags & ISP1760_FLAG_BUS_WIDTH_16 ? 16 : 32, > isp->devflags & ISP1760_FLAG_ANALOG_OC ? "analog" : "digital"); > } > > +void isp1760_set_pullup(struct isp1760_device *isp, bool enable) > +{ > + isp1760_write32(isp->regs, HW_OTG_CTRL_SET, > + enable ? HW_DP_PULLUP : HW_DP_PULLUP << 16); > +} > + > int isp1760_register(struct resource *mem, int irq, unsigned long irqflags, > int rst_gpio, struct device *dev, unsigned int devflags) > { > @@ -126,6 +163,13 @@ int isp1760_register(struct resource *mem, int irq, unsigned long irqflags, > if (ret < 0) > goto error; > > + if (devflags & ISP1760_FLAG_ISP1761) { > + ret = isp1760_udc_register(isp, irq, irqflags | IRQF_SHARED | > + IRQF_DISABLED); > + if (ret < 0) > + goto error; > + } > + > dev_set_drvdata(dev, isp); > > return 0; > @@ -148,6 +192,9 @@ void isp1760_unregister(struct device *dev) > > release_mem_region(isp->mem_start, isp->mem_size); > > + if (isp->devflags & ISP1760_FLAG_ISP1761) > + isp1760_udc_unregister(isp); > + > isp1760_hcd_unregister(&isp->hcd); > > iounmap(isp->regs); > @@ -162,6 +209,11 @@ void isp1760_shutdown(struct device *dev) > { > struct isp1760_device *isp = dev_get_drvdata(dev); > > + if (isp->devflags & ISP1760_FLAG_ISP1761) { > + isp1760_write32(isp->regs, DC_MODE, DC_SFRESET); > + isp1760_write32(isp->regs, DC_MODE, 0); > + } looks like you need to fix commit log of previous patch where you state you'd maybe reset device side too, but that code only comes in here. > + > isp1760_write32(isp->regs, HC_RESET_REG, SW_RESET_RESET_HC); > } > > diff --git a/drivers/usb/host/isp1760-core.h b/drivers/usb/host/isp1760-core.h > index fea1622..7d3aad4 100644 > --- a/drivers/usb/host/isp1760-core.h > +++ b/drivers/usb/host/isp1760-core.h > @@ -19,6 +19,7 @@ > #include <linux/ioport.h> > > #include "isp1760-hcd.h" > +#include "isp1760-udc.h" > > struct device; > > @@ -48,6 +49,7 @@ struct isp1760_device { > int rst_gpio; > > struct isp1760_hcd hcd; > + struct isp1760_udc udc; > }; > > int isp1760_register(struct resource *mem, int irq, unsigned long irqflags, > @@ -55,6 +57,8 @@ int isp1760_register(struct resource *mem, int irq, unsigned long irqflags, > void isp1760_unregister(struct device *dev); > void isp1760_shutdown(struct device *dev); > > +void isp1760_set_pullup(struct isp1760_device *isp, bool enable); > + > static inline u32 isp1760_read32(void __iomem *base, u32 reg) > { > return readl(base + reg); > diff --git a/drivers/usb/host/isp1760-hcd.c b/drivers/usb/host/isp1760-hcd.c > index a6081d5..1e3d1d9 100644 > --- a/drivers/usb/host/isp1760-hcd.c > +++ b/drivers/usb/host/isp1760-hcd.c > @@ -502,15 +502,6 @@ static int isp1760_hc_setup(struct usb_hcd *hcd) > > reg_write32(hcd->regs, HC_INTERRUPT_ENABLE, INTERRUPT_ENABLE_MASK); > > - /* > - * PORT 1 Control register of the ISP1760 is the OTG control > - * register on ISP1761. Since there is no OTG or device controller > - * support in this driver, we use port 1 as a "normal" USB host port on > - * both chips. > - */ > - reg_write32(hcd->regs, HC_PORT1_CTRL, PORT1_POWER | PORT1_INIT2); > - mdelay(10); > - looks like moving this code should be part of its own patch. > priv->hcs_params = reg_read32(hcd->regs, HC_HCSPARAMS); > > return priv_init(hcd); > diff --git a/drivers/usb/host/isp1760-regs.h b/drivers/usb/host/isp1760-regs.h > index c2a3900..b67095c 100644 > --- a/drivers/usb/host/isp1760-regs.h > +++ b/drivers/usb/host/isp1760-regs.h > @@ -16,6 +16,10 @@ > #ifndef _ISP1760_REGS_H_ > #define _ISP1760_REGS_H_ > > +/* ----------------------------------------------------------------------------- > + * Host Controller > + */ > + > /* EHCI capability registers */ > #define HC_CAPLENGTH 0x000 > #define HC_LENGTH(p) (((p) >> 00) & 0x00ff) /* bits 7:0 */ > @@ -70,6 +74,9 @@ > #define HC_HW_MODE_CTRL 0x300 > #define ALL_ATX_RESET (1 << 31) > #define HW_ANA_DIGI_OC (1 << 15) > +#define HW_DEV_DMA (1 << 11) > +#define HW_COMN_IRQ (1 << 10) > +#define HW_COMN_DMA (1 << 9) > #define HW_DATA_BUS_32BIT (1 << 8) > #define HW_DACK_POL_HIGH (1 << 6) > #define HW_DREQ_POL_HIGH (1 << 5) > @@ -98,6 +105,17 @@ > #define PORT1_INIT2 (1 << 23) > #define HW_OTG_CTRL_SET 0x374 > #define HW_OTG_CTRL_CLR 0x376 > +#define HW_OTG_DISABLE (1 << 10) > +#define HW_OTG_SE0_EN (1 << 9) > +#define HW_BDIS_ACON_EN (1 << 8) > +#define HW_SW_SEL_HC_DC (1 << 7) > +#define HW_VBUS_CHRG (1 << 6) > +#define HW_VBUS_DISCHRG (1 << 5) > +#define HW_VBUS_DRV (1 << 4) > +#define HW_SEL_CP_EXT (1 << 3) > +#define HW_DM_PULLDOWN (1 << 2) > +#define HW_DP_PULLDOWN (1 << 1) > +#define HW_DP_PULLUP (1 << 0) > > /* Interrupt Register */ > #define HC_INTERRUPT_REG 0x310 > @@ -117,4 +135,96 @@ > #define HC_INT_IRQ_MASK_AND_REG 0x328 > #define HC_ATL_IRQ_MASK_AND_REG 0x32c > > +/* ----------------------------------------------------------------------------- > + * Peripheral Controller > + */ > + > +/* Initialization Registers */ > +#define DC_ADDRESS 0x0200 > +#define DC_DEVEN (1 << 7) > + > +#define DC_MODE 0x020c > +#define DC_DMACLKON (1 << 9) > +#define DC_VBUSSTAT (1 << 8) > +#define DC_CLKAON (1 << 7) > +#define DC_SNDRSU (1 << 6) > +#define DC_GOSUSP (1 << 5) > +#define DC_SFRESET (1 << 4) > +#define DC_GLINTENA (1 << 3) > +#define DC_WKUPCS (1 << 2) > + > +#define DC_INTCONF 0x0210 > +#define DC_CDBGMOD_ACK_NAK (0 << 6) > +#define DC_CDBGMOD_ACK (1 << 6) > +#define DC_CDBGMOD_ACK_1NAK (2 << 6) > +#define DC_DDBGMODIN_ACK_NAK (0 << 4) > +#define DC_DDBGMODIN_ACK (1 << 4) > +#define DC_DDBGMODIN_ACK_1NAK (2 << 4) > +#define DC_DDBGMODOUT_ACK_NYET_NAK (0 << 2) > +#define DC_DDBGMODOUT_ACK_NYET (1 << 2) > +#define DC_DDBGMODOUT_ACK_NYET_1NAK (2 << 2) > +#define DC_INTLVL (1 << 1) > +#define DC_INTPOL (1 << 0) > + > +#define DC_DEBUG 0x0212 > +#define DC_INTENABLE 0x0214 > +#define DC_IEPTX(n) (1 << (11 + 2 * (n))) > +#define DC_IEPRX(n) (1 << (10 + 2 * (n))) > +#define DC_IEPRXTX(n) (3 << (10 + 2 * (n))) > +#define DC_IEP0SETUP (1 << 8) > +#define DC_IEVBUS (1 << 7) > +#define DC_IEDMA (1 << 6) > +#define DC_IEHS_STA (1 << 5) > +#define DC_IERESM (1 << 4) > +#define DC_IESUSP (1 << 3) > +#define DC_IEPSOF (1 << 2) > +#define DC_IESOF (1 << 1) > +#define DC_IEBRST (1 << 0) > + > +/* Data Flow Registers */ > +#define DC_EPINDEX 0x022c > +#define DC_EP0SETUP (1 << 5) > +#define DC_ENDPIDX(n) ((n) << 1) > +#define DC_EPDIR (1 << 0) > + > +#define DC_CTRLFUNC 0x0228 > +#define DC_CLBUF (1 << 4) > +#define DC_VENDP (1 << 3) > +#define DC_DSEN (1 << 2) > +#define DC_STATUS (1 << 1) > +#define DC_STALL (1 << 0) > + > +#define DC_DATAPORT 0x0220 > +#define DC_BUFLEN 0x021c > +#define DC_DATACOUNT_MASK 0xffff > +#define DC_BUFSTAT 0x021e > +#define DC_EPMAXPKTSZ 0x0204 > + > +#define DC_EPTYPE 0x0208 > +#define DC_NOEMPKT (1 << 4) > +#define DC_EPENABLE (1 << 3) > +#define DC_DBLBUF (1 << 2) > +#define DC_ENDPTYP_ISOC (1 << 0) > +#define DC_ENDPTYP_BULK (2 << 0) > +#define DC_ENDPTYP_INTERRUPT (3 << 0) > + > +/* DMA Registers */ > +#define DC_DMACMD 0x0230 > +#define DC_DMATXCOUNT 0x0234 > +#define DC_DMACONF 0x0238 > +#define DC_DMAHW 0x023c > +#define DC_DMAINTREASON 0x0250 > +#define DC_DMAINTEN 0x0254 > +#define DC_DMAEP 0x0258 > +#define DC_DMABURSTCOUNT 0x0264 > + > +/* General Registers */ > +#define DC_INTERRUPT 0x0218 > +#define DC_CHIPID 0x0270 > +#define DC_FRAMENUM 0x0274 > +#define DC_SCRATCH 0x0278 > +#define DC_UNLOCKDEV 0x027c > +#define DC_INTPULSEWIDTH 0x0280 > +#define DC_TESTMODE 0x0284 > + > #endif > diff --git a/drivers/usb/host/isp1760-udc.c b/drivers/usb/host/isp1760-udc.c > new file mode 100644 > index 0000000..2f447ed > --- /dev/null > +++ b/drivers/usb/host/isp1760-udc.c > @@ -0,0 +1,1418 @@ > +/* > + * Driver for the NXP ISP1761 device controller > + * > + * Copyright 2014 Ideas on Board Oy > + * > + * Contacts: > + * Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > + * > + * This program is free software; you can redistribute it and/or > + * modify it under the terms of the GNU General Public License > + * version 2 as published by the Free Software Foundation. > + */ > + > +#include <linux/interrupt.h> > +#include <linux/io.h> > +#include <linux/kernel.h> > +#include <linux/list.h> > +#include <linux/module.h> > +#include <linux/slab.h> > +#include <linux/usb.h> > + > +#include "isp1760-core.h" > +#include "isp1760-regs.h" > +#include "isp1760-udc.h" > + > +struct isp1760_request { > + struct usb_request req; > + struct list_head queue; > + struct isp1760_ep *ep; > + unsigned int packet_size; > +}; > + > +static inline struct isp1760_udc *gadget_to_udc(struct usb_gadget *gadget) > +{ > + return container_of(gadget, struct isp1760_udc, gadget); > +} > + > +static inline struct isp1760_ep *ep_to_udc_ep(struct usb_ep *ep) > +{ > + return container_of(ep, struct isp1760_ep, ep); > +} > + > +static inline struct isp1760_request *req_to_udc_req(struct usb_request *req) > +{ > + return container_of(req, struct isp1760_request, req); > +} > + > +static inline u32 isp1760_udc_read(struct isp1760_udc *udc, u16 reg) > +{ > + return isp1760_read32(udc->regs, reg); > +} > + > +static inline void isp1760_udc_write(struct isp1760_udc *udc, u16 reg, u32 val) > +{ > + isp1760_write32(udc->regs, reg, val); > +} > + > +/* ----------------------------------------------------------------------------- > + * Endpoint Management > + */ > + > +static struct isp1760_ep *isp1760_udc_find_ep(struct isp1760_udc *udc, > + u16 index) > +{ > + unsigned int i; > + > + if (index == 0) > + return &udc->ep[0]; > + > + for (i = 1; i < ARRAY_SIZE(udc->ep); ++i) { > + if (udc->ep[i].addr == index) > + return udc->ep[i].desc ? &udc->ep[i] : NULL; > + } > + > + return NULL; > +} > + > +static void __isp1760_udc_select_ep(struct isp1760_ep *ep, int dir) > +{ > + isp1760_udc_write(ep->udc, DC_EPINDEX, > + DC_ENDPIDX(ep->addr & USB_ENDPOINT_NUMBER_MASK) | > + (dir == USB_DIR_IN ? DC_EPDIR : 0)); > +} > + > +/** > + * isp1760_udc_select_ep - Select an endpoint for register access > + * @ep: The endpoint > + * > + * The ISP1761 endpoint registers are banked. This function selects the target > + * endpoint for banked register access. The selection remains valid until the > + * next call to this function, the next direct access to the EPINDEX register > + * or the next reset, whichever comes first. > + * > + * Called with the UDC spinlock held. > + */ > +static void isp1760_udc_select_ep(struct isp1760_ep *ep) > +{ > + __isp1760_udc_select_ep(ep, ep->addr & USB_ENDPOINT_DIR_MASK); > +} > + > +/* Called with the UDC spinlock held. */ > +static void isp1760_udc_ctrl_send_status(struct isp1760_ep *ep, int dir) > +{ > + struct isp1760_udc *udc = ep->udc; > + > + /* > + * Proceed to the status stage. The status stage data packet flows in > + * the direction opposite to the data stage data packets, we thus need > + * to select the OUT/IN endpoint for IN/OUT transfers. > + */ > + isp1760_udc_write(udc, DC_EPINDEX, DC_ENDPIDX(0) | > + (dir == USB_DIR_IN ? 0 : DC_EPDIR)); > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_STATUS); > + > + udc->ep0_state = ISP1760_CTRL_SETUP; > +} > + > +/* Called without the UDC spinlock held. */ > +static void isp1760_udc_request_complete(struct isp1760_ep *ep, > + struct isp1760_request *req, > + int status) > +{ > + struct isp1760_udc *udc = ep->udc; > + unsigned long flags; > + > + dev_dbg(ep->udc->isp->dev, "completing request %p with status %d\n", > + req, status); > + > + req->ep = NULL; > + req->req.status = status; > + req->req.complete(&ep->ep, &req->req); > + > + spin_lock_irqsave(&udc->lock, flags); > + > + /* > + * When completing control OUT requests, move to the status stage after > + * calling the request complete callback. This gives the gadget an > + * opportunity to stall the control transfer if needed. > + */ > + if (status == 0 && ep->addr == 0 && > + udc->ep0_state == ISP1760_CTRL_DATA_OUT) > + isp1760_udc_ctrl_send_status(ep, USB_DIR_OUT); > + > + spin_unlock_irqrestore(&udc->lock, flags); > +} > + > +static void isp1760_udc_ctrl_send_stall(struct isp1760_ep *ep) > +{ > + struct isp1760_udc *udc = ep->udc; > + unsigned long flags; > + > + dev_dbg(ep->udc->isp->dev, "%s(ep%02x)\n", __func__, ep->addr); > + > + spin_lock_irqsave(&udc->lock, flags); > + > + /* Stall both the IN and OUT endpoints. */ > + __isp1760_udc_select_ep(ep, USB_DIR_OUT); > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_STALL); > + __isp1760_udc_select_ep(ep, USB_DIR_IN); > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_STALL); > + > + /* A protocol stall completes the control transaction. */ > + udc->ep0_state = ISP1760_CTRL_SETUP; > + > + spin_unlock_irqrestore(&udc->lock, flags); > +} > + > +/* ----------------------------------------------------------------------------- > + * Data Endpoints > + */ > + > +/* Called with the UDC spinlock held. */ > +static bool isp1760_udc_receive(struct isp1760_ep *ep, > + struct isp1760_request *req) > +{ > + struct isp1760_udc *udc = ep->udc; > + unsigned int len; > + u32 *buf; > + int i; > + > + isp1760_udc_select_ep(ep); > + len = isp1760_udc_read(udc, DC_BUFLEN) & DC_DATACOUNT_MASK; > + > + dev_dbg(udc->isp->dev, "%s: received %u bytes (%u/%u done)\n", > + __func__, len, req->req.actual, req->req.length); > + > + len = min(len, req->req.length - req->req.actual); > + > + if (!len) { > + /* > + * There's no data to be read from the FIFO, acknowledge the RX > + * interrupt by clearing the buffer. > + * > + * FIXME: What if another packet arrives in the meantime ? > + */ > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_CLBUF); > + return false; > + } > + > + buf = req->req.buf + req->req.actual; > + > + /* > + * Make sure not to read more than one extra byte, otherwise data from > + * the next packet might be removed from the FIFO. > + */ > + for (i = len; i > 2; i -= 4, ++buf) > + *buf = le32_to_cpu(isp1760_udc_read(udc, DC_DATAPORT)); > + if (i > 0) > + *(u16 *)buf = le16_to_cpu(readw(udc->regs + DC_DATAPORT)); > + > + req->req.actual += len; > + > + /* TODO Support short_not_ok */ > + > + dev_dbg(udc->isp->dev, > + "%s: req %p actual/length %u/%u maxpacket %u packet size %u\n", > + __func__, req, req->req.actual, req->req.length, ep->maxpacket, > + len); > + > + ep->rx_pending = false; > + > + /* > + * Complete the request if all data has been received or if a short > + * packet has been received. > + */ > + if (req->req.actual == req->req.length || len < ep->maxpacket) { > + list_del(&req->queue); > + return true; > + } > + > + return false; > +} > + > +static void isp1760_udc_transmit(struct isp1760_ep *ep, > + struct isp1760_request *req) > +{ > + struct isp1760_udc *udc = ep->udc; > + u32 *buf = req->req.buf + req->req.actual; > + int i; > + > + req->packet_size = min(req->req.length - req->req.actual, > + ep->maxpacket); > + > + dev_dbg(udc->isp->dev, "%s: transferring %u bytes (%u/%u done)\n", > + __func__, req->packet_size, req->req.actual, > + req->req.length); > + > + __isp1760_udc_select_ep(ep, USB_DIR_IN); > + > + if (req->packet_size) > + isp1760_udc_write(udc, DC_BUFLEN, req->packet_size); > + > + /* > + * Make sure not to write more than one extra byte, otherwise extra data > + * will stay in the FIFO and will be transmitted during the next control > + * request. The endpoint control CLBUF bit is supposed to allow flushing > + * the FIFO for this kind of conditions, but doesn't seem to work. > + */ > + for (i = req->packet_size; i > 2; i -= 4, ++buf) > + isp1760_udc_write(udc, DC_DATAPORT, cpu_to_le32(*buf)); > + if (i > 0) > + writew(cpu_to_le16(*(u16 *)buf), udc->regs + DC_DATAPORT); > + > + if (ep->addr == 0) > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_DSEN); > + if (!req->packet_size) > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_VENDP); > +} > + > +static void isp1760_ep_rx_ready(struct isp1760_ep *ep) > +{ > + struct isp1760_udc *udc = ep->udc; > + struct isp1760_request *req; > + bool complete; > + > + spin_lock(&udc->lock); > + > + if (ep->addr == 0 && udc->ep0_state != ISP1760_CTRL_DATA_OUT) { > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "%s: invalid ep0 state %u\n", __func__, > + udc->ep0_state); > + return; > + } > + > + if (ep->addr != 0 && !ep->desc) { > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "%s: ep%02x is disabled\n", __func__, > + ep->addr); > + return; > + } > + > + if (list_empty(&ep->queue)) { > + ep->rx_pending = true; > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "%s: ep%02x (%p) has no request queued\n", > + __func__, ep->addr, ep); > + return; > + } > + > + req = list_first_entry(&ep->queue, struct isp1760_request, > + queue); > + complete = isp1760_udc_receive(ep, req); > + > + spin_unlock(&udc->lock); > + > + if (complete) > + isp1760_udc_request_complete(ep, req, 0); > +} > + > +static void isp1760_ep_tx_complete(struct isp1760_ep *ep) > +{ > + struct isp1760_udc *udc = ep->udc; > + struct isp1760_request *complete = NULL; > + struct isp1760_request *req; > + bool need_zlp; > + > + spin_lock(&udc->lock); > + > + if (ep->addr == 0 && udc->ep0_state != ISP1760_CTRL_DATA_IN) { > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "TX IRQ: invalid endpoint state %u\n", > + udc->ep0_state); > + return; > + } > + > + if (list_empty(&ep->queue)) { > + /* > + * This can happen for the control endpoint when the reply to > + * the GET_STATUS IN control request is sent directly by the > + * setup IRQ handler. Just proceed to the status stage. > + */ > + if (ep->addr == 0) { > + isp1760_udc_ctrl_send_status(ep, USB_DIR_IN); > + spin_unlock(&udc->lock); > + return; > + } > + > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "%s: ep%02x has no request queued\n", > + __func__, ep->addr); > + return; > + } > + > + req = list_first_entry(&ep->queue, struct isp1760_request, > + queue); > + req->req.actual += req->packet_size; > + > + need_zlp = req->req.actual == req->req.length && > + !(req->req.length % ep->maxpacket) && > + req->packet_size && req->req.zero; > + > + dev_dbg(udc->isp->dev, > + "TX IRQ: req %p actual/length %u/%u maxpacket %u packet size %u zero %u need zlp %u\n", > + req, req->req.actual, req->req.length, ep->maxpacket, > + req->packet_size, req->req.zero, need_zlp); > + > + /* > + * Complete the request if all data has been sent and we don't need to > + * transmit a zero length packet. > + */ > + if (req->req.actual == req->req.length && !need_zlp) { > + complete = req; > + list_del(&req->queue); > + > + if (ep->addr == 0) > + isp1760_udc_ctrl_send_status(ep, USB_DIR_IN); > + > + if (!list_empty(&ep->queue)) > + req = list_first_entry(&ep->queue, > + struct isp1760_request, queue); > + else > + req = NULL; > + } > + > + /* Transmit the next packet or start the next request, if any. */ > + /* > + * TODO: If the endpoint is stalled the next request shouldn't be > + * started, but what about the next packet ? > + */ > + if (req) > + isp1760_udc_transmit(ep, req); > + > + spin_unlock(&udc->lock); > + > + if (complete) > + isp1760_udc_request_complete(ep, complete, 0); > +} > + > +static int __isp1760_udc_set_halt(struct isp1760_ep *ep, bool halt) > +{ > + struct isp1760_udc *udc = ep->udc; > + > + dev_dbg(udc->isp->dev, "%s: %s halt on ep%02x\n", __func__, > + halt ? "set" : "clear", ep->addr); > + > + if (ep->desc && usb_endpoint_xfer_isoc(ep->desc)) { > + dev_dbg(udc->isp->dev, "%s: ep%02x is isochronous\n", __func__, > + ep->addr); > + return -EINVAL; > + } > + > + isp1760_udc_select_ep(ep); > + isp1760_udc_write(udc, DC_CTRLFUNC, halt ? DC_STALL : 0); > + > + if (ep->addr == 0) { > + /* When halting the control endpoint, stall both IN and OUT. */ > + __isp1760_udc_select_ep(ep, USB_DIR_IN); > + isp1760_udc_write(udc, DC_CTRLFUNC, halt ? DC_STALL : 0); > + } else if (!halt) { > + /* Reset the data PID by cycling the endpoint enable bit. */ > + u16 eptype = isp1760_udc_read(udc, DC_EPTYPE); > + > + isp1760_udc_write(udc, DC_EPTYPE, eptype & ~DC_EPENABLE); > + isp1760_udc_write(udc, DC_EPTYPE, eptype); > + > + /* > + * Disabling the endpoint emptied the transmit FIFO, fill it > + * again if a request is pending. > + * > + * TODO: Synchronize with the TX IRQ handler ? > + */ > + if ((ep->addr & USB_DIR_IN) && !list_empty(&ep->queue)) { > + struct isp1760_request *req; > + > + req = list_first_entry(&ep->queue, > + struct isp1760_request, queue); > + isp1760_udc_transmit(ep, req); > + } > + } > + > + ep->halted = halt; > + > + return 0; > +} > + > +/* ----------------------------------------------------------------------------- > + * Control Endpoint > + */ > + > +static int isp1760_udc_get_status(struct isp1760_udc *udc, > + const struct usb_ctrlrequest *req) > +{ > + struct isp1760_ep *ep; > + u16 status; > + > + if (req->wLength != cpu_to_le16(2) || req->wValue != cpu_to_le16(0)) > + return -EINVAL; > + > + switch (req->bRequestType) { > + case USB_DIR_IN | USB_RECIP_DEVICE: > + status = udc->devstatus; > + break; > + > + case USB_DIR_IN | USB_RECIP_INTERFACE: > + status = 0; > + break; > + > + case USB_DIR_IN | USB_RECIP_ENDPOINT: > + ep = isp1760_udc_find_ep(udc, le16_to_cpu(req->wIndex)); > + if (!ep) > + return -EINVAL; > + > + status = 0; > + if (ep->halted) > + status |= 1 << USB_ENDPOINT_HALT; > + break; > + > + default: > + return -EINVAL; > + } > + > + isp1760_udc_write(udc, DC_EPINDEX, DC_ENDPIDX(0) | DC_EPDIR); > + isp1760_udc_write(udc, DC_BUFLEN, 2); > + > + writew(cpu_to_le16(status), udc->regs + DC_DATAPORT); > + > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_DSEN); > + > + dev_dbg(udc->isp->dev, "%s: status 0x%04x\n", __func__, status); > + > + return 0; > +} > + > +static int isp1760_udc_set_address(struct isp1760_udc *udc, u16 addr) > +{ > + if (addr > 127) { > + dev_dbg(udc->isp->dev, "invalid device address %u\n", addr); > + return -EINVAL; > + } if you want to be pedantic, you could add a check here to stall SetAdress when gadget is already in CONFIGURED state. > + > + usb_gadget_set_state(&udc->gadget, addr ? USB_STATE_ADDRESS : > + USB_STATE_DEFAULT); > + > + isp1760_udc_write(udc, DC_ADDRESS, DC_DEVEN | addr); > + > + spin_lock(&udc->lock); > + isp1760_udc_ctrl_send_status(&udc->ep[0], USB_DIR_OUT); > + spin_unlock(&udc->lock); > + > + return 0; > +} > + > +static bool isp1760_ep0_setup_standard(struct isp1760_udc *udc, > + struct usb_ctrlrequest *req) > +{ > + bool stall; > + > + switch (req->bRequest) { > + case USB_REQ_GET_STATUS: > + return isp1760_udc_get_status(udc, req); > + > + case USB_REQ_CLEAR_FEATURE: > + switch (req->bRequestType) { > + case USB_DIR_OUT | USB_RECIP_DEVICE: { > + /* TODO: Handle remote wakeup feature */ > + return true; > + } > + > + case USB_DIR_OUT | USB_RECIP_ENDPOINT: { > + u16 index = le16_to_cpu(req->wIndex); > + struct isp1760_ep *ep; > + > + if (req->wLength != cpu_to_le16(0) || > + req->wValue != cpu_to_le16(USB_ENDPOINT_HALT)) > + return true; > + > + ep = isp1760_udc_find_ep(udc, index); > + if (!ep) > + return true; > + > + spin_lock(&udc->lock); > + > + /* > + * If the endpoint is wedged only the gadget can clear > + * the halt feature. Pretend success in that case, but > + * keep the endpoint halted. > + */ > + if (!ep->wedged) > + stall = __isp1760_udc_set_halt(ep, false); > + else > + stall = false; > + > + if (!stall) > + isp1760_udc_ctrl_send_status(&udc->ep[0], > + USB_DIR_OUT); > + > + spin_unlock(&udc->lock); > + return stall; > + } > + > + default: > + return true; > + } > + break; > + > + case USB_REQ_SET_FEATURE: > + switch (req->bRequestType) { > + case USB_DIR_OUT | USB_RECIP_DEVICE: { > + /* TODO: Handle remote wakeup and test mode features */ > + return true; > + } > + > + case USB_DIR_OUT | USB_RECIP_ENDPOINT: { > + u16 index = le16_to_cpu(req->wIndex); > + struct isp1760_ep *ep; > + > + if (req->wLength != cpu_to_le16(0) || > + req->wValue != cpu_to_le16(USB_ENDPOINT_HALT)) > + return true; > + > + ep = isp1760_udc_find_ep(udc, index); > + if (!ep) > + return true; > + > + /* > + * TODO The stall feature must be reset to 0 by a > + * SET_CONFIGURATION or SET_INTERFACE request. > + */ > + spin_lock(&udc->lock); > + > + stall = __isp1760_udc_set_halt(ep, true); > + if (!stall) > + isp1760_udc_ctrl_send_status(&udc->ep[0], > + USB_DIR_OUT); > + > + spin_unlock(&udc->lock); > + return stall; > + } > + > + default: > + return true; > + } > + break; > + > + case USB_REQ_SET_ADDRESS: > + if (req->bRequestType != (USB_DIR_OUT | USB_RECIP_DEVICE)) > + return true; > + > + return isp1760_udc_set_address(udc, le16_to_cpu(req->wValue)); > + > + case USB_REQ_SET_CONFIGURATION: > + if (req->bRequestType != (USB_DIR_OUT | USB_RECIP_DEVICE)) > + return true; > + > + if (udc->gadget.state != USB_STATE_ADDRESS && > + udc->gadget.state != USB_STATE_CONFIGURED) > + return true; > + > + stall = udc->driver->setup(&udc->gadget, req) < 0; > + if (stall) > + return true; > + > + usb_gadget_set_state(&udc->gadget, req->wValue ? > + USB_STATE_CONFIGURED : USB_STATE_ADDRESS); > + return false; > + > + default: > + return udc->driver->setup(&udc->gadget, req) < 0; > + } > +} > + > +static void isp1760_ep0_setup(struct isp1760_udc *udc) > +{ > + union { > + struct usb_ctrlrequest r; > + u32 data[2]; > + } req; > + unsigned int count; > + bool stall = false; > + > + spin_lock(&udc->lock); > + > + isp1760_udc_write(udc, DC_EPINDEX, DC_EP0SETUP); > + > + count = isp1760_udc_read(udc, DC_BUFLEN) & DC_DATACOUNT_MASK; > + if (count != sizeof(req)) { > + spin_unlock(&udc->lock); > + > + dev_err(udc->isp->dev, "invalid length %u for setup packet\n", > + count); > + > + isp1760_udc_ctrl_send_stall(&udc->ep[0]); > + return; > + } > + > + req.data[0] = isp1760_udc_read(udc, DC_DATAPORT); > + req.data[1] = isp1760_udc_read(udc, DC_DATAPORT); > + > + if (udc->ep0_state != ISP1760_CTRL_SETUP) { > + spin_unlock(&udc->lock); > + dev_dbg(udc->isp->dev, "unexpected SETUP packet\n"); > + return; > + } > + > + /* Move to the data stage. */ > + if (req.r.bRequestType & USB_DIR_IN) > + udc->ep0_state = ISP1760_CTRL_DATA_IN; > + else > + udc->ep0_state = ISP1760_CTRL_DATA_OUT; data phase is optional. This should be something like: if (req.r.wLength > 0) { if (req.r.bRequestType & USB_DIR_IN) udc->ep0_state = ISP1760_CTRL_DATA_IN; else udc->ep0_state = ISP1760_CTRL_DATA_OUT; } else { udc->ep0_state = ISP1760_CTRL_STATUS; } > + spin_unlock(&udc->lock); > + > + dev_dbg(udc->isp->dev, > + "%s: bRequestType 0x%02x bRequest 0x%02x wValue 0x%04x wIndex 0x%04x wLength 0x%04x\n", > + __func__, req.r.bRequestType, req.r.bRequest, > + le16_to_cpu(req.r.wValue), le16_to_cpu(req.r.wIndex), > + le16_to_cpu(req.r.wLength)); > + > + if ((req.r.bRequestType & USB_TYPE_MASK) == USB_TYPE_STANDARD) > + stall = isp1760_ep0_setup_standard(udc, &req.r); > + else > + stall = udc->driver->setup(&udc->gadget, &req.r) < 0; > + > + if (stall) > + isp1760_udc_ctrl_send_stall(&udc->ep[0]); > +} > + > +/* ----------------------------------------------------------------------------- > + * Gadget Endpoint Operations > + */ > + > +static int isp1760_ep_enable(struct usb_ep *ep, > + const struct usb_endpoint_descriptor *desc) > +{ > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + struct isp1760_udc *udc = uep->udc; > + unsigned long flags; > + unsigned int type; > + > + dev_dbg(uep->udc->isp->dev, "%s\n", __func__); > + > + /* > + * Validate the descriptor. The control endpoint can't be enabled > + * manually. > + */ > + if (desc->bDescriptorType != USB_DT_ENDPOINT || > + desc->bEndpointAddress == 0 || > + desc->bEndpointAddress != uep->addr || > + le16_to_cpu(desc->wMaxPacketSize) > ep->maxpacket) { > + dev_dbg(udc->isp->dev, > + "%s: invalid descriptor type %u addr %02x ep addr %02x max packet size %u/%u\n", > + __func__, desc->bDescriptorType, > + desc->bEndpointAddress, uep->addr, > + le16_to_cpu(desc->wMaxPacketSize), ep->maxpacket); > + return -EINVAL; > + } > + > + switch (usb_endpoint_type(desc)) { > + case USB_ENDPOINT_XFER_ISOC: > + type = DC_ENDPTYP_ISOC; > + break; > + case USB_ENDPOINT_XFER_BULK: > + type = DC_ENDPTYP_BULK; > + break; > + case USB_ENDPOINT_XFER_INT: > + type = DC_ENDPTYP_INTERRUPT; > + break; > + case USB_ENDPOINT_XFER_CONTROL: > + default: > + dev_dbg(udc->isp->dev, "%s: control endpoints unsupported\n", > + __func__); > + return -EINVAL; > + } > + > + spin_lock_irqsave(&udc->lock, flags); > + > + uep->desc = desc; > + uep->maxpacket = le16_to_cpu(desc->wMaxPacketSize); > + uep->rx_pending = false; > + uep->halted = false; > + uep->wedged = false; > + > + isp1760_udc_select_ep(uep); > + isp1760_udc_write(udc, DC_EPMAXPKTSZ, uep->maxpacket); > + isp1760_udc_write(udc, DC_BUFLEN, uep->maxpacket); > + isp1760_udc_write(udc, DC_EPTYPE, DC_EPENABLE | type); > + > + spin_unlock_irqrestore(&udc->lock, flags); > + > + return 0; > +} > + > +static int isp1760_ep_disable(struct usb_ep *ep) > +{ > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + struct isp1760_udc *udc = uep->udc; > + struct isp1760_request *req, *nreq; > + LIST_HEAD(req_list); > + unsigned long flags; > + > + dev_dbg(udc->isp->dev, "%s\n", __func__); > + > + spin_lock_irqsave(&udc->lock, flags); > + > + if (!uep->desc) { > + dev_dbg(udc->isp->dev, "%s: endpoint not enabled\n", __func__); > + spin_unlock_irqrestore(&udc->lock, flags); > + return -EINVAL; > + } > + > + uep->desc = NULL; > + uep->maxpacket = 0; > + > + isp1760_udc_select_ep(uep); > + isp1760_udc_write(udc, DC_EPTYPE, 0); > + > + /* TODO Synchronize with the IRQ handler */ > + > + list_splice_init(&uep->queue, &req_list); > + > + spin_unlock_irqrestore(&udc->lock, flags); > + > + list_for_each_entry_safe(req, nreq, &req_list, queue) { > + list_del(&req->queue); > + isp1760_udc_request_complete(uep, req, -ESHUTDOWN); > + } > + > + return 0; > +} > + > +static struct usb_request *isp1760_ep_alloc_request(struct usb_ep *ep, > + gfp_t gfp_flags) > +{ > + struct isp1760_request *req; > + > + req = kzalloc(sizeof(*req), gfp_flags); > + > + return &req->req; > +} > + > +static void isp1760_ep_free_request(struct usb_ep *ep, struct usb_request *_req) > +{ > + struct isp1760_request *req = req_to_udc_req(_req); > + > + kfree(req); > +} > + > +static int isp1760_ep_queue(struct usb_ep *ep, struct usb_request *_req, > + gfp_t gfp_flags) > +{ > + struct isp1760_request *req = req_to_udc_req(_req); > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + struct isp1760_udc *udc = uep->udc; > + bool complete = false; > + unsigned long flags; > + int ret = 0; > + > + _req->status = -EINPROGRESS; > + _req->actual = 0; > + > + spin_lock_irqsave(&udc->lock, flags); > + > + dev_dbg(udc->isp->dev, > + "%s: req %p (%u bytes%s) ep %p(0x%02x)\n", __func__, _req, > + _req->length, _req->zero ? " (zlp)" : "", uep, uep->addr); > + > + req->ep = uep; > + > + if (uep->addr == 0) { > + switch (udc->ep0_state) { > + case ISP1760_CTRL_DATA_IN: > + dev_dbg(udc->isp->dev, "%s: transmitting req %p\n", > + __func__, req); > + > + list_add_tail(&req->queue, &uep->queue); > + isp1760_udc_transmit(uep, req); > + break; > + > + case ISP1760_CTRL_DATA_OUT: > + if (_req->length) { > + list_add_tail(&req->queue, &uep->queue); > + __isp1760_udc_select_ep(uep, USB_DIR_OUT); > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_DSEN); > + } else { > + complete = true; > + } > + break; > + > + default: > + dev_dbg(udc->isp->dev, "%s: invalid ep0 state\n", > + __func__); > + ret = -EINVAL; > + break; > + } > + } else if (uep->desc) { > + bool empty = list_empty(&uep->queue); > + > + list_add_tail(&req->queue, &uep->queue); > + if ((uep->addr & USB_DIR_IN) && !uep->halted && empty) > + isp1760_udc_transmit(uep, req); > + else if (!(uep->addr & USB_DIR_IN) && uep->rx_pending) > + complete = isp1760_udc_receive(uep, req); > + } else { > + dev_dbg(udc->isp->dev, > + "%s: can't queue request to disabled ep%02x\n", > + __func__, uep->addr); > + ret = -ESHUTDOWN; > + } > + > + if (ret < 0) > + req->ep = NULL; > + > + spin_unlock_irqrestore(&udc->lock, flags); > + > + if (complete) > + isp1760_udc_request_complete(uep, req, 0); > + > + return ret; > +} > + > +static int isp1760_ep_dequeue(struct usb_ep *ep, struct usb_request *_req) > +{ > + struct isp1760_request *req = req_to_udc_req(_req); > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + struct isp1760_udc *udc = uep->udc; > + unsigned long flags; > + > + dev_dbg(uep->udc->isp->dev, "%s(ep%02x)\n", __func__, uep->addr); > + > + spin_lock_irqsave(&udc->lock, flags); > + > + if (req->ep != uep) > + req = NULL; > + else > + list_del(&req->queue); > + > + spin_unlock_irqrestore(&udc->lock, flags); > + > + if (!req) > + return -EINVAL; > + > + isp1760_udc_request_complete(uep, req, -ECONNRESET); > + return 0; > +} > + > +static int __isp1760_ep_set_halt(struct isp1760_ep *uep, bool stall, bool wedge) > +{ > + struct isp1760_udc *udc = uep->udc; > + int ret; > + > + if (!uep->addr) { > + /* > + * Halting the control endpoint is only valid as a delayed error > + * response to a SETUP packet. Make sure EP0 is in the right > + * stage and that the gadget isn't trying to clear the halt > + * condition. > + */ > + if (WARN_ON((udc->ep0_state != ISP1760_CTRL_DATA_IN && > + udc->ep0_state != ISP1760_CTRL_DATA_OUT) || no, this isn't really correct. You can very well stall a CTRL request. Imagine you get a ctrl request which isn't valid at all, you can stall right there. A data phase can also be stalled if gadget expected for e.g. DATA IN but host tried to move DATA OUT. > + !stall || wedge)) { > + return -EINVAL; > + } > + } > + > + if (uep->addr && !uep->desc) { > + dev_dbg(udc->isp->dev, "%s: ep%02x is disabled\n", __func__, > + uep->addr); > + return -EINVAL; > + } > + > + if (uep->addr & USB_DIR_IN) { > + /* Refuse to halt IN endpoints with active transfers. */ > + if (!list_empty(&uep->queue)) { > + dev_dbg(udc->isp->dev, > + "%s: ep%02x has request pending\n", __func__, > + uep->addr); > + return -EAGAIN; > + } > + } > + > + ret = __isp1760_udc_set_halt(uep, stall); > + if (ret < 0) > + return ret; > + > + if (!uep->addr) { > + /* > + * Stalling EP0 completes the control transaction, move back to > + * the SETUP state. > + */ > + udc->ep0_state = ISP1760_CTRL_SETUP; > + return 0; > + } > + > + if (wedge) > + uep->wedged = true; > + else if (!stall) > + uep->wedged = false; > + > + return 0; > +} > + > +static int isp1760_ep_set_halt(struct usb_ep *ep, int value) > +{ > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + unsigned long flags; > + int ret; > + > + dev_dbg(uep->udc->isp->dev, "%s: %s halt on ep%02x\n", __func__, > + value ? "set" : "clear", uep->addr); > + > + spin_lock_irqsave(&uep->udc->lock, flags); > + ret = __isp1760_ep_set_halt(uep, value, false); > + spin_unlock_irqrestore(&uep->udc->lock, flags); > + > + return ret; > +} > + > +static int isp1760_ep_set_wedge(struct usb_ep *ep) > +{ > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + unsigned long flags; > + int ret; > + > + dev_dbg(uep->udc->isp->dev, "%s: set wedge on ep%02x)\n", __func__, > + uep->addr); > + > + spin_lock_irqsave(&uep->udc->lock, flags); > + ret = __isp1760_ep_set_halt(uep, true, true); > + spin_unlock_irqrestore(&uep->udc->lock, flags); > + > + return ret; > +} > + > +static void isp1760_ep_fifo_flush(struct usb_ep *ep) > +{ > + struct isp1760_ep *uep = ep_to_udc_ep(ep); > + struct isp1760_udc *udc = uep->udc; > + unsigned long flags; > + > + spin_lock_irqsave(&udc->lock, flags); > + > + isp1760_udc_select_ep(uep); > + > + /* > + * Set the CLBUF bit twice to flush both buffers in case double > + * buffering is enabled. > + */ > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_CLBUF); > + isp1760_udc_write(udc, DC_CTRLFUNC, DC_CLBUF); > + > + spin_unlock_irqrestore(&udc->lock, flags); > +} > + > +static const struct usb_ep_ops isp1760_ep_ops = { > + .enable = isp1760_ep_enable, > + .disable = isp1760_ep_disable, > + .alloc_request = isp1760_ep_alloc_request, > + .free_request = isp1760_ep_free_request, > + .queue = isp1760_ep_queue, > + .dequeue = isp1760_ep_dequeue, > + .set_halt = isp1760_ep_set_halt, > + .set_wedge = isp1760_ep_set_wedge, > + .fifo_flush = isp1760_ep_fifo_flush, > +}; > + > +/* ----------------------------------------------------------------------------- > + * Device States > + */ > + > +static void isp1760_udc_disconnect(struct isp1760_udc *udc) > +{ > + if (udc->gadget.state < USB_STATE_POWERED) > + return; > + > + dev_dbg(udc->isp->dev, "Device disconnected in state %u\n", > + udc->gadget.state); > + > + udc->gadget.speed = USB_SPEED_UNKNOWN; > + usb_gadget_set_state(&udc->gadget, USB_STATE_ATTACHED); > + > + if (udc->driver->disconnect) > + udc->driver->disconnect(&udc->gadget); alright, so this can be called from your reset IRQ handler. You need a little more inteligence on this branch because if you're coming from reset IRQ you only want to call gadget driver's disconnect in case speed != UNKNOWN. That has become easier, however, because of a recent patch adding ->reset() method which should be called from reset unconditionally and ->disconnect() will only be called from disconnect IRQ. > + > + /* TODO Reset all endpoints ? */ > +} > + > +static void isp1760_udc_reset(struct isp1760_udc *udc) > +{ > + unsigned long flags; > + > + /* > + * The device controller currently shares its interrupt with the host > + * controller, the DC_IRQ polarity and signaling mode are ignored. Set > + * the to active-low level-triggered. > + * > + * Configure the control, in and out pipes to generate interrupts on > + * ACK tokens only (and NYET for the out pipe). The default > + * configuration also generates an interrupt on the first NACK token. > + */ > + isp1760_udc_write(udc, DC_INTCONF, DC_CDBGMOD_ACK | DC_DDBGMODIN_ACK | > + DC_DDBGMODOUT_ACK_NYET); > + > + isp1760_udc_write(udc, DC_INTENABLE, DC_IEPRXTX(7) | DC_IEPRXTX(6) | > + DC_IEPRXTX(5) | DC_IEPRXTX(4) | DC_IEPRXTX(3) | > + DC_IEPRXTX(2) | DC_IEPRXTX(1) | DC_IEPRXTX(0) | > + DC_IEP0SETUP | DC_IEVBUS | DC_IERESM | DC_IESUSP | > + DC_IEHS_STA | DC_IEBRST); > + > + isp1760_udc_disconnect(udc); > + > + if (udc->connected) > + isp1760_set_pullup(udc->isp, true); > + > + spin_lock_irqsave(&udc->lock, flags); > + > + isp1760_udc_write(udc, DC_ADDRESS, DC_DEVEN); > + > + usb_gadget_set_state(&udc->gadget, USB_STATE_DEFAULT); > + > + udc->ep0_state = ISP1760_CTRL_SETUP; > + udc->gadget.speed = USB_SPEED_FULL; > + > + spin_unlock_irqrestore(&udc->lock, flags); > +} > + > +static void isp1760_udc_suspend(struct isp1760_udc *udc) > +{ > + if (udc->gadget.state < USB_STATE_DEFAULT) > + return; > + > + if (udc->driver->suspend) > + udc->driver->suspend(&udc->gadget); > +} > + > +static void isp1760_udc_resume(struct isp1760_udc *udc) > +{ > + if (udc->gadget.state < USB_STATE_DEFAULT) > + return; > + > + if (udc->driver->resume) > + udc->driver->resume(&udc->gadget); > +} > + > +/* ----------------------------------------------------------------------------- > + * Gadget Operations > + */ > + > +static int isp1760_udc_get_frame(struct usb_gadget *gadget) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + return isp1760_udc_read(udc, DC_FRAMENUM) & ((1 << 11) - 1); > +} > + > +static int isp1760_udc_wakeup(struct usb_gadget *gadget) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + dev_dbg(udc->isp->dev, "%s\n", __func__); > + return -ENOTSUPP; > +} > + > +static int isp1760_udc_set_selfpowered(struct usb_gadget *gadget, > + int is_selfpowered) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + if (is_selfpowered) > + udc->devstatus |= 1 << USB_DEVICE_SELF_POWERED; > + else > + udc->devstatus &= ~(1 << USB_DEVICE_SELF_POWERED); > + > + return 0; > +} > + > +static int isp1760_udc_pullup(struct usb_gadget *gadget, int is_on) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + isp1760_set_pullup(udc->isp, is_on); > + udc->connected = is_on; > + > + return 0; > +} > + > +static int isp1760_udc_start(struct usb_gadget *gadget, > + struct usb_gadget_driver *driver) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + /* The hardware doesn't support low speed. */ > + if (driver->max_speed < USB_SPEED_FULL) { > + dev_err(udc->isp->dev, "Invalid gadget driver\n"); > + return -EINVAL; > + } > + > + spin_lock(&udc->lock); > + > + if (udc->driver) { > + dev_err(udc->isp->dev, "UDC already has a gadget driver\n"); > + spin_unlock(&udc->lock); > + return -EBUSY; > + } > + > + udc->driver = driver; > + > + spin_unlock(&udc->lock); > + > + dev_dbg(udc->isp->dev, "starting UDC with driver %s\n", > + driver->function); > + > + udc->devstatus = 0; > + udc->connected = true; > + > + usb_gadget_set_state(&udc->gadget, USB_STATE_ATTACHED); > + > + /* DMA isn't supported yet, don't enable the DMA clock. */ > + isp1760_udc_write(udc, DC_MODE, DC_GLINTENA); > + > + isp1760_udc_reset(udc); > + > + dev_dbg(udc->isp->dev, "UDC started with driver %s\n", > + driver->function); > + > + return 0; > +} > + > +static int isp1760_udc_stop(struct usb_gadget *gadget, > + struct usb_gadget_driver *driver) > +{ > + struct isp1760_udc *udc = gadget_to_udc(gadget); > + > + dev_dbg(udc->isp->dev, "%s\n", __func__); > + > + isp1760_udc_write(udc, DC_MODE, 0); > + > + spin_lock(&udc->lock); > + udc->driver = NULL; > + spin_unlock(&udc->lock); > + > + return 0; > +} > + > +static struct usb_gadget_ops isp1760_udc_ops = { > + .get_frame = isp1760_udc_get_frame, > + .wakeup = isp1760_udc_wakeup, > + .set_selfpowered = isp1760_udc_set_selfpowered, > + .pullup = isp1760_udc_pullup, > + .udc_start = isp1760_udc_start, > + .udc_stop = isp1760_udc_stop, > +}; > + > +/* ----------------------------------------------------------------------------- > + * Interrupt Handling > + */ > + > +static irqreturn_t isp1760_udc_irq(int irq, void *dev) > +{ > + struct isp1760_udc *udc = dev; > + unsigned int i; > + u32 status; > + > + status = isp1760_udc_read(udc, DC_INTERRUPT) > + & isp1760_udc_read(udc, DC_INTENABLE); > + isp1760_udc_write(udc, DC_INTERRUPT, status); > + > + if (status & DC_IEBRST) { > + dev_dbg(udc->isp->dev, "%s(BRST)\n", __func__); > + > + isp1760_udc_reset(udc); > + } > + > + for (i = 0; i <= 7; ++i) { > + struct isp1760_ep *ep = &udc->ep[i*2]; > + > + if (status & DC_IEPTX(i)) { > + dev_dbg(udc->isp->dev, "%s(EPTX%u)\n", __func__, i); > + isp1760_ep_tx_complete(ep); > + } > + > + if (status & DC_IEPRX(i)) { > + dev_dbg(udc->isp->dev, "%s(EPRX%u)\n", __func__, i); > + isp1760_ep_rx_ready(i ? ep - 1 : ep); > + } > + } > + > + if (status & DC_IEP0SETUP) { > + dev_dbg(udc->isp->dev, "%s(EP0SETUP)\n", __func__); > + > + isp1760_ep0_setup(udc); > + } > + > + if (status & DC_IEVBUS) { > + dev_dbg(udc->isp->dev, "%s(VBUS)\n", __func__); > + /* The VBUS interrupt is only triggered when VBUS appearsr */ > + usb_gadget_set_state(&udc->gadget, USB_STATE_POWERED); > + } > + > + if (status & DC_IERESM) { > + dev_dbg(udc->isp->dev, "%s(RESM)\n", __func__); > + isp1760_udc_resume(udc); > + } > + > + if (status & DC_IESUSP) { > + dev_dbg(udc->isp->dev, "%s(SUSP)\n", __func__); > + > + if (!(isp1760_udc_read(udc, DC_MODE) & DC_VBUSSTAT)) > + isp1760_udc_disconnect(udc); > + else > + isp1760_udc_suspend(udc); > + } > + > + if (status & DC_IEHS_STA) { > + dev_dbg(udc->isp->dev, "%s(HS_STA)\n", __func__); > + udc->gadget.speed = USB_SPEED_HIGH; > + } > + > + return status ? IRQ_HANDLED : IRQ_NONE; > +} > + > +/* ----------------------------------------------------------------------------- > + * Registration > + */ > + > +static void isp1760_udc_init_eps(struct isp1760_udc *udc) > +{ > + unsigned int i; > + > + INIT_LIST_HEAD(&udc->gadget.ep_list); > + > + for (i = 0; i < ARRAY_SIZE(udc->ep); ++i) { > + struct isp1760_ep *ep = &udc->ep[i]; > + unsigned int ep_num = (i + 1) / 2; > + bool is_in = !(i & 1); > + > + ep->udc = udc; > + > + INIT_LIST_HEAD(&ep->queue); > + > + ep->addr = (ep_num && is_in ? USB_DIR_IN : USB_DIR_OUT) > + | ep_num; > + ep->desc = NULL; > + > + sprintf(ep->name, "ep%u%s", ep_num, > + ep_num ? (is_in ? "in" : "out") : ""); > + > + ep->ep.ops = &isp1760_ep_ops; > + ep->ep.name = ep->name; > + > + /* > + * Hardcode the maximum packet sizes for now, to 64 bytes for > + * the control endpoint and 512 bytes for all other endpoints. > + * This fits in the 8kB FIFO without double-buffering. > + */ > + if (ep_num == 0) { > + ep->ep.maxpacket = 64; > + ep->maxpacket = 64; > + udc->gadget.ep0 = &ep->ep; > + } else { > + ep->ep.maxpacket = 512; > + ep->maxpacket = 0; > + list_add_tail(&ep->ep.ep_list, &udc->gadget.ep_list); > + } > + } > +} > + > +static int isp1760_udc_init(struct isp1760_udc *udc) > +{ > + u16 scratch; > + u32 chipid; > + > + /* > + * Check that the controller is present by writing to the scratch > + * register, modifying the bus pattern by reading from the chip ID > + * register, and reading the scratch register value back. The chip ID > + * and scratch register contents must match the expected values. > + */ > + isp1760_udc_write(udc, DC_SCRATCH, 0xbabe); heh -- balbi
Attachment:
signature.asc
Description: Digital signature