usb.h 8.71 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2008  Free Software Foundation, Inc.
 *
 *  GRUB is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#ifndef	GRUB_USB_H
#define	GRUB_USB_H	1

22
#include <grub/err.h>
23 24 25 26 27 28 29 30 31 32
#include <grub/usbdesc.h>
#include <grub/usbtrans.h>

typedef struct grub_usb_device *grub_usb_device_t;
typedef struct grub_usb_controller *grub_usb_controller_t;
typedef struct grub_usb_controller_dev *grub_usb_controller_dev_t;

typedef enum
  {
    GRUB_USB_ERR_NONE,
33
    GRUB_USB_ERR_WAIT,
34 35 36 37 38 39
    GRUB_USB_ERR_INTERNAL,
    GRUB_USB_ERR_STALL,
    GRUB_USB_ERR_DATA,
    GRUB_USB_ERR_NAK,
    GRUB_USB_ERR_BABBLE,
    GRUB_USB_ERR_TIMEOUT,
40
    GRUB_USB_ERR_BITSTUFF,
41 42
    GRUB_USB_ERR_UNRECOVERABLE,
    GRUB_USB_ERR_BADDEVICE
43 44 45 46 47 48 49 50 51 52
  } grub_usb_err_t;

typedef enum
  {
    GRUB_USB_SPEED_NONE,
    GRUB_USB_SPEED_LOW,
    GRUB_USB_SPEED_FULL,
    GRUB_USB_SPEED_HIGH
  } grub_usb_speed_t;

53 54 55 56
typedef int (*grub_usb_iterate_hook_t) (grub_usb_device_t dev, void *data);
typedef int (*grub_usb_controller_iterate_hook_t) (grub_usb_controller_t dev,
						   void *data);

57
/* Call HOOK with each device, until HOOK returns non-zero.  */
58
int grub_usb_iterate (grub_usb_iterate_hook_t hook, void *hook_data);
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75

grub_usb_err_t grub_usb_device_initialize (grub_usb_device_t dev);

grub_usb_err_t grub_usb_get_descriptor (grub_usb_device_t dev,
					grub_uint8_t type, grub_uint8_t index,
					grub_size_t size, char *data);

grub_usb_err_t grub_usb_clear_halt (grub_usb_device_t dev, int endpoint);


grub_usb_err_t grub_usb_set_configuration (grub_usb_device_t dev,
					   int configuration);

void grub_usb_controller_dev_register (grub_usb_controller_dev_t usb);

void grub_usb_controller_dev_unregister (grub_usb_controller_dev_t usb);

76 77
int grub_usb_controller_iterate (grub_usb_controller_iterate_hook_t hook,
				 void *hook_data);
78 79 80 81 82 83 84 85 86


grub_usb_err_t grub_usb_control_msg (grub_usb_device_t dev, grub_uint8_t reqtype,
				     grub_uint8_t request, grub_uint16_t value,
				     grub_uint16_t index, grub_size_t size,
				     char *data);

grub_usb_err_t
grub_usb_bulk_read (grub_usb_device_t dev,
87 88
		    struct grub_usb_desc_endp *endpoint,
		    grub_size_t size, char *data);
89 90
grub_usb_err_t
grub_usb_bulk_write (grub_usb_device_t dev,
91 92
		     struct grub_usb_desc_endp *endpoint,
		     grub_size_t size, char *data);
93 94 95 96 97

grub_usb_err_t
grub_usb_root_hub (grub_usb_controller_t controller);


98

99 100 101 102 103 104
/* XXX: All handled by libusb for now.  */
struct grub_usb_controller_dev
{
  /* The device name.  */
  const char *name;

105
  int (*iterate) (grub_usb_controller_iterate_hook_t hook, void *hook_data);
106

107 108 109 110 111 112 113 114 115
  grub_usb_err_t (*setup_transfer) (grub_usb_controller_t dev,
				    grub_usb_transfer_t transfer);

  grub_usb_err_t (*check_transfer) (grub_usb_controller_t dev,
				    grub_usb_transfer_t transfer,
				    grub_size_t *actual);

  grub_usb_err_t (*cancel_transfer) (grub_usb_controller_t dev,
				     grub_usb_transfer_t transfer);
116 117 118

  int (*hubports) (grub_usb_controller_t dev);

119 120
  grub_usb_err_t (*portstatus) (grub_usb_controller_t dev, unsigned int port,
				unsigned int enable);
121

starous's avatar
starous committed
122
  grub_usb_speed_t (*detect_dev) (grub_usb_controller_t dev, int port, int *changed);
123

Aleš Nesrsta's avatar
Aleš Nesrsta committed
124 125
  /* Per controller flag - port reset pending, don't do another reset */
  grub_uint64_t pending_reset;
126 127 128 129 130 131 132

  /* Max. number of transfer descriptors used per one bulk transfer */
  /* The reason is to prevent "exhausting" of TD by large bulk */
  /* transfer - number of TD is limited in USB host driver */
  /* Value is calculated/estimated in driver - some TDs should be */
  /* reserved for posible concurrent control or "interrupt" transfers */
  grub_size_t max_bulk_tds;
Aleš Nesrsta's avatar
Aleš Nesrsta committed
133
  
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
  /* The next host controller.  */
  struct grub_usb_controller_dev *next;
};

struct grub_usb_controller
{
  /* The underlying USB Host Controller device.  */
  grub_usb_controller_dev_t dev;

  /* Data used by the USB Host Controller Driver.  */
  void *data;
};


struct grub_usb_interface
{
  struct grub_usb_desc_if *descif;

  struct grub_usb_desc_endp *descendp;
153 154 155 156 157 158 159

  /* A driver is handling this interface. Do we need to support multiple drivers
     for single interface?
   */
  int attached;

  void (*detach_hook) (struct grub_usb_device *dev, int config, int interface);
160 161

  void *detach_data;
162 163 164 165 166 167 168 169 170 171 172
};

struct grub_usb_configuration
{
  /* Configuration descriptors .  */
  struct grub_usb_desc_config *descconf;

  /* Interfaces associated to this configuration.  */
  struct grub_usb_interface interf[32];
};

173 174 175 176 177 178 179 180 181 182 183 184
struct grub_usb_hub_port
{
  grub_uint64_t soft_limit_time;
  grub_uint64_t hard_limit_time;
  enum { 
    PORT_STATE_NORMAL = 0,
    PORT_STATE_WAITING_FOR_STABLE_POWER = 1,
    PORT_STATE_FAILED_DEVICE = 2,
    PORT_STATE_STABLE_POWER = 3,
  } state;
};

185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
struct grub_usb_device
{
  /* The device descriptor of this device.  */
  struct grub_usb_desc_device descdev;

  /* The controller the device is connected to.  */
  struct grub_usb_controller controller;

  /* Device configurations (after opening the device).  */
  struct grub_usb_configuration config[8];

  /* Device address.  */
  int addr;

  /* Device speed.  */
  grub_usb_speed_t speed;

202
  /* All descriptors are read if this is set to 1.  */
203 204 205
  int initialized;

  /* Data toggle values (used for bulk transfers only).  */
206
  int toggle[256];
207

208 209 210
  /* Used by libusb wrapper.  Schedulded for removal. */
  void *data;

211 212
  /* Hub information.  */

213 214 215 216 217
  /* Array of children for a hub.  */
  grub_usb_device_t *children;

  /* Number of hub ports.  */
  unsigned nports;
218

219 220
  struct grub_usb_hub_port *ports;

221 222 223 224 225
  grub_usb_transfer_t hub_transfer;

  grub_uint32_t statuschange;

  struct grub_usb_desc_endp *hub_endpoint;
226 227

  /* EHCI Split Transfer information */
228
  int split_hubport;
229

230
  int split_hubaddr;
231 232 233 234
};



235 236 237 238 239 240 241 242 243 244 245 246 247 248
typedef enum grub_usb_ep_type
  {
    GRUB_USB_EP_CONTROL,
    GRUB_USB_EP_ISOCHRONOUS,
    GRUB_USB_EP_BULK,
    GRUB_USB_EP_INTERRUPT
  } grub_usb_ep_type_t;

static inline enum grub_usb_ep_type
grub_usb_get_ep_type (struct grub_usb_desc_endp *ep)
{
  return ep->attrib & 3;
}

249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268
typedef enum
  {
    GRUB_USB_CLASS_NOTHERE,
    GRUB_USB_CLASS_AUDIO,
    GRUB_USB_CLASS_COMMUNICATION,
    GRUB_USB_CLASS_HID,
    GRUB_USB_CLASS_XXX,
    GRUB_USB_CLASS_PHYSICAL,
    GRUB_USB_CLASS_IMAGE,
    GRUB_USB_CLASS_PRINTER,
    GRUB_USB_CLASS_MASS_STORAGE,
    GRUB_USB_CLASS_HUB,
    GRUB_USB_CLASS_DATA_INTERFACE,
    GRUB_USB_CLASS_SMART_CARD,
    GRUB_USB_CLASS_CONTENT_SECURITY,
    GRUB_USB_CLASS_VIDEO
  } grub_usb_classes_t;

typedef enum
  {
269 270 271 272 273 274
    GRUB_USBMS_SUBCLASS_BULK = 0x06,
  	/* Experimental support for non-pure SCSI devices */
    GRUB_USBMS_SUBCLASS_RBC = 0x01,
    GRUB_USBMS_SUBCLASS_MMC2 = 0x02,
    GRUB_USBMS_SUBCLASS_UFI = 0x04,
    GRUB_USBMS_SUBCLASS_SFF8070 = 0x05
275 276 277 278
  } grub_usbms_subclass_t;

typedef enum
  {
starous's avatar
starous committed
279 280 281 282
    GRUB_USBMS_PROTOCOL_BULK = 0x50,
    /* Experimental support for Control/Bulk/Interrupt (CBI) devices */
    GRUB_USBMS_PROTOCOL_CBI = 0x00, /* CBI with interrupt */
    GRUB_USBMS_PROTOCOL_CB = 0x01  /* CBI wthout interrupt */
283 284 285 286 287 288 289 290 291 292 293
  } grub_usbms_protocol_t;

static inline struct grub_usb_desc_if *
grub_usb_get_config_interface (struct grub_usb_desc_config *config)
{
  struct grub_usb_desc_if *interf;

  interf = (struct grub_usb_desc_if *) (sizeof (*config) + (char *) config);
  return interf;
}

294 295 296 297 298 299
typedef int (*grub_usb_attach_hook_class) (grub_usb_device_t usbdev,
					   int configno, int interfno);

struct grub_usb_attach_desc
{
  struct grub_usb_attach_desc *next;
300
  struct grub_usb_attach_desc **prev;
301 302 303 304 305 306 307
  int class;
  grub_usb_attach_hook_class hook;
};

void grub_usb_register_attach_hook_class (struct grub_usb_attach_desc *desc);
void grub_usb_unregister_attach_hook_class (struct grub_usb_attach_desc *desc);

308
void grub_usb_poll_devices (int wait_for_completion);
309

starous's avatar
starous committed
310
void grub_usb_device_attach (grub_usb_device_t dev);
311
grub_usb_err_t
312
grub_usb_bulk_read_extended (grub_usb_device_t dev,
313 314
			     struct grub_usb_desc_endp *endpoint,
			     grub_size_t size, char *data,
315
			     int timeout, grub_size_t *actual);
316 317
grub_usb_transfer_t
grub_usb_bulk_read_background (grub_usb_device_t dev,
318 319
			       struct grub_usb_desc_endp *endpoint,
			       grub_size_t size, void *data);
320 321
grub_usb_err_t
grub_usb_check_transfer (grub_usb_transfer_t trans, grub_size_t *actual);
322 323
void
grub_usb_cancel_transfer (grub_usb_transfer_t trans);
starous's avatar
starous committed
324

325
#endif /* GRUB_USB_H */