2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* arch/mips/src/pic32mx/pic32mx_usbdev.c
|
|
|
|
*
|
2012-01-07 00:17:03 +01:00
|
|
|
* Copyright (C) 2011-2012 Gregory Nutt. All rights reserved.
|
|
|
|
* Author: Gregory Nutt <gnutt@nuttx.org>
|
2011-12-28 01:49:48 +01:00
|
|
|
*
|
|
|
|
* References:
|
2011-12-29 13:52:15 +01:00
|
|
|
* This file derives from the STM32 USB device driver with modifications
|
|
|
|
* based on additional information from:
|
|
|
|
*
|
2011-12-28 01:49:48 +01:00
|
|
|
* - "USB On-The-Go (OTG)", DS61126E, Microchip Technology Inc., 2009
|
2011-12-29 13:52:15 +01:00
|
|
|
* - Sample code provided with the Sure Electronics PIC32 board
|
|
|
|
* (which seems to have derived from Microchip PICDEM PIC18 code).
|
2011-12-28 01:49:48 +01:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
|
|
|
* used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
#include <nuttx/arch.h>
|
|
|
|
#include <nuttx/usb/usb.h>
|
|
|
|
#include <nuttx/usb/usbdev.h>
|
|
|
|
#include <nuttx/usb/usbdev_trace.h>
|
|
|
|
|
|
|
|
#include <arch/irq.h>
|
|
|
|
|
|
|
|
#include "up_arch.h"
|
|
|
|
#include "pic32mx-internal.h"
|
|
|
|
#include "pic32mx-usbotg.h"
|
|
|
|
|
|
|
|
#if defined(CONFIG_USBDEV) && defined(CONFIG_PIC32MX_USBDEV)
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Pre-processor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
/* Configuration ************************************************************/
|
|
|
|
|
|
|
|
#ifndef CONFIG_USBDEV_EP0_MAXSIZE
|
|
|
|
# define CONFIG_USBDEV_EP0_MAXSIZE 64
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Extremely detailed register debug that you would normally never want
|
|
|
|
* enabled.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CONFIG_DEBUG
|
|
|
|
# undef CONFIG_PIC32MX_USBDEV_REGDEBUG
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Interrupts ***************************************************************/
|
|
|
|
/* Initial interrupt sets */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_SOFINTS
|
|
|
|
# define USB_SOF_INTERRUPT USB_INT_SOF
|
|
|
|
#else
|
|
|
|
# define USB_SOF_INTERRUPT 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define ERROR_INTERRUPTS (USB_EINT_PID|USB_EINT_CRC5|USB_EINT_EOF|\
|
|
|
|
USB_EINT_CRC16|USB_EINT_DFN8|USB_EINT_BTO|\
|
|
|
|
USB_EINT_BTS)
|
|
|
|
#define NORMAL_INTERRUPTS (USB_INT_URST|USB_INT_UERR|USB_SOF_INTERRUPT|\
|
|
|
|
USB_INT_TRN|USB_INT_IDLE|USB_INT_STALL)
|
|
|
|
|
|
|
|
/* Endpoints ****************************************************************/
|
|
|
|
/* Endpoint identifiers. The PIC32MX supports up to 16 mono-directional or 8
|
|
|
|
* bidirectional endpoints. However, when you take into account PMA buffer
|
|
|
|
* usage (see below) and the fact that EP0 is bidirectional, then there is
|
|
|
|
* a functional limitation of EP0 + 5 mono-directional endpoints = 6. We'll
|
|
|
|
* define PIC32MX_NENDPOINTS to be 8, however, because that is how many
|
|
|
|
* endpoint register sets there are.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define PIC32MX_NENDPOINTS (16)
|
|
|
|
#define EP0 (0)
|
|
|
|
|
|
|
|
#define PIC32MX_ENDP_BIT(ep) (1 << (ep))
|
2011-12-30 01:09:43 +01:00
|
|
|
#define PIC32MX_ENDP_ALLSET 0xffff
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Endpoint Definitions */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
#ifndef CONFIG_USB_PINGPONG
|
2011-12-30 01:09:43 +01:00
|
|
|
# define USB_NEXT_PINGPONG (0)
|
2011-12-29 21:46:38 +01:00
|
|
|
# define EP0_OUT_EVEN (0)
|
|
|
|
# define EP0_OUT_ODD (0)
|
|
|
|
# define EP0_IN_EVEN (1)
|
|
|
|
# define EP0_IN_ODD (1)
|
|
|
|
# define EP_OUT_EVEN(ep) ((ep) << 1)
|
|
|
|
# define EP_OUT_ODD(ep) ((ep) << 1)
|
|
|
|
# define EP_IN_EVEN(ep) (((ep) << 1) + 1)
|
|
|
|
# define EP_IN_ODD(ep) (((ep) << 1) + 1)
|
|
|
|
# define EP(ep,dir,pp) (((ep) << 1) + (dir))
|
2011-12-28 20:40:37 +01:00
|
|
|
#else
|
2011-12-30 01:09:43 +01:00
|
|
|
# define USB_NEXT_PINGPONG (USB_BDT_DTS)
|
2011-12-29 21:46:38 +01:00
|
|
|
# define EP0_OUT_EVEN (0)
|
|
|
|
# define EP0_OUT_ODD (1)
|
|
|
|
# define EP0_IN_EVEN (2)
|
|
|
|
# define EP0_IN_ODD (3)
|
|
|
|
# define EP_OUT_EVEN(ep) ((ep) << 2)
|
|
|
|
# define EP_OUT_ODD(ep) (((ep) << 2) + 1)
|
|
|
|
# define EP_IN_EVEN(ep) (((ep) << 2) + 2)
|
|
|
|
# define EP_IN_ODD(ep) (((ep) << 2) + 3)
|
|
|
|
# define EP(ep,dir,pp) (((ep) << 2) + ((dir) << 1) + (pp))
|
2011-12-28 20:40:37 +01:00
|
|
|
#endif
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Packet sizes. We us a fixed 64 max packet size for all endpoint types */
|
|
|
|
|
|
|
|
#define PIC32MX_MAXPACKET_SHIFT (6)
|
|
|
|
#define PIC32MX_MAXPACKET_SIZE (1 << (PIC32MX_MAXPACKET_SHIFT))
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
#define PIC32MX_EP0MAXPACKET PIC32MX_MAXPACKET_SIZE
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Endpoint register initialization parameters */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
#define PIC32MX_EP_CONTROL (USB_EP_EPHSHK|USB_EP_EPTXEN|USB_EP_EPRXEN)
|
|
|
|
#define PIC32MX_EP_BULKIN (USB_EP_EPTXEN|USB_EP_EPCONDIS|USB_EP_EPHSHK)
|
|
|
|
#define PIC32MX_EP_BULKOUT (USB_EP_EPRXEN|USB_EP_EPCONDIS|USB_EP_EPHSHK)
|
|
|
|
#define PIC32MX_EP_INTIN (USB_EP_EPTXEN|USB_EP_EPCONDIS|USB_EP_EPHSHK)
|
|
|
|
#define PIC32MX_EP_INTOUT (USB_EP_EPRXEN|USB_EP_EPCONDIS|USB_EP_EPHSHK)
|
|
|
|
#define PIC32MX_EP_ISOCIN (USB_EP_EPTXEN|USB_EP_EPCONDIS)
|
|
|
|
#define PIC32MX_EP_ISOCOUT (USB_EP_EPRXEN|USB_EP_EPCONDIS)
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* USB-related masks */
|
|
|
|
|
|
|
|
#define REQRECIPIENT_MASK (USB_REQ_TYPE_MASK | USB_REQ_RECIPIENT_MASK)
|
|
|
|
|
|
|
|
/* Request queue operations *************************************************/
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
#define pic32mx_rqempty(ep) ((ep)->head == NULL)
|
|
|
|
#define pic32mx_rqpeek(ep) ((ep)->head)
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* USB trace ****************************************************************/
|
|
|
|
/* Trace error codes */
|
|
|
|
|
|
|
|
#define PIC32MX_TRACEERR_ALLOCFAIL 0x0001
|
|
|
|
#define PIC32MX_TRACEERR_BADCLEARFEATURE 0x0002
|
|
|
|
#define PIC32MX_TRACEERR_BADDEVGETSTATUS 0x0003
|
|
|
|
#define PIC32MX_TRACEERR_BADEPGETSTATUS 0x0004
|
|
|
|
#define PIC32MX_TRACEERR_BADEPNO 0x0005
|
|
|
|
#define PIC32MX_TRACEERR_BADEPTYPE 0x0006
|
|
|
|
#define PIC32MX_TRACEERR_BADGETCONFIG 0x0007
|
|
|
|
#define PIC32MX_TRACEERR_BADGETSETDESC 0x0008
|
|
|
|
#define PIC32MX_TRACEERR_BADGETSTATUS 0x0009
|
|
|
|
#define PIC32MX_TRACEERR_BADSETADDRESS 0x000a
|
|
|
|
#define PIC32MX_TRACEERR_BADSETCONFIG 0x000b
|
|
|
|
#define PIC32MX_TRACEERR_BADSETFEATURE 0x000c
|
|
|
|
#define PIC32MX_TRACEERR_BINDFAILED 0x000d
|
|
|
|
#define PIC32MX_TRACEERR_DISPATCHSTALL 0x000e
|
|
|
|
#define PIC32MX_TRACEERR_DRIVER 0x000f
|
|
|
|
#define PIC32MX_TRACEERR_DRIVERREGISTERED 0x0010
|
2011-12-30 01:09:43 +01:00
|
|
|
#define PIC32MX_TRACEERR_EP0SETUPSTALLED 0x0011
|
|
|
|
#define PIC32MX_TRACEERR_EPDISABLED 0x0012
|
|
|
|
#define PIC32MX_TRACEERR_EPOUTNULLPACKET 0x0013
|
|
|
|
#define PIC32MX_TRACEERR_EPRESERVE 0x0014
|
|
|
|
#define PIC32MX_TRACEERR_INVALIDCTRLREQ 0x0015
|
|
|
|
#define PIC32MX_TRACEERR_INVALIDPARMS 0x0016
|
|
|
|
#define PIC32MX_TRACEERR_IRQREGISTRATION 0x0017
|
|
|
|
#define PIC32MX_TRACEERR_NOTCONFIGURED 0x0018
|
|
|
|
#define PIC32MX_TRACEERR_REQABORTED 0x0019
|
|
|
|
#define PIC32MX_TRACEERR_INVALIDSTATE 0x001a
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Trace interrupt codes */
|
|
|
|
|
|
|
|
#define PIC32MX_TRACEINTID_CLEARFEATURE 0x0001
|
|
|
|
#define PIC32MX_TRACEINTID_DEVGETSTATUS 0x0002
|
|
|
|
#define PIC32MX_TRACEINTID_DISPATCH 0x0003
|
|
|
|
#define PIC32MX_TRACEINTID_EP0IN 0x0004
|
|
|
|
#define PIC32MX_TRACEINTID_EP0INDONE 0x0005
|
|
|
|
#define PIC32MX_TRACEINTID_EP0OUTDONE 0x0006
|
|
|
|
#define PIC32MX_TRACEINTID_EP0SETUPDONE 0x0007
|
|
|
|
#define PIC32MX_TRACEINTID_EP0SETUPSETADDRESS 0x0008
|
|
|
|
#define PIC32MX_TRACEINTID_EPGETSTATUS 0x0009
|
|
|
|
#define PIC32MX_TRACEINTID_EPINDONE 0x000a
|
|
|
|
#define PIC32MX_TRACEINTID_EPINQEMPTY 0x000b
|
|
|
|
#define PIC32MX_TRACEINTID_EPOUTDONE 0x000c
|
|
|
|
#define PIC32MX_TRACEINTID_EPOUTPENDING 0x000d
|
|
|
|
#define PIC32MX_TRACEINTID_EPOUTQEMPTY 0x000e
|
2011-12-28 20:40:37 +01:00
|
|
|
#define PIC32MX_TRACEINTID_SOF 0x000f
|
2011-12-28 01:49:48 +01:00
|
|
|
#define PIC32MX_TRACEINTID_GETCONFIG 0x0010
|
|
|
|
#define PIC32MX_TRACEINTID_GETSETDESC 0x0011
|
|
|
|
#define PIC32MX_TRACEINTID_GETSETIF 0x0012
|
|
|
|
#define PIC32MX_TRACEINTID_GETSTATUS 0x0013
|
2011-12-30 01:09:43 +01:00
|
|
|
#define PIC32MX_TRACEINTID_IFGETSTATUS 0x0014
|
|
|
|
#define PIC32MX_TRACEINTID_TRNC 0x0015
|
|
|
|
#define PIC32MX_TRACEINTID_INTERRUPT 0x0016
|
|
|
|
#define PIC32MX_TRACEINTID_NOSTDREQ 0x0017
|
|
|
|
#define PIC32MX_TRACEINTID_RESET 0x0018
|
|
|
|
#define PIC32MX_TRACEINTID_SETCONFIG 0x0019
|
|
|
|
#define PIC32MX_TRACEINTID_SETFEATURE 0x001a
|
|
|
|
#define PIC32MX_TRACEINTID_IDLE 0x001b
|
|
|
|
#define PIC32MX_TRACEINTID_SYNCHFRAME 0x001c
|
|
|
|
#define PIC32MX_TRACEINTID_WKUP 0x001d
|
|
|
|
#define PIC32MX_TRACEINTID_T1MSEC 0x001e
|
|
|
|
#define PIC32MX_TRACEINTID_OTGID 0x001f
|
|
|
|
#define PIC32MX_TRACEINTID_STALL 0x0020
|
|
|
|
#define PIC32MX_TRACEINTID_UERR 0x0021
|
2012-01-07 14:35:58 +01:00
|
|
|
#define PIC32MX_TRACEINTID_SUSPENDED 0x0022
|
|
|
|
#define PIC32MX_TRACEINTID_WAITRESET 0x0023
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Misc Helper Macros *******************************************************/
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
#define PHYS_ADDR(va) ((uint32_t)(va) & 0x1fffffff)
|
|
|
|
#define VIRT_ADDR(pa) (KSEG1_BASE | (uint32_t)(pa))
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Ever-present MIN and MAX macros */
|
|
|
|
|
|
|
|
#ifndef MIN
|
|
|
|
# define MIN(a,b) (a < b ? a : b)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef MAX
|
|
|
|
# define MAX(a,b) (a > b ? a : b)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Byte ordering in host-based values */
|
|
|
|
|
|
|
|
#ifdef CONFIG_ENDIAN_BIG
|
|
|
|
# define LSB 1
|
|
|
|
# define MSB 0
|
|
|
|
#else
|
|
|
|
# define LSB 0
|
|
|
|
# define MSB 1
|
|
|
|
#endif
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Debug ********************************************************************/
|
|
|
|
|
|
|
|
#ifdef CONFIG_PIC32MX_USBDEV_REGDEBUG
|
|
|
|
# define regdbg lldbg
|
|
|
|
# ifdef CONFIG_DEBUG_VERBOSE
|
|
|
|
# define regvdbg lldbg
|
|
|
|
# else
|
|
|
|
# define regvdbg(x...)
|
|
|
|
# endif
|
|
|
|
#else
|
|
|
|
# define pic32mx_getreg(addr) getreg16(addr)
|
|
|
|
# define pic32mx_putreg(val,addr) putreg16(val,addr)
|
|
|
|
# define regdbg(x...)
|
|
|
|
# define regvdbg(x...)
|
|
|
|
#endif
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Private Type Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* Overvall device state */
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
enum pic32mx_devstate_e
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
DEVSTATE_DETACHED = 0, /* Not connected to a host */
|
|
|
|
DEVSTATE_ATTACHED, /* Connected to a host */
|
|
|
|
DEVSTATE_POWERED, /* Powered */
|
|
|
|
DEVSTATE_DEFAULT, /* Default state */
|
|
|
|
DEVSTATE_ADDRPENDING, /* Waiting for an address */
|
|
|
|
DEVSTATE_ADDRESS, /* Address received */
|
|
|
|
DEVSTATE_CONFIGURED, /* Configuration received */
|
2011-12-28 20:40:37 +01:00
|
|
|
};
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* The various states of the control pipe */
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
enum pic32mx_ctrlstate_e
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 16:55:02 +01:00
|
|
|
CTRLSTATE_WAITSETUP = 0, /* No request in progress, waiting for setup */
|
|
|
|
CTRLSTATE_RDREQUEST, /* Read request (OUT) in progress */
|
|
|
|
CTRLSTATE_WRREQUEST, /* Write request (IN) in progress */
|
2011-12-28 01:49:48 +01:00
|
|
|
CTRLSTATE_STALLED /* We are stalled */
|
|
|
|
};
|
|
|
|
|
|
|
|
union wb_u
|
|
|
|
{
|
|
|
|
uint16_t w;
|
|
|
|
uint8_t b[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
/* A container for a request so that the request make be retained in a list */
|
|
|
|
|
|
|
|
struct pic32mx_req_s
|
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
struct usbdev_req_s req; /* Standard USB request */
|
2011-12-28 20:40:37 +01:00
|
|
|
struct pic32mx_req_s *flink; /* Supports a singly linked list */
|
2011-12-28 01:49:48 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/* This is the internal representation of an endpoint */
|
|
|
|
|
|
|
|
struct pic32mx_ep_s
|
|
|
|
{
|
|
|
|
/* Common endpoint fields. This must be the first thing defined in the
|
|
|
|
* structure so that it is possible to simply cast from struct usbdev_ep_s
|
|
|
|
* to struct pic32mx_ep_s.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct usbdev_ep_s ep; /* Standard endpoint structure */
|
|
|
|
|
|
|
|
/* PIC32MX-specific fields */
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
struct pic32mx_usbdev_s *dev; /* Reference to private driver data */
|
|
|
|
struct pic32mx_req_s *head; /* Request list for this endpoint */
|
|
|
|
struct pic32mx_req_s *tail;
|
|
|
|
uint8_t stalled:1; /* true: Endpoint is stalled */
|
|
|
|
uint8_t halted:1; /* true: Endpoint feature halted */
|
|
|
|
uint8_t txbusy:1; /* true: TX endpoint FIFO full */
|
|
|
|
uint8_t txnullpkt:1; /* Null packet needed at end of transfer */
|
2011-12-29 01:33:02 +01:00
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
uint8_t ep0ready:1 /* EP0 OUT already prepared */
|
|
|
|
#endif
|
2011-12-28 20:40:37 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdtin; /* BDT entry for the IN transaction*/
|
|
|
|
volatile struct usbotg_bdtentry_s *bdtout; /* BDT entry for the OUT transaction */
|
2011-12-28 01:49:48 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s
|
|
|
|
{
|
|
|
|
/* Common device fields. This must be the first thing defined in the
|
|
|
|
* structure so that it is possible to simply cast from struct usbdev_s
|
|
|
|
* to structpic32mx_usbdev_s.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct usbdev_s usbdev;
|
|
|
|
|
|
|
|
/* The bound device class driver */
|
|
|
|
|
|
|
|
struct usbdevclass_driver_s *driver;
|
|
|
|
|
|
|
|
/* PIC32MX-specific fields */
|
|
|
|
|
|
|
|
struct usb_ctrlreq_s ctrl; /* Last EP0 request */
|
|
|
|
uint8_t devstate; /* Driver state (see enum pic32mx_devstate_e) */
|
|
|
|
uint8_t ctrlstate; /* Control EP state (see enum pic32mx_ctrlstate_e) */
|
|
|
|
uint8_t selfpowered:1; /* 1: Device is self powered */
|
2011-12-28 20:40:37 +01:00
|
|
|
uint8_t rwakeup:1; /* 1: Device supports remote wakeup */
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t epavail; /* Bitset of available endpoints */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* The endpoint list */
|
|
|
|
|
|
|
|
struct pic32mx_ep_s eplist[PIC32MX_NENDPOINTS];
|
|
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Function Prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* Register operations ******************************************************/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
#ifdef CONFIG_PIC32MX_USBDEV_REGDEBUG
|
2011-12-28 01:49:48 +01:00
|
|
|
static uint16_t pic32mx_getreg(uint32_t addr);
|
|
|
|
static void pic32mx_putreg(uint16_t val, uint32_t addr);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Suspend/Resume Helpers ***************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_suspend(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_resume(struct pic32mx_usbdev_s *priv);
|
|
|
|
|
|
|
|
/* Request Helpers **********************************************************/
|
|
|
|
|
|
|
|
static struct pic32mx_req_s *
|
|
|
|
pic32mx_rqdequeue(struct pic32mx_ep_s *privep);
|
|
|
|
static void pic32mx_rqenqueue(struct pic32mx_ep_s *privep,
|
|
|
|
struct pic32mx_req_s *req);
|
|
|
|
static inline void
|
|
|
|
pic32mx_abortrequest(struct pic32mx_ep_s *privep,
|
|
|
|
struct pic32mx_req_s *privreq, int16_t result);
|
|
|
|
static void pic32mx_reqcomplete(struct pic32mx_ep_s *privep, int16_t result);
|
2011-12-29 01:33:02 +01:00
|
|
|
static void pic32mx_epwrite(struct pic32mx_ep_s *privep,
|
|
|
|
const uint8_t *src, uint32_t nbytes);
|
2011-12-28 01:49:48 +01:00
|
|
|
static int pic32mx_wrrequest(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep);
|
2011-12-29 01:33:02 +01:00
|
|
|
static int pic32mx_rdcomplete(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep);
|
2011-12-29 16:55:02 +01:00
|
|
|
static int pic32mx_ep0rdsetup(struct pic32mx_usbdev_s *priv,
|
|
|
|
uint8_t *dest, int readlen);
|
|
|
|
static int pic32mx_rdsetup(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep, uint8_t *dest, int readlen);
|
2011-12-28 01:49:48 +01:00
|
|
|
static int pic32mx_rdrequest(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep);
|
|
|
|
static void pic32mx_cancelrequests(struct pic32mx_ep_s *privep);
|
|
|
|
|
|
|
|
/* Interrupt level processing ***********************************************/
|
|
|
|
|
|
|
|
static void pic32mx_dispatchrequest(struct pic32mx_usbdev_s *priv);
|
2011-12-28 20:40:37 +01:00
|
|
|
static void pic32mx_ep0stall(struct pic32mx_usbdev_s *priv);
|
2011-12-30 01:09:43 +01:00
|
|
|
static void pic32mx_eptransfer(struct pic32mx_usbdev_s *priv, uint8_t epno,
|
|
|
|
uint16_t status);
|
2011-12-29 21:46:38 +01:00
|
|
|
static void pic32mx_ep0nextsetup(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_ep0done(struct pic32mx_usbdev_s *priv,
|
|
|
|
uint8_t *response, int nbytes);
|
2011-12-28 01:49:48 +01:00
|
|
|
static void pic32mx_ep0setup(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_ep0out(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_ep0in(struct pic32mx_usbdev_s *priv);
|
2011-12-28 20:40:37 +01:00
|
|
|
static void pic32mx_ep0transfer(struct pic32mx_usbdev_s *priv, uint16_t status);
|
2011-12-28 01:49:48 +01:00
|
|
|
static int pic32mx_interrupt(int irq, void *context);
|
|
|
|
|
|
|
|
/* Endpoint helpers *********************************************************/
|
|
|
|
|
|
|
|
static inline struct pic32mx_ep_s *
|
|
|
|
pic32mx_epreserve(struct pic32mx_usbdev_s *priv, uint8_t epset);
|
|
|
|
static inline void
|
|
|
|
pic32mx_epunreserve(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep);
|
|
|
|
static inline bool
|
|
|
|
pic32mx_epreserved(struct pic32mx_usbdev_s *priv, int epno);
|
|
|
|
|
|
|
|
/* Endpoint operations ******************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epconfigure(struct usbdev_ep_s *ep,
|
|
|
|
const struct usb_epdesc_s *desc, bool last);
|
|
|
|
static int pic32mx_epdisable(struct usbdev_ep_s *ep);
|
|
|
|
static struct usbdev_req_s *
|
|
|
|
pic32mx_epallocreq(struct usbdev_ep_s *ep);
|
|
|
|
static void pic32mx_epfreereq(struct usbdev_ep_s *ep,
|
|
|
|
struct usbdev_req_s *);
|
|
|
|
static int pic32mx_epsubmit(struct usbdev_ep_s *ep,
|
|
|
|
struct usbdev_req_s *req);
|
|
|
|
static int pic32mx_epcancel(struct usbdev_ep_s *ep,
|
|
|
|
struct usbdev_req_s *req);
|
2011-12-29 21:46:38 +01:00
|
|
|
static int pic32mx_epbdtstall(struct usbdev_ep_s *ep,
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt, bool resume,
|
|
|
|
bool epin);
|
2011-12-28 01:49:48 +01:00
|
|
|
static int pic32mx_epstall(struct usbdev_ep_s *ep, bool resume);
|
|
|
|
|
|
|
|
/* USB device controller operations *****************************************/
|
|
|
|
|
|
|
|
static struct usbdev_ep_s *
|
|
|
|
pic32mx_allocep(struct usbdev_s *dev, uint8_t epno, bool in,
|
|
|
|
uint8_t eptype);
|
|
|
|
static void pic32mx_freeep(struct usbdev_s *dev, struct usbdev_ep_s *ep);
|
|
|
|
static int pic32mx_getframe(struct usbdev_s *dev);
|
|
|
|
static int pic32mx_wakeup(struct usbdev_s *dev);
|
|
|
|
static int pic32mx_selfpowered(struct usbdev_s *dev, bool selfpowered);
|
|
|
|
|
|
|
|
/* Initialization/Reset *****************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_reset(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_attach(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_detach(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_hwreset(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_hwsetup(struct pic32mx_usbdev_s *priv);
|
|
|
|
static void pic32mx_hwshutdown(struct pic32mx_usbdev_s *priv);
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* Since there is only a single USB interface, all status information can be
|
|
|
|
* be simply retained in a single global instance.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static struct pic32mx_usbdev_s g_usbdev;
|
|
|
|
|
|
|
|
static const struct usbdev_epops_s g_epops =
|
|
|
|
{
|
|
|
|
.configure = pic32mx_epconfigure,
|
|
|
|
.disable = pic32mx_epdisable,
|
|
|
|
.allocreq = pic32mx_epallocreq,
|
|
|
|
.freereq = pic32mx_epfreereq,
|
|
|
|
.submit = pic32mx_epsubmit,
|
|
|
|
.cancel = pic32mx_epcancel,
|
|
|
|
.stall = pic32mx_epstall,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct usbdev_ops_s g_devops =
|
|
|
|
{
|
|
|
|
.allocep = pic32mx_allocep,
|
|
|
|
.freeep = pic32mx_freeep,
|
|
|
|
.getframe = pic32mx_getframe,
|
|
|
|
.wakeup = pic32mx_wakeup,
|
|
|
|
.selfpowered = pic32mx_selfpowered,
|
|
|
|
.pullup = pic32mx_usbpullup,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Buffer Descriptor Table */
|
|
|
|
|
|
|
|
#ifndef CONFIG_USB_PINGPONG
|
|
|
|
volatile struct usbotg_bdtentry_s
|
|
|
|
g_bdt[(PIC32MX_NENDPOINTS + 1) * 2] __attribute__ ((aligned(512)));
|
|
|
|
#else
|
|
|
|
volatile struct usbotg_bdtentry_s
|
|
|
|
g_bdt[(PIC32MX_NENDPOINTS + 1) * 4] __attribute__ ((aligned(512)));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Private Functions
|
|
|
|
****************************************************************************/
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Register Operations
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_getreg
|
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
#ifdef CONFIG_PIC32MX_USBDEV_REGDEBUG
|
2011-12-28 01:49:48 +01:00
|
|
|
static uint16_t pic32mx_getreg(uint32_t addr)
|
|
|
|
{
|
|
|
|
static uint32_t prevaddr = 0;
|
|
|
|
static uint16_t preval = 0;
|
|
|
|
static uint32_t count = 0;
|
|
|
|
|
|
|
|
/* Read the value from the register */
|
|
|
|
|
|
|
|
uint16_t val = getreg16(addr);
|
|
|
|
|
|
|
|
/* Is this the same value that we read from the same register last time?
|
|
|
|
* Are we polling the register? If so, suppress some of the output.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (addr == prevaddr && val == preval)
|
|
|
|
{
|
|
|
|
if (count == 0xffffffff || ++count > 3)
|
|
|
|
{
|
|
|
|
if (count == 4)
|
|
|
|
{
|
|
|
|
lldbg("...\n");
|
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No this is a new address or value */
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Did we print "..." for the previous value? */
|
|
|
|
|
|
|
|
if (count > 3)
|
|
|
|
{
|
|
|
|
/* Yes.. then show how many times the value repeated */
|
|
|
|
|
|
|
|
lldbg("[repeats %d more times]\n", count-3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Save the new address, value, and count */
|
|
|
|
|
|
|
|
prevaddr = addr;
|
|
|
|
preval = val;
|
|
|
|
count = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Show the register value read */
|
|
|
|
|
|
|
|
lldbg("%08x->%04x\n", addr, val);
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_putreg
|
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
#ifdef CONFIG_PIC32MX_USBDEV_REGDEBUG
|
2011-12-28 01:49:48 +01:00
|
|
|
static void pic32mx_putreg(uint16_t val, uint32_t addr)
|
|
|
|
{
|
|
|
|
/* Show the register value being written */
|
|
|
|
|
|
|
|
lldbg("%08x<-%04x\n", addr, val);
|
|
|
|
|
|
|
|
/* Write the value */
|
|
|
|
|
|
|
|
putreg16(val, addr);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Request Helpers
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_rqdequeue
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static struct pic32mx_req_s *pic32mx_rqdequeue(struct pic32mx_ep_s *privep)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *ret = privep->head;
|
|
|
|
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
privep->head = ret->flink;
|
|
|
|
if (!privep->head)
|
|
|
|
{
|
|
|
|
privep->tail = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret->flink = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_rqenqueue
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_rqenqueue(struct pic32mx_ep_s *privep, struct pic32mx_req_s *req)
|
|
|
|
{
|
|
|
|
req->flink = NULL;
|
|
|
|
if (!privep->head)
|
|
|
|
{
|
|
|
|
privep->head = req;
|
|
|
|
privep->tail = req;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
privep->tail->flink = req;
|
|
|
|
privep->tail = req;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_abortrequest
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
pic32mx_abortrequest(struct pic32mx_ep_s *privep, struct pic32mx_req_s *privreq, int16_t result)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_REQABORTED), (uint16_t)USB_EPNO(privep->ep.eplog));
|
|
|
|
|
|
|
|
/* Save the result in the request structure */
|
|
|
|
|
|
|
|
privreq->req.result = result;
|
|
|
|
|
|
|
|
/* Callback to the request completion handler */
|
|
|
|
|
|
|
|
privreq->req.callback(&privep->ep, &privreq->req);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_reqcomplete
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_reqcomplete(struct pic32mx_ep_s *privep, int16_t result)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq;
|
|
|
|
irqstate_t flags;
|
|
|
|
|
|
|
|
/* Remove the completed request at the head of the endpoint request list */
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
privreq = pic32mx_rqdequeue(privep);
|
|
|
|
irqrestore(flags);
|
|
|
|
|
|
|
|
if (privreq)
|
|
|
|
{
|
|
|
|
/* If endpoint 0, temporarily reflect the state of protocol stalled
|
|
|
|
* in the callback.
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool stalled = privep->stalled;
|
|
|
|
if (USB_EPNO(privep->ep.eplog) == EP0)
|
|
|
|
{
|
|
|
|
privep->stalled = (privep->dev->ctrlstate == CTRLSTATE_STALLED);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Save the result in the request structure */
|
|
|
|
|
|
|
|
privreq->req.result = result;
|
|
|
|
|
|
|
|
/* Callback to the request completion handler */
|
|
|
|
|
|
|
|
privreq->flink = NULL;
|
|
|
|
privreq->req.callback(&privep->ep, &privreq->req);
|
|
|
|
|
|
|
|
/* Restore the stalled indication */
|
|
|
|
|
|
|
|
privep->stalled = stalled;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2011-12-29 01:33:02 +01:00
|
|
|
* Name: pic32mx_epwrite
|
2011-12-28 01:49:48 +01:00
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
static void pic32mx_epwrite(struct pic32mx_ep_s *privep,
|
|
|
|
const uint8_t *src, uint32_t nbytes)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdt = privep->bdtin;
|
2011-12-29 21:46:38 +01:00
|
|
|
uint32_t status;
|
2011-12-29 01:33:02 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
usbtrace(TRACE_WRITE(USB_EPNO(privep->ep.eplog)), nbytes);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Clear status and toggle the DTS bit if required */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status & USB_BDT_DATA01;
|
2011-12-29 01:33:02 +01:00
|
|
|
#ifndef CONFIG_USB_PINGPONG
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_BDT_DATA01;
|
2011-12-29 01:33:02 +01:00
|
|
|
#endif
|
2011-12-29 21:46:38 +01:00
|
|
|
bdt->status = status;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
/* Set the data pointer, data length, and enable the endpoint */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
bdt->addr = (uint8_t *)PHYS_ADDR(src);
|
2011-12-29 01:33:02 +01:00
|
|
|
|
|
|
|
/* Set the data length and give the BDT to USB. Preserving
|
|
|
|
* only the data toggle.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (nbytes << USB_BDT_BYTECOUNT_SHIFT) | USB_BDT_DTS;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
/* Point to the next ping pong buffer. */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_NEXT_PINGPONG;
|
2011-12-29 01:33:02 +01:00
|
|
|
#endif
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* And, finally, give the BDT to the USB */
|
|
|
|
|
|
|
|
status |= USB_BDT_UOWN;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT IN {%08x, %08x}\n",
|
|
|
|
USB_EPNO(privep->ep.eplog), status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Indicate that there is data in the TX packet memory. This will be cleared
|
|
|
|
* when the next data out interrupt is received.
|
|
|
|
*/
|
|
|
|
|
|
|
|
privep->txbusy = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_wrrequest
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_wrrequest(struct pic32mx_usbdev_s *priv, struct pic32mx_ep_s *privep)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq;
|
|
|
|
uint8_t *buf;
|
|
|
|
uint8_t epno;
|
|
|
|
int nbytes;
|
|
|
|
int bytesleft;
|
|
|
|
|
|
|
|
/* We get here when an IN endpoint interrupt occurs. So now we know that
|
|
|
|
* there is no TX transfer in progress.
|
|
|
|
*/
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
privep->txbusy = false;
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Get the endpoint number that we are servicing */
|
|
|
|
|
|
|
|
epno = USB_EPNO(privep->ep.eplog);
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Check the request from the head of the endpoint request queue */
|
|
|
|
|
|
|
|
privreq = pic32mx_rqpeek(privep);
|
|
|
|
if (!privreq)
|
|
|
|
{
|
|
|
|
/* There is no TX transfer in progress and no new pending TX
|
|
|
|
* requests to send.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPINQEMPTY), epno);
|
|
|
|
|
|
|
|
/* Special case endpoint zero. If there is nothing to be sent, then
|
|
|
|
* we need to configure to received the next SETUP packet.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (epno == 0)
|
|
|
|
{
|
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
ullvdbg("epno=%d req=%p: len=%d xfrd=%d nullpkt=%d\n",
|
|
|
|
epno, privreq, privreq->req.len, privreq->req.xfrd, privep->txnullpkt);
|
|
|
|
|
|
|
|
/* Get the number of bytes left to be sent in the packet */
|
|
|
|
|
|
|
|
bytesleft = privreq->req.len - privreq->req.xfrd;
|
|
|
|
nbytes = bytesleft;
|
|
|
|
|
|
|
|
/* Send the next packet */
|
|
|
|
|
|
|
|
if (nbytes > 0)
|
|
|
|
{
|
|
|
|
/* Either send the maxpacketsize or all of the remaining data in
|
|
|
|
* the request.
|
|
|
|
*/
|
|
|
|
|
|
|
|
privep->txnullpkt = 0;
|
|
|
|
if (nbytes >= privep->ep.maxpacket)
|
|
|
|
{
|
|
|
|
nbytes = privep->ep.maxpacket;
|
|
|
|
|
|
|
|
/* Handle the case where this packet is exactly the
|
|
|
|
* maxpacketsize. Do we need to send a zero-length packet
|
|
|
|
* in this case?
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (bytesleft == privep->ep.maxpacket &&
|
|
|
|
(privreq->req.flags & USBDEV_REQFLAGS_NULLPKT) != 0)
|
|
|
|
{
|
|
|
|
privep->txnullpkt = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Send the packet (might be a null packet nbytes == 0) */
|
|
|
|
|
|
|
|
buf = privreq->req.buf + privreq->req.xfrd;
|
2011-12-30 01:09:43 +01:00
|
|
|
|
|
|
|
/* Setup the writes to the endpoints */
|
|
|
|
|
|
|
|
pic32mx_epwrite(privep, buf, nbytes);
|
|
|
|
|
|
|
|
/* Special case endpoint 0 state information. The write request is in
|
|
|
|
* progress.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
if (epno == 0)
|
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_WRREQUEST;
|
2011-12-29 01:33:02 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Update for the next data IN interrupt */
|
|
|
|
|
|
|
|
privreq->req.xfrd += nbytes;
|
|
|
|
bytesleft = privreq->req.len - privreq->req.xfrd;
|
|
|
|
|
|
|
|
/* If all of the bytes were sent (including any final null packet)
|
|
|
|
* then we are finished with the transfer
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (bytesleft == 0 && !privep->txnullpkt)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_COMPLETE(USB_EPNO(privep->ep.eplog)), privreq->req.xfrd);
|
|
|
|
privep->txnullpkt = 0;
|
|
|
|
pic32mx_reqcomplete(privep, OK);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
/* Special case writes to endpoint zero. If there is no transfer in
|
|
|
|
* progress, then we need to configure to received the next SETUP packet.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (epno == 0)
|
|
|
|
{
|
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_rdcomplete
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_rdcomplete(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep)
|
|
|
|
{
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt = privep->bdtout;
|
|
|
|
struct pic32mx_req_s *privreq;
|
|
|
|
int readlen;
|
|
|
|
|
|
|
|
/* Check the request from the head of the endpoint request queue */
|
|
|
|
|
|
|
|
privreq = pic32mx_rqpeek(privep);
|
|
|
|
if (!privreq)
|
|
|
|
{
|
|
|
|
/* There is no packet to receive any data. Then why are we here? */
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPOUTQEMPTY),
|
|
|
|
USB_EPNO(privep->ep.eplog));
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
ullvdbg("EP%d: len=%d xfrd=%d BDT={%08x, %08x}\n",
|
|
|
|
USB_EPNO(privep->ep.eplog), privreq->req.len, privreq->req.xfrd,
|
|
|
|
bdt->status, bdt->addr);
|
2011-12-29 01:33:02 +01:00
|
|
|
|
|
|
|
/* Get the length of the data received from the BDT */
|
|
|
|
|
|
|
|
readlen = (bdt->status & USB_BDT_BYTECOUNT_MASK) >> USB_BDT_BYTECOUNT_SHIFT;
|
|
|
|
|
|
|
|
/* If the receive buffer is full or this is a partial packet,
|
|
|
|
* then we are finished with the transfer
|
|
|
|
*/
|
|
|
|
|
|
|
|
privreq->req.xfrd += readlen;
|
|
|
|
if (readlen < privep->ep.maxpacket || privreq->req.xfrd >= privreq->req.len)
|
|
|
|
{
|
|
|
|
/* Complete the transfer and mark the state IDLE. The endpoint
|
|
|
|
* RX will be marked valid when the data phase completes.
|
|
|
|
*/
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_COMPLETE(USB_EPNO(privep->ep.eplog)), privreq->req.xfrd);
|
2011-12-29 01:33:02 +01:00
|
|
|
pic32mx_reqcomplete(privep, OK);
|
|
|
|
}
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
/* Set up the next read operation */
|
|
|
|
|
|
|
|
return pic32mx_rdrequest(priv, privep);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0rdsetup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_ep0rdsetup(struct pic32mx_usbdev_s *priv, uint8_t *dest,
|
|
|
|
int readlen)
|
|
|
|
{
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt = priv->eplist[EP0].bdtout;
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Clear status bits (making sure that UOWN is cleared before doing anything
|
|
|
|
* else.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status & ~(USB_BDT_UOWN|USB_BDT_BYTECOUNT_MASK|USB_BDT_DATA01);
|
|
|
|
bdt->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Set the data pointer, data length, and enable the endpoint */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
bdt->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Select data0 or data 1 when enabling */
|
2011-12-29 01:33:02 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
if ((status & USB_BDT_DATA01) == USB_BDT_DATA0)
|
2011-12-29 16:55:02 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA1 | USB_BDT_DTS);
|
2011-12-29 16:55:02 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA0 | USB_BDT_DTS);
|
2011-12-29 16:55:02 +01:00
|
|
|
}
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_RDREQUEST;
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_rdsetup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_rdsetup(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep, uint8_t *dest, int readlen)
|
|
|
|
{
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt = privep->bdtout;
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Clear status bits (making sure that UOWN is cleared before doing anything
|
|
|
|
* else). The DATA01 is (only) is preserved.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status & USB_BDT_DATA01;
|
|
|
|
bdt->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Toggle the DATA01 bit if required for ping pong support */
|
|
|
|
|
|
|
|
#ifndef CONFIG_USB_PINGPONG
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_BDT_DATA01;
|
2011-12-29 16:55:02 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Set the data pointer, data length, and enable the endpoint */
|
|
|
|
|
|
|
|
bdt->addr = (uint8_t *)PHYS_ADDR(dest);
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Set the data length (preserving the data toggle). */
|
2011-12-29 16:55:02 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (readlen << USB_BDT_BYTECOUNT_SHIFT) | USB_BDT_DTS;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
/* Point to the next ping pong buffer. */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_NEXT_PINGPONG;
|
2011-12-29 16:55:02 +01:00
|
|
|
#endif
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
status |= USB_BDT_UOWN;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n",
|
|
|
|
USB_EPNO(privep->ep.eplog), status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
return OK;
|
2011-12-29 01:33:02 +01:00
|
|
|
}
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_rdrequest
|
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
static int pic32mx_rdrequest(struct pic32mx_usbdev_s *priv,
|
|
|
|
struct pic32mx_ep_s *privep)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq;
|
|
|
|
uint8_t *dest;
|
|
|
|
int readlen;
|
2011-12-29 16:55:02 +01:00
|
|
|
int ret;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Check the request from the head of the endpoint request queue */
|
|
|
|
|
|
|
|
privreq = pic32mx_rqpeek(privep);
|
|
|
|
if (!privreq)
|
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
/* There is no packet to receive any data. */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPOUTQEMPTY),
|
|
|
|
USB_EPNO(privep->ep.eplog));
|
2011-12-28 01:49:48 +01:00
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
ullvdbg("EP%d: len=%d xfrd=%d\n",
|
|
|
|
USB_EPNO(privep->ep.eplog), privreq->req.len, privreq->req.xfrd);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Ignore any attempt to receive a zero length packet */
|
|
|
|
|
|
|
|
if (privreq->req.len == 0)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_EPOUTNULLPACKET), 0);
|
|
|
|
pic32mx_reqcomplete(privep, OK);
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
usbtrace(TRACE_READ(USB_EPNO(privep->ep.eplog)), privreq->req.xfrd);
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
/* Get the destination transfer address and size */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
dest = privreq->req.buf + privreq->req.xfrd;
|
|
|
|
readlen = MIN(privreq->req.len, privep->ep.maxpacket);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
/* Handle EP0 in a few special ways */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
if (USB_EPNO(privep->ep.eplog) == EP0)
|
|
|
|
{
|
|
|
|
ret = pic32mx_ep0rdsetup(priv, dest, readlen);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = pic32mx_rdsetup(priv, privep, dest, readlen);
|
|
|
|
}
|
|
|
|
return ret;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_cancelrequests
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_cancelrequests(struct pic32mx_ep_s *privep)
|
|
|
|
{
|
|
|
|
while (!pic32mx_rqempty(privep))
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_COMPLETE(USB_EPNO(privep->ep.eplog)),
|
|
|
|
(pic32mx_rqpeek(privep))->req.xfrd);
|
|
|
|
pic32mx_reqcomplete(privep, -ESHUTDOWN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Interrupt Level Processing
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_dispatchrequest
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_dispatchrequest(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_DISPATCH), 0);
|
|
|
|
if (priv && priv->driver)
|
|
|
|
{
|
|
|
|
/* Forward to the control request to the class driver implementation */
|
|
|
|
|
|
|
|
ret = CLASS_SETUP(priv->driver, &priv->usbdev, &priv->ctrl);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
/* Stall on failure */
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_DISPATCHSTALL), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0stall
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
static void pic32mx_ep0stall(struct pic32mx_usbdev_s *priv)
|
2011-12-28 20:40:37 +01:00
|
|
|
{
|
|
|
|
uint16_t regval;
|
|
|
|
|
|
|
|
/* Check if EP0 is stalled */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_EP0);
|
|
|
|
if ((regval & USB_EP_EPSTALL) != 0)
|
|
|
|
{
|
|
|
|
/* Check if we own the BDTs. All BDs of endpoint zero should be owned by
|
|
|
|
* the USB. Check anyway.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
2011-12-28 20:40:37 +01:00
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
if ((ep0->bdtout->status & USB_BDT_UOWN) != 0 &&
|
|
|
|
(ep0->bdtin->status & (USB_BDT_UOWN | USB_BDT_BSTALL)) == (USB_BDT_UOWN | USB_BDT_BSTALL))
|
2011-12-28 20:40:37 +01:00
|
|
|
{
|
|
|
|
/* Set ep0 BDT status to stall also */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status = (USB_BDT_UOWN| USB_BDT_DATA0 | USB_BDT_DTS | USB_BDT_BSTALL);
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", status, ep0->bdtout->addr);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
ep0->bdtout->status = status;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Then clear the EP0 stall status */
|
|
|
|
|
|
|
|
regval &= ~USB_EP_EPSTALL;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_EP0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_eptransfer
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
static void pic32mx_eptransfer(struct pic32mx_usbdev_s *priv, uint8_t epno,
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t status)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
/* Decode and service non control endpoints interrupt */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
privep = &priv->eplist[epno];
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Check if the last transaction was an EP0 OUT transaction */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
if ((status & USB_STAT_DIR) == USB_STAT_DIR_OUT)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
/* OUT: host-to-device */
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPOUTDONE), status);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Handle read requests. First check if a read request is available to
|
|
|
|
* accept the host data.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!pic32mx_rqempty(privep))
|
|
|
|
{
|
|
|
|
/* Read host data into the current read request */
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
pic32mx_rdcomplete(priv, privep);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* NAK further OUT packets if there there no more read requests */
|
|
|
|
|
|
|
|
if (pic32mx_rqempty(privep))
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPOUTPENDING), (uint16_t)epno);
|
2011-12-28 20:40:37 +01:00
|
|
|
#warning "Missing logic"
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
}
|
2011-12-30 01:09:43 +01:00
|
|
|
else
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
/* IN: device-to-host */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPINDONE), status);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
|
|
|
/* Handle write requests */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_wrrequest(priv, privep);
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0nextsetup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_ep0nextsetup(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt = ep0->bdtin;
|
|
|
|
uint16_t status;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
|
|
|
|
|
|
|
/* Don't need to do anything to EP0 OUT buffer descriptor table (BDT) here,
|
|
|
|
* as EP0 OUT next is already configured and ready to receive a SETUP
|
|
|
|
* transaction.
|
|
|
|
*/
|
|
|
|
|
|
|
|
status = bdt->status & ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
status ^= USB_NEXT_PINGPONG;
|
|
|
|
#endif
|
|
|
|
status &= ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
regdbg("EP0 BDT IN {%08x, %08x}\n", status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0done
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_ep0done(struct pic32mx_usbdev_s *priv,
|
|
|
|
uint8_t *response, int nbytes)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
|
|
|
volatile struct usbotg_bdtentry_s *bdtout;
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
volatile struct usbotg_bdtentry_s *bdtnext;
|
|
|
|
#endif
|
|
|
|
uint16_t status;
|
|
|
|
|
|
|
|
/* Which BDT are we working on new? Which one will be next */
|
|
|
|
|
|
|
|
bdtout = ep0->bdtout;
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
if (bdtout == &g_bdt[EP0_OUT_EVEN])
|
|
|
|
{
|
|
|
|
btdnext = &g_bdt[EP0_OUT_ODD]
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DEBUGASSERT(bdtout == &g_bdt[EP0_OUT_ODD]);
|
|
|
|
btdnext = &g_bdt[EP0_OUT_EVEN]
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Send the EP0 SETUP response */
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
pic32mx_epwrite(&priv->eplist[EP0], response, nbytes);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
/* Prepare OUT EP to respond to early termination NOTE: If
|
|
|
|
* something went wrong during the control transfer, the last
|
|
|
|
* status stage may not be sent by the host. When this happens,
|
|
|
|
* two different things could happen depending on the host. a)
|
|
|
|
* The host could send out a RESET. b) The host could send out
|
|
|
|
* a new SETUP transaction without sending a RESET first. To
|
|
|
|
* properly handle case (b), the OUT EP must be setup to
|
|
|
|
* receive either a zero length OUT transaction, or a new
|
|
|
|
* SETUP transaction. Furthermore, the Cnt byte should be set
|
|
|
|
* to prepare for the SETUP data (8-byte or more), and the
|
|
|
|
* buffer address should be pointed to priv->ctrl.
|
|
|
|
*/
|
|
|
|
|
|
|
|
status = bdtout->status & ~(USB_BDT_BYTECOUNT_MASK | USB_BDT_DTS);
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
|
|
|
status |= USB_BDT_UOWN; /* Note: DTSEN is 0 */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
bdtout->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
regdbg("EP0 BDT OUT (Next) {%08x, %08x}\n", status, bdtout->addr);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
bdtout->status = status;
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
status = btdnext->status & ~USB_BDT_BYTECOUNT_MASK;
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
|
|
|
status |= USB_BDT_UOWN | USB_BDT_BSTALL;
|
|
|
|
|
|
|
|
btdnext->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", status, btdnext->addr);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
btdnext->status = status;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Prepare IN EP to transfer data */
|
|
|
|
|
|
|
|
status = ep0->bdtin->status & ~(USB_BDT_BYTECOUNT_MASK | USB_BDT_DATA01);
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
|
|
|
status |= USB_BDT_UOWN | USB_BDT_DATA1 | USB_BDT_DTS;
|
|
|
|
|
|
|
|
ep0->bdtin->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
regdbg("EP0 BDT IN {%08x, %08x}\n", status, ep0->bdtin->addr);
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
ep0->bdtin->status = status;
|
|
|
|
}
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0setup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_ep0setup(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdt;
|
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
|
|
|
struct pic32mx_req_s *privreq = pic32mx_rqpeek(ep0);
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
union wb_u value;
|
|
|
|
union wb_u index;
|
|
|
|
union wb_u len;
|
|
|
|
union wb_u response;
|
2011-12-29 21:46:38 +01:00
|
|
|
bool dispatched = false;
|
2011-12-28 20:40:37 +01:00
|
|
|
uint8_t epno;
|
|
|
|
int nbytes = 0; /* Assume zero-length packet */
|
|
|
|
int ret;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Terminate any pending requests (doesn't work if the pending request
|
|
|
|
* was a zero-length transfer!)
|
|
|
|
*/
|
|
|
|
|
|
|
|
while (!pic32mx_rqempty(ep0))
|
|
|
|
{
|
|
|
|
int16_t result = OK;
|
|
|
|
if (privreq->req.xfrd != privreq->req.len)
|
|
|
|
{
|
|
|
|
result = -EPROTO;
|
|
|
|
}
|
|
|
|
|
|
|
|
usbtrace(TRACE_COMPLETE(ep0->ep.eplog), privreq->req.xfrd);
|
|
|
|
pic32mx_reqcomplete(ep0, result);
|
|
|
|
}
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Check if the USB currently owns the buffer */
|
|
|
|
|
|
|
|
if ((ep0->bdtin->status & USB_BDT_UOWN) != 0)
|
|
|
|
{
|
|
|
|
/* Yes.. give control back to the CPU. This handles for the
|
|
|
|
* ownership after a stall.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
ep0->bdtin->status &= ~USB_BDT_UOWN;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-29 21:46:38 +01:00
|
|
|
|
|
|
|
regdbg("EP0 BDT IN {%08x, %08x}\n", ep0->bdtin->status, ep0->bdtin->addr);
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Assume NOT stalled; no TX in progress */
|
|
|
|
|
|
|
|
ep0->stalled = 0;
|
|
|
|
ep0->txbusy = 0;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Initialize for the SETUP */
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* And extract the little-endian 16-bit values to host order */
|
|
|
|
|
|
|
|
value.w = GETUINT16(priv->ctrl.value);
|
|
|
|
index.w = GETUINT16(priv->ctrl.index);
|
|
|
|
len.w = GETUINT16(priv->ctrl.len);
|
|
|
|
|
|
|
|
ullvdbg("SETUP: type=%02x req=%02x value=%04x index=%04x len=%04x\n",
|
|
|
|
priv->ctrl.type, priv->ctrl.req, value.w, index.w, len.w);
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Dispatch any non-standard requests */
|
|
|
|
|
|
|
|
if ((priv->ctrl.type & USB_REQ_TYPE_MASK) != USB_REQ_TYPE_STANDARD)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_NOSTDREQ), priv->ctrl.type);
|
|
|
|
|
|
|
|
/* Let the class implementation handle all non-standar requests */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle standard request. Pick off the things of interest to the
|
|
|
|
* USB device controller driver; pass what is left to the class driver
|
|
|
|
*/
|
|
|
|
|
|
|
|
switch (priv->ctrl.req)
|
|
|
|
{
|
|
|
|
case USB_REQ_GETSTATUS:
|
|
|
|
{
|
|
|
|
/* type: device-to-host; recipient = device, interface, endpoint
|
|
|
|
* value: 0
|
|
|
|
* index: zero interface endpoint
|
|
|
|
* len: 2; data = status
|
|
|
|
*/
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_GETSTATUS), priv->ctrl.type);
|
|
|
|
if (len.w != 2 || (priv->ctrl.type & USB_REQ_DIR_IN) == 0 ||
|
|
|
|
index.b[MSB] != 0 || value.w != 0)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADEPGETSTATUS), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (priv->ctrl.type & USB_REQ_RECIPIENT_MASK)
|
|
|
|
{
|
|
|
|
case USB_REQ_RECIPIENT_ENDPOINT:
|
|
|
|
{
|
|
|
|
epno = USB_EPNO(index.b[LSB]);
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EPGETSTATUS), epno);
|
|
|
|
if (epno >= PIC32MX_NENDPOINTS)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADEPGETSTATUS), epno);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
privep = &priv->eplist[epno];
|
|
|
|
response.w = 0; /* Not stalled */
|
|
|
|
nbytes = 2; /* Response size: 2 bytes */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
if (USB_ISEPIN(index.b[LSB]))
|
|
|
|
{
|
2012-01-07 00:41:20 +01:00
|
|
|
/* IN endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
bdt = privep->bdtin;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
else
|
|
|
|
{
|
2012-01-07 00:41:20 +01:00
|
|
|
/* OUT endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
bdt = privep->bdtout;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* BSTALL set if stalled */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
if ((bdt->status & USB_BDT_BSTALL) != 0)
|
2011-12-28 20:40:37 +01:00
|
|
|
{
|
|
|
|
response.b[LSB] = 1; /* Stalled, set bit 0 */
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_RECIPIENT_DEVICE:
|
|
|
|
{
|
|
|
|
if (index.w == 0)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_DEVGETSTATUS), 0);
|
|
|
|
|
|
|
|
/* Features: Remote Wakeup=YES; selfpowered=? */
|
|
|
|
|
|
|
|
response.w = 0;
|
|
|
|
response.b[LSB] = (priv->selfpowered << USB_FEATURE_SELFPOWERED) |
|
2011-12-28 20:40:37 +01:00
|
|
|
(priv->rwakeup << USB_FEATURE_REMOTEWAKEUP);
|
2011-12-28 01:49:48 +01:00
|
|
|
nbytes = 2; /* Response size: 2 bytes */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADDEVGETSTATUS), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_RECIPIENT_INTERFACE:
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_IFGETSTATUS), 0);
|
2011-12-28 20:40:37 +01:00
|
|
|
response.w = 0;
|
|
|
|
nbytes = 2; /* Response size: 2 bytes */
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADGETSTATUS), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
case USB_REQ_CLEARFEATURE:
|
|
|
|
{
|
|
|
|
/* type: host-to-device; recipient = device, interface or endpoint
|
|
|
|
* value: feature selector
|
|
|
|
* index: zero interface endpoint;
|
|
|
|
* len: zero, data = none
|
|
|
|
*/
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_CLEARFEATURE), priv->ctrl.type);
|
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) != USB_REQ_RECIPIENT_DEVICE)
|
|
|
|
{
|
|
|
|
/* Disable B device from performing HNP */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
if (value.w == USBOTG_FEATURE_B_HNP_ENABLE)
|
|
|
|
{
|
|
|
|
/* Disable HNP */
|
|
|
|
#warning Missing Logic
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable A device HNP support */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
else if (value.w == USBOTG_FEATURE_A_HNP_SUPPORT)
|
|
|
|
{
|
|
|
|
/* Disable HNP support*/
|
|
|
|
#warning Missing Logic
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable alternate HNP support */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
else if (value.w == USBOTG_FEATURE_A_ALT_HNP_SUPPORT)
|
|
|
|
{
|
|
|
|
/* Disable alternate HNP */
|
|
|
|
#warning Missing Logic
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
/* Disable remote wakeup */
|
|
|
|
|
|
|
|
if (value.w == USB_FEATURE_REMOTEWAKEUP)
|
|
|
|
{
|
|
|
|
priv->rwakeup = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Let the class implementation handle all other device features */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_ENDPOINT)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
epno = USB_EPNO(index.b[LSB]);
|
2011-12-28 20:40:37 +01:00
|
|
|
if (epno > 0 && epno < PIC32MX_NENDPOINTS && index.b[MSB] == 0 &&
|
2011-12-28 01:49:48 +01:00
|
|
|
value.w == USB_FEATURE_ENDPOINTHALT && len.w == 0)
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
privep = &priv->eplist[epno];
|
|
|
|
privep->halted = 0;
|
|
|
|
ret = pic32mx_epstall(&privep->ep, true);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADCLEARFEATURE), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Let the class implementation handle all other recipients (except for the
|
|
|
|
* endpoint recipient)
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_SETFEATURE:
|
|
|
|
{
|
|
|
|
/* type: host-to-device; recipient = device, interface, endpoint
|
|
|
|
* value: feature selector
|
|
|
|
* index: zero interface endpoint;
|
|
|
|
* len: 0; data = none
|
|
|
|
*/
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_SETFEATURE), priv->ctrl.type);
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_DEVICE)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Enable B device to perform HNP */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
if (value.w == USBOTG_FEATURE_B_HNP_ENABLE)
|
|
|
|
{
|
|
|
|
/* Enable HNP */
|
|
|
|
#warning "Missing logic"
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable A device HNP supports */
|
|
|
|
|
|
|
|
else if (value.w == USBOTG_FEATURE_A_HNP_SUPPORT)
|
|
|
|
{
|
|
|
|
/* Enable HNP support */
|
|
|
|
#warning "Missing logic"
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Another port on the A device supports HNP */
|
|
|
|
|
|
|
|
else if (value.w == USBOTG_FEATURE_A_ALT_HNP_SUPPORT)
|
|
|
|
{
|
|
|
|
/* Enable alternate HNP */
|
|
|
|
#warning "Missing logic"
|
|
|
|
}
|
|
|
|
else
|
2012-01-07 00:41:20 +01:00
|
|
|
#endif
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
if (value.w == USB_FEATURE_REMOTEWAKEUP)
|
|
|
|
{
|
|
|
|
priv->rwakeup = 0;
|
|
|
|
}
|
|
|
|
else if (value.w == USB_FEATURE_TESTMODE)
|
|
|
|
{
|
|
|
|
/* Special case recipient=device test mode */
|
|
|
|
|
|
|
|
ullvdbg("test mode: %d\n", index.w);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Let the class implementation handle all other device features */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_ENDPOINT)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
/* Handler recipient=endpoint */
|
|
|
|
|
|
|
|
epno = USB_EPNO(index.b[LSB]);
|
|
|
|
if (epno < PIC32MX_NENDPOINTS && index.b[MSB] == 0 &&
|
|
|
|
value.w == USB_FEATURE_ENDPOINTHALT && len.w == 0)
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
privep = &priv->eplist[epno];
|
|
|
|
privep->halted = 1;
|
|
|
|
ret = pic32mx_epstall(&privep->ep, false);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADSETFEATURE), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* The class driver handles all recipients except recipient=endpoint */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_SETADDRESS:
|
|
|
|
{
|
|
|
|
/* type: host-to-device; recipient = device
|
|
|
|
* value: device address
|
|
|
|
* index: 0
|
|
|
|
* len: 0; data = none
|
|
|
|
*/
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EP0SETUPSETADDRESS), value.w);
|
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) != USB_REQ_RECIPIENT_DEVICE ||
|
|
|
|
index.w != 0 || len.w != 0 || value.w > 127)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADSETADDRESS), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Note that setting of the device address will be deferred. A zero-length
|
|
|
|
* packet will be sent and the device address will be set when the zero-
|
|
|
|
* length packet transfer completes.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
priv->devstate = DEVSTATE_ADDRPENDING;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_GETDESCRIPTOR:
|
|
|
|
/* type: device-to-host; recipient = device
|
|
|
|
* value: descriptor type and index
|
|
|
|
* index: 0 or language ID;
|
|
|
|
* len: descriptor len; data = descriptor
|
|
|
|
*/
|
|
|
|
case USB_REQ_SETDESCRIPTOR:
|
|
|
|
/* type: host-to-device; recipient = device
|
|
|
|
* value: descriptor type and index
|
|
|
|
* index: 0 or language ID;
|
|
|
|
* len: descriptor len; data = descriptor
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_GETSETDESC), priv->ctrl.type);
|
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_DEVICE)
|
|
|
|
{
|
|
|
|
/* The request seems valid... let the class implementation handle it */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADGETSETDESC), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_GETCONFIGURATION:
|
|
|
|
/* type: device-to-host; recipient = device
|
|
|
|
* value: 0;
|
|
|
|
* index: 0;
|
|
|
|
* len: 1; data = configuration value
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_GETCONFIG), priv->ctrl.type);
|
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_DEVICE &&
|
|
|
|
value.w == 0 && index.w == 0 && len.w == 1)
|
|
|
|
{
|
|
|
|
/* The request seems valid... let the class implementation handle it */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADGETCONFIG), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_SETCONFIGURATION:
|
|
|
|
/* type: host-to-device; recipient = device
|
|
|
|
* value: configuration value
|
|
|
|
* index: 0;
|
|
|
|
* len: 0; data = none
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_SETCONFIG), priv->ctrl.type);
|
|
|
|
if ((priv->ctrl.type & USB_REQ_RECIPIENT_MASK) == USB_REQ_RECIPIENT_DEVICE &&
|
|
|
|
index.w == 0 && len.w == 0)
|
|
|
|
{
|
|
|
|
/* The request seems valid... let the class implementation handle it */
|
|
|
|
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADSETCONFIG), 0);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_GETINTERFACE:
|
|
|
|
/* type: device-to-host; recipient = interface
|
|
|
|
* value: 0
|
|
|
|
* index: interface;
|
|
|
|
* len: 1; data = alt interface
|
|
|
|
*/
|
|
|
|
case USB_REQ_SETINTERFACE:
|
|
|
|
/* type: host-to-device; recipient = interface
|
|
|
|
* value: alternate setting
|
|
|
|
* index: interface;
|
|
|
|
* len: 0; data = none
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
/* Let the class implementation handle the request */
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_GETSETIF), priv->ctrl.type);
|
|
|
|
pic32mx_dispatchrequest(priv);
|
2011-12-29 21:46:38 +01:00
|
|
|
dispatched = true;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_REQ_SYNCHFRAME:
|
|
|
|
/* type: device-to-host; recipient = endpoint
|
|
|
|
* value: 0
|
|
|
|
* index: endpoint;
|
|
|
|
* len: 2; data = frame number
|
|
|
|
*/
|
|
|
|
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_SYNCHFRAME), 0);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDCTRLREQ), priv->ctrl.req);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* At this point, the request has been handled and there are three possible
|
|
|
|
* outcomes:
|
|
|
|
*
|
|
|
|
* 1. The setup request was successfully handled above and a response packet
|
|
|
|
* must be sent (may be a zero length packet).
|
|
|
|
* 2. The request was successfully handled by the class implementation. In
|
|
|
|
* case, the EP0 IN response has already been queued and the local variable
|
2011-12-29 21:46:38 +01:00
|
|
|
* 'dispatched' will be set to true and ctrlstate != CTRLSTATE_STALLED;
|
2011-12-28 01:49:48 +01:00
|
|
|
* 3. An error was detected in either the above logic or by the class implementation
|
|
|
|
* logic. In either case, priv->state will be set CTRLSTATE_STALLED
|
|
|
|
* to indicate this case.
|
|
|
|
*
|
|
|
|
* NOTE: Non-standard requests are a special case. They are handled by the
|
|
|
|
* class implementation and this function returned early above, skipping this
|
|
|
|
* logic altogether.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
if (!dispatched)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t regval;
|
|
|
|
|
|
|
|
/* PKTDIS bit is set when a Setup Transaction is received. Clear to resume
|
|
|
|
* packet processing.
|
2011-12-28 01:49:48 +01:00
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_CON);
|
|
|
|
regval &= ~USB_CON_PKTDIS;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
|
|
|
|
|
|
|
if (priv->ctrlstate != CTRLSTATE_STALLED)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
/* We will respond. First, restrict the data length to the length
|
|
|
|
* requested in the setup packet
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
if (nbytes > len.w)
|
|
|
|
{
|
|
|
|
nbytes = len.w;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Send the response (might be a zero-length packet) */
|
|
|
|
|
|
|
|
pic32mx_ep0done(priv, response.b, nbytes);
|
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Stall EP0 */
|
|
|
|
|
|
|
|
(void)pic32mx_epstall(&ep0->ep, false);
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0in
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_ep0in(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdt;
|
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status;
|
|
|
|
int ret;
|
2011-12-29 01:33:02 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* There is no longer anything in the EP0 TX packet memory */
|
|
|
|
|
2011-12-29 01:33:02 +01:00
|
|
|
ep0->txbusy = false;
|
2011-12-29 21:46:38 +01:00
|
|
|
bdt = ep0->bdtin;
|
2011-12-29 01:33:02 +01:00
|
|
|
|
|
|
|
/* Switch to the next ping pong buffer */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status;
|
2011-12-29 01:33:02 +01:00
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_NEXT_PINGPONG;
|
2011-12-29 01:33:02 +01:00
|
|
|
#endif
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Are we processing the completion of one packet of an outgoing request
|
|
|
|
* from the class driver?
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (priv->ctrlstate == CTRLSTATE_WRREQUEST)
|
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Handle the next queue IN transfer */
|
2011-12-29 01:33:02 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
ret = pic32mx_wrrequest(priv, &priv->eplist[EP0]);
|
|
|
|
if (ret == OK)
|
2011-12-29 16:55:02 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Is there another IN transfer in-flight? */
|
2011-12-29 01:33:02 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
if (priv->ctrlstate == CTRLSTATE_WAITSETUP)
|
2011-12-29 16:55:02 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
/* No... Prepare for the next SETUP transfer */
|
|
|
|
|
|
|
|
pic32mx_ep0nextsetup(priv);
|
2011-12-29 01:33:02 +01:00
|
|
|
}
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* No.. Are we processing the completion of a status response? */
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
else if (priv->ctrlstate == CTRLSTATE_WAITSETUP)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
/* Look at the saved SETUP command. Was it a SET ADDRESS request?
|
|
|
|
* If so, then now is the time to set the address.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
if (priv->devstate == DEVSTATE_ADDRPENDING)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
uint16_t addr = GETUINT16(priv->ctrl.value);
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
/* This should be the equivalent state */
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
DEBUGASSERT(priv->ctrl.req == USB_REQ_SETADDRESS &&
|
|
|
|
(priv->ctrl.type & REQRECIPIENT_MASK) ==
|
2011-12-29 16:55:02 +01:00
|
|
|
(USB_REQ_TYPE_STANDARD | USB_REQ_RECIPIENT_DEVICE));
|
2011-12-29 01:33:02 +01:00
|
|
|
|
|
|
|
pic32mx_putreg(addr, PIC32MX_USB_ADDR);
|
|
|
|
if (addr > 0)
|
|
|
|
{
|
|
|
|
priv->devstate = DEVSTATE_ADDRESS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
priv->devstate = DEVSTATE_DEFAULT;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Prepare for the next SETUP transfer */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
pic32mx_ep0nextsetup(priv);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* No other state is expected in this context */
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
else
|
|
|
|
{
|
2011-12-29 16:55:02 +01:00
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDSTATE), priv->ctrlstate);
|
2011-12-28 01:49:48 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0out
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_ep0out(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
2011-12-29 01:33:02 +01:00
|
|
|
struct pic32mx_ep_s *ep0 = &priv->eplist[EP0];
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
switch (priv->ctrlstate)
|
|
|
|
{
|
2011-12-29 16:55:02 +01:00
|
|
|
case CTRLSTATE_RDREQUEST: /* Read request in progress */
|
2011-12-29 01:33:02 +01:00
|
|
|
|
|
|
|
/* Process the next read request for EP0 */
|
|
|
|
|
|
|
|
pic32mx_rdcomplete(priv, ep0);
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Was this the end of the OUT transfer? */
|
|
|
|
|
|
|
|
if (priv->ctrlstate == CTRLSTATE_WAITSETUP)
|
|
|
|
{
|
|
|
|
/* Prepare EP0 OUT for the next SETUP transaction. */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = ep0->bdtout->status & ~(USB_BDT_BYTECOUNT_MASK|USB_BDT_DATA01);
|
|
|
|
ep0->bdtout->status = status;
|
|
|
|
|
|
|
|
ep0->bdtout->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
|
|
|
status |= (USB_BDT_DATA0 | USB_BDT_DTS | USB_BDT_BSTALL);
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
/* Set BSTALL in case the host tries to send more data than
|
|
|
|
* it claims it was going to send.
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (USB_BDT_BSTALL | USB_BDT_UOWN);
|
|
|
|
|
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", status, ep0->bdtout->addr);
|
|
|
|
|
|
|
|
ep0->bdtout->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CTRLSTATE_WAITSETUP: /* No transfer in progress, waiting for setup */
|
|
|
|
{
|
|
|
|
/* In this case the last OUT transaction must have been a status
|
|
|
|
* stage of a CTRLSTATE_WRREQUEST
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Prepare EP0 OUT for the next SETUP transaction. However,
|
|
|
|
* EP0 may have already been prepared if ping-pong buffering
|
|
|
|
* was enabled on EP0 OUT, and the last control transfer was
|
|
|
|
* of direction: device to host, see pic32mx_ep0done(). If
|
|
|
|
* it was already prepared, we do not want to do anything to
|
|
|
|
* the buffer descriptor table (BDT).
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
pic32mx_ep0nextsetup(priv);
|
2011-12-29 16:55:02 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
if (!priv->ep0ready)
|
|
|
|
#endif
|
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
status = ep0->bdtout->status & ~(USB_BDT_BYTECOUNT_MASK|USB_BDT_DATA01);
|
|
|
|
ep0->bdtout->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
ep0->bdtout->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA0 | USB_BDT_DTS | USB_BDT_BSTALL);
|
|
|
|
|
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", status, ep0->bdtout->addr);
|
|
|
|
|
|
|
|
ep0->bdtout->status = status;
|
2011-12-29 16:55:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
priv->ep0ready = 0;
|
|
|
|
#endif
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2011-12-29 16:55:02 +01:00
|
|
|
{
|
|
|
|
/* Unexpected state OR host aborted the OUT transfer before it
|
|
|
|
* completed, STALL the endpoint in either case
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDSTATE), priv->ctrlstate);
|
|
|
|
priv->ctrlstate = CTRLSTATE_STALLED;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_ep0transfer
|
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
static void pic32mx_ep0transfer(struct pic32mx_usbdev_s *priv, uint16_t status)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdt;
|
|
|
|
|
|
|
|
/* The following information is available in the tatus register :
|
|
|
|
*
|
|
|
|
* ENDPT - The 4 bit endpoint number that cause the interrupt.
|
|
|
|
* DIR - The direction of the endpoint.
|
|
|
|
* PPBI - The ping-pong buffer used in the transaction.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Check if the last transaction was an EP0 OUT transaction */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
if ((status & USB_STAT_DIR) == USB_STAT_DIR_OUT)
|
|
|
|
{
|
2011-12-29 21:46:38 +01:00
|
|
|
int index;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* It was an EP0 OUT transaction. Get the index to the BDT. */
|
|
|
|
|
|
|
|
#if CONFIG_USB_PINGPONG
|
|
|
|
index = ((status & USB_STAT_PPBI) == 0 ? EP0_IN_EVEN : EP0_IN_ODD);
|
|
|
|
#else
|
|
|
|
index = EP0_IN_EVEN;
|
|
|
|
#endif
|
|
|
|
bdt = &g_bdt[index];
|
2011-12-28 20:40:37 +01:00
|
|
|
priv->eplist[0].bdtout = bdt;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
regdbg("EP0 BDT OUT {%08x, %08x}\n", bdt->status, bdt->addr);
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Toggle it to the next ping pong buffer (if applicable) */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
bdt->status ^= USB_NEXT_PINGPONG;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Check the current EP0 OUT buffer contains a SETUP packet */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 13:52:15 +01:00
|
|
|
if (((bdt->status & USB_BDT_PID_MASK) >> USB_BDT_PID_SHIFT) == USB_PID_SETUP_TOKEN)
|
2011-12-28 20:40:37 +01:00
|
|
|
{
|
|
|
|
/* Check if the SETUP transaction data went into the priv->ctrl
|
|
|
|
* buffer. If not, then we will need to copy it.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
if (bdt->addr != (uint8_t *)PHYS_ADDR(&priv->ctrl))
|
|
|
|
{
|
|
|
|
void *src = (void *)VIRT_ADDR(bdt->addr);
|
|
|
|
void *dest = &priv->ctrl;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
memcpy(dest, src, USB_SIZEOF_CTRLREQ);
|
|
|
|
bdt->addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Handle the control OUT transfer */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EP0SETUPDONE), bdt->status);
|
2011-12-28 01:49:48 +01:00
|
|
|
pic32mx_ep0setup(priv);
|
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
else
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Handle the data OUT transfer */
|
|
|
|
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EP0OUTDONE), status);
|
2011-12-28 01:49:48 +01:00
|
|
|
pic32mx_ep0out(priv);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* No.. it was an EP0 IN transfer */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
else /* if ((status & USB_STAT_DIR) == USB_STAT_DIR_IN) */
|
|
|
|
{
|
2011-12-30 15:54:43 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_EP0INDONE), status);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Handle the IN transfer */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
pic32mx_ep0in(priv);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Check for a stall */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
if (priv->ctrlstate == CTRLSTATE_STALLED)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Stall EP0 */
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_EP0SETUPSTALLED), priv->ctrlstate);
|
2011-12-30 01:09:43 +01:00
|
|
|
(void)pic32mx_epstall(&priv->eplist[EP0].ep, false);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_interrupt
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_interrupt(int irq, void *context)
|
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
2012-01-07 14:35:58 +01:00
|
|
|
uint16_t usbir;
|
|
|
|
uint16_t otgir;
|
2011-12-28 01:49:48 +01:00
|
|
|
uint16_t regval;
|
2011-12-28 20:40:37 +01:00
|
|
|
int i;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
/* Get the set of pending USB and OTG interrupts interrupts */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
usbir = pic32mx_getreg(PIC32MX_USB_IR) & pic32mx_getreg(PIC32MX_USB_IE);
|
|
|
|
otgir = pic32mx_getreg(PIC32MX_USBOTG_IR) & pic32mx_getreg(PIC32MX_USBOTG_IE);
|
2012-01-09 22:34:58 +01:00
|
|
|
usbtrace(TRACE_INTENTRY(PIC32MX_TRACEINTID_INTERRUPT), usbir | otgir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
/* Session Request Protocol (SRP) Time Out Check */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* if USB OTG SRP is ready */
|
|
|
|
# warning "Missing logic"
|
|
|
|
{
|
|
|
|
/* Check if the 1 millisecond timer has expired */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if (otgir & OTG_INT_T1MSEC) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_T1MSEC), otgir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Check for the USB OTG SRP timeout */
|
|
|
|
# warning "Missing logic"
|
|
|
|
{
|
|
|
|
/* Handle OTG events of the SRP timeout has expired */
|
|
|
|
# warning "Missing logic"
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Clear Interrupt 1 msec timer Flag */
|
|
|
|
|
|
|
|
pic32mx_putreg(USBOTG_INT_T1MSEC, PIC32MX_USBOTG_IR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Handle events while we are in the attached state */
|
|
|
|
|
|
|
|
if (priv->devstate == DEVSTATE_ATTACHED)
|
|
|
|
{
|
|
|
|
/* Clear all USB interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_INT_ALL, PIC32MX_USB_IR);
|
|
|
|
|
2012-01-18 00:02:40 +01:00
|
|
|
/* Make sure that the USE reset and IDLE detect interrupts are enabled */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_IE);
|
|
|
|
regval |= (USB_INT_URST|USB_INT_IDLE);
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_IE);
|
|
|
|
|
|
|
|
/* Now were are in the powered state */
|
|
|
|
|
|
|
|
priv->devstate = DEVSTATE_POWERED;
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
/* Check if the ID Pin Changed State */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if (otgir & USBOTG_INT_ID) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_OTGID), otgir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Re-detect and re-initialize */
|
|
|
|
#warning "Missing logic"
|
|
|
|
|
|
|
|
pic32mx_putreg(USBOTG_INT_ID, PIC32MX_USBOTG_IR);
|
|
|
|
}
|
|
|
|
#endif
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Service the USB Activity Interrupt */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((otgir & USBOTG_INT_ACTV) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_WKUP), otgir);
|
|
|
|
|
|
|
|
/* Wake-up from susepnd mode */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_putreg(USBOTG_INT_ACTV, PIC32MX_USBOTG_IR);
|
|
|
|
pic32mx_resume(priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* It is pointless to continue servicing if the device is in suspend mode. */
|
|
|
|
|
|
|
|
if ((pic32mx_getreg(PIC32MX_USB_PWRC) & USB_PWRC_USUSPEND) != 0)
|
|
|
|
{
|
|
|
|
/* Just clear the interrupt and return */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_SUSPENDED), pic32mx_getreg(PIC32MX_USB_PWRC));
|
|
|
|
goto interrupt_exit;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Service USB Bus Reset Interrupt. When bus reset is received during
|
|
|
|
* suspend, ACTVIF will be set first, once the UCONbits.SUSPND is clear,
|
|
|
|
* then the URSTIF bit will be asserted. This is why URSTIF is checked
|
|
|
|
* after ACTVIF. The USB reset flag is masked when the USB state is in
|
|
|
|
* DEVSTATE_DETACHED or DEVSTATE_ATTACHED, and therefore cannot cause a
|
|
|
|
* USB reset event during these two states.
|
|
|
|
*/
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_URST) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_RESET), usbir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Reset interrupt received. Restore our power-up state */
|
|
|
|
|
|
|
|
pic32mx_reset(priv);
|
|
|
|
pic32mx_attach(priv);
|
|
|
|
priv->devstate = DEVSTATE_DEFAULT;
|
|
|
|
|
|
|
|
/* Re-initialize EP0 */
|
|
|
|
|
|
|
|
g_bdt[EP0_OUT_EVEN].addr = (uint8_t *)PHYS_ADDR(&priv->ctrl);
|
|
|
|
g_bdt[EP0_OUT_EVEN].status &= ~USB_BDT_BYTECOUNT_MASK;
|
2011-12-28 20:40:37 +01:00
|
|
|
g_bdt[EP0_OUT_EVEN].status |= (USB_SIZEOF_CTRLREQ << USB_BDT_BYTECOUNT_SHIFT);
|
2011-12-28 01:49:48 +01:00
|
|
|
g_bdt[EP0_OUT_EVEN].status &= ~USB_BDT_STATUS_MASK;
|
|
|
|
g_bdt[EP0_OUT_EVEN].status |= USB_BDT_UOWN | USB_BDT_DATA0 | USB_BDT_DTS | USB_BDT_BSTALL;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
/* Disable and deactivate HNP */
|
|
|
|
#warning Missing Logic
|
|
|
|
#endif
|
|
|
|
pic32mx_putreg(USB_INT_URST, PIC32MX_USB_IR);
|
2012-01-07 14:35:58 +01:00
|
|
|
goto interrupt_exit;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Service IDLE interrupts */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_IDLE) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_IDLE), usbir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
#ifdef CONFIG_USBOTG
|
2011-12-28 01:49:48 +01:00
|
|
|
/* If Suspended, Try to switch to Host */
|
2012-01-07 00:41:20 +01:00
|
|
|
#warning "Missing logic"
|
2011-12-28 01:49:48 +01:00
|
|
|
#else
|
|
|
|
pic32mx_suspend(priv);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
#endif
|
|
|
|
pic32mx_putreg(USB_INT_IDLE, PIC32MX_USB_IR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Service SOF interrupts */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_SOFINTS
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_SOF) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_SOF), 0);
|
|
|
|
|
|
|
|
/* I am not sure why you would ever enable SOF interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_INT_SOF, PIC32MX_USB_IR);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Service stall interrupts */
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_STALL) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_STALL), usbir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
pic32mx_ep0stall(priv);
|
|
|
|
|
|
|
|
/* Clear the pending STALL interrupt */
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_INT_STALL, PIC32MX_USB_IR);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Service error interrupts */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_UERR) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_UERR), usbir);
|
2011-12-28 01:49:48 +01:00
|
|
|
ulldbg("Error: EIR=%04x\n", pic32mx_getreg(PIC32MX_USB_EIR));
|
|
|
|
|
|
|
|
/* Clear all pending USB error interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_EINT_ALL, PIC32MX_USB_EIR);
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* There is no point in continuing if the host has not sent a bus reset.
|
|
|
|
* Once bus reset is received, the device transitions into the DEFAULT
|
2011-12-28 01:49:48 +01:00
|
|
|
* state and is ready for communication.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (priv->devstate < DEVSTATE_DEFAULT)
|
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
/* Just clear the interrupt and return */
|
|
|
|
|
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_WAITRESET), priv->devstate);
|
|
|
|
goto interrupt_exit;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Service USB Transaction Complete Interrupt */
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
if ((usbir & USB_INT_TRN) != 0)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2012-01-07 14:35:58 +01:00
|
|
|
usbtrace(TRACE_INTDECODE(PIC32MX_TRACEINTID_TRNC), usbir);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Drain the USAT FIFO entries. If the USB FIFO ever gets full, USB
|
|
|
|
* bandwidth utilization can be compromised, and the device won't be
|
|
|
|
* able to receive SETUP packets.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
uint8_t epno;
|
|
|
|
|
|
|
|
/* Is token processing complete */
|
|
|
|
|
|
|
|
if ((pic32mx_getreg(PIC32MX_USB_IR) & USB_INT_TRN) != 0)
|
|
|
|
{
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_STAT);
|
|
|
|
pic32mx_putreg(USB_INT_TRN, PIC32MX_USB_IR);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Handle the endpoint tranfer complete event. */
|
|
|
|
|
|
|
|
epno = (regval & USB_STAT_ENDPT_MASK) >> USB_STAT_ENDPT_SHIFT;
|
|
|
|
if (epno == 0)
|
|
|
|
{
|
|
|
|
pic32mx_ep0transfer(priv, regval);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
pic32mx_eptransfer(priv, epno, regval);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* USTAT FIFO must be empty. */
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
/* Clear the pending USB interrupt. Goto is used in the above to assure
|
|
|
|
* that all interrupt exists pass through this logic.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 14:35:58 +01:00
|
|
|
interrupt_exit:
|
2011-12-28 01:49:48 +01:00
|
|
|
up_clrpend_irq(PIC32MX_IRQSRC_USB);
|
2012-01-09 22:34:58 +01:00
|
|
|
usbtrace(TRACE_INTEXIT(PIC32MX_TRACEINTID_INTERRUPT), usbir | otgir);
|
2011-12-28 01:49:48 +01:00
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Suspend/Resume Helpers
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_suspend
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
static void pic32mx_suspend(struct pic32mx_usbdev_s *priv)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
uint16_t regval;
|
|
|
|
|
2012-01-18 00:02:40 +01:00
|
|
|
/* Enable the ACTV interrupt.
|
|
|
|
*
|
|
|
|
* NOTE: Do not clear UIRbits.ACTVIF here! Reason: ACTVIF is only
|
2011-12-28 01:49:48 +01:00
|
|
|
* generated once an IDLEIF has been generated. This is a 1:1 ratio
|
|
|
|
* interrupt generation. For every IDLEIF, there will be only one ACTVIF
|
2012-01-07 00:41:20 +01:00
|
|
|
* regardless of the number of subsequent bus transitions. If the ACTIF
|
2011-12-28 01:49:48 +01:00
|
|
|
* is cleared here, a problem could occur. The driver services IDLEIF
|
|
|
|
* first because ACTIVIE=0. If this routine clears the only ACTIVIF,
|
|
|
|
* then it can never get out of the suspend mode.
|
|
|
|
*/
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
regval = pic32mx_getreg(PIC32MX_USBOTG_IE);
|
2011-12-28 01:49:48 +01:00
|
|
|
regval |= USBOTG_INT_ACTV;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USBOTG_IE);
|
|
|
|
|
2012-01-18 00:02:40 +01:00
|
|
|
/* Disable further IDLE interrupts. Once is enough. */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_IE);
|
|
|
|
regval &= ~USB_INT_IDLE;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_IE);
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Invoke a callback into board-specific logic. The board-specific logic
|
|
|
|
* may enter into sleep or idle modes or switch to a slower clock, etc.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_usbsuspend((struct usbdev_s *)priv, false);
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_resume
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
static void pic32mx_resume(struct pic32mx_usbdev_s *priv)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
irqstate_t flags;
|
2011-12-28 01:49:48 +01:00
|
|
|
uint16_t regval;
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
flags = irqsave();
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Start RESUME signaling */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_CON);
|
|
|
|
regval |= USB_CON_RESUME;
|
2012-01-07 00:17:03 +01:00
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 00:17:03 +01:00
|
|
|
/* Keep the RESUME line set for 1-13 ms */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
up_mdelay(10);
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
regval &= ~USB_CON_RESUME;
|
2012-01-07 00:17:03 +01:00
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* This function is called when the USB activity interrupt occurs.
|
|
|
|
* If using clock switching, this is the place to call out to
|
|
|
|
* logic to restore the original MCU core clock frequency.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_usbsuspend((struct usbdev_s *)priv, true);
|
|
|
|
|
|
|
|
/* Disable further activity interrupts */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USBOTG_IE);
|
|
|
|
regval &= ~USBOTG_INT_ACTV;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USBOTG_IE);
|
|
|
|
|
|
|
|
/* The ACTVIF bit cannot be cleared immediately after the USB module wakes
|
|
|
|
* up from Suspend or while the USB module is suspended. A few clock cycles
|
|
|
|
* are required to synchronize the internal hardware state machine before
|
|
|
|
* the ACTIVIF bit can be cleared by firmware. Clearing the ACTVIF bit
|
|
|
|
* before the internal hardware is synchronized may not have an effect on
|
|
|
|
* the value of ACTVIF. Additionally, if the USB module uses the clock from
|
|
|
|
* the 96 MHz PLL source, then after clearing the SUSPND bit, the USB
|
|
|
|
* module may not be immediately operational while waiting for the 96 MHz
|
|
|
|
* PLL to lock.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_INT_IDLE, PIC32MX_USBOTG_IR);
|
2011-12-30 01:09:43 +01:00
|
|
|
irqrestore(flags);
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Endpoint Helpers
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epreserve
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static inline struct pic32mx_ep_s *
|
|
|
|
pic32mx_epreserve(struct pic32mx_usbdev_s *priv, uint8_t epset)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep = NULL;
|
|
|
|
irqstate_t flags;
|
|
|
|
int epndx = 0;
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
epset &= priv->epavail;
|
|
|
|
if (epset)
|
|
|
|
{
|
|
|
|
/* Select the lowest bit in the set of matching, available endpoints
|
|
|
|
* (skipping EP0)
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (epndx = 1; epndx < PIC32MX_NENDPOINTS; epndx++)
|
|
|
|
{
|
|
|
|
uint8_t bit = PIC32MX_ENDP_BIT(epndx);
|
|
|
|
if ((epset & bit) != 0)
|
|
|
|
{
|
|
|
|
/* Mark the endpoint no longer available */
|
|
|
|
|
|
|
|
priv->epavail &= ~bit;
|
|
|
|
|
|
|
|
/* And return the pointer to the standard endpoint structure */
|
|
|
|
|
|
|
|
privep = &priv->eplist[epndx];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
irqrestore(flags);
|
|
|
|
return privep;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epunreserve
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
pic32mx_epunreserve(struct pic32mx_usbdev_s *priv, struct pic32mx_ep_s *privep)
|
|
|
|
{
|
|
|
|
irqstate_t flags = irqsave();
|
|
|
|
priv->epavail |= PIC32MX_ENDP_BIT(USB_EPNO(privep->ep.eplog));
|
|
|
|
irqrestore(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epreserved
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
pic32mx_epreserved(struct pic32mx_usbdev_s *priv, int epno)
|
|
|
|
{
|
|
|
|
return ((priv->epavail & PIC32MX_ENDP_BIT(epno)) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Endpoint operations
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epconfigure
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epconfigure(struct usbdev_ep_s *ep,
|
|
|
|
const struct usb_epdesc_s *desc,
|
|
|
|
bool last)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep = (struct pic32mx_ep_s *)ep;
|
2011-12-30 01:09:43 +01:00
|
|
|
volatile struct usbotg_bdtentry_s *bdt;
|
2011-12-28 01:49:48 +01:00
|
|
|
uint16_t maxpacket;
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t regval;
|
|
|
|
uint16_t status;
|
2011-12-28 01:49:48 +01:00
|
|
|
uint8_t epno;
|
2011-12-30 01:09:43 +01:00
|
|
|
bool epin;
|
|
|
|
bool bidi;
|
|
|
|
int index;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep || !desc)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
ulldbg("ERROR: ep=%p desc=%p\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Get the unadorned endpoint address */
|
|
|
|
|
|
|
|
epno = USB_EPNO(desc->addr);
|
2011-12-30 01:09:43 +01:00
|
|
|
epin = USB_ISEPIN(desc->addr);
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
usbtrace(TRACE_EPCONFIGURE, (uint16_t)epno);
|
|
|
|
DEBUGASSERT(epno == USB_EPNO(ep->eplog));
|
|
|
|
|
|
|
|
/* Set the requested type */
|
|
|
|
|
|
|
|
switch (desc->attr & USB_EP_ATTR_XFERTYPE_MASK)
|
|
|
|
{
|
|
|
|
case USB_EP_ATTR_XFER_INT: /* Interrupt endpoint */
|
2011-12-30 01:09:43 +01:00
|
|
|
regval = epin ? PIC32MX_EP_INTIN : PIC32MX_EP_INTOUT;
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_EP_ATTR_XFER_BULK: /* Bulk endpoint */
|
2011-12-30 01:09:43 +01:00
|
|
|
regval = epin ? PIC32MX_EP_BULKIN : PIC32MX_EP_BULKOUT;
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_EP_ATTR_XFER_ISOC: /* Isochronous endpoint */
|
2011-12-30 01:09:43 +01:00
|
|
|
regval = epin ? PIC32MX_EP_ISOCIN : PIC32MX_EP_ISOCOUT;
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case USB_EP_ATTR_XFER_CONTROL: /* Control endpoint */
|
2011-12-30 01:09:43 +01:00
|
|
|
regval = PIC32MX_EP_CONTROL;
|
|
|
|
bidi = true;
|
2011-12-28 01:49:48 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADEPTYPE), (uint16_t)desc->type);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Enable the endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_EP(epno));
|
|
|
|
regdbg("PIC32MX_USB_EP%d: %04x\n", epno, getreg16(PIC32MX_USB_EP(epno)));
|
|
|
|
|
|
|
|
/* Setup up buffer descriptor table (BDT) entry/ies for this endpoint */
|
|
|
|
|
|
|
|
if (epin || bidi)
|
|
|
|
{
|
|
|
|
/* Get the pointer to BDT entry */
|
|
|
|
|
|
|
|
index = EP(epno, 1, 0);
|
|
|
|
bdt = &g_bdt[index];
|
|
|
|
privep->bdtin = bdt;
|
|
|
|
|
|
|
|
/* Mark that we own the entry */
|
|
|
|
|
|
|
|
status = bdt->status & ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
/* Set DATA1 to one because the first thing we will do when transmitting is
|
|
|
|
* toggle the bit
|
|
|
|
*/
|
|
|
|
|
|
|
|
status |= USB_BDT_DATA01;
|
|
|
|
bdt->status = status;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT IN {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
|
|
|
|
/* Now do the same for the other buffer (if ping pong mode is supported).
|
|
|
|
* The only difference is the we clear DATA1 (making it DATA0)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
bdt = &g_bdt[index+1];
|
|
|
|
status = bdt->status & ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
status &= ~USB_BDT_DATA01;
|
|
|
|
bdt->status = status;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT IN {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
#endif
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
if (!epin || bidi)
|
|
|
|
{
|
|
|
|
index = EP(epno, 0, 0);
|
|
|
|
bdt = &g_bdt[index];
|
|
|
|
privep->bdtout = bdt;
|
|
|
|
|
|
|
|
/* Mark that we own the entry */
|
|
|
|
|
|
|
|
status = bdt->status & ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
/* Set DATA1 to one because the first thing we will do when transmitting is
|
|
|
|
* toggle the bit
|
|
|
|
*/
|
|
|
|
|
|
|
|
status |= USB_BDT_DATA01;
|
|
|
|
bdt->status = status;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
|
|
|
|
/* Now do the same for the other buffer (if ping pong mode is supported).
|
|
|
|
* The only difference is the we clear DATA1 (making it DATA0)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
bdt = &g_bdt[index+1];
|
|
|
|
status = bdt->status & ~USB_BDT_UOWN;
|
|
|
|
|
|
|
|
status &= ~USB_BDT_DATA01;
|
|
|
|
bdt->status = status;
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
#endif
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Get the maxpacket size of the endpoint. */
|
|
|
|
|
|
|
|
maxpacket = GETUINT16(desc->mxpacketsize);
|
|
|
|
DEBUGASSERT(maxpacket <= PIC32MX_MAXPACKET_SIZE);
|
|
|
|
ep->maxpacket = maxpacket;
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Set the full, logic EP number (that includes direction encoded in bit 7) */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
if (epin)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
ep->eplog = USB_EPIN(epno);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ep->eplog = USB_EPOUT(epno);
|
|
|
|
}
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
return OK;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epdisable
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epdisable(struct usbdev_ep_s *ep)
|
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
struct pic32mx_usbdev_s *priv;
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
volatile uint32_t *ptr;
|
|
|
|
int epno;
|
|
|
|
int i;
|
2011-12-28 01:49:48 +01:00
|
|
|
irqstate_t flags;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
ulldbg("ERROR: ep=%p\n", ep);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
priv = privep->dev;
|
|
|
|
epno = USB_EPNO(ep->eplog);
|
2011-12-28 01:49:48 +01:00
|
|
|
usbtrace(TRACE_EPDISABLE, epno);
|
|
|
|
|
|
|
|
/* Cancel any ongoing activity */
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
pic32mx_cancelrequests(privep);
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Disable the endpoint */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_EP(epno));
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Reset the BDTs */
|
|
|
|
|
|
|
|
ptr = (uint32_t*)&g_bdt[EP(epno, 0, 0)];
|
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
for (i = 0; i < (4 * sizeof(struct usbotg_bdtentry_s) / sizeof(uint32_t)); i++)
|
|
|
|
#else
|
|
|
|
for (i = 0; i < (2 * sizeof(struct usbotg_bdtentry_s) / sizeof(uint32_t)); i++)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
*ptr++ = 0;
|
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
irqrestore(flags);
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epallocreq
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static struct usbdev_req_s *pic32mx_epallocreq(struct usbdev_ep_s *ep)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
usbtrace(TRACE_EPALLOCREQ, USB_EPNO(ep->eplog));
|
|
|
|
|
|
|
|
privreq = (struct pic32mx_req_s *)malloc(sizeof(struct pic32mx_req_s));
|
|
|
|
if (!privreq)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_ALLOCFAIL), 0);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(privreq, 0, sizeof(struct pic32mx_req_s));
|
|
|
|
return &privreq->req;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epfreereq
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_epfreereq(struct usbdev_ep_s *ep, struct usbdev_req_s *req)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq = (struct pic32mx_req_s*)req;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep || !req)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
usbtrace(TRACE_EPFREEREQ, USB_EPNO(ep->eplog));
|
|
|
|
|
|
|
|
free(privreq);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epsubmit
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epsubmit(struct usbdev_ep_s *ep, struct usbdev_req_s *req)
|
|
|
|
{
|
|
|
|
struct pic32mx_req_s *privreq = (struct pic32mx_req_s *)req;
|
|
|
|
struct pic32mx_ep_s *privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
struct pic32mx_usbdev_s *priv;
|
|
|
|
irqstate_t flags;
|
|
|
|
uint8_t epno;
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!req || !req->callback || !req->buf || !ep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
ulldbg("ERROR: req=%p callback=%p buf=%p ep=%p\n", req, req->callback, req->buf, ep);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
usbtrace(TRACE_EPSUBMIT, USB_EPNO(ep->eplog));
|
|
|
|
priv = privep->dev;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!priv->driver)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_NOTCONFIGURED), priv->usbdev.speed);
|
|
|
|
ulldbg("ERROR: driver=%p\n", priv->driver);
|
|
|
|
return -ESHUTDOWN;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Handle the request from the class driver */
|
|
|
|
|
|
|
|
epno = USB_EPNO(ep->eplog);
|
|
|
|
req->result = -EINPROGRESS;
|
|
|
|
req->xfrd = 0;
|
|
|
|
flags = irqsave();
|
|
|
|
|
|
|
|
/* If we are stalled, then drop all requests on the floor */
|
|
|
|
|
|
|
|
if (privep->stalled)
|
|
|
|
{
|
|
|
|
pic32mx_abortrequest(privep, privreq, -EBUSY);
|
|
|
|
ulldbg("ERROR: stalled\n");
|
|
|
|
ret = -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle IN (device-to-host) requests. NOTE: If the class device is
|
|
|
|
* using the bi-directional EP0, then we assume that they intend the EP0
|
|
|
|
* IN functionality.
|
|
|
|
*/
|
|
|
|
|
|
|
|
else if (USB_ISEPIN(ep->eplog) || epno == EP0)
|
|
|
|
{
|
|
|
|
/* Add the new request to the request queue for the IN endpoint */
|
|
|
|
|
|
|
|
pic32mx_rqenqueue(privep, privreq);
|
|
|
|
usbtrace(TRACE_INREQQUEUED(epno), req->len);
|
|
|
|
|
|
|
|
/* If the IN endpoint FIFO is available, then transfer the data now */
|
|
|
|
|
|
|
|
if (!privep->txbusy)
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
ret = pic32mx_wrrequest(priv, privep);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle OUT (host-to-device) requests */
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Add the new request to the request queue for the OUT endpoint */
|
|
|
|
|
|
|
|
privep->txnullpkt = 0;
|
|
|
|
pic32mx_rqenqueue(privep, privreq);
|
|
|
|
usbtrace(TRACE_OUTREQQUEUED(epno), req->len);
|
|
|
|
}
|
|
|
|
|
|
|
|
irqrestore(flags);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epcancel
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epcancel(struct usbdev_ep_s *ep, struct usbdev_req_s *req)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
struct pic32mx_usbdev_s *priv;
|
|
|
|
irqstate_t flags;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep || !req)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
usbtrace(TRACE_EPCANCEL, USB_EPNO(ep->eplog));
|
|
|
|
priv = privep->dev;
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
pic32mx_cancelrequests(privep);
|
|
|
|
irqrestore(flags);
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2011-12-29 21:46:38 +01:00
|
|
|
* Name: pic32mx_epbdtstall
|
2011-12-28 01:49:48 +01:00
|
|
|
****************************************************************************/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
static int pic32mx_epbdtstall(struct usbdev_ep_s *ep,
|
|
|
|
volatile struct usbotg_bdtentry_s *bdt,
|
|
|
|
bool resume, bool epin)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
struct pic32mx_usbdev_s *priv;
|
2011-12-28 20:40:37 +01:00
|
|
|
uint32_t regaddr;
|
|
|
|
uint16_t regval;
|
2011-12-29 21:46:38 +01:00
|
|
|
uint16_t status;
|
|
|
|
uint8_t epno;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/* Recover pointers */
|
2011-12-28 20:40:37 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
priv = (struct pic32mx_usbdev_s *)privep->dev;
|
|
|
|
epno = USB_EPNO(ep->eplog);
|
|
|
|
|
|
|
|
/* Handle the resume condition */
|
|
|
|
|
|
|
|
if (resume)
|
|
|
|
{
|
|
|
|
/* Resuming a stalled endpoint */
|
|
|
|
|
|
|
|
usbtrace(TRACE_EPRESUME, epno);
|
|
|
|
privep->stalled = false;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Point to the appropriate EP register */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
regaddr = PIC32MX_USB_EP(epno);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Clear the STALL bit in the UEP register */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
regval = pic32mx_getreg(regaddr);
|
|
|
|
regval &= ~USB_EP_EPSTALL;
|
|
|
|
pic32mx_putreg(regval, regaddr);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Check for an IN endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
if (USB_ISEPIN(ep->eplog))
|
2011-12-28 20:40:37 +01:00
|
|
|
{
|
|
|
|
/* If the endpoint is an IN endpoint then we need to return it
|
|
|
|
* to the CPU and reset the DTS bit so that the next transfer
|
|
|
|
* is correct
|
|
|
|
*/
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
/* Toggle over the to the next buffer */
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_NEXT_PINGPONG;
|
2011-12-28 20:40:37 +01:00
|
|
|
#endif
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA1);
|
|
|
|
|
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
2011-12-28 20:40:37 +01:00
|
|
|
|
|
|
|
/* Restart any queued write requests */
|
|
|
|
|
|
|
|
(void)pic32mx_wrrequest(priv, privep);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-12-28 20:40:37 +01:00
|
|
|
/* If the endpoint was an OUT endpoint then we need to give
|
2011-12-29 21:46:38 +01:00
|
|
|
* control of the endpoint back to the USB so that the
|
2011-12-28 20:40:37 +01:00
|
|
|
* function driver can receive the data as they expected.
|
|
|
|
* Also need to set the DTS bit so the next packet will be
|
|
|
|
* correct.
|
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
#ifdef CONFIG_USB_PINGPONG
|
|
|
|
/* Toggle over the to the next buffer */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status ^= USB_NEXT_PINGPONG;
|
2011-12-28 20:40:37 +01:00
|
|
|
#endif
|
2011-12-29 21:46:38 +01:00
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA1 | USB_BDT_DTS);
|
|
|
|
|
|
|
|
/* Then give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
2011-12-28 20:40:37 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle the stall condition */
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_EPSTALL, epno);
|
|
|
|
privep->stalled = true;
|
|
|
|
|
2011-12-28 20:40:37 +01:00
|
|
|
/* Then STALL the endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
status = bdt->status & ~(USB_BDT_BYTECOUNT_MASK | USB_BDT_DATA01);
|
|
|
|
status |= (USB_BDT_UOWN | USB_BDT_DATA0 | USB_BDT_DTS | USB_BDT_BSTALL);
|
|
|
|
|
|
|
|
/* And give the BDT to the USB */
|
|
|
|
|
|
|
|
regdbg("EP%d BDT OUT {%08x, %08x}\n", epno, status, bdt->addr);
|
|
|
|
|
|
|
|
bdt->status = status;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2011-12-29 21:46:38 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_epstall
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_epstall(struct usbdev_ep_s *ep, bool resume)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
irqstate_t flags;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!ep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Recover pointers */
|
|
|
|
|
|
|
|
privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
|
|
|
|
/* STALL or RESUME the endpoint */
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
usbtrace(resume ? TRACE_EPRESUME : TRACE_EPSTALL, USB_EPNO(ep->eplog));
|
|
|
|
|
|
|
|
/* Special case EP0. When we stall EP0 we have to stall both the IN and
|
|
|
|
* OUT BDTs.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (USB_EPNO(ep->eplog) == 0)
|
|
|
|
{
|
|
|
|
ret = pic32mx_epbdtstall(ep, privep->bdtin, resume, true);
|
|
|
|
if (ret == OK)
|
|
|
|
{
|
|
|
|
ret = pic32mx_epbdtstall(ep, privep->bdtout, resume, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise, select the BDT for the endpoint direction */
|
|
|
|
|
|
|
|
else if (USB_ISEPIN(ep->eplog))
|
|
|
|
{
|
|
|
|
/* It is an IN endpoint */
|
|
|
|
|
|
|
|
ret = pic32mx_epbdtstall(ep, privep->bdtin, resume, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* It is an OUT endpoint */
|
|
|
|
|
|
|
|
ret = pic32mx_epbdtstall(ep, privep->bdtout, resume, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
irqrestore(flags);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Device Controller Operations
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_allocep
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static struct usbdev_ep_s *pic32mx_allocep(struct usbdev_s *dev, uint8_t epno,
|
2011-12-30 01:09:43 +01:00
|
|
|
bool epin, uint8_t eptype)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
struct pic32mx_usbdev_s *priv = (struct pic32mx_usbdev_s *)dev;
|
|
|
|
struct pic32mx_ep_s *privep = NULL;
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t epset = PIC32MX_ENDP_ALLSET;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
usbtrace(TRACE_DEVALLOCEP, (uint16_t)epno);
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!dev)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Ignore any direction bits in the logical address */
|
|
|
|
|
|
|
|
epno = USB_EPNO(epno);
|
|
|
|
|
|
|
|
/* A logical address of 0 means that any endpoint will do */
|
|
|
|
|
|
|
|
if (epno > 0)
|
|
|
|
{
|
|
|
|
/* Otherwise, we will return the endpoint structure only for the requested
|
|
|
|
* 'logical' endpoint. All of the other checks will still be performed.
|
|
|
|
*
|
|
|
|
* First, verify that the logical endpoint is in the range supported by
|
|
|
|
* by the hardware.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (epno >= PIC32MX_NENDPOINTS)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BADEPNO), (uint16_t)epno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert the logical address to a physical OUT endpoint address and
|
|
|
|
* remove all of the candidate endpoints from the bitset except for the
|
|
|
|
* the IN/OUT pair for this logical address.
|
|
|
|
*/
|
|
|
|
|
|
|
|
epset = PIC32MX_ENDP_BIT(epno);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if the selected endpoint number is available */
|
|
|
|
|
|
|
|
privep = pic32mx_epreserve(priv, epset);
|
|
|
|
if (!privep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_EPRESERVE), (uint16_t)epset);
|
2011-12-30 01:09:43 +01:00
|
|
|
return NULL;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return &privep->ep;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_freeep
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_freeep(struct usbdev_s *dev, struct usbdev_ep_s *ep)
|
|
|
|
{
|
|
|
|
struct pic32mx_usbdev_s *priv;
|
|
|
|
struct pic32mx_ep_s *privep;
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!dev || !ep)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
2011-12-30 01:09:43 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
priv = (struct pic32mx_usbdev_s *)dev;
|
|
|
|
privep = (struct pic32mx_ep_s *)ep;
|
|
|
|
usbtrace(TRACE_DEVFREEEP, (uint16_t)USB_EPNO(ep->eplog));
|
2011-12-30 01:09:43 +01:00
|
|
|
DEBUGASSERT(priv && privep);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Disable the endpoint */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
(void)pic32mx_epdisable(ep);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Mark the endpoint as available */
|
|
|
|
|
|
|
|
pic32mx_epunreserve(priv, privep);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_getframe
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_getframe(struct usbdev_s *dev)
|
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t frml;
|
|
|
|
uint16_t frmh;
|
|
|
|
uint16_t tmp;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!dev)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Return the last frame number detected by the hardware. Thr FRMH/L
|
|
|
|
* registers are updated with the current frame number whenever a SOF
|
|
|
|
* TOKEN is received.
|
|
|
|
*/
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
/* Loop until we can be sure that there was no wrap from the FRML
|
|
|
|
* to the FRMH register.
|
|
|
|
*/
|
|
|
|
|
|
|
|
frmh = pic32mx_getreg(PIC32MX_USB_FRMH) & USB_FRMH_MASK;
|
|
|
|
frml = pic32mx_getreg(PIC32MX_USB_FRML) & USB_FRML_MASK;
|
|
|
|
tmp = pic32mx_getreg(PIC32MX_USB_FRMH) & USB_FRMH_MASK;
|
|
|
|
}
|
|
|
|
while (frmh != tmp);
|
|
|
|
|
|
|
|
/* Combine to for the full 11-bit value */
|
|
|
|
|
|
|
|
tmp = (frmh) << 8 | frml;
|
|
|
|
usbtrace(TRACE_DEVGETFRAME, tmp);
|
|
|
|
return tmp;
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_wakeup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_wakeup(struct usbdev_s *dev)
|
|
|
|
{
|
|
|
|
struct pic32mx_usbdev_s *priv = (struct pic32mx_usbdev_s *)dev;
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVWAKEUP, 0);
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!dev)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Resume normal operation. */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_resume(priv);
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_selfpowered
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int pic32mx_selfpowered(struct usbdev_s *dev, bool selfpowered)
|
|
|
|
{
|
|
|
|
struct pic32mx_usbdev_s *priv = (struct pic32mx_usbdev_s *)dev;
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVSELFPOWERED, (uint16_t)selfpowered);
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!dev)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
priv->selfpowered = selfpowered;
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Initialization/Reset
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_reset
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_reset(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
int epno;
|
|
|
|
|
|
|
|
/* Tell the class driver that we are disconnected. The class driver
|
|
|
|
* should then accept any new configurations.
|
|
|
|
*/
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
if (priv->driver)
|
|
|
|
{
|
|
|
|
CLASS_DISCONNECT(priv->driver, &priv->usbdev);
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Reset the device state structure */
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
priv->ctrlstate = CTRLSTATE_WAITSETUP;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Reset endpoints */
|
|
|
|
|
|
|
|
for (epno = 0; epno < PIC32MX_NENDPOINTS; epno++)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep = &priv->eplist[epno];
|
|
|
|
|
|
|
|
/* Cancel any queued requests. Since they are canceled
|
|
|
|
* with status -ESHUTDOWN, then will not be requeued
|
|
|
|
* until the configuration is reset. NOTE: This should
|
|
|
|
* not be necessary... the CLASS_DISCONNECT above should
|
|
|
|
* result in the class implementation calling pic32mx_epdisable
|
|
|
|
* for each of its configured endpoints.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_cancelrequests(privep);
|
|
|
|
|
|
|
|
/* Reset endpoint status */
|
|
|
|
|
|
|
|
privep->stalled = false;
|
|
|
|
privep->halted = false;
|
|
|
|
privep->txbusy = false;
|
|
|
|
privep->txnullpkt = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Re-configure the USB controller in its initial, unconnected state */
|
|
|
|
|
|
|
|
pic32mx_hwreset(priv);
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_attach
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_attach(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
uint16_t regval;
|
|
|
|
|
|
|
|
/* Check if we are in the detached state */
|
|
|
|
|
|
|
|
if (priv->devstate == DEVSTATE_DETACHED)
|
|
|
|
{
|
|
|
|
/* Initialize registers to known states. */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_CON);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Mask all USB interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_IE);
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
/* Configure things like: pull ups, full/low-speed mode,
|
2011-12-28 01:49:48 +01:00
|
|
|
* set the ping pong mode, and set internal transceiver
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_CNFG1);
|
|
|
|
pic32mx_putreg(ERROR_INTERRUPTS, PIC32MX_USB_EIE);
|
|
|
|
pic32mx_putreg(NORMAL_INTERRUPTS, PIC32MX_USB_IE);
|
|
|
|
|
|
|
|
/* Enable the USB module and attach to bus */
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_CON);
|
|
|
|
if ((regval & USB_CON_USBEN) == 0)
|
|
|
|
{
|
|
|
|
pic32mx_putreg(regval | USB_CON_USBEN, PIC32MX_USB_CON);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while ((regval & USB_CON_USBEN) == 0);
|
|
|
|
|
|
|
|
/* Enable OTG */
|
|
|
|
|
|
|
|
#ifdef CONFIG_USBOTG
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USBOTG_CON);
|
|
|
|
regval |= (USBOTG_CON_DPPULUP | USBOTG_CON_OTGEN);
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USBOTG_CON);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Transition to the attached state */
|
|
|
|
|
|
|
|
priv->devstate = DEVSTATE_ATTACHED;
|
|
|
|
priv->usbdev.speed = USB_SPEED_FULL;
|
|
|
|
|
|
|
|
/* Enable USB interrupts */
|
|
|
|
|
|
|
|
up_enable_irq(PIC32MX_IRQSRC_USB);
|
|
|
|
|
|
|
|
/* Enable pull-up to connect the device. The host should enumerate us
|
|
|
|
* some time after this
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_usbpullup(&priv->usbdev, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_detach
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_detach(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
/* Disable USB interrupts at the interrupt controller */
|
|
|
|
|
|
|
|
up_disable_irq(PIC32MX_IRQSRC_USB);
|
|
|
|
|
|
|
|
/* Disable the USB controller and detach from the bus. */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_CON);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Mask all USB interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_IE);
|
|
|
|
|
|
|
|
/* We are now in the detached state */
|
|
|
|
|
|
|
|
priv->devstate = DEVSTATE_DETACHED;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_USBOTG
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Disable the D+ Pullup */
|
|
|
|
|
|
|
|
U1OTGCONbits.DPPULUP = 0;
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Disable and deactivate HNP */
|
|
|
|
#warning Missing Logic
|
|
|
|
|
|
|
|
/* Check if the ID Pin Changed State */
|
|
|
|
|
|
|
|
if ((pic32mx_getreg(PIC32MX_USBOTG_IR) & pic32mx_getreg(PIC32MX_USBOTG_IE) & USBOTG_INT_ID) != 0)
|
|
|
|
{
|
|
|
|
/* Re-detect & Initialize */
|
|
|
|
#warning "Missing logic"
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Clear ID Interrupt Flag */
|
|
|
|
|
|
|
|
pic32mx_putreg(USBOTG_INT_ID, PIC32MX_USBOTG_IR);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_hwreset
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_hwreset(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
uint16_t regval;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Clear all pending USB interrupts */
|
|
|
|
|
|
|
|
pic32mx_putreg(USB_EINT_ALL, PIC32MX_USB_EIR);
|
|
|
|
pic32mx_putreg(USB_INT_ALL, PIC32MX_USB_IR);
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
/* Reset configuration and enable interrrupts */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_CNFG1);
|
|
|
|
pic32mx_putreg(ERROR_INTERRUPTS, PIC32MX_USB_EIE);
|
2012-01-18 00:02:40 +01:00
|
|
|
pic32mx_putreg(NORMAL_INTERRUPTS, PIC32MX_USB_IE);
|
2011-12-30 01:09:43 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Power up the USB module */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_PWRC);
|
|
|
|
regval |= USB_PWRC_USBPWR;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_PWRC);
|
|
|
|
|
|
|
|
/* Set the address of the buffer descriptor table (BDT) */
|
|
|
|
|
|
|
|
regval = (uint16_t)((PHYS_ADDR(g_bdt)) >> 8);
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_BDTP1);
|
|
|
|
|
|
|
|
/* Assert reset request to all of the Ping Pong buffer pointers */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_CON);
|
|
|
|
regval |= USB_CON_PPBRST;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
|
|
|
|
|
|
|
/* Reset to default address */
|
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_ADDR);
|
2011-12-28 20:40:37 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Clear all of the endpoint control registers */
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
for (i = 0; i < PIC32MX_NENDPOINTS; i++)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_EP(i));
|
|
|
|
}
|
2011-12-28 20:40:37 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Bring the ping pong buffer pointers out of reset */
|
|
|
|
|
|
|
|
regval &= ~USB_CON_PPBRST;
|
2011-12-28 20:40:37 +01:00
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Clear all of the buffer descriptor table (BDT) entries */
|
|
|
|
|
|
|
|
memset((void*)g_bdt, 0, sizeof(g_bdt));
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Initialize EP0 as a Ctrl EP */
|
|
|
|
|
2011-12-30 01:09:43 +01:00
|
|
|
pic32mx_putreg(PIC32MX_EP_CONTROL, PIC32MX_USB_EP0);
|
2011-12-30 15:54:43 +01:00
|
|
|
regdbg("PIC32MX_USB_EP0: %04x\n", getreg16(PIC32MX_USB_EP0));
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Flush any pending transactions */
|
|
|
|
|
|
|
|
while ((pic32mx_getreg(PIC32MX_USB_IR) & USB_INT_TRN) != 0)
|
|
|
|
{
|
|
|
|
pic32mx_putreg(USB_INT_TRN, PIC32MX_USB_IR);
|
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Make sure packet processing is enabled */
|
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_CON);
|
|
|
|
regval &= ~USB_CON_PKTDIS;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_CON);
|
|
|
|
|
|
|
|
/* Get ready for the first packet */
|
|
|
|
|
2011-12-29 16:55:02 +01:00
|
|
|
priv->eplist[EP0].bdtin = &g_bdt[EP0_IN_EVEN];
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Indicate that we are now in the detached state */
|
|
|
|
|
|
|
|
priv->devstate = DEVSTATE_DETACHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_hwsetup
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_hwsetup(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
|
|
|
int epno;
|
|
|
|
|
|
|
|
/* Disconnect the device / disable the pull-up. We don't want the
|
|
|
|
* host to enumerate us until the class driver is registered.
|
2012-01-07 00:41:20 +01:00
|
|
|
*/
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_usbpullup(&priv->usbdev, false);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Initialize the device state structure. NOTE: many fields
|
|
|
|
* have the initial value of zero and, hence, are not explicitly
|
|
|
|
* initialized here.
|
|
|
|
*/
|
|
|
|
|
|
|
|
memset(priv, 0, sizeof(struct pic32mx_usbdev_s));
|
|
|
|
priv->usbdev.ops = &g_devops;
|
|
|
|
priv->usbdev.ep0 = &priv->eplist[EP0].ep;
|
|
|
|
priv->epavail = PIC32MX_ENDP_ALLSET & ~PIC32MX_ENDP_BIT(EP0);
|
2011-12-28 20:40:37 +01:00
|
|
|
priv->rwakeup = 1;
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/* Initialize the endpoint list */
|
|
|
|
|
|
|
|
for (epno = 0; epno < PIC32MX_NENDPOINTS; epno++)
|
|
|
|
{
|
|
|
|
struct pic32mx_ep_s *privep = &priv->eplist[epno];
|
|
|
|
|
|
|
|
/* Set endpoint operations, reference to driver structure (not
|
|
|
|
* really necessary because there is only one controller), and
|
|
|
|
* the (physical) endpoint number which is just the index to the
|
|
|
|
* endpoint.
|
|
|
|
*/
|
|
|
|
|
|
|
|
privep->ep.ops = &g_epops;
|
|
|
|
privep->dev = priv;
|
|
|
|
privep->ep.eplog = epno;
|
|
|
|
|
|
|
|
/* We will use a fixed maxpacket size for all endpoints (perhaps
|
|
|
|
* ISOC endpoints could have larger maxpacket???). A smaller
|
|
|
|
* packet size can be selected when the endpoint is configured.
|
|
|
|
*/
|
|
|
|
|
|
|
|
privep->ep.maxpacket = PIC32MX_MAXPACKET_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Select a smaller endpoint size for EP0 */
|
|
|
|
|
|
|
|
#if PIC32MX_EP0MAXPACKET < PIC32MX_MAXPACKET_SIZE
|
|
|
|
priv->eplist[EP0].ep.maxpacket = PIC32MX_EP0MAXPACKET;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_hwshutdown
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void pic32mx_hwshutdown(struct pic32mx_usbdev_s *priv)
|
|
|
|
{
|
2011-12-30 01:09:43 +01:00
|
|
|
uint16_t regval;
|
|
|
|
|
|
|
|
/* Put the hardware in its normal, unconnected state */
|
|
|
|
|
|
|
|
pic32mx_reset(priv);
|
2011-12-28 01:49:48 +01:00
|
|
|
priv->usbdev.speed = USB_SPEED_UNKNOWN;
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
/* Disable all interrupts and force the USB controller into reset */
|
2011-12-30 01:09:43 +01:00
|
|
|
|
|
|
|
pic32mx_putreg(0, PIC32MX_USB_EIE);
|
2012-01-18 00:02:40 +01:00
|
|
|
pic32mx_putreg(0, PIC32MX_USB_IE);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
/* Clear any pending interrupts */
|
2011-12-30 01:09:43 +01:00
|
|
|
|
|
|
|
pic32mx_putreg(USB_EINT_ALL, PIC32MX_USB_EIR);
|
|
|
|
pic32mx_putreg(USB_INT_ALL, PIC32MX_USB_IR);
|
2011-12-28 01:49:48 +01:00
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
/* Disconnect the device / disable the pull-up */
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
pic32mx_usbpullup(&priv->usbdev, false);
|
2012-01-07 00:41:20 +01:00
|
|
|
|
2011-12-28 01:49:48 +01:00
|
|
|
/* Power down the USB controller */
|
2011-12-30 01:09:43 +01:00
|
|
|
|
|
|
|
regval = pic32mx_getreg(PIC32MX_USB_PWRC);
|
|
|
|
regval &= ~USB_PWRC_USBPWR;
|
|
|
|
pic32mx_putreg(regval, PIC32MX_USB_PWRC);
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: up_usbinitialize
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Initialize the USB driver
|
|
|
|
*
|
|
|
|
* Input Parameters:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Returned Value:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
void up_usbinitialize(void)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVINIT, 0);
|
|
|
|
|
|
|
|
/* Power up the USB controller, but leave it in the reset state */
|
|
|
|
|
|
|
|
pic32mx_hwsetup(priv);
|
|
|
|
|
2012-01-05 00:06:20 +01:00
|
|
|
/* Attach USB controller interrupt handler. The hardware will not be
|
2011-12-28 01:49:48 +01:00
|
|
|
* initialized and interrupts will not be enabled until the class device
|
|
|
|
* driver is bound. Getting the IRQs here only makes sure that we have
|
|
|
|
* them when we need them later.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (irq_attach(PIC32MX_IRQ_USB, pic32mx_interrupt) != 0)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_IRQREGISTRATION),
|
|
|
|
(uint16_t)PIC32MX_IRQ_USB);
|
2012-01-05 00:06:20 +01:00
|
|
|
up_usbuninitialize();
|
2011-12-28 01:49:48 +01:00
|
|
|
}
|
2012-01-07 00:41:20 +01:00
|
|
|
}
|
2011-12-28 01:49:48 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: up_usbuninitialize
|
|
|
|
* Description:
|
|
|
|
* Initialize the USB driver
|
|
|
|
* Input Parameters:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Returned Value:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-01-07 00:41:20 +01:00
|
|
|
void up_usbuninitialize(void)
|
2011-12-28 01:49:48 +01:00
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
irqstate_t flags;
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
usbtrace(TRACE_DEVUNINIT, 0);
|
|
|
|
|
|
|
|
/* Disable and detach the USB IRQs */
|
|
|
|
|
|
|
|
up_disable_irq(PIC32MX_IRQSRC_USB);
|
|
|
|
irq_detach(PIC32MX_IRQ_USB);
|
|
|
|
|
|
|
|
if (priv->driver)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_DRIVERREGISTERED), 0);
|
|
|
|
usbdev_unregister(priv->driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put the hardware in an inactive state */
|
|
|
|
|
|
|
|
pic32mx_hwshutdown(priv);
|
|
|
|
irqrestore(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: usbdev_register
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Register a USB device class driver. The class driver's bind() method will be
|
|
|
|
* called to bind it to a USB device driver.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int usbdev_register(struct usbdevclass_driver_s *driver)
|
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVREGISTER, 0);
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (!driver || !driver->ops->bind || !driver->ops->unbind ||
|
|
|
|
!driver->ops->disconnect || !driver->ops->setup)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->driver)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_DRIVER), 0);
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* First hook up the driver */
|
|
|
|
|
|
|
|
priv->driver = driver;
|
|
|
|
|
|
|
|
/* Then bind the class driver */
|
|
|
|
|
|
|
|
ret = CLASS_BIND(driver, &priv->usbdev);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_BINDFAILED), (uint16_t)-ret);
|
|
|
|
priv->driver = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Setup the USB controller in it initial unconnected state */
|
|
|
|
|
|
|
|
pic32mx_hwreset(priv);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: usbdev_unregister
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Un-register usbdev class driver. If the USB device is connected to a
|
|
|
|
* USB host, it will first disconnect(). The driver is also requested to
|
|
|
|
* unbind() and clean up any device state, before this procedure finally
|
|
|
|
* returns.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int usbdev_unregister(struct usbdevclass_driver_s *driver)
|
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
irqstate_t flags;
|
|
|
|
|
|
|
|
usbtrace(TRACE_DEVUNREGISTER, 0);
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
|
|
if (driver != priv->driver)
|
|
|
|
{
|
|
|
|
usbtrace(TRACE_DEVERROR(PIC32MX_TRACEERR_INVALIDPARMS), 0);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Reset the hardware and cancel all requests. All requests must be
|
|
|
|
* canceled while the class driver is still bound.
|
|
|
|
*/
|
|
|
|
|
|
|
|
flags = irqsave();
|
|
|
|
pic32mx_reset(priv);
|
|
|
|
|
|
|
|
/* Unbind the class driver */
|
|
|
|
|
|
|
|
CLASS_UNBIND(driver, &priv->usbdev);
|
|
|
|
|
|
|
|
/* Disable USB controller interrupts (but keep them attached) */
|
|
|
|
|
|
|
|
up_disable_irq(PIC32MX_IRQSRC_USB);
|
|
|
|
|
|
|
|
/* Put the hardware in an inactive state. Then bring the hardware back up
|
|
|
|
* in the reset state (this is probably not necessary, the pic32mx_reset()
|
|
|
|
* call above was probably sufficient).
|
|
|
|
*/
|
|
|
|
|
|
|
|
pic32mx_hwshutdown(priv);
|
|
|
|
pic32mx_hwsetup(priv);
|
|
|
|
|
|
|
|
/* Unhook the driver */
|
|
|
|
|
|
|
|
priv->driver = NULL;
|
|
|
|
irqrestore(flags);
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pic32mx_usbattach and pic32mx_usbdetach
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* The USB stack must be notified when the device is attached or detached
|
|
|
|
* by calling one of these functions.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
void pic32mx_usbattach(void)
|
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
pic32mx_attach(priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pic32mx_usbdetach(void)
|
|
|
|
{
|
|
|
|
/* For now there is only one USB controller, but we will always refer to
|
|
|
|
* it using a pointer to make any future ports to multiple USB controllers
|
|
|
|
* easier.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct pic32mx_usbdev_s *priv = &g_usbdev;
|
|
|
|
pic32mx_detach(priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_USBDEV && CONFIG_PIC32MX_USB */
|