#ifndef _USBDIVAR_H_
#define _USBDIVAR_H_
#include "bpfilter.h"
#if NBPFILTER > 0
#include <net/bpf.h>
#endif
#include <sys/timeout.h>
struct usb_dma_block;
struct usb_dma {
struct usb_dma_block *block;
u_int offs;
#define USB_DMA_COHERENT (1 << 0)
};
struct usbd_xfer;
struct usbd_pipe;
struct usbd_endpoint {
usb_endpoint_descriptor_t *edesc;
usb_endpoint_ss_comp_descriptor_t *esscd;
int refcnt;
int savedtoggle;
};
struct usbd_bus_methods {
usbd_status (*open_pipe)(struct usbd_pipe *);
int (*dev_setaddr)(struct usbd_device *, int);
void (*soft_intr)(void *);
void (*do_poll)(struct usbd_bus *);
struct usbd_xfer * (*allocx)(struct usbd_bus *);
void (*freex)(struct usbd_bus *, struct usbd_xfer *);
};
struct usbd_pipe_methods {
usbd_status (*transfer)(struct usbd_xfer *);
usbd_status (*start)(struct usbd_xfer *);
void (*abort)(struct usbd_xfer *);
void (*close)(struct usbd_pipe *);
void (*cleartoggle)(struct usbd_pipe *);
void (*done)(struct usbd_xfer *);
};
struct usbd_tt {
struct usbd_hub *hub;
void *hcpriv;
};
struct usbd_port {
usb_port_status_t status;
u_int16_t power;
u_int8_t portno;
u_int8_t restartcnt;
#define USBD_RESTART_MAX 5
u_int8_t reattach;
struct usbd_device *device;
struct usbd_device *parent;
struct usbd_tt *tt;
};
struct usbd_hub {
int (*explore)(struct usbd_device *);
void *hubsoftc;
struct usbd_port *ports;
int nports;
u_int8_t powerdelay;
u_int8_t ttthink;
u_int8_t multi;
};
struct usbd_bus {
struct device bdev;
const struct usbd_bus_methods *methods;
#if NBPFILTER > 0
void *bpfif;
caddr_t bpf;
#endif
u_int32_t pipe_size;
struct usbd_device *root_hub;
struct usbd_device *devices[USB_MAX_DEVICES];
char use_polling;
char dying;
int flags;
#define USB_BUS_CONFIG_PENDING 0x01
#define USB_BUS_DISCONNECTING 0x02
struct device *usbctl;
struct usb_device_stats stats;
int intr_context;
u_int no_intrs;
int usbrev;
#define USBREV_UNKNOWN 0
#define USBREV_PRE_1_0 1
#define USBREV_1_0 2
#define USBREV_1_1 3
#define USBREV_2_0 4
#define USBREV_3_0 5
#define USBREV_STR { "unknown", "pre 1.0", "1.0", "1.1", "2.0", "3.0" }
void *soft;
bus_dma_tag_t dmatag;
int dmaflags;
};
struct usbd_device {
struct usbd_bus *bus;
struct usbd_pipe *default_pipe;
u_int8_t dying;
u_int8_t ref_cnt;
u_int8_t address;
u_int8_t config;
u_int8_t depth;
u_int8_t speed;
u_int8_t self_powered;
u_int16_t power;
int16_t langid;
#define USBD_NOLANG (-1)
struct usbd_port *powersrc;
struct usbd_device *myhub;
struct usbd_port *myhsport;
struct usbd_endpoint def_ep;
usb_endpoint_descriptor_t def_ep_desc;
struct usbd_interface *ifaces;
usb_device_descriptor_t ddesc;
usb_config_descriptor_t *cdesc;
const struct usbd_quirks *quirks;
struct usbd_hub *hub;
struct device **subdevs;
int nsubdev;
int ndevs;
char *serial;
char *vendor;
char *product;
};
struct usbd_interface {
struct usbd_device *device;
usb_interface_descriptor_t *idesc;
int index;
int altindex;
struct usbd_endpoint *endpoints;
void *priv;
LIST_HEAD(, usbd_pipe) pipes;
uint8_t claimed;
uint8_t nendpt;
};
struct usbd_pipe {
struct usbd_interface *iface;
struct usbd_device *device;
struct usbd_endpoint *endpoint;
size_t pipe_size;
char running;
char aborting;
SIMPLEQ_HEAD(, usbd_xfer) queue;
LIST_ENTRY(usbd_pipe) next;
struct usbd_xfer *intrxfer;
char repeat;
int interval;
const struct usbd_pipe_methods *methods;
};
struct usbd_xfer {
struct usbd_pipe *pipe;
void *priv;
char *buffer;
u_int32_t length;
u_int32_t actlen;
u_int16_t flags;
u_int32_t timeout;
usbd_status status;
usbd_callback callback;
volatile char done;
#ifdef DIAGNOSTIC
u_int32_t busy_free;
#define XFER_FREE 0x42555359
#define XFER_ONQU 0x4f4e5155
#endif
usb_device_request_t request;
u_int16_t *frlengths;
int nframes;
struct usbd_device *device;
struct usb_dma dmabuf;
int rqflags;
#define URQ_REQUEST 0x01
#define URQ_AUTO_DMABUF 0x10
#define URQ_DEV_DMABUF 0x20
SIMPLEQ_ENTRY(usbd_xfer) next;
void *hcpriv;
struct usb_task abort_task;
struct timeout timeout_handle;
};
void usbd_dump_iface(struct usbd_interface *);
void usbd_dump_device(struct usbd_device *);
void usbd_dump_endpoint(struct usbd_endpoint *);
void usbd_dump_queue(struct usbd_pipe *);
void usbd_dump_pipe(struct usbd_pipe *);
int usbctlprint(void *, const char *);
void usb_delay_ms(struct usbd_bus *, u_int);
usbd_status usbd_port_disown_to_1_1(struct usbd_device *, int);
int usbd_reset_port(struct usbd_device *, int);
usbd_status usbd_setup_pipe(struct usbd_device *,
struct usbd_interface *, struct usbd_endpoint *, int,
struct usbd_pipe **);
int usbd_set_address(struct usbd_device *, int);
usbd_status usbd_new_device(struct device *, struct usbd_bus *,
int, int, int, struct usbd_port *);
usbd_status usbd_fill_iface_data(struct usbd_device *, int, int);
usbd_status usb_insert_transfer(struct usbd_xfer *);
void usb_transfer_complete(struct usbd_xfer *);
int usbd_detach(struct usbd_device *, struct device *);
void usb_needs_explore(struct usbd_device *, int);
void usb_needs_reattach(struct usbd_device *);
void usb_schedsoftintr(struct usbd_bus *);
void usb_tap(struct usbd_bus *, struct usbd_xfer *, uint8_t);
#define USBTAP_DIR_OUT 0
#define USBTAP_DIR_IN 1
#define UHUB_UNK_CONFIGURATION -1
#define UHUB_UNK_INTERFACE -1
static inline int
usbd_xfer_isread(struct usbd_xfer *xfer)
{
if (xfer->rqflags & URQ_REQUEST)
return (xfer->request.bmRequestType & UT_READ);
return (xfer->pipe->endpoint->edesc->bEndpointAddress & UE_DIR_IN);
}
#endif