| /* Copyright (c) 2014 The Chromium OS Authors. All rights reserved. |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "common.h" |
| #include "link_defs.h" |
| #include "registers.h" |
| #include "spi.h" |
| #include "usb_descriptor.h" |
| #include "usb_spi.h" |
| #include "util.h" |
| |
| static int16_t usb_spi_map_error(int error) |
| { |
| switch (error) { |
| case EC_SUCCESS: return USB_SPI_SUCCESS; |
| case EC_ERROR_TIMEOUT: return USB_SPI_TIMEOUT; |
| case EC_ERROR_BUSY: return USB_SPI_BUSY; |
| default: return USB_SPI_UNKNOWN_ERROR | (error & 0x7fff); |
| } |
| } |
| |
| static uint16_t usb_spi_read_packet(struct usb_spi_config const *config) |
| { |
| size_t i; |
| uint16_t bytes = btable_ep[config->endpoint].rx_count & 0x3ff; |
| size_t count = MAX((bytes + 1) / 2, USB_MAX_PACKET_SIZE / 2); |
| |
| /* |
| * The USB peripheral doesn't support DMA access to its packet |
| * RAM so we have to copy messages out into a bounce buffer. |
| */ |
| for (i = 0; i < count; ++i) |
| config->buffer[i] = config->rx_ram[i]; |
| |
| /* |
| * RX packet consumed, mark the packet as VALID. The master |
| * could queue up the next command while we process this SPI |
| * transaction and prepare the response. |
| */ |
| STM32_TOGGLE_EP(config->endpoint, EP_RX_MASK, EP_RX_VALID, 0); |
| |
| return bytes; |
| } |
| |
| static void usb_spi_write_packet(struct usb_spi_config const *config, |
| uint8_t count) |
| { |
| size_t i; |
| |
| /* |
| * Copy read bytes and status back out of bounce buffer and |
| * update TX packet state (mark as VALID for master to read). |
| */ |
| for (i = 0; i < (count + 1) / 2; ++i) |
| config->tx_ram[i] = config->buffer[i]; |
| |
| btable_ep[config->endpoint].tx_count = count; |
| |
| STM32_TOGGLE_EP(config->endpoint, EP_TX_MASK, EP_TX_VALID, 0); |
| } |
| |
| static int rx_valid(struct usb_spi_config const *config) |
| { |
| return (STM32_USB_EP(config->endpoint) & EP_RX_MASK) == EP_RX_VALID; |
| } |
| |
| void usb_spi_deferred(struct usb_spi_config const *config) |
| { |
| /* |
| * If our overall enabled state has changed we call the board specific |
| * enable or disable routines and save our new state. |
| */ |
| int enabled = (config->state->enabled_host && |
| config->state->enabled_device); |
| |
| if (enabled ^ config->state->enabled) { |
| if (enabled) usb_spi_board_enable(config); |
| else usb_spi_board_disable(config); |
| |
| config->state->enabled = enabled; |
| } |
| |
| /* |
| * And if there is a USB packet waiting we process it and generate a |
| * response. |
| */ |
| if (!rx_valid(config)) { |
| uint16_t count = usb_spi_read_packet(config); |
| uint8_t write_count = (config->buffer[0] >> 0) & 0xff; |
| uint8_t read_count = (config->buffer[0] >> 8) & 0xff; |
| |
| if (!config->state->enabled) { |
| config->buffer[0] = USB_SPI_DISABLED; |
| } else if (write_count > USB_SPI_MAX_WRITE_COUNT || |
| write_count != (count - 2)) { |
| config->buffer[0] = USB_SPI_WRITE_COUNT_INVALID; |
| } else if (read_count > USB_SPI_MAX_READ_COUNT) { |
| config->buffer[0] = USB_SPI_READ_COUNT_INVALID; |
| } else { |
| config->buffer[0] = usb_spi_map_error( |
| spi_transaction(SPI_FLASH_DEVICE, |
| (uint8_t *)(config->buffer + 1), |
| write_count, |
| (uint8_t *)(config->buffer + 1), |
| read_count)); |
| } |
| |
| usb_spi_write_packet(config, read_count + 2); |
| } |
| } |
| |
| void usb_spi_tx(struct usb_spi_config const *config) |
| { |
| STM32_TOGGLE_EP(config->endpoint, EP_TX_MASK, EP_TX_NAK, 0); |
| } |
| |
| void usb_spi_rx(struct usb_spi_config const *config) |
| { |
| STM32_TOGGLE_EP(config->endpoint, EP_RX_MASK, EP_RX_NAK, 0); |
| |
| hook_call_deferred(config->deferred, 0); |
| } |
| |
| void usb_spi_reset(struct usb_spi_config const *config) |
| { |
| int endpoint = config->endpoint; |
| |
| btable_ep[endpoint].tx_addr = usb_sram_addr(config->tx_ram); |
| btable_ep[endpoint].tx_count = 0; |
| |
| btable_ep[endpoint].rx_addr = usb_sram_addr(config->rx_ram); |
| btable_ep[endpoint].rx_count = |
| 0x8000 | ((USB_MAX_PACKET_SIZE / 32 - 1) << 10); |
| |
| STM32_USB_EP(endpoint) = ((endpoint << 0) | /* Endpoint Addr*/ |
| (2 << 4) | /* TX NAK */ |
| (0 << 9) | /* Bulk EP */ |
| (3 << 12)); /* RX Valid */ |
| } |
| |
| int usb_spi_interface(struct usb_spi_config const *config, |
| usb_uint *rx_buf, |
| usb_uint *tx_buf) |
| { |
| struct usb_setup_packet setup; |
| |
| usb_read_setup_packet(rx_buf, &setup); |
| |
| if (setup.bmRequestType != (USB_DIR_OUT | |
| USB_TYPE_VENDOR | |
| USB_RECIP_INTERFACE)) |
| return 1; |
| |
| if (setup.wValue != 0 || |
| setup.wIndex != config->interface || |
| setup.wLength != 0) |
| return 1; |
| |
| if (!config->state->enabled_device) |
| return 1; |
| |
| switch (setup.bRequest) { |
| case USB_SPI_REQ_ENABLE: |
| config->state->enabled_host = 1; |
| break; |
| |
| case USB_SPI_REQ_DISABLE: |
| config->state->enabled_host = 0; |
| break; |
| |
| default: return 1; |
| } |
| |
| /* |
| * Our state has changed, call the deferred function to handle the |
| * state change. |
| */ |
| hook_call_deferred(config->deferred, 0); |
| |
| btable_ep[0].tx_count = 0; |
| STM32_TOGGLE_EP(0, EP_TX_RX_MASK, EP_TX_RX_VALID, EP_STATUS_OUT); |
| return 0; |
| } |
| |
| void usb_spi_enable(struct usb_spi_config const *config, int enabled) |
| { |
| config->state->enabled_device = enabled; |
| |
| hook_call_deferred(config->deferred, 0); |
| } |