You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					443 lines
				
				12 KiB
			
		
		
			
		
	
	
					443 lines
				
				12 KiB
			| 
											1 week ago
										 | /*
 | ||
|  |  * Copyright (c) 2006-2021, 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 <dfs_fs.h>
 | ||
|  | #include <drivers/usb_host.h>
 | ||
|  | #include "mass.h"
 | ||
|  | 
 | ||
|  | #ifdef RT_USBH_MSTORAGE
 | ||
|  | 
 | ||
|  | #define UDISK_MAX_COUNT        8
 | ||
|  | static rt_uint8_t _udisk_idset = 0; | ||
|  | 
 | ||
|  | static int udisk_get_id(void) | ||
|  | { | ||
|  |     int i; | ||
|  | 
 | ||
|  |     for(i=0; i< UDISK_MAX_COUNT; i++) | ||
|  |     { | ||
|  |         if((_udisk_idset & (1 << i)) != 0) continue; | ||
|  |         else break; | ||
|  |     } | ||
|  | 
 | ||
|  |     /* it should not happen */ | ||
|  |     if(i == UDISK_MAX_COUNT) RT_ASSERT(0); | ||
|  | 
 | ||
|  |     _udisk_idset |= (1 << i); | ||
|  |     return i; | ||
|  | } | ||
|  | 
 | ||
|  | static void udisk_free_id(int id) | ||
|  | { | ||
|  |     RT_ASSERT(id < UDISK_MAX_COUNT) | ||
|  | 
 | ||
|  |     _udisk_idset &= ~(1 << id); | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * This function will initialize the udisk device | ||
|  |  * | ||
|  |  * @param dev the pointer of device driver structure | ||
|  |  * | ||
|  |  * @return RT_EOK | ||
|  |  */ | ||
|  | static rt_err_t rt_udisk_init(rt_device_t dev) | ||
|  | { | ||
|  |     return RT_EOK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * This function will read some data from a device. | ||
|  |  * | ||
|  |  * @param dev the pointer of device driver structure | ||
|  |  * @param pos the position of reading | ||
|  |  * @param buffer the data buffer to save read data | ||
|  |  * @param size the size of buffer | ||
|  |  * | ||
|  |  * @return the actually read size on successful, otherwise negative returned. | ||
|  |  */ | ||
|  | static rt_size_t rt_udisk_read(rt_device_t dev, rt_off_t pos, void* buffer, | ||
|  |     rt_size_t size) | ||
|  | { | ||
|  |     rt_err_t ret; | ||
|  |     struct uhintf* intf; | ||
|  |     struct ustor_data* data; | ||
|  |     int timeout = USB_TIMEOUT_LONG; | ||
|  | 
 | ||
|  |     /* check parameter */ | ||
|  |     RT_ASSERT(dev != RT_NULL); | ||
|  |     RT_ASSERT(buffer != RT_NULL); | ||
|  | 
 | ||
|  |     if(size > 4096) timeout *= 2; | ||
|  | 
 | ||
|  |     data = (struct ustor_data*)dev->user_data; | ||
|  |     intf = data->intf; | ||
|  | 
 | ||
|  |     ret = rt_usbh_storage_read10(intf, (rt_uint8_t*)buffer, pos, size, timeout); | ||
|  | 
 | ||
|  |     if (ret != RT_EOK) | ||
|  |     { | ||
|  |         rt_kprintf("usb mass_storage read failed\n"); | ||
|  |         return 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     return size; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * This function will write some data to a device. | ||
|  |  * | ||
|  |  * @param dev the pointer of device driver structure | ||
|  |  * @param pos the position of written | ||
|  |  * @param buffer the data buffer to be written to device | ||
|  |  * @param size the size of buffer | ||
|  |  * | ||
|  |  * @return the actually written size on successful, otherwise negative returned. | ||
|  |  */ | ||
|  | static rt_size_t rt_udisk_write (rt_device_t dev, rt_off_t pos, const void* buffer, | ||
|  |     rt_size_t size) | ||
|  | { | ||
|  |     rt_err_t ret; | ||
|  |     struct uhintf* intf; | ||
|  |     struct ustor_data* data; | ||
|  |     int timeout = USB_TIMEOUT_LONG; | ||
|  | 
 | ||
|  |     /* check parameter */ | ||
|  |     RT_ASSERT(dev != RT_NULL); | ||
|  |     RT_ASSERT(buffer != RT_NULL); | ||
|  | 
 | ||
|  |     if(size * SECTOR_SIZE > 4096) timeout *= 2; | ||
|  | 
 | ||
|  |     data = (struct ustor_data*)dev->user_data; | ||
|  |     intf = data->intf; | ||
|  | 
 | ||
|  |     ret = rt_usbh_storage_write10(intf, (rt_uint8_t*)buffer, pos, size, timeout); | ||
|  |     if (ret != RT_EOK) | ||
|  |     { | ||
|  |         rt_kprintf("usb mass_storage write %d sector failed\n", size); | ||
|  |         return 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     return size; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * 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. | ||
|  |  */ | ||
|  | static rt_err_t rt_udisk_control(rt_device_t dev, int cmd, void *args) | ||
|  | { | ||
|  |     ustor_t stor; | ||
|  |     struct ustor_data* data; | ||
|  | 
 | ||
|  |     /* check parameter */ | ||
|  |     RT_ASSERT(dev != RT_NULL); | ||
|  | 
 | ||
|  |     data = (struct ustor_data*)dev->user_data; | ||
|  |     stor = (ustor_t)data->intf->user_data; | ||
|  | 
 | ||
|  |     if (cmd == RT_DEVICE_CTRL_BLK_GETGEOME) | ||
|  |     { | ||
|  |         struct rt_device_blk_geometry *geometry; | ||
|  | 
 | ||
|  |         geometry = (struct rt_device_blk_geometry *)args; | ||
|  |         if (geometry == RT_NULL) return -RT_ERROR; | ||
|  | 
 | ||
|  |         geometry->bytes_per_sector = SECTOR_SIZE; | ||
|  |         geometry->block_size = stor->capicity[1]; | ||
|  |         geometry->sector_count = stor->capicity[0]; | ||
|  |     } | ||
|  | 
 | ||
|  |     return RT_EOK; | ||
|  | } | ||
|  | 
 | ||
|  | #ifdef RT_USING_DEVICE_OPS
 | ||
|  | const static struct rt_device_ops udisk_device_ops = | ||
|  | { | ||
|  |     rt_udisk_init, | ||
|  |     RT_NULL, | ||
|  |     RT_NULL, | ||
|  |     rt_udisk_read, | ||
|  |     rt_udisk_write, | ||
|  |     rt_udisk_control | ||
|  | }; | ||
|  | #endif
 | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * This function will run udisk driver when usb disk is detected. | ||
|  |  * | ||
|  |  * @param intf the usb interface instance. | ||
|  |  * | ||
|  |  * @return the error code, RT_EOK on successfully. | ||
|  |  */ | ||
|  | rt_err_t rt_udisk_run(struct uhintf* intf) | ||
|  | { | ||
|  |     int i = 0; | ||
|  |     rt_err_t ret; | ||
|  |     char dname[8]; | ||
|  |     char sname[8]; | ||
|  |     rt_uint8_t max_lun, *sector, sense[18], inquiry[36]; | ||
|  |     struct dfs_partition part[MAX_PARTITION_COUNT]; | ||
|  |     ustor_t stor; | ||
|  | 
 | ||
|  |     /* check parameter */ | ||
|  |     RT_ASSERT(intf != RT_NULL); | ||
|  | 
 | ||
|  |     /* set interface */ | ||
|  | //    ret = rt_usbh_set_interface(intf->device, intf->intf_desc->bInterfaceNumber);
 | ||
|  | //    if(ret != RT_EOK)
 | ||
|  | //        rt_usbh_clear_feature(intf->device, 0, USB_FEATURE_ENDPOINT_HALT);
 | ||
|  |     /* reset mass storage class device */ | ||
|  |     ret = rt_usbh_storage_reset(intf); | ||
|  |     if(ret != RT_EOK) return ret; | ||
|  | 
 | ||
|  |     stor = (ustor_t)intf->user_data; | ||
|  | 
 | ||
|  |     /* get max logic unit number */ | ||
|  |     ret = rt_usbh_storage_get_max_lun(intf, &max_lun); | ||
|  |     if(ret != RT_EOK) | ||
|  |         rt_usbh_clear_feature(intf->device, 0, USB_FEATURE_ENDPOINT_HALT); | ||
|  | 
 | ||
|  |     /* reset pipe in endpoint */ | ||
|  |     if(stor->pipe_in->status == UPIPE_STATUS_STALL) | ||
|  |     { | ||
|  |         ret = rt_usbh_clear_feature(intf->device, | ||
|  |         stor->pipe_in->ep.bEndpointAddress, USB_FEATURE_ENDPOINT_HALT); | ||
|  |         if(ret != RT_EOK) return ret; | ||
|  |     } | ||
|  | 
 | ||
|  | 
 | ||
|  |     /* reset pipe out endpoint */ | ||
|  |     if(stor->pipe_out->status == UPIPE_STATUS_STALL) | ||
|  |     { | ||
|  |         ret = rt_usbh_clear_feature(intf->device, | ||
|  |         stor->pipe_out->ep.bEndpointAddress, USB_FEATURE_ENDPOINT_HALT); | ||
|  |         if(ret != RT_EOK) return ret; | ||
|  |     } | ||
|  | 
 | ||
|  |     while((ret = rt_usbh_storage_inquiry(intf, inquiry)) != RT_EOK) | ||
|  |     { | ||
|  |         if(ret == -RT_EIO) return ret; | ||
|  | 
 | ||
|  |         rt_thread_delay(5); | ||
|  |         if(i++ < 10) continue; | ||
|  |         rt_kprintf("rt_usbh_storage_inquiry error\n"); | ||
|  |         return -RT_ERROR; | ||
|  |     } | ||
|  | 
 | ||
|  |     i = 0; | ||
|  | 
 | ||
|  |     /* wait device ready */ | ||
|  |     while((ret = rt_usbh_storage_test_unit_ready(intf)) != RT_EOK) | ||
|  |     { | ||
|  |         if(ret == -RT_EIO) return ret; | ||
|  | 
 | ||
|  |         ret = rt_usbh_storage_request_sense(intf, sense); | ||
|  |         if(ret == -RT_EIO) return ret; | ||
|  | 
 | ||
|  |         rt_thread_delay(10); | ||
|  |         if(i++ < 10) continue; | ||
|  | 
 | ||
|  |         rt_kprintf("rt_usbh_storage_test_unit_ready error\n"); | ||
|  |         return -RT_ERROR; | ||
|  |     } | ||
|  | 
 | ||
|  |     i = 0; | ||
|  |     rt_memset(stor->capicity, 0, sizeof(stor->capicity)); | ||
|  | 
 | ||
|  |     /* get storage capacity */ | ||
|  |     while((ret = rt_usbh_storage_get_capacity(intf, | ||
|  |         (rt_uint8_t*)stor->capicity)) != RT_EOK) | ||
|  |     { | ||
|  |         if(ret == -RT_EIO) return ret; | ||
|  | 
 | ||
|  |         rt_thread_delay(50); | ||
|  |         if(i++ < 10) continue; | ||
|  | 
 | ||
|  |         stor->capicity[0] = 2880; | ||
|  |         stor->capicity[1] = 0x200; | ||
|  | 
 | ||
|  |         rt_kprintf("rt_usbh_storage_get_capacity error\n"); | ||
|  |         break; | ||
|  |     } | ||
|  | 
 | ||
|  |     stor->capicity[0] = uswap_32(stor->capicity[0]); | ||
|  |     stor->capicity[1] = uswap_32(stor->capicity[1]); | ||
|  |     stor->capicity[0] += 1; | ||
|  | 
 | ||
|  |     RT_DEBUG_LOG(RT_DEBUG_USB, ("capicity %d, block size %d\n", | ||
|  |         stor->capicity[0], stor->capicity[1])); | ||
|  | 
 | ||
|  |     /* get the first sector to read partition table */ | ||
|  |     sector = (rt_uint8_t*) rt_malloc (SECTOR_SIZE); | ||
|  |     if (sector == RT_NULL) | ||
|  |     { | ||
|  |         rt_kprintf("allocate partition sector buffer failed\n"); | ||
|  |         return -RT_ERROR; | ||
|  |     } | ||
|  | 
 | ||
|  |     rt_memset(sector, 0, SECTOR_SIZE); | ||
|  | 
 | ||
|  |     RT_DEBUG_LOG(RT_DEBUG_USB, ("read partition table\n")); | ||
|  | 
 | ||
|  |     /* get the partition table */ | ||
|  |     ret = rt_usbh_storage_read10(intf, sector, 0, 1, USB_TIMEOUT_LONG); | ||
|  |     if(ret != RT_EOK) | ||
|  |     { | ||
|  |         rt_kprintf("read parition table error\n"); | ||
|  | 
 | ||
|  |         rt_free(sector); | ||
|  |         return -RT_ERROR; | ||
|  |     } | ||
|  | 
 | ||
|  |     RT_DEBUG_LOG(RT_DEBUG_USB, ("finished reading partition\n")); | ||
|  | 
 | ||
|  |     for(i=0; i<MAX_PARTITION_COUNT; i++) | ||
|  |     { | ||
|  |         /* get the first partition */ | ||
|  |         ret = dfs_filesystem_get_partition(&part[i], sector, i); | ||
|  |         if (ret == RT_EOK) | ||
|  |         { | ||
|  |             struct ustor_data* data = rt_malloc(sizeof(struct ustor_data)); | ||
|  |             rt_memset(data, 0, sizeof(struct ustor_data)); | ||
|  |             data->intf = intf; | ||
|  |             data->udisk_id = udisk_get_id(); | ||
|  |             rt_snprintf(dname, 6, "ud%d-%d", data->udisk_id, i); | ||
|  |             rt_snprintf(sname, 8, "sem_ud%d",  i); | ||
|  |             data->part.lock = rt_sem_create(sname, 1, RT_IPC_FLAG_FIFO); | ||
|  | 
 | ||
|  |             /* register sdcard device */ | ||
|  |             stor->dev[i].type    = RT_Device_Class_Block; | ||
|  | #ifdef RT_USING_DEVICE_OPS
 | ||
|  |             stor->dev[i].ops     = &udisk_device_ops; | ||
|  | #else
 | ||
|  |             stor->dev[i].init    = rt_udisk_init; | ||
|  |             stor->dev[i].read    = rt_udisk_read; | ||
|  |             stor->dev[i].write   = rt_udisk_write; | ||
|  |             stor->dev[i].control = rt_udisk_control; | ||
|  | #endif
 | ||
|  |             stor->dev[i].user_data = (void*)data; | ||
|  | 
 | ||
|  |             rt_device_register(&stor->dev[i], dname, RT_DEVICE_FLAG_RDWR | | ||
|  |                 RT_DEVICE_FLAG_REMOVABLE | RT_DEVICE_FLAG_STANDALONE); | ||
|  | 
 | ||
|  |             stor->dev_cnt++; | ||
|  |             if (dfs_mount(stor->dev[i].parent.name, UDISK_MOUNTPOINT, "elm", | ||
|  |                 0, 0) == 0) | ||
|  |             { | ||
|  |                 RT_DEBUG_LOG(RT_DEBUG_USB, ("udisk part %d mount successfully\n", i)); | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 RT_DEBUG_LOG(RT_DEBUG_USB, ("udisk part %d mount failed\n", i)); | ||
|  |             } | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             if(i == 0) | ||
|  |             { | ||
|  |                 struct ustor_data* data = rt_malloc(sizeof(struct ustor_data)); | ||
|  |                 rt_memset(data, 0, sizeof(struct ustor_data)); | ||
|  |                 data->udisk_id = udisk_get_id(); | ||
|  | 
 | ||
|  |                 /* there is no partition table */ | ||
|  |                 data->part.offset = 0; | ||
|  |                 data->part.size   = 0; | ||
|  |                 data->intf = intf; | ||
|  |                 data->part.lock = rt_sem_create("sem_ud", 1, RT_IPC_FLAG_FIFO); | ||
|  | 
 | ||
|  |                 rt_snprintf(dname, 7, "udisk%d", data->udisk_id); | ||
|  | 
 | ||
|  |                 /* register sdcard device */ | ||
|  |                 stor->dev[0].type    = RT_Device_Class_Block; | ||
|  | #ifdef RT_USING_DEVICE_OPS
 | ||
|  |                 stor->dev[i].ops     = &udisk_device_ops; | ||
|  | #else
 | ||
|  |                 stor->dev[0].init    = rt_udisk_init; | ||
|  |                 stor->dev[0].read    = rt_udisk_read; | ||
|  |                 stor->dev[0].write   = rt_udisk_write; | ||
|  |                 stor->dev[0].control = rt_udisk_control; | ||
|  | #endif
 | ||
|  |                 stor->dev[0].user_data = (void*)data; | ||
|  | 
 | ||
|  |                 rt_device_register(&stor->dev[0], dname, | ||
|  |                     RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_REMOVABLE | ||
|  |                     | RT_DEVICE_FLAG_STANDALONE); | ||
|  | 
 | ||
|  |                 stor->dev_cnt++; | ||
|  |                 if (dfs_mount(stor->dev[0].parent.name, UDISK_MOUNTPOINT, | ||
|  |                     "elm", 0, 0) == 0) | ||
|  |                 { | ||
|  |                     rt_kprintf("Mount FAT on Udisk successful.\n"); | ||
|  |                 } | ||
|  |                 else | ||
|  |                 { | ||
|  |                     rt_kprintf("Mount FAT on Udisk failed.\n"); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             break; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     rt_free(sector); | ||
|  | 
 | ||
|  |     return RT_EOK; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * This function will be invoked when usb disk plug out is detected and it would clean | ||
|  |  * and release all udisk related resources. | ||
|  |  * | ||
|  |  * @param intf the usb interface instance. | ||
|  |  * | ||
|  |  * @return the error code, RT_EOK on successfully. | ||
|  |  */ | ||
|  | rt_err_t rt_udisk_stop(struct uhintf* intf) | ||
|  | { | ||
|  |     int i; | ||
|  |     ustor_t stor; | ||
|  |     struct ustor_data* data; | ||
|  | 
 | ||
|  |     /* check parameter */ | ||
|  |     RT_ASSERT(intf != RT_NULL); | ||
|  |     RT_ASSERT(intf->device != RT_NULL); | ||
|  | 
 | ||
|  |     stor = (ustor_t)intf->user_data; | ||
|  |     RT_ASSERT(stor != RT_NULL); | ||
|  | 
 | ||
|  |     for(i=0; i<stor->dev_cnt; i++) | ||
|  |     { | ||
|  |         rt_device_t dev = &stor->dev[i]; | ||
|  |         data = (struct ustor_data*)dev->user_data; | ||
|  | 
 | ||
|  |         /* unmount filesystem */ | ||
|  |         dfs_unmount(UDISK_MOUNTPOINT); | ||
|  | 
 | ||
|  |         /* delete semaphore */ | ||
|  |         rt_sem_delete(data->part.lock); | ||
|  |         udisk_free_id(data->udisk_id); | ||
|  |         rt_free(data); | ||
|  | 
 | ||
|  |         /* unregister device */ | ||
|  |         rt_device_unregister(&stor->dev[i]); | ||
|  |     } | ||
|  | 
 | ||
|  |     return RT_EOK; | ||
|  | } | ||
|  | 
 | ||
|  | #endif
 | ||
|  | 
 |