Changeset bdc8ab1 in mainline


Ignore:
Timestamp:
2011-03-03T23:18:05Z (14 years ago)
Author:
Jan Vesely <jano.vesely@…>
Branches:
lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
Children:
9445aad
Parents:
dc75234
Message:

Fixed: setup write sends data rounded up to max_packet_size

Unified setup read and write routines

File:
1 edited

Legend:

Unmodified
Added
Removed
  • uspace/drv/uhci-hcd/batch.c

    rdc75234 rbdc8ab1  
    3535#include <str_error.h>
    3636
     37#include <usb/usb.h>
    3738#include <usb/debug.h>
    3839
     
    4647static int batch_schedule(batch_t *instance);
    4748
     49static void batch_control(
     50    batch_t *instance, int data_stage, int status_stage);
    4851static void batch_call_in(batch_t *instance);
    4952static void batch_call_out(batch_t *instance);
     
    9396        instance->transport_buffer =
    9497           (size > 0) ? malloc32(transport_size) : NULL;
     98
    9599        if ((size > 0) && (instance->transport_buffer == NULL)) {
    96100                usb_log_error("Failed to allocate device accessible buffer.\n");
     
    168172{
    169173        assert(instance);
    170 
    171174        /* we are data out, we are supposed to provide data */
    172175        memcpy(instance->transport_buffer, instance->buffer, instance->buffer_size);
    173 
    174         const bool low_speed = instance->speed == USB_SPEED_LOW;
    175         int toggle = 0;
    176         /* setup stage */
    177         transfer_descriptor_init(instance->tds, DEFAULT_ERROR_COUNT,
    178             instance->setup_size, toggle, false, low_speed,
    179             instance->target, USB_PID_SETUP, instance->setup_buffer,
    180             &instance->tds[1]);
    181 
    182         /* data stage */
    183         size_t i = 1;
    184         for (;i < instance->packets - 1; ++i) {
    185                 char *data =
    186                     instance->transport_buffer + ((i - 1) * instance->max_packet_size);
    187                 toggle = 1 - toggle;
    188 
    189                 transfer_descriptor_init(&instance->tds[i], DEFAULT_ERROR_COUNT,
    190                     instance->max_packet_size, toggle++, false, low_speed,
    191                     instance->target, USB_PID_OUT, data, &instance->tds[i + 1]);
    192         }
    193 
    194         /* status stage */
    195         i = instance->packets - 1;
    196         transfer_descriptor_init(&instance->tds[i], DEFAULT_ERROR_COUNT,
    197             0, 1, false, low_speed, instance->target, USB_PID_IN, NULL, NULL);
    198 
    199         instance->tds[i].status |= TD_STATUS_COMPLETE_INTERRUPT_FLAG;
    200         usb_log_debug2("Control write last TD status: %x.\n",
    201                 instance->tds[i].status);
    202 
     176        batch_control(instance, USB_PID_OUT, USB_PID_IN);
    203177        instance->next_step = batch_call_out_and_dispose;
    204178        usb_log_debug("Batch(%p) CONTROL WRITE initialized.\n", instance);
     
    209183{
    210184        assert(instance);
    211 
    212         const bool low_speed = instance->speed == USB_SPEED_LOW;
    213         int toggle = 0;
    214         /* setup stage */
    215         transfer_descriptor_init(instance->tds, DEFAULT_ERROR_COUNT,
    216             instance->setup_size, toggle, false, low_speed, instance->target,
    217             USB_PID_SETUP, instance->setup_buffer, &instance->tds[1]);
    218 
    219         /* data stage */
    220         size_t i = 1;
    221         for (;i < instance->packets - 1; ++i) {
    222                 char *data =
    223                     instance->transport_buffer + ((i - 1) * instance->max_packet_size);
    224                 toggle = 1 - toggle;
    225 
    226                 transfer_descriptor_init(&instance->tds[i], DEFAULT_ERROR_COUNT,
    227                     instance->max_packet_size, toggle, false, low_speed,
    228                                 instance->target, USB_PID_IN, data, &instance->tds[i + 1]);
    229         }
    230 
    231         /* status stage */
    232         i = instance->packets - 1;
    233         transfer_descriptor_init(&instance->tds[i], DEFAULT_ERROR_COUNT,
    234             0, 1, false, low_speed, instance->target, USB_PID_OUT, NULL, NULL);
    235 
    236         instance->tds[i].status |= TD_STATUS_COMPLETE_INTERRUPT_FLAG;
    237         usb_log_debug2("Control read last TD status: %x.\n",
    238                 instance->tds[i].status);
    239 
     185        batch_control(instance, USB_PID_IN, USB_PID_OUT);
    240186        instance->next_step = batch_call_in_and_dispose;
    241187        usb_log_debug("Batch(%p) CONTROL READ initialized.\n", instance);
     
    272218{
    273219        assert(instance);
    274 
    275220        memcpy(instance->transport_buffer, instance->buffer, instance->buffer_size);
    276221
     
    297242}
    298243/*----------------------------------------------------------------------------*/
     244static void batch_control(
     245    batch_t *instance, int data_stage, int status_stage)
     246{
     247        assert(instance);
     248
     249        const bool low_speed = instance->speed == USB_SPEED_LOW;
     250        int toggle = 0;
     251        /* setup stage */
     252        transfer_descriptor_init(instance->tds, DEFAULT_ERROR_COUNT,
     253            instance->setup_size, toggle, false, low_speed, instance->target,
     254            USB_PID_SETUP, instance->setup_buffer, &instance->tds[1]);
     255
     256        /* data stage */
     257        size_t packet = 1;
     258        size_t remain_size = instance->buffer_size;
     259        while (remain_size > 0) {
     260                char *data =
     261                    instance->transport_buffer + instance->buffer_size
     262                    - remain_size;
     263
     264                toggle = 1 - toggle;
     265
     266                const size_t packet_size =
     267                    (instance->max_packet_size > remain_size) ?
     268                    remain_size : instance->max_packet_size;
     269
     270                transfer_descriptor_init(&instance->tds[packet],
     271                    DEFAULT_ERROR_COUNT, packet_size, toggle, false, low_speed,
     272                    instance->target, data_stage, data,
     273                    &instance->tds[packet + 1]);
     274
     275                ++packet;
     276                assert(packet < instance->packets);
     277                assert(packet_size <= remain_size);
     278                remain_size -= packet_size;
     279        }
     280
     281        /* status stage */
     282        assert(packet == instance->packets - 1);
     283        transfer_descriptor_init(&instance->tds[packet], DEFAULT_ERROR_COUNT,
     284            0, 1, false, low_speed, instance->target, status_stage, NULL, NULL);
     285
     286
     287        instance->tds[packet].status |= TD_STATUS_COMPLETE_INTERRUPT_FLAG;
     288        usb_log_debug2("Control last TD status: %x.\n",
     289            instance->tds[packet].status);
     290}
     291/*----------------------------------------------------------------------------*/
    299292void batch_call_in(batch_t *instance)
    300293{
Note: See TracChangeset for help on using the changeset viewer.