/* * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved. * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. 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. * * 3. Neither the name of the copyright holder nor the names of its contributors may be used * to endorse or promote products derived from this software without specific prior written * permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 COPYRIGHT HOLDER OR * CONTRIBUTORS 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. */ #include "disk.h" #include "stdio.h" #include "stdlib.h" #include "unistd.h" #include "sys/mount.h" #include "linux/spinlock.h" #include "path_cache.h" #ifndef LOSCFG_FS_FAT_CACHE #include "los_vm_common.h" #include "user_copy.h" #endif los_disk g_sysDisk[SYS_MAX_DISK]; los_part g_sysPart[SYS_MAX_PART]; UINT32 g_uwFatSectorsPerBlock = CONFIG_FS_FAT_SECTOR_PER_BLOCK; UINT32 g_uwFatBlockNums = CONFIG_FS_FAT_BLOCK_NUMS; spinlock_t g_diskSpinlock; spinlock_t g_diskFatBlockSpinlock; UINT32 g_usbMode = 0; #define MEM_ADDR_ALIGN_BYTE 64 #define RWE_RW_RW 0755 #define DISK_LOCK(mux) do { \ if (pthread_mutex_lock(mux) != 0) { \ PRINT_ERR("%s %d, mutex lock failed\n", __FUNCTION__, __LINE__); \ } \ } while (0) #define DISK_UNLOCK(mux) do { \ if (pthread_mutex_unlock(mux) != 0) { \ PRINT_ERR("%s %d, mutex unlock failed\n", __FUNCTION__, __LINE__); \ } \ } while (0) typedef VOID *(*StorageHookFunction)(VOID *); #ifdef LOSCFG_FS_FAT_CACHE static UINT32 OsReHookFuncAddDiskRef(StorageHookFunction handler, VOID *param) __attribute__((weakref("osReHookFuncAdd"))); static UINT32 OsReHookFuncDelDiskRef(StorageHookFunction handler) __attribute__((weakref("osReHookFuncDel"))); UINT32 GetFatBlockNums(VOID) { return g_uwFatBlockNums; } VOID SetFatBlockNums(UINT32 blockNums) { g_uwFatBlockNums = blockNums; } UINT32 GetFatSectorsPerBlock(VOID) { return g_uwFatSectorsPerBlock; } VOID SetFatSectorsPerBlock(UINT32 sectorsPerBlock) { if (((sectorsPerBlock % UNSIGNED_INTEGER_BITS) == 0) && ((sectorsPerBlock >> UNINT_LOG2_SHIFT) <= BCACHE_BLOCK_FLAGS)) { g_uwFatSectorsPerBlock = sectorsPerBlock; } } #endif INT32 los_alloc_diskid_byname(const CHAR *diskName) { INT32 diskID; los_disk *disk = NULL; UINT32 intSave; size_t nameLen; if (diskName == NULL) { PRINT_ERR("The paramter disk_name is NULL"); return VFS_ERROR; } nameLen = strlen(diskName); if (nameLen > DISK_NAME) { PRINT_ERR("diskName is too long!\n"); return VFS_ERROR; } spin_lock_irqsave(&g_diskSpinlock, intSave); for (diskID = 0; diskID < SYS_MAX_DISK; diskID++) { disk = get_disk(diskID); if ((disk != NULL) && (disk->disk_status == STAT_UNUSED)) { disk->disk_status = STAT_UNREADY; break; } } spin_unlock_irqrestore(&g_diskSpinlock, intSave); if ((disk == NULL) || (diskID == SYS_MAX_DISK)) { PRINT_ERR("los_alloc_diskid_byname failed %d!\n", diskID); return VFS_ERROR; } if (disk->disk_name != NULL) { LOS_MemFree(m_aucSysMem0, disk->disk_name); disk->disk_name = NULL; } disk->disk_name = LOS_MemAlloc(m_aucSysMem0, (nameLen + 1)); if (disk->disk_name == NULL) { PRINT_ERR("los_alloc_diskid_byname alloc disk name failed\n"); return VFS_ERROR; } if (strncpy_s(disk->disk_name, (nameLen + 1), diskName, nameLen) != EOK) { PRINT_ERR("The strncpy_s failed.\n"); LOS_MemFree(m_aucSysMem0, disk->disk_name); disk->disk_name = NULL; return VFS_ERROR; } disk->disk_name[nameLen] = '\0'; return diskID; } INT32 los_get_diskid_byname(const CHAR *diskName) { INT32 diskID; los_disk *disk = NULL; size_t diskNameLen; if (diskName == NULL) { PRINT_ERR("The paramter diskName is NULL"); return VFS_ERROR; } diskNameLen = strlen(diskName); if (diskNameLen > DISK_NAME) { PRINT_ERR("diskName is too long!\n"); return VFS_ERROR; } for (diskID = 0; diskID < SYS_MAX_DISK; diskID++) { disk = get_disk(diskID); if ((disk != NULL) && (disk->disk_name != NULL) && (disk->disk_status == STAT_INUSED)) { if (strlen(disk->disk_name) != diskNameLen) { continue; } if (strcmp(diskName, disk->disk_name) == 0) { break; } } } if ((disk == NULL) || (diskID == SYS_MAX_DISK)) { PRINT_ERR("los_get_diskid_byname failed!\n"); return VFS_ERROR; } return diskID; } los_disk *los_get_mmcdisk_bytype(UINT8 type) { const CHAR *mmcDevHead = "/dev/mmcblk"; for (INT32 diskId = 0; diskId < SYS_MAX_DISK; diskId++) { los_disk *disk = get_disk(diskId); if (disk == NULL) { continue; } else if ((disk->type == type) && (strncmp(disk->disk_name, mmcDevHead, strlen(mmcDevHead)) == 0)) { return disk; } } PRINT_ERR("Cannot find the mmc disk!\n"); return NULL; } VOID OsSetUsbStatus(UINT32 diskID) { if (diskID < SYS_MAX_DISK) { g_usbMode |= (1u << diskID) & UINT_MAX; } } VOID OsClearUsbStatus(UINT32 diskID) { if (diskID < SYS_MAX_DISK) { g_usbMode &= ~((1u << diskID) & UINT_MAX); } } #ifdef LOSCFG_FS_FAT_CACHE static BOOL GetDiskUsbStatus(UINT32 diskID) { return (g_usbMode & (1u << diskID)) ? TRUE : FALSE; } #endif los_disk *get_disk(INT32 id) { if ((id >= 0) && (id < SYS_MAX_DISK)) { return &g_sysDisk[id]; } return NULL; } los_part *get_part(INT32 id) { if ((id >= 0) && (id < SYS_MAX_PART)) { return &g_sysPart[id]; } return NULL; } static UINT64 GetFirstPartStart(const los_part *part) { los_part *firstPart = NULL; los_disk *disk = get_disk((INT32)part->disk_id); firstPart = (disk == NULL) ? NULL : LOS_DL_LIST_ENTRY(disk->head.pstNext, los_part, list); return (firstPart == NULL) ? 0 : firstPart->sector_start; } static VOID DiskPartAddToDisk(los_disk *disk, los_part *part) { part->disk_id = disk->disk_id; part->part_no_disk = disk->part_count; LOS_ListTailInsert(&disk->head, &part->list); disk->part_count++; } static VOID DiskPartDelFromDisk(los_disk *disk, los_part *part) { LOS_ListDelete(&part->list); disk->part_count--; } static los_part *DiskPartAllocate(struct Vnode *dev, UINT64 start, UINT64 count) { UINT32 i; los_part *part = get_part(0); /* traversing from the beginning of the array */ if (part == NULL) { return NULL; } for (i = 0; i < SYS_MAX_PART; i++) { if (part->dev == NULL) { part->part_id = i; part->part_no_mbr = 0; part->dev = dev; part->sector_start = start; part->sector_count = count; part->part_name = NULL; LOS_ListInit(&part->list); return part; } part++; } return NULL; } static VOID DiskPartRelease(los_part *part) { part->dev = NULL; part->part_no_disk = 0; part->part_no_mbr = 0; if (part->part_name != NULL) { free(part->part_name); part->part_name = NULL; } } /* * name is a combination of disk_name, 'p' and part_count, such as "/dev/mmcblk0p0" * disk_name : DISK_NAME + 1 * 'p' : 1 * part_count: 1 */ #define DEV_NAME_BUFF_SIZE (DISK_NAME + 3) static INT32 DiskAddPart(los_disk *disk, UINT64 sectorStart, UINT64 sectorCount, BOOL IsValidPart) { CHAR devName[DEV_NAME_BUFF_SIZE]; struct Vnode *diskDev = NULL; struct Vnode *partDev = NULL; los_part *part = NULL; INT32 ret; if ((disk == NULL) || (disk->disk_status == STAT_UNUSED) || (disk->dev == NULL)) { return VFS_ERROR; } if ((sectorCount > disk->sector_count) || ((disk->sector_count - sectorCount) < sectorStart)) { PRINT_ERR("DiskAddPart failed: sector start is %llu, sector count is %llu\n", sectorStart, sectorCount); return VFS_ERROR; } diskDev = disk->dev; if (IsValidPart == TRUE) { ret = snprintf_s(devName, sizeof(devName), sizeof(devName) - 1, "%s%c%u", ((disk->disk_name == NULL) ? "null" : disk->disk_name), 'p', disk->part_count); if (ret < 0) { return VFS_ERROR; } if (register_blockdriver(devName, ((struct drv_data *)diskDev->data)->ops, RWE_RW_RW, ((struct drv_data *)diskDev->data)->priv)) { PRINT_ERR("DiskAddPart : register %s fail!\n", devName); return VFS_ERROR; } VnodeHold(); VnodeLookup(devName, &partDev, 0); part = DiskPartAllocate(partDev, sectorStart, sectorCount); VnodeDrop(); if (part == NULL) { (VOID)unregister_blockdriver(devName); return VFS_ERROR; } } else { part = DiskPartAllocate(diskDev, sectorStart, sectorCount); if (part == NULL) { return VFS_ERROR; } } DiskPartAddToDisk(disk, part); if (disk->type == EMMC) { part->type = EMMC; } return (INT32)part->part_id; } static INT32 DiskDivide(los_disk *disk, struct disk_divide_info *info) { UINT32 i; INT32 ret; disk->type = info->part[0].type; for (i = 0; i < info->part_count; i++) { if (info->sector_count < info->part[i].sector_start) { return VFS_ERROR; } if (info->part[i].sector_count > (info->sector_count - info->part[i].sector_start)) { PRINT_ERR("Part[%u] sector_start:%llu, sector_count:%llu, exceed emmc sector_count:%llu.\n", i, info->part[i].sector_start, info->part[i].sector_count, (info->sector_count - info->part[i].sector_start)); info->part[i].sector_count = info->sector_count - info->part[i].sector_start; PRINT_ERR("Part[%u] sector_count change to %llu.\n", i, info->part[i].sector_count); ret = DiskAddPart(disk, info->part[i].sector_start, info->part[i].sector_count, TRUE); if (ret == VFS_ERROR) { return VFS_ERROR; } break; } ret = DiskAddPart(disk, info->part[i].sector_start, info->part[i].sector_count, TRUE); if (ret == VFS_ERROR) { return VFS_ERROR; } } return ENOERR; } static CHAR GPTPartitionTypeRecognition(const CHAR *parBuf) { const CHAR *buf = parBuf; const CHAR *fsType = "FAT"; const CHAR *str = "\xEB\x52\x90" "NTFS "; /* NTFS Boot entry point */ if (((LD_DWORD_DISK(&buf[BS_FILSYSTEMTYPE32]) & BS_FS_TYPE_MASK) == BS_FS_TYPE_VALUE) || (strncmp(&buf[BS_FILSYSTYPE], fsType, strlen(fsType)) == 0)) { return BS_FS_TYPE_FAT; } else if (strncmp(&buf[BS_JMPBOOT], str, strlen(str)) == 0) { return BS_FS_TYPE_NTFS; } return ENOERR; } static INT32 DiskPartitionMemZalloc(size_t boundary, size_t size, CHAR **gptBuf, CHAR **partitionBuf) { CHAR *buffer1 = NULL; CHAR *buffer2 = NULL; buffer1 = (CHAR *)memalign(boundary, size); if (buffer1 == NULL) { PRINT_ERR("%s buffer1 malloc %lu failed! %d\n", __FUNCTION__, size, __LINE__); return -ENOMEM; } buffer2 = (CHAR *)memalign(boundary, size); if (buffer2 == NULL) { PRINT_ERR("%s buffer2 malloc %lu failed! %d\n", __FUNCTION__, size, __LINE__); free(buffer1); return -ENOMEM; } (VOID)memset_s(buffer1, size, 0, size); (VOID)memset_s(buffer2, size, 0, size); *gptBuf = buffer1; *partitionBuf = buffer2; return ENOERR; } static INT32 GPTInfoGet(struct Vnode *blkDrv, CHAR *gptBuf) { INT32 ret; struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)gptBuf, 1, 1); /* Read the device first sector */ if (ret != 1) { /* Read failed */ PRINT_ERR("%s %d\n", __FUNCTION__, __LINE__); return -EIO; } if (!VERIFY_GPT(gptBuf)) { PRINT_ERR("%s %d\n", __FUNCTION__, __LINE__); return VFS_ERROR; } return ENOERR; } static INT32 OsGPTPartitionRecognitionSub(struct disk_divide_info *info, const CHAR *partitionBuf, UINT32 *partitionCount, UINT64 partitionStart, UINT64 partitionEnd) { CHAR partitionType; if (VERIFY_FS(partitionBuf)) { partitionType = GPTPartitionTypeRecognition(partitionBuf); if (partitionType) { if (*partitionCount >= MAX_DIVIDE_PART_PER_DISK) { return VFS_ERROR; } info->part[*partitionCount].type = partitionType; info->part[*partitionCount].sector_start = partitionStart; info->part[*partitionCount].sector_count = (partitionEnd - partitionStart) + 1; (*partitionCount)++; } else { PRINT_ERR("The partition type is not allowed to use!\n"); } } else { PRINT_ERR("Do not support the partition type!\n"); } return ENOERR; } static INT32 OsGPTPartitionRecognition(struct Vnode *blkDrv, struct disk_divide_info *info, const CHAR *gptBuf, CHAR *partitionBuf, UINT32 *partitionCount) { UINT32 j; INT32 ret = VFS_ERROR; UINT64 partitionStart, partitionEnd; struct block_operations *bops = NULL; for (j = 0; j < PAR_ENTRY_NUM_PER_SECTOR; j++) { if (!VERITY_AVAILABLE_PAR(&gptBuf[j * TABLE_SIZE])) { PRINTK("The partition type is ESP or MSR!\n"); continue; } if (!VERITY_PAR_VALID(&gptBuf[j * TABLE_SIZE])) { return VFS_ERROR; } partitionStart = LD_QWORD_DISK(&gptBuf[(j * TABLE_SIZE) + GPT_PAR_START_OFFSET]); partitionEnd = LD_QWORD_DISK(&gptBuf[(j * TABLE_SIZE) + GPT_PAR_END_OFFSET]); if ((partitionStart >= partitionEnd) || (partitionEnd > info->sector_count)) { PRINT_ERR("GPT partition %u recognition failed : partitionStart = %llu, partitionEnd = %llu\n", j, partitionStart, partitionEnd); return VFS_ERROR; } (VOID)memset_s(partitionBuf, info->sector_size, 0, info->sector_size); bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)partitionBuf, partitionStart, 1); if (ret != 1) { /* read failed */ PRINT_ERR("%s %d\n", __FUNCTION__, __LINE__); return -EIO; } ret = OsGPTPartitionRecognitionSub(info, partitionBuf, partitionCount, partitionStart, partitionEnd); if (ret != ENOERR) { return VFS_ERROR; } } return ret; } static INT32 DiskGPTPartitionRecognition(struct Vnode *blkDrv, struct disk_divide_info *info) { CHAR *gptBuf = NULL; CHAR *partitionBuf = NULL; UINT32 tableNum, i, index; UINT32 partitionCount = 0; INT32 ret; ret = DiskPartitionMemZalloc(MEM_ADDR_ALIGN_BYTE, info->sector_size, &gptBuf, &partitionBuf); if (ret != ENOERR) { return ret; } ret = GPTInfoGet(blkDrv, gptBuf); if (ret < 0) { goto OUT_WITH_MEM; } tableNum = LD_DWORD_DISK(&gptBuf[TABLE_NUM_OFFSET]); if (tableNum > TABLE_MAX_NUM) { tableNum = TABLE_MAX_NUM; } index = (tableNum % PAR_ENTRY_NUM_PER_SECTOR) ? ((tableNum / PAR_ENTRY_NUM_PER_SECTOR) + 1) : (tableNum / PAR_ENTRY_NUM_PER_SECTOR); for (i = 0; i < index; i++) { (VOID)memset_s(gptBuf, info->sector_size, 0, info->sector_size); struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)gptBuf, TABLE_START_SECTOR + i, 1); if (ret != 1) { /* read failed */ PRINT_ERR("%s %d\n", __FUNCTION__, __LINE__); ret = -EIO; goto OUT_WITH_MEM; } ret = OsGPTPartitionRecognition(blkDrv, info, gptBuf, partitionBuf, &partitionCount); if (ret < 0) { if (ret == VFS_ERROR) { ret = (INT32)partitionCount; } goto OUT_WITH_MEM; } } ret = (INT32)partitionCount; OUT_WITH_MEM: free(gptBuf); free(partitionBuf); return ret; } static INT32 OsMBRInfoGet(struct Vnode *blkDrv, CHAR *mbrBuf) { INT32 ret; /* read MBR, start from sector 0, length is 1 sector */ struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)mbrBuf, 0, 1); if (ret != 1) { /* read failed */ PRINT_ERR("driver read return error: %d\n", ret); return -EIO; } /* Check boot record signature. */ if (LD_WORD_DISK(&mbrBuf[BS_SIG55AA]) != BS_SIG55AA_VALUE) { return VFS_ERROR; } return ENOERR; } static INT32 OsEBRInfoGet(struct Vnode *blkDrv, const struct disk_divide_info *info, CHAR *ebrBuf, const CHAR *mbrBuf) { INT32 ret; if (VERIFY_FS(mbrBuf)) { if (info->sector_count <= LD_DWORD_DISK(&mbrBuf[PAR_OFFSET + PAR_START_OFFSET])) { return VFS_ERROR; } struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)ebrBuf, LD_DWORD_DISK(&mbrBuf[PAR_OFFSET + PAR_START_OFFSET]), 1); if ((ret != 1) || (!VERIFY_FS(ebrBuf))) { /* read failed */ PRINT_ERR("OsEBRInfoGet, verify_fs error, ret = %d\n", ret); return -EIO; } } return ENOERR; } static INT32 OsPrimaryPartitionRecognition(const CHAR *mbrBuf, struct disk_divide_info *info, INT32 *extendedPos, INT32 *mbrCount) { INT32 i; CHAR mbrPartitionType; INT32 extendedFlag = 0; INT32 count = 0; for (i = 0; i < MAX_PRIMARY_PART_PER_DISK; i++) { mbrPartitionType = mbrBuf[PAR_OFFSET + PAR_TYPE_OFFSET + (i * PAR_TABLE_SIZE)]; if (mbrPartitionType) { info->part[i].type = mbrPartitionType; info->part[i].sector_start = LD_DWORD_DISK(&mbrBuf[PAR_OFFSET + PAR_START_OFFSET + (i * PAR_TABLE_SIZE)]); info->part[i].sector_count = LD_DWORD_DISK(&mbrBuf[PAR_OFFSET + PAR_COUNT_OFFSET + (i * PAR_TABLE_SIZE)]); if ((mbrPartitionType == EXTENDED_PAR) || (mbrPartitionType == EXTENDED_8G)) { extendedFlag = 1; *extendedPos = i; continue; } count++; } } *mbrCount = count; return extendedFlag; } static INT32 OsLogicalPartitionRecognition(struct Vnode *blkDrv, struct disk_divide_info *info, UINT32 extendedAddress, CHAR *ebrBuf, INT32 mbrCount) { INT32 ret; UINT32 extendedOffset = 0; CHAR ebrPartitionType; INT32 ebrCount = 0; do { (VOID)memset_s(ebrBuf, info->sector_size, 0, info->sector_size); if (((UINT64)(extendedAddress) + extendedOffset) >= info->sector_count) { PRINT_ERR("extended partition is out of disk range: extendedAddress = %u, extendedOffset = %u\n", extendedAddress, extendedOffset); break; } struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; ret = bops->read(blkDrv, (UINT8 *)ebrBuf, extendedAddress + extendedOffset, 1); if (ret != 1) { /* read failed */ PRINT_ERR("driver read return error: %d, extendedAddress = %u, extendedOffset = %u\n", ret, extendedAddress, extendedOffset); return -EIO; } ebrPartitionType = ebrBuf[PAR_OFFSET + PAR_TYPE_OFFSET]; if (ebrPartitionType && ((mbrCount + ebrCount) < MAX_DIVIDE_PART_PER_DISK)) { info->part[MAX_PRIMARY_PART_PER_DISK + ebrCount].type = ebrPartitionType; info->part[MAX_PRIMARY_PART_PER_DISK + ebrCount].sector_start = extendedAddress + extendedOffset + LD_DWORD_DISK(&ebrBuf[PAR_OFFSET + PAR_START_OFFSET]); info->part[MAX_PRIMARY_PART_PER_DISK + ebrCount].sector_count = LD_DWORD_DISK(&ebrBuf[PAR_OFFSET + PAR_COUNT_OFFSET]); ebrCount++; } extendedOffset = LD_DWORD_DISK(&ebrBuf[PAR_OFFSET + PAR_START_OFFSET + PAR_TABLE_SIZE]); } while ((ebrBuf[PAR_OFFSET + PAR_TYPE_OFFSET + PAR_TABLE_SIZE] != 0) && ((mbrCount + ebrCount) < MAX_DIVIDE_PART_PER_DISK)); return ebrCount; } static INT32 DiskPartitionRecognition(struct Vnode *blkDrv, struct disk_divide_info *info) { INT32 ret; INT32 extendedFlag; INT32 extendedPos = 0; INT32 mbrCount = 0; UINT32 extendedAddress; CHAR *mbrBuf = NULL; CHAR *ebrBuf = NULL; if (blkDrv == NULL) { return -EINVAL; } struct block_operations *bops = (struct block_operations *)((struct drv_data *)blkDrv->data)->ops; if ((bops == NULL) || (bops->read == NULL)) { return -EINVAL; } ret = DiskPartitionMemZalloc(MEM_ADDR_ALIGN_BYTE, info->sector_size, &mbrBuf, &ebrBuf); if (ret != ENOERR) { return ret; } ret = OsMBRInfoGet(blkDrv, mbrBuf); if (ret < 0) { goto OUT_WITH_MEM; } /* The partition type is GPT */ if (mbrBuf[PARTION_MODE_BTYE] == (CHAR)PARTION_MODE_GPT) { ret = DiskGPTPartitionRecognition(blkDrv, info); goto OUT_WITH_MEM; } ret = OsEBRInfoGet(blkDrv, info, ebrBuf, mbrBuf); if (ret < 0) { ret = 0; /* no mbr */ goto OUT_WITH_MEM; } extendedFlag = OsPrimaryPartitionRecognition(mbrBuf, info, &extendedPos, &mbrCount); if (extendedFlag) { extendedAddress = LD_DWORD_DISK(&mbrBuf[PAR_OFFSET + PAR_START_OFFSET + (extendedPos * PAR_TABLE_SIZE)]); ret = OsLogicalPartitionRecognition(blkDrv, info, extendedAddress, ebrBuf, mbrCount); if (ret <= 0) { goto OUT_WITH_MEM; } } ret += mbrCount; OUT_WITH_MEM: free(ebrBuf); free(mbrBuf); return ret; } INT32 DiskPartitionRegister(los_disk *disk) { INT32 count; UINT32 i, partSize; los_part *part = NULL; struct disk_divide_info parInfo; /* Fill disk_divide_info structure to set partition's infomation. */ (VOID)memset_s(parInfo.part, sizeof(parInfo.part), 0, sizeof(parInfo.part)); partSize = sizeof(parInfo.part) / sizeof(parInfo.part[0]); parInfo.sector_size = disk->sector_size; parInfo.sector_count = disk->sector_count; count = DiskPartitionRecognition(disk->dev, &parInfo); if (count == VFS_ERROR) { part = get_part(DiskAddPart(disk, 0, disk->sector_count, FALSE)); if (part == NULL) { return VFS_ERROR; } part->part_no_mbr = 0; PRINTK("Disk %s doesn't contain a valid partition table.\n", disk->disk_name); return ENOERR; } else if (count < 0) { return VFS_ERROR; } parInfo.part_count = count; if (count == 0) { part = get_part(DiskAddPart(disk, 0, disk->sector_count, TRUE)); if (part == NULL) { return VFS_ERROR; } part->part_no_mbr = 0; PRINTK("No MBR detected.\n"); return ENOERR; } for (i = 0; i < partSize; i++) { /* Read the disk_divide_info structure to get partition's infomation. */ if ((parInfo.part[i].type != 0) && (parInfo.part[i].type != EXTENDED_PAR) && (parInfo.part[i].type != EXTENDED_8G)) { part = get_part(DiskAddPart(disk, parInfo.part[i].sector_start, parInfo.part[i].sector_count, TRUE)); if (part == NULL) { return VFS_ERROR; } part->part_no_mbr = i + 1; part->filesystem_type = parInfo.part[i].type; } } return ENOERR; } #ifndef LOSCFG_FS_FAT_CACHE static INT32 disk_read_directly(los_disk *disk, VOID *buf, UINT64 sector, UINT32 count) { INT32 result = VFS_ERROR; struct block_operations *bops = (struct block_operations *)((struct drv_data *)disk->dev->data)->ops; if ((bops == NULL) || (bops->read == NULL)) { return VFS_ERROR; } if (LOS_IsUserAddressRange((VADDR_T)buf, count * disk->sector_size)) { UINT32 cnt = 0; UINT8 *buffer = disk->buff; for (; count != 0; count -= cnt) { cnt = (count > DISK_DIRECT_BUFFER_SIZE) ? DISK_DIRECT_BUFFER_SIZE : count; result = bops->read(disk->dev, buffer, sector, cnt); if (result == (INT32)cnt) { result = ENOERR; } else { break; } if (LOS_CopyFromKernel(buf, disk->sector_size * cnt, buffer, disk->sector_size * cnt)) { result = VFS_ERROR; break; } buf = (UINT8 *)buf + disk->sector_size * cnt; sector += cnt; } } else { result = bops->read(disk->dev, buf, sector, count); if (result == count) { result = ENOERR; } } return result; } static INT32 disk_write_directly(los_disk *disk, const VOID *buf, UINT64 sector, UINT32 count) { struct block_operations *bops = (struct block_operations *)((struct drv_data *)disk->dev->data)->ops; INT32 result = VFS_ERROR; if ((bops == NULL) || (bops->read == NULL)) { return VFS_ERROR; } if (LOS_IsUserAddressRange((VADDR_T)buf, count * disk->sector_size)) { UINT32 cnt = 0; UINT8 *buffer = disk->buff; for (; count != 0; count -= cnt) { cnt = (count > DISK_DIRECT_BUFFER_SIZE) ? DISK_DIRECT_BUFFER_SIZE : count; if (LOS_CopyToKernel(buffer, disk->sector_size * cnt, buf, disk->sector_size * cnt)) { result = VFS_ERROR; break; } result = bops->write(disk->dev, buffer, sector, cnt); if (result == (INT32)cnt) { result = ENOERR; } else { break; } buf = (UINT8 *)buf + disk->sector_size * cnt; sector += cnt; } } else { result = bops->write(disk->dev, buf, sector, count); if (result == count) { result = ENOERR; } } return result; } #endif INT32 los_disk_read(INT32 drvID, VOID *buf, UINT64 sector, UINT32 count, BOOL useRead) { #ifdef LOSCFG_FS_FAT_CACHE UINT32 len; #endif INT32 result = VFS_ERROR; los_disk *disk = get_disk(drvID); if ((buf == NULL) || (count == 0)) { /* buff equal to NULL or count equal to 0 */ return result; } if (disk == NULL) { return result; } DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { goto ERROR_HANDLE; } if ((count > disk->sector_count) || ((disk->sector_count - count) < sector)) { goto ERROR_HANDLE; } #ifdef LOSCFG_FS_FAT_CACHE if (disk->bcache != NULL) { if (((UINT64)(disk->bcache->sectorSize) * count) > UINT_MAX) { goto ERROR_HANDLE; } len = disk->bcache->sectorSize * count; /* useRead should be FALSE when reading large contiguous data */ result = BlockCacheRead(disk->bcache, (UINT8 *)buf, &len, sector, useRead); if (result != ENOERR) { PRINT_ERR("los_disk_read read err = %d, sector = %llu, len = %u\n", result, sector, len); } } else { result = VFS_ERROR; } #else if (disk->dev == NULL) { goto ERROR_HANDLE; } result = disk_read_directly(disk, buf, sector, count); #endif if (result != ENOERR) { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 los_disk_write(INT32 drvID, const VOID *buf, UINT64 sector, UINT32 count) { #ifdef LOSCFG_FS_FAT_CACHE UINT32 len; #endif INT32 result = VFS_ERROR; los_disk *disk = get_disk(drvID); if (disk == NULL || disk->dev == NULL || disk->dev->data == NULL) { return result; } if ((buf == NULL) || (count == 0)) { /* buff equal to NULL or count equal to 0 */ return result; } DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { goto ERROR_HANDLE; } if ((count > disk->sector_count) || ((disk->sector_count - count) < sector)) { goto ERROR_HANDLE; } #ifdef LOSCFG_FS_FAT_CACHE if (disk->bcache != NULL) { if (((UINT64)(disk->bcache->sectorSize) * count) > UINT_MAX) { goto ERROR_HANDLE; } len = disk->bcache->sectorSize * count; result = BlockCacheWrite(disk->bcache, (const UINT8 *)buf, &len, sector); if (result != ENOERR) { PRINT_ERR("los_disk_write write err = %d, sector = %llu, len = %u\n", result, sector, len); } } else { result = VFS_ERROR; } #else if (disk->dev == NULL) { goto ERROR_HANDLE; } result = disk_write_directly(disk, buf, sector, count); #endif if (result != ENOERR) { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 los_disk_ioctl(INT32 drvID, INT32 cmd, VOID *buf) { struct geometry info; los_disk *disk = get_disk(drvID); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); if ((disk->dev == NULL) || (disk->disk_status != STAT_INUSED)) { goto ERROR_HANDLE; } if (cmd == DISK_CTRL_SYNC) { DISK_UNLOCK(&disk->disk_mutex); return ENOERR; } if (buf == NULL) { goto ERROR_HANDLE; } (VOID)memset_s(&info, sizeof(info), 0, sizeof(info)); struct block_operations *bops = (struct block_operations *)((struct drv_data *)disk->dev->data)->ops; if ((bops == NULL) || (bops->geometry == NULL) || (bops->geometry(disk->dev, &info) != 0)) { goto ERROR_HANDLE; } if (cmd == DISK_GET_SECTOR_COUNT) { *(UINT64 *)buf = info.geo_nsectors; if (info.geo_nsectors == 0) { goto ERROR_HANDLE; } } else if (cmd == DISK_GET_SECTOR_SIZE) { *(size_t *)buf = info.geo_sectorsize; } else if (cmd == DISK_GET_BLOCK_SIZE) { /* Get erase block size in unit of sectors (UINT32) */ /* Block Num SDHC == 512, SD can be set to 512 or other */ *(size_t *)buf = DISK_MAX_SECTOR_SIZE / info.geo_sectorsize; } else { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 los_part_read(INT32 pt, VOID *buf, UINT64 sector, UINT32 count, BOOL useRead) { const los_part *part = get_part(pt); los_disk *disk = NULL; INT32 ret; if (part == NULL) { return VFS_ERROR; } disk = get_disk((INT32)part->disk_id); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); if ((part->dev == NULL) || (disk->disk_status != STAT_INUSED)) { goto ERROR_HANDLE; } if (count > part->sector_count) { PRINT_ERR("los_part_read failed, invaild count, count = %u\n", count); goto ERROR_HANDLE; } /* Read from absolute sector. */ if (part->type == EMMC) { if ((disk->sector_count - part->sector_start) > sector) { sector += part->sector_start; } else { PRINT_ERR("los_part_read failed, invaild sector, sector = %llu\n", sector); goto ERROR_HANDLE; } } if ((sector >= GetFirstPartStart(part)) && (((sector + count) > (part->sector_start + part->sector_count)) || (sector < part->sector_start))) { PRINT_ERR("los_part_read error, sector = %llu, count = %u, part->sector_start = %llu, " "part->sector_count = %llu\n", sector, count, part->sector_start, part->sector_count); goto ERROR_HANDLE; } /* useRead should be FALSE when reading large contiguous data */ ret = los_disk_read((INT32)part->disk_id, buf, sector, count, useRead); if (ret < 0) { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 los_part_write(INT32 pt, const VOID *buf, UINT64 sector, UINT32 count) { const los_part *part = get_part(pt); los_disk *disk = NULL; INT32 ret; if (part == NULL) { return VFS_ERROR; } disk = get_disk((INT32)part->disk_id); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); if ((part->dev == NULL) || (disk->disk_status != STAT_INUSED)) { goto ERROR_HANDLE; } if (count > part->sector_count) { PRINT_ERR("los_part_write failed, invaild count, count = %u\n", count); goto ERROR_HANDLE; } /* Write to absolute sector. */ if (part->type == EMMC) { if ((disk->sector_count - part->sector_start) > sector) { sector += part->sector_start; } else { PRINT_ERR("los_part_write failed, invaild sector, sector = %llu\n", sector); goto ERROR_HANDLE; } } if ((sector >= GetFirstPartStart(part)) && (((sector + count) > (part->sector_start + part->sector_count)) || (sector < part->sector_start))) { PRINT_ERR("los_part_write, sector = %llu, count = %u, part->sector_start = %llu, " "part->sector_count = %llu\n", sector, count, part->sector_start, part->sector_count); goto ERROR_HANDLE; } ret = los_disk_write((INT32)part->disk_id, buf, sector, count); if (ret < 0) { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } #define GET_ERASE_BLOCK_SIZE 0x2 INT32 los_part_ioctl(INT32 pt, INT32 cmd, VOID *buf) { struct geometry info; los_part *part = get_part(pt); los_disk *disk = NULL; if (part == NULL) { return VFS_ERROR; } disk = get_disk((INT32)part->disk_id); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); if ((part->dev == NULL) || (disk->disk_status != STAT_INUSED)) { goto ERROR_HANDLE; } if (cmd == DISK_CTRL_SYNC) { DISK_UNLOCK(&disk->disk_mutex); return ENOERR; } if (buf == NULL) { goto ERROR_HANDLE; } (VOID)memset_s(&info, sizeof(info), 0, sizeof(info)); struct block_operations *bops = (struct block_operations *)((struct drv_data *)part->dev->data)->ops; if ((bops == NULL) || (bops->geometry == NULL) || (bops->geometry(part->dev, &info) != 0)) { goto ERROR_HANDLE; } if (cmd == DISK_GET_SECTOR_COUNT) { *(UINT64 *)buf = part->sector_count; if (*(UINT64 *)buf == 0) { goto ERROR_HANDLE; } } else if (cmd == DISK_GET_SECTOR_SIZE) { *(size_t *)buf = info.geo_sectorsize; } else if (cmd == DISK_GET_BLOCK_SIZE) { /* Get erase block size in unit of sectors (UINT32) */ if ((bops->ioctl == NULL) || (bops->ioctl(part->dev, GET_ERASE_BLOCK_SIZE, (UINTPTR)buf) != 0)) { goto ERROR_HANDLE; } } else { goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 los_disk_cache_clear(INT32 drvID) { INT32 result = ENOERR; #ifdef LOSCFG_FS_FAT_CACHE los_part *part = get_part(drvID); los_disk *disk = NULL; if (part == NULL) { return VFS_ERROR; } result = OsSdSync(part->disk_id); if (result != ENOERR) { PRINTK("[ERROR]disk_cache_clear SD sync failed!\n"); return result; } disk = get_disk(part->disk_id); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); result = BcacheClearCache(disk->bcache); DISK_UNLOCK(&disk->disk_mutex); #endif return result; } #ifdef LOSCFG_FS_FAT_CACHE static VOID DiskCacheThreadInit(UINT32 diskID, OsBcache *bc) { bc->prereadFun = NULL; if (GetDiskUsbStatus(diskID) == FALSE) { if (BcacheAsyncPrereadInit(bc) == LOS_OK) { bc->prereadFun = ResumeAsyncPreread; } #ifdef LOSCFG_FS_FAT_CACHE_SYNC_THREAD BcacheSyncThreadInit(bc, diskID); #endif } if (OsReHookFuncAddDiskRef != NULL) { (VOID)OsReHookFuncAddDiskRef((StorageHookFunction)OsSdSync, (VOID *)0); (VOID)OsReHookFuncAddDiskRef((StorageHookFunction)OsSdSync, (VOID *)1); } } static OsBcache *DiskCacheInit(UINT32 diskID, const struct geometry *diskInfo, struct Vnode *blkDriver) { #define SECTOR_SIZE 512 OsBcache *bc = NULL; UINT32 sectorPerBlock = diskInfo->geo_sectorsize / SECTOR_SIZE; if (sectorPerBlock != 0) { sectorPerBlock = g_uwFatSectorsPerBlock / sectorPerBlock; if (sectorPerBlock != 0) { bc = BlockCacheInit(blkDriver, diskInfo->geo_sectorsize, sectorPerBlock, g_uwFatBlockNums, diskInfo->geo_nsectors / sectorPerBlock); } } if (bc == NULL) { PRINT_ERR("disk_init : disk have not init bcache cache!\n"); return NULL; } DiskCacheThreadInit(diskID, bc); return bc; } static VOID DiskCacheDeinit(los_disk *disk) { UINT32 diskID = disk->disk_id; if (GetDiskUsbStatus(diskID) == FALSE) { if (BcacheAsyncPrereadDeinit(disk->bcache) != LOS_OK) { PRINT_ERR("Blib async preread deinit failed in %s, %d\n", __FUNCTION__, __LINE__); } #ifdef LOSCFG_FS_FAT_CACHE_SYNC_THREAD BcacheSyncThreadDeinit(disk->bcache); #endif } BlockCacheDeinit(disk->bcache); disk->bcache = NULL; if (OsReHookFuncDelDiskRef != NULL) { (VOID)OsReHookFuncDelDiskRef((StorageHookFunction)OsSdSync); } } #endif static VOID DiskStructInit(const CHAR *diskName, INT32 diskID, const struct geometry *diskInfo, struct Vnode *blkDriver, los_disk *disk) { size_t nameLen; disk->disk_id = diskID; disk->dev = blkDriver; disk->sector_start = 0; disk->sector_size = diskInfo->geo_sectorsize; disk->sector_count = diskInfo->geo_nsectors; nameLen = strlen(diskName); /* caller los_disk_init has chek name */ if (disk->disk_name != NULL) { LOS_MemFree(m_aucSysMem0, disk->disk_name); disk->disk_name = NULL; } disk->disk_name = LOS_MemAlloc(m_aucSysMem0, (nameLen + 1)); if (disk->disk_name == NULL) { PRINT_ERR("DiskStructInit alloc memory failed.\n"); return; } if (strncpy_s(disk->disk_name, (nameLen + 1), diskName, nameLen) != EOK) { PRINT_ERR("DiskStructInit strncpy_s failed.\n"); LOS_MemFree(m_aucSysMem0, disk->disk_name); disk->disk_name = NULL; return; } disk->disk_name[nameLen] = '\0'; LOS_ListInit(&disk->head); } static INT32 DiskDivideAndPartitionRegister(struct disk_divide_info *info, los_disk *disk) { INT32 ret; if (info != NULL) { ret = DiskDivide(disk, info); if (ret != ENOERR) { PRINT_ERR("DiskDivide failed, ret = %d\n", ret); return ret; } } else { ret = DiskPartitionRegister(disk); if (ret != ENOERR) { PRINT_ERR("DiskPartitionRegister failed, ret = %d\n", ret); return ret; } } return ENOERR; } static INT32 DiskDeinit(los_disk *disk) { los_part *part = NULL; char *diskName = NULL; CHAR devName[DEV_NAME_BUFF_SIZE]; INT32 ret; if (LOS_ListEmpty(&disk->head) == FALSE) { part = LOS_DL_LIST_ENTRY(disk->head.pstNext, los_part, list); while (&part->list != &disk->head) { diskName = (disk->disk_name == NULL) ? "null" : disk->disk_name; ret = snprintf_s(devName, sizeof(devName), sizeof(devName) - 1, "%s%c%d", diskName, 'p', disk->part_count - 1); if (ret < 0) { return -ENAMETOOLONG; } DiskPartDelFromDisk(disk, part); (VOID)unregister_blockdriver(devName); DiskPartRelease(part); part = LOS_DL_LIST_ENTRY(disk->head.pstNext, los_part, list); } } DISK_LOCK(&disk->disk_mutex); #ifdef LOSCFG_FS_FAT_CACHE DiskCacheDeinit(disk); #else if (disk->buff != NULL) { free(disk->buff); } #endif disk->dev = NULL; DISK_UNLOCK(&disk->disk_mutex); (VOID)unregister_blockdriver(disk->disk_name); if (disk->disk_name != NULL) { LOS_MemFree(m_aucSysMem0, disk->disk_name); disk->disk_name = NULL; } ret = pthread_mutex_destroy(&disk->disk_mutex); if (ret != 0) { PRINT_ERR("%s %d, mutex destroy failed, ret = %d\n", __FUNCTION__, __LINE__, ret); return -EFAULT; } disk->disk_status = STAT_UNUSED; return ENOERR; } static UINT32 OsDiskInitSub(const CHAR *diskName, INT32 diskID, los_disk *disk, struct geometry *diskInfo, struct Vnode *blkDriver) { pthread_mutexattr_t attr; #ifdef LOSCFG_FS_FAT_CACHE OsBcache *bc = DiskCacheInit((UINT32)diskID, diskInfo, blkDriver); if (bc == NULL) { return VFS_ERROR; } disk->bcache = bc; #endif (VOID)pthread_mutexattr_init(&attr); attr.type = PTHREAD_MUTEX_RECURSIVE; (VOID)pthread_mutex_init(&disk->disk_mutex, &attr); DiskStructInit(diskName, diskID, diskInfo, blkDriver, disk); #ifndef LOSCFG_FS_FAT_CACHE disk->buff = malloc(diskInfo->geo_sectorsize * DISK_DIRECT_BUFFER_SIZE); if (disk->buff == NULL) { PRINT_ERR("OsDiskInitSub: direct buffer of disk init failed\n"); return VFS_ERROR; } #endif return ENOERR; } INT32 los_disk_init(const CHAR *diskName, const struct block_operations *bops, VOID *priv, INT32 diskID, VOID *info) { struct geometry diskInfo; struct Vnode *blkDriver = NULL; los_disk *disk = get_disk(diskID); INT32 ret; if ((diskName == NULL) || (disk == NULL) || (disk->disk_status != STAT_UNREADY) || (strlen(diskName) > DISK_NAME)) { return VFS_ERROR; } if (register_blockdriver(diskName, bops, RWE_RW_RW, priv) != 0) { PRINT_ERR("disk_init : register %s fail!\n", diskName); return VFS_ERROR; } VnodeHold(); ret = VnodeLookup(diskName, &blkDriver, 0); if (ret < 0) { VnodeDrop(); ret = ENOENT; goto DISK_FIND_ERROR; } struct block_operations *bops2 = (struct block_operations *)((struct drv_data *)blkDriver->data)->ops; if ((bops2 == NULL) || (bops2->geometry == NULL) || (bops2->geometry(blkDriver, &diskInfo) != 0)) { goto DISK_BLKDRIVER_ERROR; } if (diskInfo.geo_sectorsize < DISK_MAX_SECTOR_SIZE) { goto DISK_BLKDRIVER_ERROR; } ret = OsDiskInitSub(diskName, diskID, disk, &diskInfo, blkDriver); if (ret != ENOERR) { (VOID)DiskDeinit(disk); VnodeDrop(); return VFS_ERROR; } VnodeDrop(); if (DiskDivideAndPartitionRegister(info, disk) != ENOERR) { (VOID)DiskDeinit(disk); return VFS_ERROR; } disk->disk_status = STAT_INUSED; if (info != NULL) { disk->type = EMMC; } else { disk->type = OTHERS; } return ENOERR; DISK_BLKDRIVER_ERROR: PRINT_ERR("disk_init : register %s ok but get disk info fail!\n", diskName); VnodeDrop(); DISK_FIND_ERROR: (VOID)unregister_blockdriver(diskName); return VFS_ERROR; } INT32 los_disk_deinit(INT32 diskID) { int ret; los_disk *disk = get_disk(diskID); if (disk == NULL) { return -EINVAL; } ret = ForceUmountDev(disk->dev); PRINTK("warning: %s lost, force umount ret = %d\n", disk->disk_name, ret); DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { DISK_UNLOCK(&disk->disk_mutex); return -EINVAL; } disk->disk_status = STAT_UNREADY; DISK_UNLOCK(&disk->disk_mutex); return DiskDeinit(disk); } INT32 los_disk_sync(INT32 drvID) { INT32 ret = ENOERR; los_disk *disk = get_disk(drvID); if (disk == NULL) { return EINVAL; } DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { DISK_UNLOCK(&disk->disk_mutex); return EINVAL; } #ifdef LOSCFG_FS_FAT_CACHE if (disk->bcache != NULL) { ret = BlockCacheSync(disk->bcache); } #endif DISK_UNLOCK(&disk->disk_mutex); return ret; } INT32 los_disk_set_bcache(INT32 drvID, UINT32 sectorPerBlock, UINT32 blockNum) { #ifdef LOSCFG_FS_FAT_CACHE INT32 ret; UINT32 intSave; OsBcache *bc = NULL; los_disk *disk = get_disk(drvID); if ((disk == NULL) || (sectorPerBlock == 0)) { return EINVAL; } /* * Because we use UINT32 flag[BCACHE_BLOCK_FLAGS] in bcache for sectors bitmap tag, so it must * be less than 32 * BCACHE_BLOCK_FLAGS. */ if (((sectorPerBlock % UNSIGNED_INTEGER_BITS) != 0) || ((sectorPerBlock >> UNINT_LOG2_SHIFT) > BCACHE_BLOCK_FLAGS)) { return EINVAL; } DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { goto ERROR_HANDLE; } if (disk->bcache != NULL) { ret = BlockCacheSync(disk->bcache); if (ret != ENOERR) { DISK_UNLOCK(&disk->disk_mutex); return ret; } } spin_lock_irqsave(&g_diskFatBlockSpinlock, intSave); DiskCacheDeinit(disk); g_uwFatBlockNums = blockNum; g_uwFatSectorsPerBlock = sectorPerBlock; bc = BlockCacheInit(disk->dev, disk->sector_size, sectorPerBlock, blockNum, disk->sector_count / sectorPerBlock); if ((bc == NULL) && (blockNum != 0)) { spin_unlock_irqrestore(&g_diskFatBlockSpinlock, intSave); DISK_UNLOCK(&disk->disk_mutex); return ENOMEM; } if (bc != NULL) { DiskCacheThreadInit((UINT32)drvID, bc); } disk->bcache = bc; spin_unlock_irqrestore(&g_diskFatBlockSpinlock, intSave); DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return EINVAL; #else return VFS_ERROR; #endif } static los_part *OsPartFind(los_disk *disk, const struct Vnode *blkDriver) { los_part *part = NULL; DISK_LOCK(&disk->disk_mutex); if ((disk->disk_status != STAT_INUSED) || (LOS_ListEmpty(&disk->head) == TRUE)) { goto EXIT; } part = LOS_DL_LIST_ENTRY(disk->head.pstNext, los_part, list); if (disk->dev == blkDriver) { goto EXIT; } while (&part->list != &disk->head) { if (part->dev == blkDriver) { goto EXIT; } part = LOS_DL_LIST_ENTRY(part->list.pstNext, los_part, list); } part = NULL; EXIT: DISK_UNLOCK(&disk->disk_mutex); return part; } los_part *los_part_find(struct Vnode *blkDriver) { INT32 i; los_disk *disk = NULL; los_part *part = NULL; if (blkDriver == NULL) { return NULL; } for (i = 0; i < SYS_MAX_DISK; i++) { disk = get_disk(i); if (disk == NULL) { continue; } part = OsPartFind(disk, blkDriver); if (part != NULL) { return part; } } return NULL; } INT32 los_part_access(const CHAR *dev, mode_t mode) { los_part *part = NULL; struct Vnode *node = NULL; VnodeHold(); if (VnodeLookup(dev, &node, 0) < 0) { VnodeDrop(); return VFS_ERROR; } part = los_part_find(node); VnodeDrop(); if (part == NULL) { return VFS_ERROR; } return ENOERR; } INT32 SetDiskPartName(los_part *part, const CHAR *src) { size_t len; los_disk *disk = NULL; if ((part == NULL) || (src == NULL)) { return VFS_ERROR; } len = strlen(src); if ((len == 0) || (len >= DISK_NAME)) { return VFS_ERROR; } disk = get_disk((INT32)part->disk_id); if (disk == NULL) { return VFS_ERROR; } DISK_LOCK(&disk->disk_mutex); if (disk->disk_status != STAT_INUSED) { goto ERROR_HANDLE; } part->part_name = (CHAR *)zalloc(len + 1); if (part->part_name == NULL) { PRINT_ERR("%s[%d] zalloc failure\n", __FUNCTION__, __LINE__); goto ERROR_HANDLE; } if (strcpy_s(part->part_name, len + 1, src) != EOK) { free(part->part_name); part->part_name = NULL; goto ERROR_HANDLE; } DISK_UNLOCK(&disk->disk_mutex); return ENOERR; ERROR_HANDLE: DISK_UNLOCK(&disk->disk_mutex); return VFS_ERROR; } INT32 add_mmc_partition(struct disk_divide_info *info, size_t sectorStart, size_t sectorCount) { UINT32 index, i; if (info == NULL) { return VFS_ERROR; } if ((info->part_count >= MAX_DIVIDE_PART_PER_DISK) || (sectorCount == 0)) { return VFS_ERROR; } if ((sectorCount > info->sector_count) || ((info->sector_count - sectorCount) < sectorStart)) { return VFS_ERROR; } index = info->part_count; for (i = 0; i < index; i++) { if (sectorStart < (info->part[i].sector_start + info->part[i].sector_count)) { return VFS_ERROR; } } info->part[index].sector_start = sectorStart; info->part[index].sector_count = sectorCount; info->part[index].type = EMMC; info->part_count++; return ENOERR; } VOID show_part(los_part *part) { if ((part == NULL) || (part->dev == NULL)) { PRINT_ERR("part is NULL\n"); return; } PRINTK("\npart info :\n"); PRINTK("disk id : %u\n", part->disk_id); PRINTK("part_id in system: %u\n", part->part_id); PRINTK("part no in disk : %u\n", part->part_no_disk); PRINTK("part no in mbr : %u\n", part->part_no_mbr); PRINTK("part filesystem : %02X\n", part->filesystem_type); PRINTK("part sec start : %llu\n", part->sector_start); PRINTK("part sec count : %llu\n", part->sector_count); } #ifdef LOSCFG_DRIVERS_MMC ssize_t StorageBlockMmcErase(uint32_t blockId, size_t secStart, size_t secNr); #endif INT32 EraseDiskByID(UINT32 diskID, size_t startSector, UINT32 sectors) { INT32 ret = VFS_ERROR; #ifdef LOSCFG_DRIVERS_MMC los_disk *disk = get_disk((INT32)diskID); if (disk != NULL) { ret = StorageBlockMmcErase(diskID, startSector, sectors); } #endif return ret; }