HelenOS sources

root/uspace/lib/drv/include/usbhc_iface.h

/* [<][>][^][v][top][bottom][index][help] */

INCLUDED FROM


/*
 * Copyright (c) 2010 Vojtech Horky
 * Copyright (c) 2017 Ondrej Hlavaty
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 * - 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.
 * - The name of the author may not be used to endorse or promote products
 *   derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
 */

/** @addtogroup libdrv
 * @addtogroup usb
 * @{
 */
/** @file
 * @brief USB host controler interface definition. This is the interface of
 * USB host controller function, which can be used by usb device drivers.
 */

#ifndef LIBDRV_USBHC_IFACE_H_
#define LIBDRV_USBHC_IFACE_H_

#include "ddf/driver.h"
#include <async.h>

/** USB speeds. */
typedef enum {
        /** USB 1.1 low speed (1.5Mbits/s). */
        USB_SPEED_LOW,
        /** USB 1.1 full speed (12Mbits/s). */
        USB_SPEED_FULL,
        /** USB 2.0 high speed (480Mbits/s). */
        USB_SPEED_HIGH,
        /** USB 3.0 super speed (5Gbits/s). */
        USB_SPEED_SUPER,
        /** Psuedo-speed serving as a boundary. */
        USB_SPEED_MAX
} usb_speed_t;

/** USB endpoint number type.
 * Negative values could be used to indicate error.
 */
typedef uint16_t usb_endpoint_t;

/** USB address type.
 * Negative values could be used to indicate error.
 */
typedef uint16_t usb_address_t;

/**
 * USB Stream ID type.
 */
typedef uint16_t usb_stream_t;

/** USB transfer type. */
typedef enum {
        USB_TRANSFER_CONTROL = 0,
        USB_TRANSFER_ISOCHRONOUS = 1,
        USB_TRANSFER_BULK = 2,
        USB_TRANSFER_INTERRUPT = 3,
} usb_transfer_type_t;

#define USB_TRANSFER_COUNT  (USB_TRANSFER_INTERRUPT + 1)

/** USB data transfer direction. */
typedef enum {
        USB_DIRECTION_IN,
        USB_DIRECTION_OUT,
        USB_DIRECTION_BOTH,
} usb_direction_t;

#define USB_DIRECTION_COUNT  (USB_DIRECTION_BOTH + 1)

/** USB complete address type.
 * Pair address + endpoint is identification of transaction recipient.
 */
typedef union {
        struct {
                usb_address_t address;
                usb_endpoint_t endpoint;
                usb_stream_t stream;
        } __attribute__((packed));
        uint64_t packed;
} usb_target_t;

// FIXME: DMA buffers shall be part of libdrv anyway.
typedef uintptr_t dma_policy_t;

typedef struct dma_buffer {
        void *virt;
        dma_policy_t policy;
} dma_buffer_t;

typedef struct usb_pipe_desc {
        /** Endpoint number. */
        usb_endpoint_t endpoint_no;

        /** Endpoint transfer type. */
        usb_transfer_type_t transfer_type;

        /** Endpoint direction. */
        usb_direction_t direction;

        /**
         * Maximum size of one transfer. Non-periodic endpoints may handle
         * bigger transfers, but those can be split into multiple USB transfers.
         */
        size_t max_transfer_size;

        /** Constraints on buffers to be transferred without copying */
        dma_policy_t transfer_buffer_policy;
} usb_pipe_desc_t;

typedef struct usb_pipe_transfer_request {
        usb_direction_t dir;
        usb_endpoint_t endpoint;
        usb_stream_t stream;

        uint64_t setup;                 /**< Valid iff the transfer is of control type */

        /**
         * The DMA buffer to share. Must be at least offset + size large. Is
         * patched after being transmitted over IPC, so the pointer is still
         * valid.
         */
        dma_buffer_t buffer;
        size_t offset;                  /**< Offset to the buffer */
        size_t size;                    /**< Requested size. */
} usbhc_iface_transfer_request_t;

/** This structure follows standard endpoint descriptor + superspeed companion
 * descriptor, and exists to avoid dependency of libdrv on libusb. Keep the
 * internal fields named exactly like their source (because we want to use the
 * same macros to access them).
 * Callers shall fill it with bare contents of respective descriptors (in usb endianity).
 */
typedef struct usb_endpoint_descriptors {
        struct {
                uint8_t endpoint_address;
                uint8_t attributes;
                uint16_t max_packet_size;
                uint8_t poll_interval;
        } endpoint;

        /* Superspeed companion descriptor */
        struct companion_desc_t {
                uint8_t max_burst;
                uint8_t attributes;
                uint16_t bytes_per_interval;
        } companion;
} usb_endpoint_descriptors_t;

extern errno_t usbhc_reserve_default_address(async_exch_t *);
extern errno_t usbhc_release_default_address(async_exch_t *);

extern errno_t usbhc_device_enumerate(async_exch_t *, unsigned, usb_speed_t);
extern errno_t usbhc_device_remove(async_exch_t *, unsigned);

extern errno_t usbhc_register_endpoint(async_exch_t *, usb_pipe_desc_t *, const usb_endpoint_descriptors_t *);
extern errno_t usbhc_unregister_endpoint(async_exch_t *, const usb_pipe_desc_t *);

extern errno_t usbhc_transfer(async_exch_t *, const usbhc_iface_transfer_request_t *, size_t *);

/** Callback for outgoing transfer */
typedef errno_t (*usbhc_iface_transfer_callback_t)(void *, int, size_t);

/** USB device communication interface. */
typedef struct {
        errno_t (*default_address_reservation)(ddf_fun_t *, bool);

        errno_t (*device_enumerate)(ddf_fun_t *, unsigned, usb_speed_t);
        errno_t (*device_remove)(ddf_fun_t *, unsigned);

        errno_t (*register_endpoint)(ddf_fun_t *, usb_pipe_desc_t *, const usb_endpoint_descriptors_t *);
        errno_t (*unregister_endpoint)(ddf_fun_t *, const usb_pipe_desc_t *);

        errno_t (*transfer)(ddf_fun_t *, const usbhc_iface_transfer_request_t *,
            usbhc_iface_transfer_callback_t, void *);
} usbhc_iface_t;

#endif
/**
 * @}
 */

/* [<][>][^][v][top][bottom][index][help] */
HelenOS homepage, sources at GitHub