| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 | /* * Copyright (c) 2006-2018, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date           Author       Notes * 2011-12-12     Yi Qiu      first version */#include <rtthread.h>#include <drivers/usb_host.h>#include "mass.h"#ifdef RT_USBH_MSTORAGEextern rt_err_t rt_udisk_run(struct uhintf* intf);extern rt_err_t rt_udisk_stop(struct uhintf* intf);static struct uclass_driver storage_driver;/** * This function will do USBREQ_GET_MAX_LUN request for the usb interface instance. * * @param intf the interface instance. * @param max_lun the buffer to save max_lun. *  * @return the error code, RT_EOK on successfully. */static rt_err_t _pipe_check(struct uhintf* intf, upipe_t pipe){    struct uinstance* device;            rt_err_t ret;    ustor_t stor;    int size = 0;     struct ustorage_csw csw;    if(intf == RT_NULL || pipe == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }        /* get usb device instance from the interface instance */        device = intf->device;        /* get storage instance from the interface instance */            stor = (ustor_t)intf->user_data;        /* check pipe status */    if(pipe->status == UPIPE_STATUS_OK) return RT_EOK;    if(pipe->status == UPIPE_STATUS_ERROR)    {        rt_kprintf("pipe status error\n");        return -RT_EIO;    }    if(pipe->status == UPIPE_STATUS_STALL)    {        /* clear the pipe stall status */        ret = rt_usbh_clear_feature(device, pipe->ep.bEndpointAddress,             USB_FEATURE_ENDPOINT_HALT);        if(ret != RT_EOK) return ret;    }        rt_thread_delay(50);    rt_kprintf("pipes1 0x%x, 0x%x\n", stor->pipe_in, stor->pipe_out);    stor->pipe_in->status = UPIPE_STATUS_OK;    RT_DEBUG_LOG(RT_DEBUG_USB, ("clean storage in pipe stall\n"));    /* it should receive csw after clear the stall feature */    size = rt_usb_hcd_pipe_xfer(stor->pipe_in->inst->hcd,         stor->pipe_in, &csw, SIZEOF_CSW, 100);    if(size != SIZEOF_CSW)     {        rt_kprintf("receive the csw after stall failed\n");        return -RT_EIO;    }        return -RT_ERROR;}/** * This function will do USBREQ_GET_MAX_LUN request for the usb interface instance. * * @param intf the interface instance. * @param max_lun the buffer to save max_lun. *  * @return the error code, RT_EOK on successfully. */static rt_err_t rt_usb_bulk_only_xfer(struct uhintf* intf,     ustorage_cbw_t cmd, rt_uint8_t* buffer, int timeout){    rt_size_t size;    rt_err_t ret;        upipe_t pipe;    struct ustorage_csw csw;    ustor_t stor;    RT_ASSERT(cmd != RT_NULL);        if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }        /* get storage instance from the interface instance */        stor = (ustor_t)intf->user_data;    do    {        /* send the cbw */        size = rt_usb_hcd_pipe_xfer(stor->pipe_out->inst->hcd, stor->pipe_out,             cmd, SIZEOF_CBW, timeout);        if(size != SIZEOF_CBW)         {            rt_kprintf("CBW size error\n");            return -RT_EIO;        }        if(cmd->xfer_len != 0)        {            pipe = (cmd->dflags == CBWFLAGS_DIR_IN) ? stor->pipe_in :                stor->pipe_out;            size = rt_usb_hcd_pipe_xfer(pipe->inst->hcd, pipe, (void*)buffer,                 cmd->xfer_len, timeout);            if(size != cmd->xfer_len)            {                rt_kprintf("request size %d, transfer size %d\n",                     cmd->xfer_len, size);                break;            }            }                /* receive the csw */        size = rt_usb_hcd_pipe_xfer(stor->pipe_in->inst->hcd, stor->pipe_in,             &csw, SIZEOF_CSW, timeout);        if(size != SIZEOF_CSW)         {            rt_kprintf("csw size error\n");            return -RT_EIO;        }    }while(0);    /* check in pipes status */    ret = _pipe_check(intf, stor->pipe_in);    if(ret != RT_EOK)     {        rt_kprintf("in pipe error\n");        return ret;    }        /* check out pipes status */    ret = _pipe_check(intf, stor->pipe_out);    if(ret != RT_EOK)    {        rt_kprintf("out pipe error\n");        return ret;    }        /* check csw status */    if(csw.signature != CSW_SIGNATURE || csw.tag != CBW_TAG_VALUE)    {        rt_kprintf("csw signature error\n");        return -RT_EIO;    }        if(csw.status != 0)    {        //rt_kprintf("csw status error:%d\n",csw.status);        return -RT_ERROR;    }        return RT_EOK;}/** * This function will do USBREQ_GET_MAX_LUN request for the usb interface instance. * * @param intf the interface instance. * @param max_lun the buffer to save max_lun. *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_get_max_lun(struct uhintf* intf, rt_uint8_t* max_lun){    struct uinstance* device;        struct urequest setup;    int timeout = USB_TIMEOUT_BASIC;    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }        /* parameter check */    RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_get_max_lun\n"));    /* get usb device instance from the interface instance */        device = intf->device;        /* construct the request */    setup.request_type = USB_REQ_TYPE_DIR_IN | USB_REQ_TYPE_CLASS |         USB_REQ_TYPE_INTERFACE;    setup.bRequest = USBREQ_GET_MAX_LUN;    setup.wValue = intf->intf_desc->bInterfaceNumber;    setup.wIndex = 0;    setup.wLength = 1;    /* do control transfer request */    if(rt_usb_hcd_setup_xfer(device->hcd, device->pipe_ep0_out, &setup, timeout) != 8)    {        return -RT_EIO;    }    if(rt_usb_hcd_pipe_xfer(device->hcd, device->pipe_ep0_in, max_lun, 1, timeout) != 1)    {        return -RT_EIO;    }    if(rt_usb_hcd_pipe_xfer(device->hcd, device->pipe_ep0_out, RT_NULL, 0, timeout) != 0)    {        return -RT_EIO;    }    return RT_EOK;}/** * This function will do USBREQ_MASS_STORAGE_RESET request for the usb interface instance. * * @param intf the interface instance. *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_reset(struct uhintf* intf){    struct urequest setup;    struct uinstance* device;        int timeout = USB_TIMEOUT_BASIC;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }        RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_reset\n"));    /* get usb device instance from the interface instance */        device = intf->device;        /* construct the request */    setup.request_type = USB_REQ_TYPE_DIR_OUT | USB_REQ_TYPE_CLASS |         USB_REQ_TYPE_INTERFACE;    setup.bRequest = USBREQ_MASS_STORAGE_RESET;    setup.wIndex = intf->intf_desc->bInterfaceNumber;    setup.wLength = 0;    setup.wValue = 0;    if(rt_usb_hcd_setup_xfer(device->hcd, device->pipe_ep0_out, &setup, timeout) != 8)    {        return -RT_EIO;    }    if(rt_usb_hcd_pipe_xfer(device->hcd, device->pipe_ep0_in, RT_NULL, 0, timeout) != 0)    {        return -RT_EIO;    }    return RT_EOK;}/** * This function will execute SCSI_READ_10 command to read data from the usb device. * * @param intf the interface instance. * @param buffer the data buffer to save read data * @param sector the start sector address to read.  * @param sector the sector count to read. *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_read10(struct uhintf* intf, rt_uint8_t *buffer,     rt_uint32_t sector, rt_size_t count, int timeout){    struct ustorage_cbw cmd;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("interface is not available\n");        return -RT_EIO;    }            RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_read10\n"));    /* construct the command block wrapper */    rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = SECTOR_SIZE * count;    cmd.dflags = CBWFLAGS_DIR_IN;    cmd.lun = 0;    cmd.cb_len = 10;    cmd.cb[0] = SCSI_READ_10;    cmd.cb[1] = 0;    cmd.cb[2] = (rt_uint8_t)(sector >> 24);    cmd.cb[3] = (rt_uint8_t)(sector >> 16);    cmd.cb[4] = (rt_uint8_t)(sector >> 8);    cmd.cb[5] = (rt_uint8_t)sector;    cmd.cb[6] = 0;    cmd.cb[7] = (count & 0xff00) >> 8;    cmd.cb[8] = (rt_uint8_t) count & 0xff;    return rt_usb_bulk_only_xfer(intf, &cmd, buffer, timeout);}/** * This function will execute SCSI_WRITE_10 command to write data to the usb device. * * @param intf the interface instance. * @param buffer the data buffer to save write data * @param sector the start sector address to write.  * @param sector the sector count to write. *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_write10(struct uhintf* intf, rt_uint8_t *buffer,     rt_uint32_t sector, rt_size_t count, int timeout){    struct ustorage_cbw cmd;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }            RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_write10\n"));    /* construct the command block wrapper */    rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = SECTOR_SIZE * count;    cmd.dflags = CBWFLAGS_DIR_OUT;    cmd.lun = 0;    cmd.cb_len = 10;    cmd.cb[0] = SCSI_WRITE_10;    cmd.cb[1] = 0;    cmd.cb[2] = (rt_uint8_t)(sector >> 24);    cmd.cb[3] = (rt_uint8_t)(sector >> 16);    cmd.cb[4] = (rt_uint8_t)(sector >> 8);    cmd.cb[5] = (rt_uint8_t)sector;    cmd.cb[6] = 0;    cmd.cb[7] = (count & 0xff00) >> 8;    cmd.cb[8] = (rt_uint8_t) count & 0xff;    return rt_usb_bulk_only_xfer(intf, &cmd, buffer, timeout);}/** * This function will execute SCSI_REQUEST_SENSE command to get sense data. * * @param intf the interface instance. * @param buffer the data buffer to save sense data *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_request_sense(struct uhintf* intf, rt_uint8_t* buffer){    struct ustorage_cbw cmd;    int timeout = USB_TIMEOUT_LONG;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }            RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_request_sense\n"));    /* construct the command block wrapper */        rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = 18;    cmd.dflags = CBWFLAGS_DIR_IN;    cmd.lun = 0;    cmd.cb_len = 6;    cmd.cb[0] = SCSI_REQUEST_SENSE;    cmd.cb[4] = 18;    return rt_usb_bulk_only_xfer(intf, &cmd, buffer, timeout);}/** * This function will execute SCSI_TEST_UNIT_READY command to get unit ready status. * * @param intf the interface instance. *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_test_unit_ready(struct uhintf* intf){    struct ustorage_cbw cmd;    int timeout = USB_TIMEOUT_LONG;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }            RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_test_unit_ready\n"));    /* construct the command block wrapper */    rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = 0;    cmd.dflags = CBWFLAGS_DIR_OUT;    cmd.lun = 0;    cmd.cb_len = 12;    cmd.cb[0] = SCSI_TEST_UNIT_READY;     return rt_usb_bulk_only_xfer(intf, &cmd, RT_NULL, timeout);}/** * This function will execute SCSI_INQUIRY_CMD command to get inquiry data. * * @param intf the interface instance. * @param buffer the data buffer to save inquiry data *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_inquiry(struct uhintf* intf, rt_uint8_t* buffer){    struct ustorage_cbw cmd;    int timeout = USB_TIMEOUT_LONG;        /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }            RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_inquiry\n"));    /* construct the command block wrapper */    rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = 36;    cmd.dflags = CBWFLAGS_DIR_IN;    cmd.lun = 0;    cmd.cb_len = 6;//12    cmd.cb[0] = SCSI_INQUIRY_CMD;    cmd.cb[4] = 36;    return rt_usb_bulk_only_xfer(intf, &cmd, buffer, timeout);}/** * This function will execute SCSI_READ_CAPACITY command to get capacity data. * * @param intf the interface instance. * @param buffer the data buffer to save capacity data *  * @return the error code, RT_EOK on successfully. */rt_err_t rt_usbh_storage_get_capacity(struct uhintf* intf, rt_uint8_t* buffer){    struct ustorage_cbw cmd;    int timeout = USB_TIMEOUT_LONG;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }        RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_get_capacity\n"));    /* construct the command block wrapper */    rt_memset(&cmd, 0, sizeof(struct ustorage_cbw));    cmd.signature = CBW_SIGNATURE;    cmd.tag = CBW_TAG_VALUE;    cmd.xfer_len = 8;    cmd.dflags = CBWFLAGS_DIR_IN;    cmd.lun = 0;    cmd.cb_len = 12;    cmd.cb[0] = SCSI_READ_CAPACITY;    return rt_usb_bulk_only_xfer(intf, &cmd, buffer, timeout);}/** * This function will run mass storage class driver when usb device is detected * and identified as a mass storage class device, it will continue to do the enumulate  * process. * * @param arg the argument. *  * @return the error code, RT_EOK on successfully. */static rt_err_t rt_usbh_storage_enable(void* arg){    int i = 0;    rt_err_t ret;        ustor_t stor;    struct uhintf* intf = (struct uhintf*)arg;    /* parameter check */    if(intf == RT_NULL)    {        rt_kprintf("the interface is not available\n");        return -RT_EIO;    }    RT_DEBUG_LOG(RT_DEBUG_USB, ("subclass %d, protocal %d\n",         intf->intf_desc->bInterfaceSubClass,        intf->intf_desc->bInterfaceProtocol));            RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_run\n"));    /* only support SCSI subclass and bulk only protocal */        stor = rt_malloc(sizeof(struct ustor));    RT_ASSERT(stor != RT_NULL);    /* initilize the data structure */    rt_memset(stor, 0, sizeof(struct ustor));        intf->user_data = (void*)stor;    for(i=0; i<intf->intf_desc->bNumEndpoints; i++)    {                uep_desc_t ep_desc;                /* get endpoint descriptor from interface descriptor */        rt_usbh_get_endpoint_descriptor(intf->intf_desc, i, &ep_desc);        if(ep_desc == RT_NULL)        {            rt_kprintf("rt_usb_get_endpoint_descriptor error\n");            return -RT_ERROR;        }                /* the endpoint type of mass storage class should be BULK */            if((ep_desc->bmAttributes & USB_EP_ATTR_TYPE_MASK) != USB_EP_ATTR_BULK)            continue;                /* allocate pipes according to the endpoint type */        if(ep_desc->bEndpointAddress & USB_DIR_IN)        {            /* alloc an in pipe for the storage instance */            stor->pipe_in = rt_usb_instance_find_pipe(intf->device,ep_desc->bEndpointAddress);        }        else        {                    /* alloc an output pipe for the storage instance */            stor->pipe_out = rt_usb_instance_find_pipe(intf->device,ep_desc->bEndpointAddress);        }    }    /* check pipes infomation */    if(stor->pipe_in == RT_NULL || stor->pipe_out == RT_NULL)    {        rt_kprintf("pipe error, unsupported device\n");        return -RT_ERROR;    }            /* should implement as callback */    ret = rt_udisk_run(intf);    if(ret != RT_EOK) return ret;    return RT_EOK;}/** * This function will be invoked when usb device plug out is detected and it would clean  * and release all mass storage class related resources. * * @param arg the argument. *  * @return the error code, RT_EOK on successfully. */static rt_err_t rt_usbh_storage_disable(void* arg){    ustor_t stor;    struct uhintf* intf = (struct uhintf*)arg;    /* parameter check */    RT_ASSERT(intf != RT_NULL);    RT_ASSERT(intf->user_data != RT_NULL);    RT_ASSERT(intf->device != RT_NULL);    RT_DEBUG_LOG(RT_DEBUG_USB, ("rt_usbh_storage_stop\n"));    /* get storage instance from interface instance */    stor = (ustor_t)intf->user_data;    rt_udisk_stop(intf);        /* free storage instance */    if(stor != RT_NULL) rt_free(stor);    return RT_EOK;}/** * This function will register mass storage class driver to the usb class driver manager. * and it should be invoked in the usb system initialization. *  * @return the error code, RT_EOK on successfully. */ucd_t rt_usbh_class_driver_storage(void){    storage_driver.class_code = USB_CLASS_MASS_STORAGE;        storage_driver.enable = rt_usbh_storage_enable;    storage_driver.disable = rt_usbh_storage_disable;    return &storage_driver;}#endif
 |