/* * Copyright (c) 2011 Jan Vesely * 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 drvusbuhcihc * @{ */ /** @file * @brief UHCI host controller driver structure */ #include #include #include int endpoint_init(endpoint_t *instance, usb_address_t address, usb_endpoint_t endpoint, usb_direction_t direction, usb_transfer_type_t type, usb_speed_t speed, size_t max_packet_size) { assert(instance); instance->address = address; instance->endpoint = endpoint; instance->direction = direction; instance->transfer_type = type; instance->speed = speed; instance->max_packet_size = max_packet_size; instance->toggle = 0; instance->active = false; fibril_mutex_initialize(&instance->guard); fibril_condvar_initialize(&instance->avail); link_initialize(&instance->same_device_eps); return EOK; } /*----------------------------------------------------------------------------*/ void endpoint_destroy(endpoint_t *instance) { assert(instance); assert(!instance->active); list_remove(&instance->same_device_eps); free(instance); } /*----------------------------------------------------------------------------*/ void endpoint_use(endpoint_t *instance) { assert(instance); fibril_mutex_lock(&instance->guard); while (instance->active) fibril_condvar_wait(&instance->avail, &instance->guard); instance->active = true; fibril_mutex_unlock(&instance->guard); } /*----------------------------------------------------------------------------*/ void endpoint_release(endpoint_t *instance) { assert(instance); fibril_mutex_lock(&instance->guard); instance->active = false; fibril_mutex_unlock(&instance->guard); fibril_condvar_signal(&instance->avail); } /*----------------------------------------------------------------------------*/ int endpoint_toggle_get(endpoint_t *instance) { assert(instance); return (int)instance->toggle; } /*----------------------------------------------------------------------------*/ void endpoint_toggle_set(endpoint_t *instance, int toggle) { assert(instance); assert(toggle == 0 || toggle == 1); instance->toggle = toggle; } /*----------------------------------------------------------------------------*/ void endpoint_toggle_reset(link_t *ep) { endpoint_t *instance = list_get_instance(ep, endpoint_t, same_device_eps); assert(instance); instance->toggle = 0; } /*----------------------------------------------------------------------------*/ void endpoint_toggle_reset_filtered(link_t *ep, usb_endpoint_t epn) { endpoint_t *instance = list_get_instance(ep, endpoint_t, same_device_eps); assert(instance); if (instance->endpoint == epn) instance->toggle = 0; } /** * @} */