USB Host Shield 2.0
 All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
UsbCore.h
Go to the documentation of this file.
1 /*
2  * File: UsbCore.h
3  * Author: xxxajk
4  *
5  * Created on September 29, 2013, 9:25 PM
6  */
7 
8 #if !defined(_usb_h_) || defined(USBCORE_H)
9 #error "Never include UsbCore.h directly; include Usb.h instead"
10 #else
11 #define USBCORE_H
12 
13 // Not used anymore? If anyone uses this, please let us know so that this may be
14 // moved to the proper place, settings.h.
15 //#define USB_METHODS_INLINE
16 
17 /* shield pins. First parameter - SS pin, second parameter - INT pin */
18 #ifdef BOARD_BLACK_WIDDOW
19 typedef MAX3421e<P6, P3> MAX3421E; // Black Widow
20 #elif defined(CORE_TEENSY) && (defined(__AVR_AT90USB646__) || defined(__AVR_AT90USB1286__))
21 typedef MAX3421e<P9, P8> MAX3421E; // Teensy++ 1.0 and 2.0
22 #elif defined(BOARD_MEGA_ADK)
23 typedef MAX3421e<P53, P54> MAX3421E; // Arduino Mega ADK
24 #elif defined(ARDUINO_AVR_BALANDUINO)
25 typedef MAX3421e<P20, P19> MAX3421E; // Balanduino
26 #else
27 typedef MAX3421e<P10, P9> MAX3421E; // Official Arduinos (UNO, Duemilanove, Mega, 2560, Leonardo etc.) or Teensy 2.0 and 3.0
28 #endif
29 
30 /* Common setup data constant combinations */
31 #define bmREQ_GET_DESCR USB_SETUP_DEVICE_TO_HOST|USB_SETUP_TYPE_STANDARD|USB_SETUP_RECIPIENT_DEVICE //get descriptor request type
32 #define bmREQ_SET USB_SETUP_HOST_TO_DEVICE|USB_SETUP_TYPE_STANDARD|USB_SETUP_RECIPIENT_DEVICE //set request type for all but 'set feature' and 'set interface'
33 #define bmREQ_CL_GET_INTF USB_SETUP_DEVICE_TO_HOST|USB_SETUP_TYPE_CLASS|USB_SETUP_RECIPIENT_INTERFACE //get interface request type
34 
35 // D7 data transfer direction (0 - host-to-device, 1 - device-to-host)
36 // D6-5 Type (0- standard, 1 - class, 2 - vendor, 3 - reserved)
37 // D4-0 Recipient (0 - device, 1 - interface, 2 - endpoint, 3 - other, 4..31 - reserved)
38 
39 // USB Device Classes
40 #define USB_CLASS_USE_CLASS_INFO 0x00 // Use Class Info in the Interface Descriptors
41 #define USB_CLASS_AUDIO 0x01 // Audio
42 #define USB_CLASS_COM_AND_CDC_CTRL 0x02 // Communications and CDC Control
43 #define USB_CLASS_HID 0x03 // HID
44 #define USB_CLASS_PHYSICAL 0x05 // Physical
45 #define USB_CLASS_IMAGE 0x06 // Image
46 #define USB_CLASS_PRINTER 0x07 // Printer
47 #define USB_CLASS_MASS_STORAGE 0x08 // Mass Storage
48 #define USB_CLASS_HUB 0x09 // Hub
49 #define USB_CLASS_CDC_DATA 0x0a // CDC-Data
50 #define USB_CLASS_SMART_CARD 0x0b // Smart-Card
51 #define USB_CLASS_CONTENT_SECURITY 0x0d // Content Security
52 #define USB_CLASS_VIDEO 0x0e // Video
53 #define USB_CLASS_PERSONAL_HEALTH 0x0f // Personal Healthcare
54 #define USB_CLASS_DIAGNOSTIC_DEVICE 0xdc // Diagnostic Device
55 #define USB_CLASS_WIRELESS_CTRL 0xe0 // Wireless Controller
56 #define USB_CLASS_MISC 0xef // Miscellaneous
57 #define USB_CLASS_APP_SPECIFIC 0xfe // Application Specific
58 #define USB_CLASS_VENDOR_SPECIFIC 0xff // Vendor Specific
59 
60 // Additional Error Codes
61 #define USB_DEV_CONFIG_ERROR_DEVICE_NOT_SUPPORTED 0xD1
62 #define USB_DEV_CONFIG_ERROR_DEVICE_INIT_INCOMPLETE 0xD2
63 #define USB_ERROR_UNABLE_TO_REGISTER_DEVICE_CLASS 0xD3
64 #define USB_ERROR_OUT_OF_ADDRESS_SPACE_IN_POOL 0xD4
65 #define USB_ERROR_HUB_ADDRESS_OVERFLOW 0xD5
66 #define USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL 0xD6
67 #define USB_ERROR_EPINFO_IS_NULL 0xD7
68 #define USB_ERROR_INVALID_ARGUMENT 0xD8
69 #define USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE 0xD9
70 #define USB_ERROR_INVALID_MAX_PKT_SIZE 0xDA
71 #define USB_ERROR_EP_NOT_FOUND_IN_TBL 0xDB
72 #define USB_ERROR_CONFIG_REQUIRES_ADDITIONAL_RESET 0xE0
73 #define USB_ERROR_FailGetDevDescr 0xE1
74 #define USB_ERROR_FailSetDevTblEntry 0xE2
75 #define USB_ERROR_FailGetConfDescr 0xE3
76 #define USB_ERROR_TRANSFER_TIMEOUT 0xFF
77 
78 #define USB_XFER_TIMEOUT 10000 //30000 // (5000) USB transfer timeout in milliseconds, per section 9.2.6.1 of USB 2.0 spec
79 //#define USB_NAK_LIMIT 32000 //NAK limit for a transfer. 0 means NAKs are not counted
80 #define USB_RETRY_LIMIT 3 // 3 retry limit for a transfer
81 #define USB_SETTLE_DELAY 200 //settle delay in milliseconds
82 
83 #define USB_NUMDEVICES 16 //number of USB devices
84 //#define HUB_MAX_HUBS 7 // maximum number of hubs that can be attached to the host controller
85 #define HUB_PORT_RESET_DELAY 20 // hub port reset delay 10 ms recomended, can be up to 20 ms
86 
87 /* USB state machine states */
88 #define USB_STATE_MASK 0xf0
89 
90 #define USB_STATE_DETACHED 0x10
91 #define USB_DETACHED_SUBSTATE_INITIALIZE 0x11
92 #define USB_DETACHED_SUBSTATE_WAIT_FOR_DEVICE 0x12
93 #define USB_DETACHED_SUBSTATE_ILLEGAL 0x13
94 #define USB_ATTACHED_SUBSTATE_SETTLE 0x20
95 #define USB_ATTACHED_SUBSTATE_RESET_DEVICE 0x30
96 #define USB_ATTACHED_SUBSTATE_WAIT_RESET_COMPLETE 0x40
97 #define USB_ATTACHED_SUBSTATE_WAIT_SOF 0x50
98 #define USB_ATTACHED_SUBSTATE_WAIT_RESET 0x51
99 #define USB_ATTACHED_SUBSTATE_GET_DEVICE_DESCRIPTOR_SIZE 0x60
100 #define USB_STATE_ADDRESSING 0x70
101 #define USB_STATE_CONFIGURING 0x80
102 #define USB_STATE_RUNNING 0x90
103 #define USB_STATE_ERROR 0xa0
104 
106 public:
107  virtual uint8_t Init(uint8_t parent, uint8_t port, bool lowspeed) { return 0; }
108  virtual uint8_t ConfigureDevice(uint8_t parent, uint8_t port, bool lowspeed) {return 0; }
109  virtual uint8_t Release() { return 0; }
110  virtual uint8_t Poll() { return 0; }
111  virtual uint8_t GetAddress() { return 0; }
112  virtual void ResetHubPort(uint8_t port) { return; } // Note used for hubs only!
113  virtual boolean VIDPIDOK(uint16_t vid, uint16_t pid) { return false; }
114  virtual boolean DEVCLASSOK(uint8_t klass) { return false; }
115 };
116 
117 /* USB Setup Packet Structure */
118 typedef struct {
119 
120  union { // offset description
121  uint8_t bmRequestType; // 0 Bit-map of request type
122 
123  struct {
124  uint8_t recipient : 5; // Recipient of the request
125  uint8_t type : 2; // Type of request
126  uint8_t direction : 1; // Direction of data X-fer
127  } __attribute__((packed));
128  } ReqType_u;
129  uint8_t bRequest; // 1 Request
130 
131  union {
132  uint16_t wValue; // 2 Depends on bRequest
133 
134  struct {
135  uint8_t wValueLo;
136  uint8_t wValueHi;
137  } __attribute__((packed));
138  } wVal_u;
139  uint16_t wIndex; // 4 Depends on bRequest
140  uint16_t wLength; // 6 Depends on bRequest
141 } SETUP_PKT, *PSETUP_PKT __attribute__((packed));
142 
143 
144 
145 // Base class for incoming data parser
146 
148 public:
149  virtual void Parse(const uint16_t len, const uint8_t *pbuf, const uint16_t &offset) = 0;
150 };
151 
152 class USB : public MAX3421E {
154  USBDeviceConfig* devConfig[USB_NUMDEVICES];
155  uint8_t bmHubPre;
156 
157 public:
158  USB(void);
159 
160  void SetHubPreMask() {
161  bmHubPre |= bmHUBPRE;
162  };
163 
165  bmHubPre &= (~bmHUBPRE);
166  };
167 
169  return(AddressPool&) addrPool;
170  };
171 
173  for(uint8_t i = 0; i < USB_NUMDEVICES; i++) {
174  if(!devConfig[i]) {
175  devConfig[i] = pdev;
176  return 0;
177  }
178  }
180  };
181 
183  addrPool.ForEachUsbDevice(pfunc);
184  };
185  uint8_t getUsbTaskState(void);
186  void setUsbTaskState(uint8_t state);
187 
188  EpInfo* getEpInfoEntry(uint8_t addr, uint8_t ep);
189  uint8_t setEpInfoEntry(uint8_t addr, uint8_t epcount, EpInfo* eprecord_ptr);
190 
191  /* Control requests */
192  uint8_t getDevDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t* dataptr);
193  uint8_t getConfDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t conf, uint8_t* dataptr);
194 
195  uint8_t getConfDescr(uint8_t addr, uint8_t ep, uint8_t conf, USBReadParser *p);
196 
197  uint8_t getStrDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t index, uint16_t langid, uint8_t* dataptr);
198  uint8_t setAddr(uint8_t oldaddr, uint8_t ep, uint8_t newaddr);
199  uint8_t setConf(uint8_t addr, uint8_t ep, uint8_t conf_value);
200 
201  uint8_t ctrlData(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t* dataptr, boolean direction);
202  uint8_t ctrlStatus(uint8_t ep, boolean direction, uint16_t nak_limit);
203  uint8_t inTransfer(uint8_t addr, uint8_t ep, uint16_t *nbytesptr, uint8_t* data);
204  uint8_t outTransfer(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t* data);
205  uint8_t dispatchPkt(uint8_t token, uint8_t ep, uint16_t nak_limit);
206 
207  void Task(void);
208 
209  uint8_t DefaultAddressing(uint8_t parent, uint8_t port, bool lowspeed);
210  uint8_t Configuring(uint8_t parent, uint8_t port, bool lowspeed);
211  uint8_t ReleaseDevice(uint8_t addr);
212 
213  uint8_t ctrlReq(uint8_t addr, uint8_t ep, uint8_t bmReqType, uint8_t bRequest, uint8_t wValLo, uint8_t wValHi,
214  uint16_t wInd, uint16_t total, uint16_t nbytes, uint8_t* dataptr, USBReadParser *p);
215 
216 private:
217  void init();
218  uint8_t SetAddress(uint8_t addr, uint8_t ep, EpInfo **ppep, uint16_t &nak_limit);
219  uint8_t OutTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t nbytes, uint8_t *data);
220  uint8_t InTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t *nbytesptr, uint8_t *data);
221  uint8_t AttemptConfig(uint8_t driver, uint8_t parent, uint8_t port, bool lowspeed);
222 };
223 
224 #if 0 //defined(USB_METHODS_INLINE)
225 //get device descriptor
226 
227 inline uint8_t USB::getDevDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t* dataptr) {
228  return( ctrlReq(addr, ep, bmREQ_GET_DESCR, USB_REQUEST_GET_DESCRIPTOR, 0x00, USB_DESCRIPTOR_DEVICE, 0x0000, nbytes, dataptr));
229 }
230 //get configuration descriptor
231 
232 inline uint8_t USB::getConfDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t conf, uint8_t* dataptr) {
233  return( ctrlReq(addr, ep, bmREQ_GET_DESCR, USB_REQUEST_GET_DESCRIPTOR, conf, USB_DESCRIPTOR_CONFIGURATION, 0x0000, nbytes, dataptr));
234 }
235 //get string descriptor
236 
237 inline uint8_t USB::getStrDescr(uint8_t addr, uint8_t ep, uint16_t nuint8_ts, uint8_t index, uint16_t langid, uint8_t* dataptr) {
238  return( ctrlReq(addr, ep, bmREQ_GET_DESCR, USB_REQUEST_GET_DESCRIPTOR, index, USB_DESCRIPTOR_STRING, langid, nuint8_ts, dataptr));
239 }
240 //set address
241 
242 inline uint8_t USB::setAddr(uint8_t oldaddr, uint8_t ep, uint8_t newaddr) {
243  return( ctrlReq(oldaddr, ep, bmREQ_SET, USB_REQUEST_SET_ADDRESS, newaddr, 0x00, 0x0000, 0x0000, NULL));
244 }
245 //set configuration
246 
247 inline uint8_t USB::setConf(uint8_t addr, uint8_t ep, uint8_t conf_value) {
248  return( ctrlReq(addr, ep, bmREQ_SET, USB_REQUEST_SET_CONFIGURATION, conf_value, 0x00, 0x0000, 0x0000, NULL));
249 }
250 
251 #endif // defined(USB_METHODS_INLINE)
252 
253 #endif /* USBCORE_H */
254 
uint8_t getConfDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t conf, uint8_t *dataptr)
Definition: Usb.cpp:761
uint16_t wValue
Definition: UsbCore.h:132
virtual uint8_t Poll()
Definition: UsbCore.h:110
#define bmHUBPRE
Definition: max3421e.h:170
virtual uint8_t GetAddress()
Definition: UsbCore.h:111
EpInfo * getEpInfoEntry(uint8_t addr, uint8_t ep)
Definition: Usb.cpp:44
#define USB_DESCRIPTOR_STRING
Definition: usb_ch9.h:65
uint16_t wLength
Definition: UsbCore.h:140
virtual void Parse(const uint16_t len, const uint8_t *pbuf, const uint16_t &offset)=0
#define USB_REQUEST_GET_DESCRIPTOR
Definition: usb_ch9.h:37
#define USB_DESCRIPTOR_DEVICE
Definition: usb_ch9.h:63
uint8_t setConf(uint8_t addr, uint8_t ep, uint8_t conf_value)
Definition: Usb.cpp:795
#define USB_NUMDEVICES
Definition: UsbCore.h:83
uint8_t setEpInfoEntry(uint8_t addr, uint8_t epcount, EpInfo *eprecord_ptr)
Definition: Usb.cpp:64
virtual uint8_t ConfigureDevice(uint8_t parent, uint8_t port, bool lowspeed)
Definition: UsbCore.h:108
uint8_t bmRequestType
Definition: UsbCore.h:121
uint8_t ctrlReq(uint8_t addr, uint8_t ep, uint8_t bmReqType, uint8_t bRequest, uint8_t wValLo, uint8_t wValHi, uint16_t wInd, uint16_t total, uint16_t nbytes, uint8_t *dataptr, USBReadParser *p)
Definition: Usb.cpp:126
#define USB_DESCRIPTOR_CONFIGURATION
Definition: usb_ch9.h:64
void ResetHubPreMask()
Definition: UsbCore.h:164
uint8_t setAddr(uint8_t oldaddr, uint8_t ep, uint8_t newaddr)
Definition: Usb.cpp:790
uint8_t ctrlStatus(uint8_t ep, boolean direction, uint16_t nak_limit)
USB(void)
Definition: Usb.cpp:25
void(* UsbDeviceHandleFunc)(UsbDevice *pdev)
Definition: address.h:90
uint16_t wIndex
Definition: UsbCore.h:139
void SetHubPreMask()
Definition: UsbCore.h:160
Definition: address.h:32
void setUsbTaskState(uint8_t state)
Definition: Usb.cpp:40
uint8_t outTransfer(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t *data)
Definition: Usb.cpp:290
#define bmREQ_GET_DESCR
Definition: UsbCore.h:31
virtual void ResetHubPort(uint8_t port)
Definition: UsbCore.h:112
#define USB_ERROR_UNABLE_TO_REGISTER_DEVICE_CLASS
Definition: UsbCore.h:63
uint8_t getStrDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t index, uint16_t langid, uint8_t *dataptr)
Definition: Usb.cpp:785
#define USB_REQUEST_SET_ADDRESS
Definition: usb_ch9.h:36
MAX3421e< P10, P9 > MAX3421E
Definition: UsbCore.h:27
uint8_t getUsbTaskState(void)
Definition: Usb.cpp:36
uint8_t inTransfer(uint8_t addr, uint8_t ep, uint16_t *nbytesptr, uint8_t *data)
Definition: Usb.cpp:206
void Task(void)
Definition: Usb.cpp:422
virtual boolean DEVCLASSOK(uint8_t klass)
Definition: UsbCore.h:114
void ForEachUsbDevice(UsbDeviceHandleFunc pfunc)
Definition: address.h:185
uint8_t ctrlData(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t *dataptr, boolean direction)
uint8_t dispatchPkt(uint8_t token, uint8_t ep, uint16_t nak_limit)
Definition: Usb.cpp:374
virtual uint8_t Init(uint8_t parent, uint8_t port, bool lowspeed)
Definition: UsbCore.h:107
AddressPool & GetAddressPool()
Definition: UsbCore.h:168
uint8_t Configuring(uint8_t parent, uint8_t port, bool lowspeed)
Definition: Usb.cpp:647
Definition: UsbCore.h:152
virtual boolean VIDPIDOK(uint16_t vid, uint16_t pid)
Definition: UsbCore.h:113
uint8_t bRequest
Definition: UsbCore.h:129
virtual uint8_t Release()
Definition: UsbCore.h:109
uint8_t RegisterDeviceClass(USBDeviceConfig *pdev)
Definition: UsbCore.h:172
#define bmREQ_SET
Definition: UsbCore.h:32
void ForEachUsbDevice(UsbDeviceHandleFunc pfunc)
Definition: UsbCore.h:182
#define USB_REQUEST_SET_CONFIGURATION
Definition: usb_ch9.h:40
uint8_t wValueHi
Definition: UsbCore.h:136
uint8_t DefaultAddressing(uint8_t parent, uint8_t port, bool lowspeed)
Definition: Usb.cpp:528
uint8_t getDevDescr(uint8_t addr, uint8_t ep, uint16_t nbytes, uint8_t *dataptr)
defined(USB_METHODS_INLINE)
Definition: Usb.cpp:756
uint8_t wValueLo
Definition: UsbCore.h:135
uint8_t ReleaseDevice(uint8_t addr)
Definition: Usb.cpp:741