102ac6454SAndrew Thompson /*- 2*4d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause 3718cf2ccSPedro F. Giffuni * 402ac6454SAndrew Thompson * Copyright (c) 2008 Hans Petter Selasky. All rights reserved. 502ac6454SAndrew Thompson * 602ac6454SAndrew Thompson * Redistribution and use in source and binary forms, with or without 702ac6454SAndrew Thompson * modification, are permitted provided that the following conditions 802ac6454SAndrew Thompson * are met: 902ac6454SAndrew Thompson * 1. Redistributions of source code must retain the above copyright 1002ac6454SAndrew Thompson * notice, this list of conditions and the following disclaimer. 1102ac6454SAndrew Thompson * 2. Redistributions in binary form must reproduce the above copyright 1202ac6454SAndrew Thompson * notice, this list of conditions and the following disclaimer in the 1302ac6454SAndrew Thompson * documentation and/or other materials provided with the distribution. 1402ac6454SAndrew Thompson * 1502ac6454SAndrew Thompson * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 1602ac6454SAndrew Thompson * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 1702ac6454SAndrew Thompson * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 1802ac6454SAndrew Thompson * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 1902ac6454SAndrew Thompson * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 2002ac6454SAndrew Thompson * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 2102ac6454SAndrew Thompson * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2202ac6454SAndrew Thompson * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2302ac6454SAndrew Thompson * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2402ac6454SAndrew Thompson * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 2502ac6454SAndrew Thompson * SUCH DAMAGE. 2602ac6454SAndrew Thompson */ 2702ac6454SAndrew Thompson 2802ac6454SAndrew Thompson /* 29fde87526SAndrew Thompson * Including this file is mandatory for all USB related c-files in the kernel. 3002ac6454SAndrew Thompson */ 3102ac6454SAndrew Thompson 3275973647SAndrew Thompson #ifndef _USB_CORE_H_ 3375973647SAndrew Thompson #define _USB_CORE_H_ 3402ac6454SAndrew Thompson 3502ac6454SAndrew Thompson /* 3602ac6454SAndrew Thompson * The following macro will tell if an USB transfer is currently 3702ac6454SAndrew Thompson * receiving or transferring data. 3802ac6454SAndrew Thompson */ 39f29a0724SAndrew Thompson #define USB_GET_DATA_ISREAD(xfer) ((xfer)->flags_int.usb_mode == \ 40ae60fdfbSAndrew Thompson USB_MODE_DEVICE ? (((xfer)->endpointno & UE_DIR_IN) ? 0 : 1) : \ 41ae60fdfbSAndrew Thompson (((xfer)->endpointno & UE_DIR_IN) ? 1 : 0)) 4202ac6454SAndrew Thompson 430eb8d462SHans Petter Selasky /* locking wrappers for BUS lock */ 440eb8d462SHans Petter Selasky #define USB_BUS_LOCK(_b) USB_MTX_LOCK(&(_b)->bus_mtx) 450eb8d462SHans Petter Selasky #define USB_BUS_UNLOCK(_b) USB_MTX_UNLOCK(&(_b)->bus_mtx) 460eb8d462SHans Petter Selasky #define USB_BUS_LOCK_ASSERT(_b, _t) USB_MTX_ASSERT(&(_b)->bus_mtx, _t) 4702ac6454SAndrew Thompson 480eb8d462SHans Petter Selasky /* locking wrappers for BUS spin lock */ 490eb8d462SHans Petter Selasky #define USB_BUS_SPIN_LOCK(_b) USB_MTX_LOCK_SPIN(&(_b)->bus_spin_lock) 500eb8d462SHans Petter Selasky #define USB_BUS_SPIN_UNLOCK(_b) USB_MTX_UNLOCK_SPIN(&(_b)->bus_spin_lock) 510eb8d462SHans Petter Selasky #define USB_BUS_SPIN_LOCK_ASSERT(_b, _t) USB_MTX_ASSERT(&(_b)->bus_spin_lock, _t) 520eb8d462SHans Petter Selasky 530eb8d462SHans Petter Selasky /* locking wrappers for XFER lock */ 540eb8d462SHans Petter Selasky #define USB_XFER_LOCK(_x) USB_MTX_LOCK((_x)->xroot->xfer_mtx) 550eb8d462SHans Petter Selasky #define USB_XFER_UNLOCK(_x) USB_MTX_UNLOCK((_x)->xroot->xfer_mtx) 560eb8d462SHans Petter Selasky #define USB_XFER_LOCK_ASSERT(_x, _t) USB_MTX_ASSERT((_x)->xroot->xfer_mtx, _t) 57ed6d949aSAndrew Thompson 58ed6d949aSAndrew Thompson /* helper for converting pointers to integers */ 59ed6d949aSAndrew Thompson #define USB_P2U(ptr) \ 603ab53b27SJohn Baldwin ((uintptr_t)(ptr)) 61ed6d949aSAndrew Thompson 62ed6d949aSAndrew Thompson /* helper for computing offsets */ 63ed6d949aSAndrew Thompson #define USB_ADD_BYTES(ptr,size) \ 643ab53b27SJohn Baldwin ((void *)(__DECONST(char *, (ptr)) + (size))) 65ed6d949aSAndrew Thompson 66ed6d949aSAndrew Thompson /* debug macro */ 67ed6d949aSAndrew Thompson #define USB_ASSERT KASSERT 68ed6d949aSAndrew Thompson 6902ac6454SAndrew Thompson /* structure prototypes */ 7002ac6454SAndrew Thompson 7102ac6454SAndrew Thompson struct file; 72760bc48eSAndrew Thompson struct usb_bus; 73760bc48eSAndrew Thompson struct usb_device; 74760bc48eSAndrew Thompson struct usb_device_request; 75760bc48eSAndrew Thompson struct usb_page; 76760bc48eSAndrew Thompson struct usb_page_cache; 77760bc48eSAndrew Thompson struct usb_xfer; 78760bc48eSAndrew Thompson struct usb_xfer_root; 7923ab0871SHans Petter Selasky struct usb_string_lang; 8002ac6454SAndrew Thompson 8102ac6454SAndrew Thompson /* typedefs */ 8202ac6454SAndrew Thompson 8302ac6454SAndrew Thompson /* structures */ 8402ac6454SAndrew Thompson 8502ac6454SAndrew Thompson /* 8602ac6454SAndrew Thompson * The following structure defines a set of internal USB transfer 8702ac6454SAndrew Thompson * flags. 8802ac6454SAndrew Thompson */ 89760bc48eSAndrew Thompson struct usb_xfer_flags_int { 90f29a0724SAndrew Thompson enum usb_hc_mode usb_mode; /* shadow copy of "udev->usb_mode" */ 9102ac6454SAndrew Thompson uint16_t control_rem; /* remainder in bytes */ 9202ac6454SAndrew Thompson 9302ac6454SAndrew Thompson uint8_t open:1; /* set if USB pipe has been opened */ 9402ac6454SAndrew Thompson uint8_t transferring:1; /* set if an USB transfer is in 9502ac6454SAndrew Thompson * progress */ 9602ac6454SAndrew Thompson uint8_t did_dma_delay:1; /* set if we waited for HW DMA */ 9702ac6454SAndrew Thompson uint8_t did_close:1; /* set if we closed the USB transfer */ 9802ac6454SAndrew Thompson uint8_t draining:1; /* set if we are draining an USB 9902ac6454SAndrew Thompson * transfer */ 10002ac6454SAndrew Thompson uint8_t started:1; /* keeps track of started or stopped */ 10102ac6454SAndrew Thompson uint8_t bandwidth_reclaimed:1; 10202ac6454SAndrew Thompson uint8_t control_xfr:1; /* set if control transfer */ 10302ac6454SAndrew Thompson uint8_t control_hdr:1; /* set if control header should be 10402ac6454SAndrew Thompson * sent */ 10502ac6454SAndrew Thompson uint8_t control_act:1; /* set if control transfer is active */ 106476183dfSAndrew Thompson uint8_t control_stall:1; /* set if control transfer should be stalled */ 107add9e3e5SHans Petter Selasky uint8_t control_did_data:1; /* set if control DATA has been transferred */ 10802ac6454SAndrew Thompson 10902ac6454SAndrew Thompson uint8_t short_frames_ok:1; /* filtered version */ 11002ac6454SAndrew Thompson uint8_t short_xfer_ok:1; /* filtered version */ 111bdc081c6SAndrew Thompson #if USB_HAVE_BUSDMA 11202ac6454SAndrew Thompson uint8_t bdma_enable:1; /* filtered version (only set if 11302ac6454SAndrew Thompson * hardware supports DMA) */ 11402ac6454SAndrew Thompson uint8_t bdma_no_post_sync:1; /* set if the USB callback wrapper 11502ac6454SAndrew Thompson * should not do the BUS-DMA post sync 11602ac6454SAndrew Thompson * operation */ 11702ac6454SAndrew Thompson uint8_t bdma_setup:1; /* set if BUS-DMA has been setup */ 118bdc081c6SAndrew Thompson #endif 11902ac6454SAndrew Thompson uint8_t isochronous_xfr:1; /* set if isochronous transfer */ 12002ac6454SAndrew Thompson uint8_t curr_dma_set:1; /* used by USB HC/DC driver */ 12102ac6454SAndrew Thompson uint8_t can_cancel_immed:1; /* set if USB transfer can be 12202ac6454SAndrew Thompson * cancelled immediately */ 123bb9c7a8bSAndrew Thompson uint8_t doing_callback:1; /* set if executing the callback */ 124e91fe3a9SHans Petter Selasky uint8_t maxp_was_clamped:1; /* set if the max packet size 125e91fe3a9SHans Petter Selasky * was outside its allowed range */ 12602ac6454SAndrew Thompson }; 12702ac6454SAndrew Thompson 12802ac6454SAndrew Thompson /* 12902ac6454SAndrew Thompson * The following structure defines an USB transfer. 13002ac6454SAndrew Thompson */ 131760bc48eSAndrew Thompson struct usb_xfer { 132760bc48eSAndrew Thompson struct usb_callout timeout_handle; 133760bc48eSAndrew Thompson TAILQ_ENTRY(usb_xfer) wait_entry; /* used at various places */ 13402ac6454SAndrew Thompson 135760bc48eSAndrew Thompson struct usb_page_cache *buf_fixup; /* fixup buffer(s) */ 136760bc48eSAndrew Thompson struct usb_xfer_queue *wait_queue; /* pointer to queue that we 13702ac6454SAndrew Thompson * are waiting on */ 138760bc48eSAndrew Thompson struct usb_page *dma_page_ptr; 139ae60fdfbSAndrew Thompson struct usb_endpoint *endpoint; /* our USB endpoint */ 140760bc48eSAndrew Thompson struct usb_xfer_root *xroot; /* used by HC driver */ 14102ac6454SAndrew Thompson void *qh_start[2]; /* used by HC driver */ 14202ac6454SAndrew Thompson void *td_start[2]; /* used by HC driver */ 14302ac6454SAndrew Thompson void *td_transfer_first; /* used by HC driver */ 14402ac6454SAndrew Thompson void *td_transfer_last; /* used by HC driver */ 14502ac6454SAndrew Thompson void *td_transfer_cache; /* used by HC driver */ 14602ac6454SAndrew Thompson void *priv_sc; /* device driver data pointer 1 */ 14702ac6454SAndrew Thompson void *priv_fifo; /* device driver data pointer 2 */ 14802ac6454SAndrew Thompson void *local_buffer; 149e0a69b51SAndrew Thompson usb_frlength_t *frlengths; 150760bc48eSAndrew Thompson struct usb_page_cache *frbuffers; 151e0a69b51SAndrew Thompson usb_callback_t *callback; 15202ac6454SAndrew Thompson 153e0a69b51SAndrew Thompson usb_frlength_t max_hc_frame_size; 154e0a69b51SAndrew Thompson usb_frlength_t max_data_length; 155e0a69b51SAndrew Thompson usb_frlength_t sumlen; /* sum of all lengths in bytes */ 156e0a69b51SAndrew Thompson usb_frlength_t actlen; /* actual length in bytes */ 157e0a69b51SAndrew Thompson usb_timeout_t timeout; /* milliseconds */ 15802ac6454SAndrew Thompson 159e0a69b51SAndrew Thompson usb_frcount_t max_frame_count; /* initial value of "nframes" after 16002ac6454SAndrew Thompson * setup */ 161e0a69b51SAndrew Thompson usb_frcount_t nframes; /* number of USB frames to transfer */ 162e0a69b51SAndrew Thompson usb_frcount_t aframes; /* actual number of USB frames 16302ac6454SAndrew Thompson * transferred */ 164a5cf1aaaSHans Petter Selasky usb_stream_t stream_id; /* USB3.0 specific field */ 16502ac6454SAndrew Thompson 16602ac6454SAndrew Thompson uint16_t max_packet_size; 16702ac6454SAndrew Thompson uint16_t max_frame_size; 16802ac6454SAndrew Thompson uint16_t qh_pos; 16902ac6454SAndrew Thompson uint16_t isoc_time_complete; /* in ms */ 170e0a69b51SAndrew Thompson usb_timeout_t interval; /* milliseconds */ 17102ac6454SAndrew Thompson 17202ac6454SAndrew Thompson uint8_t address; /* physical USB address */ 173ae60fdfbSAndrew Thompson uint8_t endpointno; /* physical USB endpoint */ 17402ac6454SAndrew Thompson uint8_t max_packet_count; 175a593f6b8SAndrew Thompson uint8_t usb_state; 176883bb300SAndrew Thompson uint8_t fps_shift; /* down shift of FPS, 0..3 */ 17702ac6454SAndrew Thompson 178e0a69b51SAndrew Thompson usb_error_t error; 17902ac6454SAndrew Thompson 180760bc48eSAndrew Thompson struct usb_xfer_flags flags; 181760bc48eSAndrew Thompson struct usb_xfer_flags_int flags_int; 18202ac6454SAndrew Thompson }; 18302ac6454SAndrew Thompson 18402ac6454SAndrew Thompson /* external variables */ 18502ac6454SAndrew Thompson 186a593f6b8SAndrew Thompson extern struct mtx usb_ref_lock; 18723ab0871SHans Petter Selasky extern const struct usb_string_lang usb_string_lang_en; 18802ac6454SAndrew Thompson 189ed6d949aSAndrew Thompson /* typedefs */ 19002ac6454SAndrew Thompson 191ed6d949aSAndrew Thompson typedef struct malloc_type *usb_malloc_type; 192ed6d949aSAndrew Thompson 193ed6d949aSAndrew Thompson /* prototypes */ 19402ac6454SAndrew Thompson 19575973647SAndrew Thompson #endif /* _USB_CORE_H_ */ 196