Changeset dcaf819 in mainline for uspace/lib/usb/src/host/usb_endpoint_manager.c
- Timestamp:
- 2011-04-06T22:33:08Z (14 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- 9a7e5b4
- Parents:
- 6bf9bc4 (diff), 87305bb (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the(diff)
links above to see all the changes relative to each parent. - File:
-
- 1 moved
Legend:
- Unmodified
- Added
- Removed
-
uspace/lib/usb/src/host/usb_endpoint_manager.c
r6bf9bc4 rdcaf819 1 1 /* 2 2 * Copyright (c) 2011 Jan Vesely 3 * All rights reserved.3 * All rights eps. 4 4 * 5 5 * Redistribution and use in source and binary forms, with or without … … 27 27 */ 28 28 29 #include <bool.h> 29 30 #include <assert.h> 30 31 #include <errno.h> 31 #include <usb/host/bandwidth.h> 32 33 typedef struct { 32 33 #include <usb/host/usb_endpoint_manager.h> 34 35 #define BUCKET_COUNT 7 36 37 typedef struct { 34 38 usb_address_t address; 35 39 usb_endpoint_t endpoint; 36 40 usb_direction_t direction; 37 } __attribute__((aligned (sizeof(unsigned long)))) transfer_t;38 /*----------------------------------------------------------------------------*/ 41 } __attribute__((aligned (sizeof(unsigned long)))) id_t; 42 #define MAX_KEYS (sizeof(id_t) / sizeof(unsigned long)) 39 43 typedef struct { 40 transfer_t transfer; 44 union { 45 id_t id; 46 unsigned long key[MAX_KEYS]; 47 }; 41 48 link_t link; 42 bool used; 43 size_t required; 44 } transfer_status_t; 45 /*----------------------------------------------------------------------------*/ 46 #define BUCKET_COUNT 7 47 #define MAX_KEYS (sizeof(transfer_t) / sizeof(unsigned long)) 48 /*----------------------------------------------------------------------------*/ 49 static hash_index_t transfer_hash(unsigned long key[]) 49 size_t bw; 50 endpoint_t *ep; 51 } node_t; 52 /*----------------------------------------------------------------------------*/ 53 static hash_index_t node_hash(unsigned long key[]) 50 54 { 51 55 hash_index_t hash = 0; … … 58 62 } 59 63 /*----------------------------------------------------------------------------*/ 60 static int transfer_compare( 61 unsigned long key[], hash_count_t keys, link_t *item) 64 static int node_compare(unsigned long key[], hash_count_t keys, link_t *item) 62 65 { 63 66 assert(item); 64 transfer_status_t *status = 65 hash_table_get_instance(item, transfer_status_t, link); 66 const size_t bytes = 67 keys < MAX_KEYS ? keys * sizeof(unsigned long) : sizeof(transfer_t); 68 return bcmp(key, &status->transfer, bytes) == 0; 69 } 70 /*----------------------------------------------------------------------------*/ 71 static void transfer_remove(link_t *item) 67 node_t *node = hash_table_get_instance(item, node_t, link); 68 hash_count_t i = 0; 69 for (; i < keys; ++i) { 70 if (key[i] != node->key[i]) 71 return false; 72 } 73 return true; 74 } 75 /*----------------------------------------------------------------------------*/ 76 static void node_remove(link_t *item) 72 77 { 73 78 assert(item); 74 transfer_status_t *status = 75 hash_table_get_instance(item, transfer_status_t, link); 76 assert(status); 77 free(status); 78 } 79 /*----------------------------------------------------------------------------*/ 80 hash_table_operations_t op = { 81 .hash = transfer_hash, 82 .compare = transfer_compare, 83 .remove_callback = transfer_remove, 79 node_t *node = hash_table_get_instance(item, node_t, link); 80 endpoint_destroy(node->ep); 81 free(node); 82 } 83 /*----------------------------------------------------------------------------*/ 84 static hash_table_operations_t op = { 85 .hash = node_hash, 86 .compare = node_compare, 87 .remove_callback = node_remove, 84 88 }; 85 89 /*----------------------------------------------------------------------------*/ … … 120 124 } 121 125 /*----------------------------------------------------------------------------*/ 122 int bandwidth_init(bandwidth_t *instance, size_t bandwidth,123 size_t (*usage_fnc)(usb_speed_t, usb_transfer_type_t, size_t, size_t))126 int usb_endpoint_manager_init(usb_endpoint_manager_t *instance, 127 size_t available_bandwidth) 124 128 { 125 129 assert(instance); 126 130 fibril_mutex_initialize(&instance->guard); 127 instance->free = bandwidth;128 instance-> usage_fnc = usage_fnc;131 fibril_condvar_initialize(&instance->change); 132 instance->free_bw = available_bandwidth; 129 133 bool ht = 130 hash_table_create(&instance-> reserved, BUCKET_COUNT, MAX_KEYS, &op);134 hash_table_create(&instance->ep_table, BUCKET_COUNT, MAX_KEYS, &op); 131 135 return ht ? EOK : ENOMEM; 132 136 } 133 137 /*----------------------------------------------------------------------------*/ 134 void bandwidth_destroy(bandwidth_t *instance) 135 { 136 hash_table_destroy(&instance->reserved); 137 } 138 /*----------------------------------------------------------------------------*/ 139 int bandwidth_reserve(bandwidth_t *instance, usb_address_t address, 140 usb_endpoint_t endpoint, usb_direction_t direction, usb_speed_t speed, 141 usb_transfer_type_t transfer_type, size_t max_packet_size, size_t size, 142 unsigned interval) 143 { 144 if (transfer_type != USB_TRANSFER_ISOCHRONOUS && 145 transfer_type != USB_TRANSFER_INTERRUPT) { 146 return ENOTSUP; 147 } 148 149 assert(instance); 150 assert(instance->usage_fnc); 151 152 transfer_t trans = { 138 void usb_endpoint_manager_destroy(usb_endpoint_manager_t *instance) 139 { 140 hash_table_destroy(&instance->ep_table); 141 } 142 /*----------------------------------------------------------------------------*/ 143 int usb_endpoint_manager_register_ep(usb_endpoint_manager_t *instance, 144 usb_address_t address, usb_endpoint_t endpoint, usb_direction_t direction, 145 endpoint_t *ep, size_t data_size) 146 { 147 assert(ep); 148 size_t bw = bandwidth_count_usb11(ep->speed, ep->transfer_type, 149 data_size, ep->max_packet_size); 150 assert(instance); 151 152 id_t id = { 153 153 .address = address, 154 154 .endpoint = endpoint, … … 156 156 }; 157 157 fibril_mutex_lock(&instance->guard); 158 const size_t required = 159 instance->usage_fnc(speed, transfer_type, size, max_packet_size); 160 161 if (required > instance->free) { 158 159 link_t *item = 160 hash_table_find(&instance->ep_table, (unsigned long*)&id); 161 if (item != NULL) { 162 fibril_mutex_unlock(&instance->guard); 163 return EEXISTS; 164 } 165 166 if (bw > instance->free_bw) { 162 167 fibril_mutex_unlock(&instance->guard); 163 168 return ENOSPC; 164 169 } 165 170 166 link_t *item = 167 hash_table_find(&instance->reserved, (unsigned long*)&trans); 168 if (item != NULL) { 169 fibril_mutex_unlock(&instance->guard); 170 return EEXISTS; 171 } 172 173 transfer_status_t *status = malloc(sizeof(transfer_status_t)); 174 if (status == NULL) { 171 node_t *node = malloc(sizeof(node_t)); 172 if (node == NULL) { 175 173 fibril_mutex_unlock(&instance->guard); 176 174 return ENOMEM; 177 175 } 178 176 179 status->transfer = trans;180 status->required = required;181 status->used = false;182 link_initialize(& status->link);183 184 hash_table_insert(&instance-> reserved,185 (unsigned long*)& status->transfer, &status->link);186 instance->free -= required;177 node->id = id; 178 node->bw = bw; 179 node->ep = ep; 180 link_initialize(&node->link); 181 182 hash_table_insert(&instance->ep_table, 183 (unsigned long*)&id, &node->link); 184 instance->free_bw -= bw; 187 185 fibril_mutex_unlock(&instance->guard); 186 fibril_condvar_broadcast(&instance->change); 188 187 return EOK; 189 /* TODO: compute bandwidth used */ 190 } 191 /*----------------------------------------------------------------------------*/ 192 int bandwidth_release(bandwidth_t *instance, usb_address_t address, 193 usb_endpoint_t endpoint, usb_direction_t direction) 194 { 195 assert(instance); 196 transfer_t trans = { 188 } 189 /*----------------------------------------------------------------------------*/ 190 int usb_endpoint_manager_unregister_ep(usb_endpoint_manager_t *instance, 191 usb_address_t address, usb_endpoint_t endpoint, usb_direction_t direction) 192 { 193 assert(instance); 194 id_t id = { 197 195 .address = address, 198 196 .endpoint = endpoint, … … 201 199 fibril_mutex_lock(&instance->guard); 202 200 link_t *item = 203 hash_table_find(&instance-> reserved, (unsigned long*)&trans);201 hash_table_find(&instance->ep_table, (unsigned long*)&id); 204 202 if (item == NULL) { 205 203 fibril_mutex_unlock(&instance->guard); … … 207 205 } 208 206 209 transfer_status_t *status = 210 hash_table_get_instance(item, transfer_status_t, link); 211 212 instance->free += status->required; 213 214 hash_table_remove(&instance->reserved, 215 (unsigned long*)&trans, MAX_KEYS); 207 node_t *node = hash_table_get_instance(item, node_t, link); 208 instance->free_bw += node->bw; 209 hash_table_remove(&instance->ep_table, (unsigned long*)&id, MAX_KEYS); 216 210 217 211 fibril_mutex_unlock(&instance->guard); 212 fibril_condvar_broadcast(&instance->change); 218 213 return EOK; 219 /* TODO: compute bandwidth freed */ 220 } 221 /*----------------------------------------------------------------------------*/ 222 int bandwidth_use(bandwidth_t *instance, usb_address_t address,223 usb_endpoint_t endpoint, usb_direction_t direction, size_tbw)224 { 225 assert(instance); 226 transfer_t trans= {214 } 215 /*----------------------------------------------------------------------------*/ 216 endpoint_t * usb_endpoint_manager_get_ep(usb_endpoint_manager_t *instance, 217 usb_address_t address, usb_endpoint_t endpoint, usb_direction_t direction, 218 size_t *bw) 219 { 220 assert(instance); 221 id_t id = { 227 222 .address = address, 228 223 .endpoint = endpoint, … … 231 226 fibril_mutex_lock(&instance->guard); 232 227 link_t *item = 233 hash_table_find(&instance->reserved, (unsigned long*)&trans); 234 int ret = EOK; 235 if (item != NULL) { 236 transfer_status_t *status = 237 hash_table_get_instance(item, transfer_status_t, link); 238 assert(status); 239 if (status->required >= bw) { 240 if (status->used) { 241 ret = EINPROGRESS; 242 } 243 status->used = true; 244 } else { 245 ret = ENOSPC; 246 } 247 } else { 248 ret = EINVAL; 249 } 228 hash_table_find(&instance->ep_table, (unsigned long*)&id); 229 if (item == NULL) { 230 fibril_mutex_unlock(&instance->guard); 231 return NULL; 232 } 233 node_t *node = hash_table_get_instance(item, node_t, link); 234 if (bw) 235 *bw = node->bw; 236 250 237 fibril_mutex_unlock(&instance->guard); 251 return ret; 252 } 253 /*----------------------------------------------------------------------------*/ 254 int bandwidth_free(bandwidth_t *instance, usb_address_t address, 255 usb_endpoint_t endpoint, usb_direction_t direction) 256 { 257 assert(instance); 258 transfer_t trans = { 259 .address = address, 260 .endpoint = endpoint, 261 .direction = direction, 262 }; 263 fibril_mutex_lock(&instance->guard); 264 link_t *item = 265 hash_table_find(&instance->reserved, (unsigned long*)&trans); 266 int ret = EOK; 267 if (item != NULL) { 268 transfer_status_t *status = 269 hash_table_get_instance(item, transfer_status_t, link); 270 assert(status); 271 if (!status->used) { 272 ret = ENOENT; 273 } 274 status->used = false; 275 } else { 276 ret = EINVAL; 277 } 278 fibril_mutex_unlock(&instance->guard); 279 return ret; 280 } 238 return node->ep; 239 }
Note:
See TracChangeset
for help on using the changeset viewer.