| /* |
| * Copyright (c) 2014, The Linux Foundation. All rights reserved. |
| * Copyright (c) 2001-2004 by David Brownell |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 and |
| * only version 2 as published by the Free Software Foundation. |
| * |
| * This program 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 General Public License for more details. |
| * |
| */ |
| |
| /* |
| * Root HUB management and Asynchronous scheduling traversal |
| * Based on ehci-hub.c and ehci-q.c |
| */ |
| |
| #define pr_fmt(fmt) "%s: " fmt, __func__ |
| |
| #include <linux/kernel.h> |
| #include <linux/module.h> |
| #include <linux/err.h> |
| #include <linux/ktime.h> |
| #include <linux/uaccess.h> |
| #include <linux/debugfs.h> |
| #include <linux/pm_runtime.h> |
| #include <linux/regulator/consumer.h> |
| #include <linux/gpio.h> |
| #include <linux/of_gpio.h> |
| #include <linux/spinlock.h> |
| #include <linux/firmware.h> |
| #include <linux/spi/spi.h> |
| #include <linux/usb.h> |
| #include <linux/usb/hcd.h> |
| #include <linux/usb/ch9.h> |
| #include <linux/usb/ch11.h> |
| |
| #include <asm/unaligned.h> |
| #include <mach/gpiomux.h> |
| |
| #define CREATE_TRACE_POINTS |
| #include <trace/events/ice40.h> |
| |
| #define FADDR_REG 0x00 /* R/W: Device address */ |
| #define HCMD_REG 0x01 /* R/W: Host transfer command */ |
| #define XFRST_REG 0x02 /* R: Transfer status */ |
| #define IRQ_REG 0x03 /* R/C: IRQ status */ |
| #define IEN_REG 0x04 /* R/W: IRQ enable */ |
| #define CTRL0_REG 0x05 /* R/W: Host control command */ |
| #define CTRL1_REG 0x06 /* R/W: Host control command */ |
| #define WBUF0_REG 0x10 /* W: Tx fifo 0 */ |
| #define WBUF1_REG 0x11 /* W: Tx fifo 1 */ |
| #define SUBUF_REG 0x12 /* W: SETUP fifo */ |
| #define WBLEN_REG 0x13 /* W: Tx fifo size */ |
| #define RBUF0_REG 0x18 /* R: Rx fifo 0 */ |
| #define RBUF1_REG 0x19 /* R: Rx fifo 1 */ |
| #define RBLEN_REG 0x1B /* R: Rx fifo size */ |
| |
| #define WRITE_CMD(addr) ((addr << 3) | 1) |
| #define READ_CMD(addr) ((addr << 3) | 0) |
| |
| /* Host controller command register definitions */ |
| #define HCMD_EP(ep) (ep & 0xF) |
| #define HCMD_BSEL(sel) (sel << 4) |
| #define HCMD_TOGV(toggle) (toggle << 5) |
| #define HCMD_PT(token) (token << 6) |
| |
| /* Transfer status register definitions */ |
| #define XFR_MASK(xfr) (xfr & 0xF) |
| #define XFR_SUCCESS 0x0 |
| #define XFR_BUSY 0x1 |
| #define XFR_PKTERR 0x2 |
| #define XFR_PIDERR 0x3 |
| #define XFR_NAK 0x4 |
| #define XFR_STALL 0x5 |
| #define XFR_WRONGPID 0x6 |
| #define XFR_CRCERR 0x7 |
| #define XFR_TOGERR 0x8 |
| #define XFR_BADLEN 0x9 |
| #define XFR_TIMEOUT 0xA |
| |
| #define LINE_STATE(xfr) ((xfr & 0x30) >> 4) /* D+, D- */ |
| #define DPST BIT(5) |
| #define DMST BIT(4) |
| #define PLLOK BIT(6) |
| #define R64B BIT(7) |
| |
| /* Interrupt enable/status register definitions */ |
| #define RESET_IRQ BIT(0) |
| #define RESUME_IRQ BIT(1) |
| #define SUSP_IRQ BIT(3) |
| #define DISCONNECT_IRQ BIT(4) |
| #define CONNECT_IRQ BIT(5) |
| #define FRAME_IRQ BIT(6) |
| #define XFR_IRQ BIT(7) |
| |
| /* Control 0 register definitions */ |
| #define RESET_CTRL BIT(0) |
| #define FRAME_RESET_CTRL BIT(1) |
| #define DET_BUS_CTRL BIT(2) |
| #define RESUME_CTRL BIT(3) |
| #define SOFEN_CTRL BIT(4) |
| #define DM_PD_CTRL BIT(6) |
| #define DP_PD_CTRL BIT(7) |
| #define HRST_CTRL BIT(5) |
| |
| /* Control 1 register definitions */ |
| #define INT_EN_CTRL BIT(0) |
| |
| enum ice40_xfr_type { |
| FIRMWARE_XFR, |
| REG_WRITE_XFR, |
| REG_READ_XFR, |
| SETUP_XFR, |
| DATA_IN_XFR, |
| DATA_OUT_XFR, |
| }; |
| |
| enum ice40_ep_phase { |
| SETUP_PHASE = 1, |
| DATA_PHASE, |
| STATUS_PHASE, |
| }; |
| |
| struct ice40_ep { |
| u8 xcat_err; |
| bool unlinking; |
| bool halted; |
| struct usb_host_endpoint *ep; |
| struct list_head ep_list; |
| }; |
| |
| struct ice40_hcd { |
| spinlock_t lock; |
| |
| struct mutex wlock; |
| struct mutex rlock; |
| |
| u8 devnum; |
| u32 port_flags; |
| u8 ctrl0; |
| u8 wblen0; |
| |
| enum ice40_ep_phase ep0_state; |
| struct usb_hcd *hcd; |
| |
| struct list_head async_list; |
| struct workqueue_struct *wq; |
| struct work_struct async_work; |
| |
| int reset_gpio; |
| int slave_select_gpio; |
| int config_done_gpio; |
| int vcc_en_gpio; |
| int clk_en_gpio; |
| |
| struct regulator *core_vcc; |
| struct regulator *spi_vcc; |
| struct regulator *gpio_vcc; |
| bool powered; |
| |
| struct dentry *dbg_root; |
| bool pcd_pending; |
| |
| /* SPI stuff later */ |
| struct spi_device *spi; |
| |
| struct spi_message *fmsg; |
| struct spi_transfer *fmsg_xfr; /* size 1 */ |
| |
| struct spi_message *wmsg; |
| struct spi_transfer *wmsg_xfr; /* size 1 */ |
| u8 *w_tx_buf; |
| u8 *w_rx_buf; |
| |
| struct spi_message *rmsg; |
| struct spi_transfer *rmsg_xfr; /* size 1 */ |
| u8 *r_tx_buf; |
| u8 *r_rx_buf; |
| |
| struct spi_message *setup_msg; |
| struct spi_transfer *setup_xfr; /* size 2 */ |
| u8 *setup_buf; /* size 1 for SUBUF */ |
| |
| struct spi_message *in_msg; |
| struct spi_transfer *in_xfr; /* size 2 */ |
| u8 *in_buf; /* size 2 for reading from RBUF0 */ |
| |
| struct spi_message *out_msg; |
| struct spi_transfer *out_xfr; /* size 2 */ |
| u8 *out_buf; /* size 1 for writing WBUF0 */ |
| }; |
| |
| static char fw_name[16] = "ice40.bin"; |
| module_param_string(fw, fw_name, sizeof(fw_name), S_IRUGO | S_IWUSR); |
| MODULE_PARM_DESC(fw, "firmware blob file name"); |
| |
| static bool debugger; |
| module_param(debugger, bool, S_IRUGO | S_IWUSR); |
| MODULE_PARM_DESC(debugger, "true to use the debug port"); |
| |
| static inline struct ice40_hcd *hcd_to_ihcd(struct usb_hcd *hcd) |
| { |
| return *((struct ice40_hcd **) hcd->hcd_priv); |
| } |
| |
| static void ice40_spi_reg_write(struct ice40_hcd *ihcd, u8 val, u8 addr) |
| { |
| int ret; |
| |
| /* |
| * Register Write Pattern: |
| * TX: 1st byte is CMD (register + write), 2nd byte is value |
| * RX: Ignore |
| * |
| * The Mutex is to protect concurrent register writes as |
| * we have only 1 SPI message struct. |
| */ |
| |
| mutex_lock(&ihcd->wlock); |
| |
| ihcd->w_tx_buf[0] = WRITE_CMD(addr); |
| ihcd->w_tx_buf[1] = val; |
| ret = spi_sync(ihcd->spi, ihcd->wmsg); |
| if (ret < 0) /* should not happen */ |
| pr_err("failed. val = %d addr = %d\n", val, addr); |
| |
| trace_ice40_reg_write(addr, val, ihcd->w_tx_buf[0], |
| ihcd->w_tx_buf[1], ret); |
| |
| mutex_unlock(&ihcd->wlock); |
| } |
| |
| static int ice40_spi_reg_read(struct ice40_hcd *ihcd, u8 addr) |
| { |
| int ret; |
| |
| /* |
| * Register Read Pattern: |
| * TX: 1st byte is CMD (register + read) |
| * RX: 1st, 2nd byte Ignore, 3rd byte value. |
| * |
| * The Mutex is to protect concurrent register reads as |
| * we have only 1 SPI message struct. |
| */ |
| |
| mutex_lock(&ihcd->rlock); |
| |
| ihcd->r_tx_buf[0] = READ_CMD(addr); |
| ret = spi_sync(ihcd->spi, ihcd->rmsg); |
| if (ret < 0) |
| pr_err("failed. addr = %d\n", addr); |
| else |
| ret = ihcd->r_rx_buf[2]; |
| |
| trace_ice40_reg_read(addr, ihcd->r_tx_buf[0], ret); |
| |
| mutex_unlock(&ihcd->rlock); |
| |
| return ret; |
| } |
| |
| static int ice40_poll_xfer(struct ice40_hcd *ihcd, int usecs) |
| { |
| ktime_t start = ktime_get(); |
| u8 val, retry = 0; |
| u8 ret = ~0; /* time out */ |
| |
| again: |
| |
| /* |
| * The SPI transaction may take tens of usec. Use ktime |
| * based checks rather than loop count. |
| */ |
| do { |
| val = ice40_spi_reg_read(ihcd, XFRST_REG); |
| |
| if (XFR_MASK(val) != XFR_BUSY) |
| return val; |
| |
| } while (ktime_us_delta(ktime_get(), start) < usecs); |
| |
| /* |
| * The SPI transaction involves a context switch. For any |
| * reason, if we are scheduled out more than usecs after |
| * the 1st read, this extra read will help. |
| */ |
| if (!retry) { |
| retry = 1; |
| goto again; |
| } |
| |
| return ret; |
| } |
| |
| static int |
| ice40_handshake(struct ice40_hcd *ihcd, u8 reg, u8 mask, u8 done, int usecs) |
| { |
| ktime_t start = ktime_get(); |
| u8 val, retry = 0; |
| |
| again: |
| do { |
| val = ice40_spi_reg_read(ihcd, reg); |
| val &= mask; |
| |
| if (val == done) |
| return 0; |
| |
| } while (ktime_us_delta(ktime_get(), start) < usecs); |
| |
| if (!retry) { |
| retry = 1; |
| goto again; |
| } |
| |
| return -ETIMEDOUT; |
| } |
| |
| |
| static const char hcd_name[] = "ice40-hcd"; |
| |
| static int ice40_reset(struct usb_hcd *hcd) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| u8 ctrl, status; |
| int ret = 0; |
| |
| /* |
| * Program the defualt address 0. The device address is |
| * re-programmed after SET_ADDRESS in URB handling path. |
| */ |
| ihcd->devnum = 0; |
| ice40_spi_reg_write(ihcd, 0, FADDR_REG); |
| |
| ihcd->wblen0 = ~0; |
| /* |
| * Read the line state. This driver is loaded after the |
| * UICC card insertion. So the line state should indicate |
| * that a Full-speed device is connected. Return error |
| * if there is no device connected. |
| * |
| * There can be no device connected during debug. A debugfs |
| * file is provided to sample the bus line and update the |
| * port flags accordingly. |
| */ |
| |
| if (debugger) |
| goto out; |
| |
| ctrl = ice40_spi_reg_read(ihcd, CTRL0_REG); |
| ice40_spi_reg_write(ihcd, ctrl | DET_BUS_CTRL, CTRL0_REG); |
| |
| ret = ice40_handshake(ihcd, CTRL0_REG, DET_BUS_CTRL, 0, 5000); |
| if (ret) { |
| pr_err("bus detection failed\n"); |
| goto out; |
| } |
| |
| status = ice40_spi_reg_read(ihcd, XFRST_REG); |
| pr_debug("line state (D+, D-) is %d\n", LINE_STATE(status)); |
| |
| if (status & DPST) { |
| pr_debug("Full speed device connected\n"); |
| ihcd->port_flags |= USB_PORT_STAT_CONNECTION; |
| } else { |
| pr_err("No device connected\n"); |
| ret = -ENODEV; |
| } |
| out: |
| return ret; |
| } |
| |
| static int ice40_run(struct usb_hcd *hcd) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| |
| /* |
| * HCD_FLAG_POLL_RH flag is not set by us. Core will not poll |
| * for the port status periodically. This uses_new_polling |
| * flag tells core that this hcd will call usb_hcd_poll_rh_status |
| * upon port change. |
| */ |
| hcd->uses_new_polling = 1; |
| |
| /* |
| * Cache the ctrl0 register to avoid multiple reads. This register |
| * is written during reset and resume. |
| */ |
| ihcd->ctrl0 = ice40_spi_reg_read(ihcd, CTRL0_REG); |
| ihcd->ctrl0 |= SOFEN_CTRL; |
| ice40_spi_reg_write(ihcd, ihcd->ctrl0, CTRL0_REG); |
| |
| return 0; |
| } |
| |
| static void ice40_stop(struct usb_hcd *hcd) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| |
| cancel_work_sync(&ihcd->async_work); |
| } |
| |
| /* |
| * The _Error looks odd. But very helpful when looking for |
| * any errors in logs. |
| */ |
| static char __maybe_unused *xfr_status_string(int status) |
| { |
| switch (XFR_MASK(status)) { |
| case XFR_SUCCESS: |
| return "Ack"; |
| case XFR_BUSY: |
| return "Busy_Error"; |
| case XFR_PKTERR: |
| return "Pkt_Error"; |
| case XFR_PIDERR: |
| return "PID_Error"; |
| case XFR_NAK: |
| return "Nak"; |
| case XFR_STALL: |
| return "Stall_Error"; |
| case XFR_WRONGPID: |
| return "WrongPID_Error"; |
| case XFR_CRCERR: |
| return "CRC_Error"; |
| case XFR_TOGERR: |
| return "Togg_Error"; |
| case XFR_BADLEN: |
| return "BadLen_Error"; |
| case XFR_TIMEOUT: |
| return "Timeout_Error"; |
| default: |
| return "Unknown_Error"; |
| } |
| } |
| |
| static int ice40_xfer_setup(struct ice40_hcd *ihcd, struct urb *urb) |
| { |
| struct usb_host_endpoint *ep = urb->ep; |
| struct ice40_ep *iep = ep->hcpriv; |
| void *buf = urb->setup_packet; |
| int ret, status; |
| u8 cmd; |
| |
| /* |
| * SETUP transaction Handling: |
| * - copy the setup buffer to SUBUF fifo |
| * - Program HCMD register to initiate the SETP transaction. |
| * - poll for completion by reading XFRST register. |
| * - Interpret the result. |
| */ |
| |
| ihcd->setup_buf[0] = WRITE_CMD(SUBUF_REG); |
| ihcd->setup_xfr[1].tx_buf = buf; |
| ihcd->setup_xfr[1].len = sizeof(struct usb_ctrlrequest); |
| |
| ret = spi_sync(ihcd->spi, ihcd->setup_msg); |
| if (ret < 0) { |
| pr_err("SPI transfer failed\n"); |
| status = ret = -EIO; |
| goto out; |
| } |
| |
| cmd = HCMD_PT(2) | HCMD_TOGV(0) | HCMD_BSEL(0) | HCMD_EP(0); |
| ice40_spi_reg_write(ihcd, cmd, HCMD_REG); |
| |
| status = ice40_poll_xfer(ihcd, 1000); |
| switch (XFR_MASK(status)) { |
| case XFR_SUCCESS: |
| iep->xcat_err = 0; |
| ret = 0; |
| break; |
| case XFR_NAK: /* Device should not return Nak for SETUP */ |
| case XFR_STALL: |
| iep->xcat_err = 0; |
| ret = -EPIPE; |
| break; |
| case XFR_PKTERR: |
| case XFR_PIDERR: |
| case XFR_WRONGPID: |
| case XFR_CRCERR: |
| case XFR_TIMEOUT: |
| if (++iep->xcat_err < 8) |
| ret = -EINPROGRESS; |
| else |
| ret = -EPROTO; |
| break; |
| default: |
| pr_err("transaction timed out\n"); |
| ret = -EIO; |
| } |
| |
| out: |
| trace_ice40_setup(xfr_status_string(status), ret); |
| return ret; |
| } |
| |
| static int ice40_xfer_in(struct ice40_hcd *ihcd, struct urb *urb) |
| { |
| struct usb_host_endpoint *ep = urb->ep; |
| struct usb_device *udev = urb->dev; |
| u32 total_len = urb->transfer_buffer_length; |
| u16 maxpacket = usb_endpoint_maxp(&ep->desc); |
| u8 epnum = usb_pipeendpoint(urb->pipe); |
| bool is_out = usb_pipeout(urb->pipe); |
| struct ice40_ep *iep = ep->hcpriv; |
| u8 cmd, status, len = 0, t, expected_len; |
| void *buf; |
| int ret; |
| bool short_packet = true; |
| |
| if (epnum == 0 && ihcd->ep0_state == STATUS_PHASE) { |
| expected_len = 0; |
| buf = NULL; |
| t = 1; /* STATUS PHASE is always DATA1 */ |
| } else { |
| expected_len = min_t(u32, maxpacket, |
| total_len - urb->actual_length); |
| buf = urb->transfer_buffer + urb->actual_length; |
| t = usb_gettoggle(udev, epnum, is_out); |
| } |
| |
| /* |
| * IN transaction Handling: |
| * - Program HCMD register to initiate the IN transaction. |
| * - poll for completion by reading XFRST register. |
| * - Interpret the result. |
| * - If ACK is received and we expect some data, read RBLEN |
| * - Read the data from RBUF |
| */ |
| |
| cmd = HCMD_PT(0) | HCMD_TOGV(t) | HCMD_BSEL(0) | HCMD_EP(epnum); |
| ice40_spi_reg_write(ihcd, cmd, HCMD_REG); |
| |
| status = ice40_poll_xfer(ihcd, 1000); |
| switch (XFR_MASK(status)) { |
| case XFR_SUCCESS: |
| usb_dotoggle(udev, epnum, is_out); |
| iep->xcat_err = 0; |
| ret = 0; |
| if ((expected_len == 64) && (status & R64B)) |
| short_packet = false; |
| break; |
| case XFR_NAK: |
| iep->xcat_err = 0; |
| ret = -EINPROGRESS; |
| break; |
| case XFR_TOGERR: |
| /* |
| * Peripheral had missed the previous Ack and sent |
| * the same packet again. Ack is sent by the hardware. |
| * As the data is received already, ignore this |
| * event. |
| */ |
| ret = -EINPROGRESS; |
| break; |
| case XFR_PKTERR: |
| case XFR_PIDERR: |
| case XFR_WRONGPID: |
| case XFR_CRCERR: |
| case XFR_TIMEOUT: |
| if (++iep->xcat_err < 8) |
| ret = -EINPROGRESS; |
| else |
| ret = -EPROTO; |
| break; |
| case XFR_STALL: |
| ret = -EPIPE; |
| break; |
| case XFR_BADLEN: |
| ret = -EOVERFLOW; |
| break; |
| default: |
| pr_err("transaction timed out\n"); |
| ret = -EIO; |
| } |
| |
| /* |
| * Proceed further only if Ack is received and |
| * we are expecting some data. |
| */ |
| if (ret || !expected_len) |
| goto out; |
| |
| if (short_packet) |
| len = ice40_spi_reg_read(ihcd, RBLEN_REG); |
| else |
| len = 64; |
| |
| /* babble condition */ |
| if (len > expected_len) { |
| pr_err("overflow condition\n"); |
| ret = -EOVERFLOW; |
| goto out; |
| } |
| |
| /* |
| * zero len packet received. nothing to read from |
| * FIFO. |
| */ |
| if (len == 0) { |
| ret = 0; |
| goto out; |
| } |
| |
| ihcd->in_buf[0] = READ_CMD(RBUF0_REG); |
| |
| ihcd->in_xfr[1].rx_buf = buf; |
| ihcd->in_xfr[1].len = len; |
| |
| ret = spi_sync(ihcd->spi, ihcd->in_msg); |
| if (ret < 0) { |
| pr_err("SPI transfer failed\n"); |
| ret = -EIO; |
| goto out; |
| } |
| |
| urb->actual_length += len; |
| if ((urb->actual_length == total_len) || |
| (len < expected_len)) |
| ret = 0; /* URB completed */ |
| else |
| ret = -EINPROGRESS; /* still pending */ |
| out: |
| trace_ice40_in(epnum, xfr_status_string(status), len, |
| expected_len, ret); |
| return ret; |
| } |
| |
| static int ice40_xfer_out(struct ice40_hcd *ihcd, struct urb *urb) |
| { |
| struct usb_host_endpoint *ep = urb->ep; |
| struct usb_device *udev = urb->dev; |
| u32 total_len = urb->transfer_buffer_length; |
| u16 maxpacket = usb_endpoint_maxp(&ep->desc); |
| u8 epnum = usb_pipeendpoint(urb->pipe); |
| bool is_out = usb_pipeout(urb->pipe); |
| struct ice40_ep *iep = ep->hcpriv; |
| u8 cmd, status, len, t; |
| void *buf; |
| int ret; |
| |
| if (epnum == 0 && ihcd->ep0_state == STATUS_PHASE) { |
| len = 0; |
| buf = NULL; |
| t = 1; /* STATUS PHASE is always DATA1 */ |
| } else { |
| len = min_t(u32, maxpacket, total_len - urb->actual_length); |
| buf = urb->transfer_buffer + urb->actual_length; |
| t = usb_gettoggle(udev, epnum, is_out); |
| } |
| |
| /* |
| * OUT transaction Handling: |
| * - If we need to send data, write the data to WBUF Fifo |
| * - Program the WBLEN register |
| * - Program HCMD register to initiate the OUT transaction. |
| * - poll for completion by reading XFRST register. |
| * - Interpret the result. |
| */ |
| |
| |
| if (!len) |
| goto no_data; |
| |
| ihcd->out_buf[0] = WRITE_CMD(WBUF0_REG); |
| |
| ihcd->out_xfr[1].tx_buf = buf; |
| ihcd->out_xfr[1].len = len; |
| |
| ret = spi_sync(ihcd->spi, ihcd->out_msg); |
| if (ret < 0) { |
| pr_err("SPI transaction failed\n"); |
| status = ret = -EIO; |
| goto out; |
| } |
| |
| no_data: |
| /* |
| * Cache the WBLEN register and update it only if it |
| * is changed from the previous value. |
| */ |
| if (len != ihcd->wblen0) { |
| ice40_spi_reg_write(ihcd, len, WBLEN_REG); |
| ihcd->wblen0 = len; |
| } |
| |
| cmd = HCMD_PT(1) | HCMD_TOGV(t) | HCMD_BSEL(0) | HCMD_EP(epnum); |
| ice40_spi_reg_write(ihcd, cmd, HCMD_REG); |
| |
| status = ice40_poll_xfer(ihcd, 1000); |
| switch (XFR_MASK(status)) { |
| case XFR_SUCCESS: |
| usb_dotoggle(udev, epnum, is_out); |
| urb->actual_length += len; |
| iep->xcat_err = 0; |
| if (!len || (urb->actual_length == total_len)) |
| ret = 0; /* URB completed */ |
| else |
| ret = -EINPROGRESS; /* pending */ |
| break; |
| case XFR_NAK: |
| iep->xcat_err = 0; |
| ret = -EINPROGRESS; |
| break; |
| case XFR_PKTERR: |
| case XFR_PIDERR: |
| case XFR_WRONGPID: |
| case XFR_CRCERR: |
| case XFR_TIMEOUT: |
| if (++iep->xcat_err < 8) |
| ret = -EINPROGRESS; |
| else |
| ret = -EPROTO; |
| break; |
| case XFR_STALL: |
| ret = -EPIPE; |
| break; |
| case XFR_BADLEN: |
| ret = -EOVERFLOW; |
| break; |
| default: |
| pr_err("transaction timed out\n"); |
| ret = -EIO; |
| } |
| |
| out: |
| trace_ice40_out(epnum, xfr_status_string(status), len, ret); |
| return ret; |
| } |
| |
| static int ice40_process_urb(struct ice40_hcd *ihcd, struct urb *urb) |
| { |
| struct usb_device *udev = urb->dev; |
| u8 devnum = usb_pipedevice(urb->pipe); |
| bool is_out = usb_pipeout(urb->pipe); |
| u32 total_len = urb->transfer_buffer_length; |
| int ret = 0; |
| |
| /* |
| * The USB device address can be reset to 0 by core temporarily |
| * during reset recovery process. Don't assume anything about |
| * device address. The device address is programmed as 0 by |
| * default. If the device address is different to the previous |
| * cached value, re-program it here before proceeding. The device |
| * address register (FADDR) holds the value across multiple |
| * transactions and we support only one device. |
| */ |
| if (ihcd->devnum != devnum) { |
| ice40_spi_reg_write(ihcd, devnum, FADDR_REG); |
| ihcd->devnum = devnum; |
| } |
| |
| switch (usb_pipetype(urb->pipe)) { |
| case PIPE_CONTROL: |
| switch (ihcd->ep0_state) { |
| case SETUP_PHASE: |
| trace_ice40_ep0("SETUP"); |
| ret = ice40_xfer_setup(ihcd, urb); |
| if (ret) |
| break; |
| if (total_len) { |
| ihcd->ep0_state = DATA_PHASE; |
| /* |
| * Data stage always begin with |
| * DATA1 PID. |
| */ |
| usb_settoggle(udev, 0, is_out, 1); |
| } else { |
| ihcd->ep0_state = STATUS_PHASE; |
| goto do_status; |
| } |
| /* fall through */ |
| case DATA_PHASE: |
| trace_ice40_ep0("DATA"); |
| if (is_out) |
| ret = ice40_xfer_out(ihcd, urb); |
| else |
| ret = ice40_xfer_in(ihcd, urb); |
| if (ret) |
| break; |
| /* DATA Phase is completed successfully */ |
| ihcd->ep0_state = STATUS_PHASE; |
| /* fall through */ |
| case STATUS_PHASE: |
| do_status: |
| trace_ice40_ep0("STATUS"); |
| /* zero len DATA transfers have IN status */ |
| if (!total_len || is_out) |
| ret = ice40_xfer_in(ihcd, urb); |
| else |
| ret = ice40_xfer_out(ihcd, urb); |
| if (ret) |
| break; |
| ihcd->ep0_state = SETUP_PHASE; |
| break; |
| default: |
| pr_err("unknown stage for a control transfer\n"); |
| break; |
| } |
| break; |
| case PIPE_BULK: |
| if (is_out) |
| ret = ice40_xfer_out(ihcd, urb); |
| else |
| ret = ice40_xfer_in(ihcd, urb); |
| /* |
| * We may have to support zero len packet terminations |
| * for URB_ZERO_PACKET URBs. |
| */ |
| break; |
| default: |
| pr_err("IN/ISO transfers not supported\n"); |
| break; |
| } |
| |
| return ret; |
| } |
| |
| /* Must be called with spin lock and interrupts disabled */ |
| static void ice40_complete_urb(struct usb_hcd *hcd, struct urb *urb, int status) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| struct usb_host_endpoint *ep = urb->ep; |
| struct ice40_ep *iep = ep->hcpriv; |
| struct urb *first_urb; |
| bool needs_update = false; |
| bool control = usb_pipecontrol(urb->pipe); |
| |
| /* |
| * If the active URB i.e the first URB in the ep list is being |
| * removed, clear the transaction error count. If it is a control |
| * URB ep0_state needs to be reset to SETUP_PHASE. |
| */ |
| first_urb = list_first_entry(&ep->urb_list, struct urb, urb_list); |
| if (urb == first_urb) |
| needs_update = true; |
| |
| usb_hcd_unlink_urb_from_ep(hcd, urb); |
| spin_unlock(&ihcd->lock); |
| trace_ice40_urb_done(urb, status); |
| usb_hcd_giveback_urb(ihcd->hcd, urb, status); |
| spin_lock(&ihcd->lock); |
| |
| if (needs_update) { |
| iep->xcat_err = 0; |
| if (control) |
| ihcd->ep0_state = SETUP_PHASE; |
| } |
| } |
| |
| static void ice40_async_work(struct work_struct *work) |
| { |
| struct ice40_hcd *ihcd = container_of(work, |
| struct ice40_hcd, async_work); |
| struct usb_hcd *hcd = ihcd->hcd; |
| struct list_head *tmp, *uent, *utmp; |
| struct ice40_ep *iep; |
| struct usb_host_endpoint *ep; |
| struct urb *urb; |
| unsigned long flags; |
| int status; |
| |
| /* |
| * Traverse the active endpoints circularly and process URBs. |
| * If any endpoint is marked for unlinking, the URBs are |
| * completed here. The endpoint is removed from active list |
| * if a URB is retired with -EPIPE/-EPROTO errors. |
| */ |
| |
| spin_lock_irqsave(&ihcd->lock, flags); |
| |
| if (list_empty(&ihcd->async_list)) |
| goto out; |
| |
| iep = list_first_entry(&ihcd->async_list, struct ice40_ep, ep_list); |
| while (1) { |
| ep = iep->ep; |
| |
| urb = list_first_entry(&ep->urb_list, struct urb, urb_list); |
| if (urb->unlinked) { |
| status = urb->unlinked; |
| } else { |
| spin_unlock_irqrestore(&ihcd->lock, flags); |
| status = ice40_process_urb(ihcd, urb); |
| spin_lock_irqsave(&ihcd->lock, flags); |
| } |
| |
| if ((status == -EPIPE) || (status == -EPROTO)) |
| iep->halted = true; |
| |
| if (status != -EINPROGRESS) |
| ice40_complete_urb(hcd, urb, status); |
| |
| if (iep->unlinking) { |
| list_for_each_safe(uent, utmp, &ep->urb_list) { |
| urb = list_entry(uent, struct urb, urb_list); |
| if (urb->unlinked) |
| ice40_complete_urb(hcd, urb, 0); |
| } |
| iep->unlinking = false; |
| } |
| |
| tmp = iep->ep_list.next; |
| if (list_empty(&ep->urb_list) || iep->halted) { |
| list_del_init(&iep->ep_list); |
| |
| if (list_empty(&ihcd->async_list)) |
| break; |
| } |
| |
| if (tmp == &ihcd->async_list) |
| tmp = tmp->next; |
| iep = list_entry(tmp, struct ice40_ep, ep_list); |
| } |
| out: |
| spin_unlock_irqrestore(&ihcd->lock, flags); |
| } |
| |
| static int |
| ice40_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, gfp_t mem_flags) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| struct usb_device *udev = urb->dev; |
| struct usb_host_endpoint *ep = urb->ep; |
| bool is_out = usb_pipeout(urb->pipe); |
| u8 epnum = usb_pipeendpoint(urb->pipe); |
| struct ice40_ep *iep; |
| unsigned long flags; |
| int ret; |
| |
| /* |
| * This bridge chip supports only Full-speed. So ISO is not |
| * supported. Interrupt support is not implemented as there |
| * is no use case. |
| */ |
| if (usb_pipeisoc(urb->pipe) || usb_pipeint(urb->pipe)) { |
| pr_debug("iso and int xfers not supported\n"); |
| ret = -ENOTSUPP; |
| goto out; |
| } |
| |
| spin_lock_irqsave(&ihcd->lock, flags); |
| |
| ret = usb_hcd_link_urb_to_ep(hcd, urb); |
| if (ret) |
| goto rel_lock; |
| |
| trace_ice40_urb_enqueue(urb); |
| |
| iep = ep->hcpriv; |
| if (!iep) { |
| iep = kzalloc(sizeof(struct ice40_ep), GFP_ATOMIC); |
| if (!iep) { |
| pr_debug("fail to allocate iep\n"); |
| ret = -ENOMEM; |
| goto unlink; |
| } |
| ep->hcpriv = iep; |
| INIT_LIST_HEAD(&iep->ep_list); |
| iep->ep = ep; |
| usb_settoggle(udev, epnum, is_out, 0); |
| if (usb_pipecontrol(urb->pipe)) |
| ihcd->ep0_state = SETUP_PHASE; |
| } |
| |
| /* |
| * We expect the interface driver to clear the stall condition |
| * before queueing another URB. For example mass storage |
| * device may STALL a bulk endpoint for un-supported command. |
| * The storage driver clear the STALL condition before queueing |
| * another URB. |
| */ |
| iep->halted = false; |
| if (list_empty(&iep->ep_list)) |
| list_add_tail(&iep->ep_list, &ihcd->async_list); |
| |
| queue_work(ihcd->wq, &ihcd->async_work); |
| |
| spin_unlock_irqrestore(&ihcd->lock, flags); |
| |
| return 0; |
| unlink: |
| usb_hcd_unlink_urb_from_ep(hcd, urb); |
| rel_lock: |
| spin_unlock_irqrestore(&ihcd->lock, flags); |
| out: |
| return ret; |
| } |
| |
| static int |
| ice40_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| struct usb_host_endpoint *ep = urb->ep; |
| struct ice40_ep *iep; |
| unsigned long flags; |
| int ret; |
| |
| spin_lock_irqsave(&ihcd->lock, flags); |
| |
| ret = usb_hcd_check_unlink_urb(hcd, urb, status); |
| if (ret) |
| goto rel_lock; |
| |
| trace_ice40_urb_dequeue(urb); |
| iep = ep->hcpriv; |
| |
| /* |
| * If the endpoint is not in asynchronous schedule, complete |
| * the URB immediately. Otherwise mark it as being unlinked. |
| * The asynchronous schedule work will take care of completing |
| * the URB when this endpoint is encountered during traversal. |
| */ |
| if (list_empty(&iep->ep_list)) |
| ice40_complete_urb(hcd, urb, status); |
| else |
| iep->unlinking = true; |
| |
| rel_lock: |
| spin_unlock_irqrestore(&ihcd->lock, flags); |
| return ret; |
| } |
| |
| static void |
| ice40_endpoint_disable(struct usb_hcd *hcd, struct usb_host_endpoint *ep) |
| { |
| struct ice40_ep *iep = ep->hcpriv; |
| |
| /* |
| * If there is no I/O on this endpoint before, ep->hcpriv |
| * will be NULL. nothing to do in this case. |
| */ |
| if (!iep) |
| return; |
| |
| if (!list_empty(&ep->urb_list)) |
| pr_err("trying to disable an non-empty endpoint\n"); |
| |
| kfree(iep); |
| ep->hcpriv = NULL; |
| } |
| |
| |
| static int ice40_hub_status_data(struct usb_hcd *hcd, char *buf) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| int ret = 0; |
| |
| /* |
| * core calls hub_status_method during suspend/resume. |
| * return 0 if there is no port change. pcd_pending |
| * is set to true when a device is connected and line |
| * state is sampled via debugfs command. clear this |
| * flag after returning the port change status. |
| */ |
| if (ihcd->pcd_pending) { |
| *buf = (1 << 1); |
| ret = 1; |
| ihcd->pcd_pending = false; |
| } |
| |
| return ret; |
| } |
| |
| static void ice40_hub_descriptor(struct usb_hub_descriptor *desc) |
| { |
| /* There is nothing special about us!! */ |
| desc->bDescLength = 9; |
| desc->bDescriptorType = 0x29; |
| desc->bNbrPorts = 1; |
| desc->wHubCharacteristics = cpu_to_le16(HUB_CHAR_NO_LPSM | |
| HUB_CHAR_NO_OCPM); |
| desc->bPwrOn2PwrGood = 0; |
| desc->bHubContrCurrent = 0; |
| desc->u.hs.DeviceRemovable[0] = 0; |
| desc->u.hs.DeviceRemovable[1] = ~0; |
| } |
| |
| static int |
| ice40_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, |
| u16 wIndex, char *buf, u16 wLength) |
| { |
| int ret = 0; |
| u8 ctrl; |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| |
| /* |
| * We have only 1 port. No special locking is required while |
| * handling root hub commands. The bridge chip does not maintain |
| * any port states. Maintain different port states in software. |
| */ |
| switch (typeReq) { |
| case ClearPortFeature: |
| if (wIndex != 1 || wLength != 0) |
| goto error; |
| switch (wValue) { |
| case USB_PORT_FEAT_SUSPEND: |
| /* |
| * The device is resumed as part of the root hub |
| * resume to simplify the resume sequence. so |
| * we may simply return from here. If device is |
| * resumed before root hub is suspended, this |
| * flags will be cleared here. |
| */ |
| if (!(ihcd->port_flags & USB_PORT_STAT_SUSPEND)) |
| break; |
| ihcd->port_flags &= ~USB_PORT_STAT_SUSPEND; |
| break; |
| case USB_PORT_FEAT_ENABLE: |
| ihcd->port_flags &= ~USB_PORT_STAT_ENABLE; |
| break; |
| case USB_PORT_FEAT_POWER: |
| ihcd->port_flags &= ~USB_PORT_STAT_POWER; |
| break; |
| case USB_PORT_FEAT_C_CONNECTION: |
| ihcd->port_flags &= ~(USB_PORT_STAT_C_CONNECTION << 16); |
| break; |
| case USB_PORT_FEAT_C_ENABLE: |
| case USB_PORT_FEAT_C_SUSPEND: |
| case USB_PORT_FEAT_C_OVER_CURRENT: |
| case USB_PORT_FEAT_C_RESET: |
| /* nothing special here */ |
| break; |
| default: |
| goto error; |
| } |
| break; |
| case GetHubDescriptor: |
| ice40_hub_descriptor((struct usb_hub_descriptor *) buf); |
| break; |
| case GetHubStatus: |
| put_unaligned_le32(0, buf); |
| break; |
| case GetPortStatus: |
| if (wIndex != 1) |
| goto error; |
| |
| /* |
| * Core resets the device and requests port status to |
| * stop the reset signaling. If there is a reset in |
| * progress, finish it here. |
| */ |
| ctrl = ice40_spi_reg_read(ihcd, CTRL0_REG); |
| if (!(ctrl & RESET_CTRL)) |
| ihcd->port_flags &= ~USB_PORT_STAT_RESET; |
| |
| put_unaligned_le32(ihcd->port_flags, buf); |
| break; |
| case SetPortFeature: |
| if (wIndex != 1 || wLength != 0) |
| goto error; |
| switch (wValue) { |
| case USB_PORT_FEAT_SUSPEND: |
| if (ihcd->port_flags & USB_PORT_STAT_RESET) |
| goto error; |
| if (!(ihcd->port_flags & USB_PORT_STAT_ENABLE)) |
| goto error; |
| /* SOFs will be stopped during root hub suspend */ |
| ihcd->port_flags |= USB_PORT_STAT_SUSPEND; |
| break; |
| case USB_PORT_FEAT_POWER: |
| ihcd->port_flags |= USB_PORT_STAT_POWER; |
| break; |
| case USB_PORT_FEAT_RESET: |
| /* Good time to enable the port */ |
| ice40_spi_reg_write(ihcd, ihcd->ctrl0 | |
| RESET_CTRL, CTRL0_REG); |
| ihcd->port_flags |= USB_PORT_STAT_RESET; |
| ihcd->port_flags |= USB_PORT_STAT_ENABLE; |
| break; |
| default: |
| goto error; |
| } |
| break; |
| default: |
| error: |
| /* "protocol stall" on error */ |
| ret = -EPIPE; |
| } |
| |
| trace_ice40_hub_control(typeReq, wValue, wIndex, wLength, ret); |
| return ret; |
| } |
| |
| static void ice40_spi_power_off(struct ice40_hcd *ihcd); |
| static int ice40_bus_suspend(struct usb_hcd *hcd) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| |
| trace_ice40_bus_suspend(0); /* start */ |
| |
| /* This happens only during debugging */ |
| if (!ihcd->devnum) { |
| pr_debug("device still not connected. abort suspend\n"); |
| trace_ice40_bus_suspend(2); /* failure */ |
| return -EAGAIN; |
| } |
| /* |
| * Stop sending the SOFs on downstream port. The device |
| * finds the bus idle and enter suspend. The device |
| * takes ~3 msec to enter suspend. |
| */ |
| ihcd->ctrl0 &= ~SOFEN_CTRL; |
| ice40_spi_reg_write(ihcd, ihcd->ctrl0, CTRL0_REG); |
| usleep_range(4500, 5000); |
| |
| /* |
| * Power collapse the bridge chip to avoid the leakage |
| * current. |
| */ |
| ice40_spi_power_off(ihcd); |
| |
| trace_ice40_bus_suspend(1); /* successful */ |
| pm_relax(&ihcd->spi->dev); |
| return 0; |
| } |
| |
| static int ice40_spi_load_fw(struct ice40_hcd *ihcd); |
| static int ice40_bus_resume(struct usb_hcd *hcd) |
| { |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| u8 ctrl0; |
| int ret, i; |
| |
| pm_stay_awake(&ihcd->spi->dev); |
| trace_ice40_bus_resume(0); /* start */ |
| /* |
| * Power up the bridge chip and load the configuration file. |
| * Re-program the previous settings. For now we need to |
| * update the device address only. |
| */ |
| |
| for (i = 0; i < 3; i++) { |
| ret = ice40_spi_load_fw(ihcd); |
| if (!ret) |
| break; |
| } |
| |
| if (ret) { |
| pr_err("Load firmware failed with ret: %d\n", ret); |
| return ret; |
| } |
| |
| ice40_spi_reg_write(ihcd, ihcd->devnum, FADDR_REG); |
| ihcd->wblen0 = ~0; |
| |
| /* |
| * Program the bridge chip to drive resume signaling. The SOFs |
| * are automatically transmitted after resume completion. It |
| * will take ~20 msec for resume completion. |
| */ |
| ice40_spi_reg_write(ihcd, ihcd->ctrl0 | RESUME_CTRL, CTRL0_REG); |
| usleep_range(20000, 21000); |
| ret = ice40_handshake(ihcd, CTRL0_REG, RESUME_CTRL, 0, 5000); |
| if (ret) { |
| pr_err("resume failed\n"); |
| trace_ice40_bus_resume(2); /* failure */ |
| return -ENODEV; |
| } |
| |
| ctrl0 = ice40_spi_reg_read(ihcd, CTRL0_REG); |
| if (!(ctrl0 & SOFEN_CTRL)) { |
| pr_err("SOFs are not transmitted after resume\n"); |
| trace_ice40_bus_resume(3); /* failure */ |
| return -ENODEV; |
| } |
| |
| ihcd->port_flags &= ~USB_PORT_STAT_SUSPEND; |
| ihcd->ctrl0 |= SOFEN_CTRL; |
| |
| trace_ice40_bus_resume(1); /* success */ |
| return 0; |
| } |
| |
| static void ice40_set_autosuspend_delay(struct usb_device *dev) |
| { |
| /* |
| * Immediate suspend for root hub and 500 msec auto-suspend |
| * timeout for the card. |
| */ |
| if (!dev->parent) |
| pm_runtime_set_autosuspend_delay(&dev->dev, 0); |
| else |
| pm_runtime_set_autosuspend_delay(&dev->dev, 500); |
| } |
| |
| static const struct hc_driver ice40_hc_driver = { |
| .description = hcd_name, |
| .product_desc = "ICE40 SPI Host Controller", |
| .hcd_priv_size = sizeof(struct ice40_hcd *), |
| .flags = HCD_USB11, |
| |
| /* setup and clean up */ |
| .reset = ice40_reset, |
| .start = ice40_run, |
| .stop = ice40_stop, |
| |
| /* endpoint and I/O routines */ |
| .urb_enqueue = ice40_urb_enqueue, |
| .urb_dequeue = ice40_urb_dequeue, |
| .endpoint_disable = ice40_endpoint_disable, |
| |
| /* Root hub operations */ |
| .hub_status_data = ice40_hub_status_data, |
| .hub_control = ice40_hub_control, |
| .bus_suspend = ice40_bus_suspend, |
| .bus_resume = ice40_bus_resume, |
| |
| .set_autosuspend_delay = ice40_set_autosuspend_delay, |
| }; |
| |
| static int ice40_spi_parse_dt(struct ice40_hcd *ihcd) |
| { |
| struct device_node *node = ihcd->spi->dev.of_node; |
| int ret = 0; |
| |
| if (!node) { |
| pr_err("device specific info missing\n"); |
| ret = -ENODEV; |
| goto out; |
| } |
| |
| ihcd->reset_gpio = of_get_named_gpio(node, "lattice,reset-gpio", 0); |
| if (ihcd->reset_gpio < 0) { |
| pr_err("reset gpio is missing\n"); |
| ret = ihcd->reset_gpio; |
| goto out; |
| } |
| |
| ihcd->slave_select_gpio = of_get_named_gpio(node, |
| "lattice,slave-select-gpio", 0); |
| if (ihcd->slave_select_gpio < 0) { |
| pr_err("slave select gpio is missing\n"); |
| ret = ihcd->slave_select_gpio; |
| goto out; |
| } |
| |
| ihcd->config_done_gpio = of_get_named_gpio(node, |
| "lattice,config-done-gpio", 0); |
| if (ihcd->config_done_gpio < 0) { |
| pr_err("config done gpio is missing\n"); |
| ret = ihcd->config_done_gpio; |
| goto out; |
| } |
| |
| ihcd->vcc_en_gpio = of_get_named_gpio(node, "lattice,vcc-en-gpio", 0); |
| if (ihcd->vcc_en_gpio < 0) { |
| pr_err("vcc enable gpio is missing\n"); |
| ret = ihcd->vcc_en_gpio; |
| goto out; |
| } |
| |
| /* |
| * When clk-en-gpio is present, it is used to enable the 19.2 MHz |
| * clock from MSM to the bridge chip. Otherwise on-board clock |
| * is used. |
| */ |
| ihcd->clk_en_gpio = of_get_named_gpio(node, "lattice,clk-en-gpio", 0); |
| if (ihcd->clk_en_gpio < 0) |
| ihcd->clk_en_gpio = 0; |
| out: |
| return ret; |
| } |
| |
| static void ice40_spi_power_off(struct ice40_hcd *ihcd) |
| { |
| if (!ihcd->powered) |
| return; |
| |
| gpio_direction_output(ihcd->vcc_en_gpio, 0); |
| regulator_disable(ihcd->core_vcc); |
| regulator_disable(ihcd->spi_vcc); |
| if (ihcd->gpio_vcc) |
| regulator_disable(ihcd->gpio_vcc); |
| if (ihcd->clk_en_gpio) |
| gpio_direction_output(ihcd->clk_en_gpio, 0); |
| |
| ihcd->powered = false; |
| } |
| |
| static int ice40_spi_power_up(struct ice40_hcd *ihcd) |
| { |
| int ret; |
| |
| if (ihcd->clk_en_gpio) { |
| ret = gpio_direction_output(ihcd->clk_en_gpio, 1); |
| if (ret < 0) { |
| pr_err("fail to enabel clk %d\n", ret); |
| goto out; |
| } |
| } |
| |
| if (ihcd->gpio_vcc) { |
| ret = regulator_enable(ihcd->gpio_vcc); /* 1.8 V */ |
| if (ret < 0) { |
| pr_err("fail to enable gpio vcc\n"); |
| goto disable_clk; |
| } |
| } |
| |
| ret = regulator_enable(ihcd->spi_vcc); /* 1.8 V */ |
| if (ret < 0) { |
| pr_err("fail to enable spi vcc\n"); |
| goto disable_gpio_vcc; |
| } |
| |
| ret = regulator_enable(ihcd->core_vcc); /* 1.2 V */ |
| if (ret < 0) { |
| pr_err("fail to enable core vcc\n"); |
| goto disable_spi_vcc; |
| } |
| |
| ret = gpio_direction_output(ihcd->vcc_en_gpio, 1); |
| if (ret < 0) { |
| pr_err("fail to assert vcc gpio\n"); |
| goto disable_core_vcc; |
| } |
| |
| ihcd->powered = true; |
| |
| return 0; |
| |
| disable_core_vcc: |
| regulator_disable(ihcd->core_vcc); |
| disable_spi_vcc: |
| regulator_disable(ihcd->spi_vcc); |
| disable_gpio_vcc: |
| if (ihcd->gpio_vcc) |
| regulator_disable(ihcd->gpio_vcc); |
| disable_clk: |
| if (ihcd->clk_en_gpio) |
| gpio_direction_output(ihcd->clk_en_gpio, 0); |
| out: |
| return ret; |
| } |
| |
| static struct gpiomux_setting slave_select_setting = { |
| .func = GPIOMUX_FUNC_GPIO, |
| .drv = GPIOMUX_DRV_2MA, |
| .pull = GPIOMUX_PULL_NONE, |
| .dir = GPIOMUX_OUT_LOW, |
| }; |
| |
| static int ice40_spi_cache_fw(struct ice40_hcd *ihcd) |
| { |
| const struct firmware *fw; |
| void *buf; |
| size_t buf_len; |
| int ret; |
| |
| ret = request_firmware(&fw, fw_name, &ihcd->spi->dev); |
| if (ret < 0) { |
| pr_err("fail to get the firmware\n"); |
| goto out; |
| } |
| |
| pr_debug("received firmware size = %zu\n", fw->size); |
| |
| /* |
| * The bridge expects additional clock cycles after |
| * receiving the configuration data. We don't have a |
| * direct control over SPI clock. Add extra bytes |
| * to the confiration data. |
| */ |
| buf_len = fw->size + 16; |
| buf = devm_kzalloc(&ihcd->spi->dev, buf_len, GFP_KERNEL); |
| if (!buf) { |
| pr_err("fail to allocate firmware buffer\n"); |
| ret = -ENOMEM; |
| goto release; |
| } |
| |
| /* |
| * The firmware buffer can not be used for DMA as it |
| * is not physically contiguous. We copy the data |
| * in kmalloc buffer. This buffer will be freed only |
| * during unbind or rmmod. |
| */ |
| memcpy(buf, fw->data, fw->size); |
| release_firmware(fw); |
| |
| /* |
| * The bridge supports only 25 MHz during configuration |
| * file loading. |
| */ |
| ihcd->fmsg_xfr[0].tx_buf = buf; |
| ihcd->fmsg_xfr[0].len = buf_len; |
| ihcd->fmsg_xfr[0].speed_hz = 25000000; |
| |
| return 0; |
| |
| release: |
| release_firmware(fw); |
| out: |
| return ret; |
| } |
| |
| static int ice40_spi_load_fw(struct ice40_hcd *ihcd) |
| { |
| int ret, i; |
| struct gpiomux_setting active_old_setting, suspend_old_setting; |
| |
| ret = gpio_direction_output(ihcd->reset_gpio, 0); |
| if (ret < 0) { |
| pr_err("fail to assert reset %d\n", ret); |
| goto out; |
| } |
| |
| ret = gpio_direction_output(ihcd->vcc_en_gpio, 0); |
| if (ret < 0) { |
| pr_err("fail to de-assert vcc_en gpio %d\n", ret); |
| goto out; |
| } |
| |
| /* |
| * The bridge chip samples the chip select signal during |
| * power-up. If it is low, it enters SPI slave mode and |
| * accepts the configuration data from us. The chip |
| * select signal is managed by the SPI controller driver. |
| * We temporarily override the chip select config to |
| * drive it low. The SPI bus needs to be locked down during |
| * this period to avoid other slave data going to our |
| * bridge chip. Disable the SPI runtime suspend for exclusive |
| * chip select access. |
| */ |
| pm_runtime_get_sync(ihcd->spi->master->dev.parent); |
| |
| spi_bus_lock(ihcd->spi->master); |
| |
| ret = msm_gpiomux_write(ihcd->slave_select_gpio, GPIOMUX_SUSPENDED, |
| &slave_select_setting, &suspend_old_setting); |
| if (ret < 0) { |
| pr_err("fail to override suspend setting and select slave %d\n", |
| ret); |
| spi_bus_unlock(ihcd->spi->master); |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| goto out; |
| } |
| |
| ret = msm_gpiomux_write(ihcd->slave_select_gpio, GPIOMUX_ACTIVE, |
| &slave_select_setting, &active_old_setting); |
| if (ret < 0) { |
| pr_err("fail to override active setting and select slave %d\n", |
| ret); |
| spi_bus_unlock(ihcd->spi->master); |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| goto out; |
| } |
| |
| ret = ice40_spi_power_up(ihcd); |
| if (ret < 0) { |
| pr_err("fail to power up the chip\n"); |
| spi_bus_unlock(ihcd->spi->master); |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| goto out; |
| } |
| |
| |
| /* |
| * The databook says 1200 usec is required before the |
| * chip becomes ready for the SPI transfer. |
| */ |
| usleep_range(1200, 1250); |
| |
| ret = msm_gpiomux_write(ihcd->slave_select_gpio, GPIOMUX_SUSPENDED, |
| &suspend_old_setting, NULL); |
| if (ret < 0) { |
| pr_err("fail to rewrite suspend setting %d\n", ret); |
| spi_bus_unlock(ihcd->spi->master); |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| goto power_off; |
| } |
| |
| ret = msm_gpiomux_write(ihcd->slave_select_gpio, GPIOMUX_ACTIVE, |
| &active_old_setting, NULL); |
| if (ret < 0) { |
| pr_err("fail to rewrite active setting %d\n", ret); |
| spi_bus_unlock(ihcd->spi->master); |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| goto power_off; |
| } |
| |
| pm_runtime_put_noidle(ihcd->spi->master->dev.parent); |
| |
| ret = spi_sync_locked(ihcd->spi, ihcd->fmsg); |
| |
| spi_bus_unlock(ihcd->spi->master); |
| |
| if (ret < 0) { |
| pr_err("spi write failed\n"); |
| goto power_off; |
| } |
| |
| for (i = 0; i < 1000; i++) { |
| ret = gpio_get_value(ihcd->config_done_gpio); |
| if (ret) { |
| pr_debug("config done asserted %d\n", i); |
| break; |
| } |
| udelay(1); |
| } |
| |
| if (ret <= 0) { |
| pr_err("config done not asserted\n"); |
| ret = -ENODEV; |
| goto power_off; |
| } |
| |
| ret = gpio_direction_output(ihcd->reset_gpio, 1); |
| if (ret < 0) { |
| pr_err("fail to assert reset %d\n", ret); |
| goto power_off; |
| } |
| udelay(50); |
| |
| ret = ice40_spi_reg_read(ihcd, XFRST_REG); |
| pr_debug("XFRST val is %x\n", ret); |
| if (!(ret & PLLOK)) { |
| pr_err("The PLL2 is not synchronized\n"); |
| goto power_off; |
| } |
| |
| pr_info("Firmware load success\n"); |
| |
| return 0; |
| |
| power_off: |
| ice40_spi_power_off(ihcd); |
| out: |
| return ret; |
| } |
| |
| static int ice40_spi_init_regulators(struct ice40_hcd *ihcd) |
| { |
| int ret; |
| |
| ihcd->spi_vcc = devm_regulator_get(&ihcd->spi->dev, "spi-vcc"); |
| if (IS_ERR(ihcd->spi_vcc)) { |
| ret = PTR_ERR(ihcd->spi_vcc); |
| if (ret != -EPROBE_DEFER) |
| pr_err("fail to get spi-vcc %d\n", ret); |
| goto out; |
| } |
| |
| ret = regulator_set_voltage(ihcd->spi_vcc, 1800000, 1800000); |
| if (ret < 0) { |
| pr_err("fail to set spi-vcc %d\n", ret); |
| goto out; |
| } |
| |
| ihcd->core_vcc = devm_regulator_get(&ihcd->spi->dev, "core-vcc"); |
| if (IS_ERR(ihcd->core_vcc)) { |
| ret = PTR_ERR(ihcd->core_vcc); |
| if (ret != -EPROBE_DEFER) |
| pr_err("fail to get core-vcc %d\n", ret); |
| goto out; |
| } |
| |
| ret = regulator_set_voltage(ihcd->core_vcc, 1200000, 1200000); |
| if (ret < 0) { |
| pr_err("fail to set core-vcc %d\n", ret); |
| goto out; |
| } |
| |
| if (!of_get_property(ihcd->spi->dev.of_node, "gpio-supply", NULL)) |
| goto out; |
| |
| ihcd->gpio_vcc = devm_regulator_get(&ihcd->spi->dev, "gpio"); |
| if (IS_ERR(ihcd->gpio_vcc)) { |
| ret = PTR_ERR(ihcd->gpio_vcc); |
| if (ret != -EPROBE_DEFER) |
| pr_err("fail to get gpio_vcc %d\n", ret); |
| goto out; |
| } |
| |
| ret = regulator_set_voltage(ihcd->gpio_vcc, 1800000, 1800000); |
| if (ret < 0) { |
| pr_err("fail to set gpio_vcc %d\n", ret); |
| goto out; |
| } |
| |
| out: |
| return ret; |
| } |
| |
| static int ice40_spi_request_gpios(struct ice40_hcd *ihcd) |
| { |
| int ret; |
| |
| ret = devm_gpio_request(&ihcd->spi->dev, ihcd->reset_gpio, |
| "ice40_reset"); |
| if (ret < 0) { |
| pr_err("fail to request reset gpio\n"); |
| goto out; |
| } |
| |
| ret = devm_gpio_request(&ihcd->spi->dev, ihcd->config_done_gpio, |
| "ice40_config_done"); |
| if (ret < 0) { |
| pr_err("fail to request config_done gpio\n"); |
| goto out; |
| } |
| |
| ret = devm_gpio_request(&ihcd->spi->dev, ihcd->vcc_en_gpio, |
| "ice40_vcc_en"); |
| if (ret < 0) { |
| pr_err("fail to request vcc_en gpio\n"); |
| goto out; |
| } |
| |
| if (ihcd->clk_en_gpio) { |
| |
| ret = devm_gpio_request(&ihcd->spi->dev, ihcd->clk_en_gpio, |
| "ice40_clk_en"); |
| if (ret < 0) |
| pr_err("fail to request clk_en gpio\n"); |
| } |
| |
| out: |
| return ret; |
| } |
| |
| static int |
| ice40_spi_init_one_xfr(struct ice40_hcd *ihcd, enum ice40_xfr_type type) |
| { |
| struct spi_message **m; |
| struct spi_transfer **t; |
| int n; |
| |
| switch (type) { |
| case FIRMWARE_XFR: |
| m = &ihcd->fmsg; |
| t = &ihcd->fmsg_xfr; |
| n = 1; |
| break; |
| case REG_WRITE_XFR: |
| m = &ihcd->wmsg; |
| t = &ihcd->wmsg_xfr; |
| n = 1; |
| break; |
| case REG_READ_XFR: |
| m = &ihcd->rmsg; |
| t = &ihcd->rmsg_xfr; |
| n = 1; |
| break; |
| case SETUP_XFR: |
| m = &ihcd->setup_msg; |
| t = &ihcd->setup_xfr; |
| n = 2; |
| break; |
| case DATA_IN_XFR: |
| m = &ihcd->in_msg; |
| t = &ihcd->in_xfr; |
| n = 2; |
| break; |
| case DATA_OUT_XFR: |
| m = &ihcd->out_msg; |
| t = &ihcd->out_xfr; |
| n = 2; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| *m = devm_kzalloc(&ihcd->spi->dev, sizeof(**m), GFP_KERNEL); |
| if (*m == NULL) |
| goto out; |
| |
| *t = devm_kzalloc(&ihcd->spi->dev, n * sizeof(**t), GFP_KERNEL); |
| if (*t == NULL) |
| goto out; |
| |
| spi_message_init_with_transfers(*m, *t, n); |
| |
| return 0; |
| out: |
| return -ENOMEM; |
| } |
| |
| static int ice40_spi_init_xfrs(struct ice40_hcd *ihcd) |
| { |
| int ret = -ENOMEM; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, FIRMWARE_XFR); |
| if (ret < 0) |
| goto out; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, REG_WRITE_XFR); |
| if (ret < 0) |
| goto out; |
| |
| ihcd->w_tx_buf = devm_kzalloc(&ihcd->spi->dev, 2, GFP_KERNEL); |
| if (!ihcd->w_tx_buf) |
| goto out; |
| |
| ihcd->w_rx_buf = devm_kzalloc(&ihcd->spi->dev, 2, GFP_KERNEL); |
| if (!ihcd->w_rx_buf) |
| goto out; |
| |
| ihcd->wmsg_xfr[0].tx_buf = ihcd->w_tx_buf; |
| ihcd->wmsg_xfr[0].rx_buf = ihcd->w_rx_buf; |
| ihcd->wmsg_xfr[0].len = 2; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, REG_READ_XFR); |
| if (ret < 0) |
| goto out; |
| |
| ihcd->r_tx_buf = devm_kzalloc(&ihcd->spi->dev, 3, GFP_KERNEL); |
| if (!ihcd->r_tx_buf) |
| goto out; |
| |
| ihcd->r_rx_buf = devm_kzalloc(&ihcd->spi->dev, 3, GFP_KERNEL); |
| if (!ihcd->r_rx_buf) |
| goto out; |
| |
| ihcd->rmsg_xfr[0].tx_buf = ihcd->r_tx_buf; |
| ihcd->rmsg_xfr[0].rx_buf = ihcd->r_rx_buf; |
| ihcd->rmsg_xfr[0].len = 3; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, SETUP_XFR); |
| if (ret < 0) |
| goto out; |
| |
| ihcd->setup_buf = devm_kzalloc(&ihcd->spi->dev, 1, GFP_KERNEL); |
| if (!ihcd->setup_buf) |
| goto out; |
| ihcd->setup_xfr[0].tx_buf = ihcd->setup_buf; |
| ihcd->setup_xfr[0].len = 1; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, DATA_IN_XFR); |
| if (ret < 0) |
| goto out; |
| ihcd->in_buf = devm_kzalloc(&ihcd->spi->dev, 2, GFP_KERNEL); |
| if (!ihcd->in_buf) |
| goto out; |
| ihcd->in_xfr[0].tx_buf = ihcd->in_buf; |
| ihcd->in_xfr[0].len = 2; |
| |
| ret = ice40_spi_init_one_xfr(ihcd, DATA_OUT_XFR); |
| if (ret < 0) |
| goto out; |
| ihcd->out_buf = devm_kzalloc(&ihcd->spi->dev, 1, GFP_KERNEL); |
| if (!ihcd->out_buf) |
| goto out; |
| ihcd->out_xfr[0].tx_buf = ihcd->out_buf; |
| ihcd->out_xfr[0].len = 1; |
| |
| return 0; |
| |
| out: |
| return -ENOMEM; |
| } |
| |
| static int ice40_dbg_cmd_open(struct inode *inode, struct file *file) |
| { |
| return single_open(file, NULL, inode->i_private); |
| } |
| |
| static ssize_t ice40_dbg_cmd_write(struct file *file, const char __user *ubuf, |
| size_t count, loff_t *ppos) |
| { |
| struct seq_file *s = file->private_data; |
| struct ice40_hcd *ihcd = s->private; |
| char buf[32]; |
| int ret; |
| u8 status, addr; |
| |
| memset(buf, 0x00, sizeof(buf)); |
| |
| if (copy_from_user(&buf, ubuf, min_t(size_t, sizeof(buf) - 1, count))) { |
| ret = -EFAULT; |
| goto out; |
| } |
| |
| if (!strcmp(buf, "poll")) { |
| if (!HCD_RH_RUNNING(ihcd->hcd)) { |
| ret = -EAGAIN; |
| goto out; |
| } |
| /* |
| * The bridge chip supports interrupt for device |
| * connect and disconnect. We don;t have a real |
| * use case of connect/disconnect. This debugfs |
| * interface provides a way to enumerate the |
| * attached device. |
| */ |
| ice40_spi_reg_write(ihcd, ihcd->ctrl0 | |
| DET_BUS_CTRL, CTRL0_REG); |
| ice40_handshake(ihcd, CTRL0_REG, DET_BUS_CTRL, 0, 5000); |
| status = ice40_spi_reg_read(ihcd, XFRST_REG); |
| if ((status & DPST)) { |
| ihcd->port_flags |= USB_PORT_STAT_CONNECTION; |
| ihcd->port_flags |= USB_PORT_STAT_C_CONNECTION << 16; |
| ihcd->pcd_pending = true; |
| usb_hcd_poll_rh_status(ihcd->hcd); |
| } else if (ihcd->port_flags & USB_PORT_STAT_CONNECTION) { |
| ihcd->port_flags &= ~USB_PORT_STAT_ENABLE; |
| ihcd->port_flags &= ~USB_PORT_STAT_CONNECTION; |
| ihcd->port_flags |= (USB_PORT_STAT_C_CONNECTION << 16); |
| ihcd->pcd_pending = true; |
| usb_hcd_poll_rh_status(ihcd->hcd); |
| } |
| } else if (!strcmp(buf, "rwtest")) { |
| ihcd->devnum = 1; |
| ice40_spi_reg_write(ihcd, 0x1, FADDR_REG); |
| addr = ice40_spi_reg_read(ihcd, FADDR_REG); |
| pr_info("addr written was 0x1 read as %x\n", addr); |
| } else if (!strcmp(buf, "force_disconnect")) { |
| if (!HCD_RH_RUNNING(ihcd->hcd)) { |
| ret = -EAGAIN; |
| goto out; |
| } |
| /* |
| * Forcfully disconnect the device. This is required |
| * for simulating the disconnect on a USB port which |
| * does not have pull-down resistors. |
| */ |
| ihcd->port_flags &= ~USB_PORT_STAT_ENABLE; |
| ihcd->port_flags &= ~USB_PORT_STAT_CONNECTION; |
| ihcd->port_flags |= (USB_PORT_STAT_C_CONNECTION << 16); |
| ihcd->pcd_pending = true; |
| usb_hcd_poll_rh_status(ihcd->hcd); |
| } else if (!strcmp(buf, "config_test")) { |
| ice40_spi_power_off(ihcd); |
| ret = ice40_spi_load_fw(ihcd); |
| if (ret) { |
| pr_err("config load failed\n"); |
| goto out; |
| } |
| } else { |
| ret = -EINVAL; |
| goto out; |
| } |
| |
| ret = count; |
| out: |
| return ret; |
| } |
| |
| const struct file_operations ice40_dbg_cmd_ops = { |
| .open = ice40_dbg_cmd_open, |
| .write = ice40_dbg_cmd_write, |
| .llseek = seq_lseek, |
| .release = single_release, |
| }; |
| |
| static int ice40_debugfs_init(struct ice40_hcd *ihcd) |
| { |
| struct dentry *dir; |
| int ret = 0; |
| |
| dir = debugfs_create_dir("ice40_hcd", NULL); |
| |
| if (!dir || IS_ERR(dir)) { |
| ret = -ENODEV; |
| goto out; |
| } |
| |
| ihcd->dbg_root = dir; |
| |
| dir = debugfs_create_file("command", S_IWUSR, ihcd->dbg_root, ihcd, |
| &ice40_dbg_cmd_ops); |
| |
| if (!dir) { |
| debugfs_remove_recursive(ihcd->dbg_root); |
| ihcd->dbg_root = NULL; |
| ret = -ENODEV; |
| } |
| |
| out: |
| return ret; |
| } |
| |
| static int ice40_spi_probe(struct spi_device *spi) |
| { |
| struct ice40_hcd *ihcd; |
| int ret; |
| |
| ihcd = devm_kzalloc(&spi->dev, sizeof(*ihcd), GFP_KERNEL); |
| if (!ihcd) { |
| pr_err("fail to allocate ihcd\n"); |
| ret = -ENOMEM; |
| goto out; |
| } |
| ihcd->spi = spi; |
| |
| ret = ice40_spi_parse_dt(ihcd); |
| if (ret) { |
| pr_err("fail to parse dt node\n"); |
| goto out; |
| } |
| |
| ret = ice40_spi_init_regulators(ihcd); |
| if (ret) { |
| pr_err("fail to init regulators\n"); |
| goto out; |
| } |
| |
| ret = ice40_spi_request_gpios(ihcd); |
| if (ret) { |
| pr_err("fail to request gpios\n"); |
| goto out; |
| } |
| |
| spin_lock_init(&ihcd->lock); |
| INIT_LIST_HEAD(&ihcd->async_list); |
| INIT_WORK(&ihcd->async_work, ice40_async_work); |
| mutex_init(&ihcd->wlock); |
| mutex_init(&ihcd->rlock); |
| |
| /* |
| * Enable all our trace points. Useful in debugging card |
| * enumeration issues. |
| */ |
| ret = trace_set_clr_event(__stringify(TRACE_SYSTEM), NULL, 1); |
| if (ret < 0) |
| pr_err("fail to enable trace points with %d\n", ret); |
| |
| ihcd->wq = create_singlethread_workqueue("ice40_wq"); |
| if (!ihcd->wq) { |
| pr_err("fail to create workqueue\n"); |
| ret = -ENOMEM; |
| goto destroy_mutex; |
| } |
| |
| ret = ice40_spi_init_xfrs(ihcd); |
| if (ret) { |
| pr_err("fail to init spi xfrs %d\n", ret); |
| goto destroy_wq; |
| } |
| |
| ret = ice40_spi_cache_fw(ihcd); |
| if (ret) { |
| pr_err("fail to cache fw %d\n", ret); |
| goto destroy_wq; |
| } |
| |
| ret = ice40_spi_load_fw(ihcd); |
| if (ret) { |
| pr_err("fail to load fw %d\n", ret); |
| goto destroy_wq; |
| } |
| |
| ihcd->hcd = usb_create_hcd(&ice40_hc_driver, &spi->dev, "ice40"); |
| if (!ihcd->hcd) { |
| pr_err("fail to alloc hcd\n"); |
| ret = -ENOMEM; |
| goto power_off; |
| } |
| *((struct ice40_hcd **) ihcd->hcd->hcd_priv) = ihcd; |
| |
| ret = usb_add_hcd(ihcd->hcd, 0, 0); |
| |
| if (ret < 0) { |
| pr_err("fail to add HCD\n"); |
| goto put_hcd; |
| } |
| |
| ice40_debugfs_init(ihcd); |
| |
| /* |
| * We manage the power states of the bridge chip |
| * as part of root hub suspend/resume. We don't |
| * need to implement any additional runtime PM |
| * methods. |
| */ |
| pm_runtime_no_callbacks(&spi->dev); |
| pm_runtime_set_active(&spi->dev); |
| pm_runtime_enable(&spi->dev); |
| |
| /* |
| * This does not mean bridge chip can wakeup the |
| * system from sleep. It's activity can prevent |
| * or abort the system sleep. The device_init_wakeup |
| * creates the wakeup source for us which we will |
| * use to control system sleep. |
| */ |
| device_init_wakeup(&spi->dev, 1); |
| pm_stay_awake(&spi->dev); |
| |
| pr_debug("success\n"); |
| |
| return 0; |
| |
| put_hcd: |
| usb_put_hcd(ihcd->hcd); |
| power_off: |
| ice40_spi_power_off(ihcd); |
| destroy_wq: |
| destroy_workqueue(ihcd->wq); |
| destroy_mutex: |
| mutex_destroy(&ihcd->rlock); |
| mutex_destroy(&ihcd->wlock); |
| out: |
| pr_info("ice40_spi_probe failed\n"); |
| return ret; |
| } |
| |
| static int ice40_spi_remove(struct spi_device *spi) |
| { |
| struct usb_hcd *hcd = spi_get_drvdata(spi); |
| struct ice40_hcd *ihcd = hcd_to_ihcd(hcd); |
| |
| debugfs_remove_recursive(ihcd->dbg_root); |
| |
| usb_remove_hcd(hcd); |
| usb_put_hcd(hcd); |
| destroy_workqueue(ihcd->wq); |
| ice40_spi_power_off(ihcd); |
| |
| pm_runtime_disable(&spi->dev); |
| pm_relax(&spi->dev); |
| |
| return 0; |
| } |
| |
| static struct of_device_id ice40_spi_of_match_table[] = { |
| { .compatible = "lattice,ice40-spi-usb", }, |
| {}, |
| }; |
| |
| static struct spi_driver ice40_spi_driver = { |
| .driver = { |
| .name = "ice40_spi", |
| .owner = THIS_MODULE, |
| .of_match_table = ice40_spi_of_match_table, |
| }, |
| .probe = ice40_spi_probe, |
| .remove = ice40_spi_remove, |
| }; |
| |
| module_spi_driver(ice40_spi_driver); |
| |
| MODULE_DESCRIPTION("ICE40 FPGA based SPI-USB bridge HCD"); |
| MODULE_LICENSE("GPL v2"); |