fat_dir.c
上传用户:yj_qqy
上传日期:2017-01-28
资源大小:2911k
文件大小:16k
源码类别:

uCOS

开发平台:

C/C++

  1. /*
  2. **********************************************************************
  3. *                          Micrium, Inc.
  4. *                      949 Crestview Circle
  5. *                     Weston,  FL 33327-1848
  6. *
  7. *                            uC/FS
  8. *
  9. *             (c) Copyright 2001 - 2003, Micrium, Inc.
  10. *                      All rights reserved.
  11. *
  12. ***********************************************************************
  13. ----------------------------------------------------------------------
  14. File        : fat_dir.c
  15. Purpose     : POSIX 1003.1 like directory support
  16. ----------------------------------------------------------------------
  17. Known problems or limitations with current version
  18. ----------------------------------------------------------------------
  19. None.
  20. ---------------------------END-OF-HEADER------------------------------
  21. */
  22. /*********************************************************************
  23. *
  24. *             #include Section
  25. *
  26. **********************************************************************
  27. */
  28. #include "fs_conf.h"
  29. #include "fs_port.h"
  30. #include "fs_dev.h"
  31. #include "fs_api.h"
  32. #include "fs_fsl.h"
  33. #include "fs_int.h"
  34. #include "fs_os.h"
  35. #include "fs_lbl.h"
  36. #include "fs_fat.h"
  37. #include "fs_clib.h"
  38. #if FS_POSIX_DIR_SUPPORT
  39. /*********************************************************************
  40. *
  41. *             _FS_fat_create_directory
  42. *
  43.   Description:
  44.   FS internal function. Create a directory in the directory specified
  45.   with DirStart. Do not call, if you have not checked before for 
  46.   existing directory with name pDirName.
  47.   Parameters:
  48.   Idx         - Index of device in the device information table 
  49.                 referred by FS__pDevInfo.
  50.   Unit        - Unit number, which is passed to the device driver.
  51.   pDirName    - Directory name. 
  52.   DirStart    - Start of directory, where to create pDirName.
  53.   DirSize     - Size of the directory starting at DirStart.
  54.   
  55.   Return value:
  56.   >=0         - Directory has been created.
  57.   <0          - An error has occured.
  58. */
  59. static int _FS_fat_create_directory(int Idx, FS_u32 Unit, const char *pDirName,
  60.                                     FS_u32 DirStart, FS_u32 DirSize) {
  61.   char *buffer;
  62.   FS__fat_dentry_type *s;
  63.   FS_u32 dirindex;
  64.   FS_u32 dsec;
  65.   FS_i32 cluster;
  66.   FS_u16 val_time;
  67.   FS_u16 val_date;
  68.   int err;
  69.   int len;
  70.   int j;
  71.   buffer = FS__fat_malloc(FS_FAT_SEC_SIZE);
  72.   if (!buffer) {
  73.     return -1;
  74.   }
  75.   len = FS__CLIB_strlen(pDirName);
  76.   if (len > 11) {
  77.     len = 11;
  78.   }
  79.   /* Read directory */
  80.   for (dirindex = 0; dirindex < DirSize; dirindex++) {
  81.     dsec = FS__fat_dir_realsec(Idx, Unit, DirStart, dirindex);
  82.     if (dsec == 0) {
  83.       /* Translation of relativ directory sector to an absolute sector failed */
  84.       FS__fat_free(buffer);
  85.       return -1;
  86.     }
  87.     err = FS__lb_read(FS__pDevInfo[Idx].devdriver, Unit, dsec, (void*)buffer); /* Read directory sector */
  88.     if (err < 0) {
  89.       /* Read error */
  90.       FS__fat_free(buffer);
  91.       return -1;
  92.     }
  93.     /* Scan the directory sector for a free or deleted entry */
  94.     s = (FS__fat_dentry_type*)buffer;
  95.     while (1) {
  96.       if (s >= (FS__fat_dentry_type*)(buffer + FS_FAT_SEC_SIZE)) {
  97.         break;  /* End of sector reached */
  98.       }
  99.       if (s->data[0] == 0x00) {
  100.         break;  /* Found a free entry */
  101.       }
  102.       if (s->data[0] == (unsigned char)0xe5) {
  103.         break;  /* Found a deleted entry */
  104.       }
  105.       s++;
  106.     }
  107.     if (s < (FS__fat_dentry_type*)(buffer + FS_FAT_SEC_SIZE)) {
  108.       /* Free entry found. Make entry and return 1st block of the file. */
  109.       FS__CLIB_strncpy((char*)s->data, pDirName, len);
  110.        s->data[11] = FS_FAT_ATTR_DIRECTORY;
  111.       cluster = FS__fat_FAT_alloc(Idx, Unit, -1);              /* Alloc block in FAT */
  112.       if (cluster >= 0) {
  113.         s->data[12]     = 0x00;                                /* Res */
  114.         s->data[13]     = 0x00;                                /* CrtTimeTenth (optional, not supported) */
  115.         s->data[14]     = 0x00;                                /* CrtTime (optional, not supported) */
  116.         s->data[15]     = 0x00;
  117.         s->data[16]     = 0x00;                                /* CrtDate (optional, not supported) */
  118.         s->data[17]     = 0x00;
  119.         s->data[18]     = 0x00;                                /* LstAccDate (optional, not supported) */
  120.         s->data[19]     = 0x00;
  121.         val_time        = FS_X_OS_GetTime();
  122.         s->data[22]     = (unsigned char)(val_time & 0xff);   /* WrtTime */
  123.         s->data[23]     = (unsigned char)(val_time / 256);
  124.         val_date        = FS_X_OS_GetDate();
  125.         s->data[24]     = (unsigned char)(val_date & 0xff);   /* WrtDate */
  126.         s->data[25]     = (unsigned char)(val_date / 256);
  127.         s->data[26]     = (unsigned char)(cluster & 0xff);    /* FstClusLo / FstClusHi */ 
  128.         s->data[27]     = (unsigned char)((cluster / 256) & 0xff);
  129.         s->data[20]     = (unsigned char)((cluster / 0x10000L) & 0xff);
  130.         s->data[21]     = (unsigned char)((cluster / 0x1000000L) & 0xff);
  131.         s->data[28]     = 0x00;                                /* FileSize */
  132.         s->data[29]     = 0x00;
  133.         s->data[30]     = 0x00;
  134.         s->data[31]     = 0x00;
  135.         err = FS__lb_write(FS__pDevInfo[Idx].devdriver, Unit, dsec, (void*)buffer); /* Write the modified directory sector */
  136.         if (err < 0) {
  137.           FS__fat_free(buffer);
  138.           return -1;
  139.         }
  140.         /* Clear new directory and make '.' and '..' entries */
  141.         /* Make "." entry */
  142.         FS__CLIB_memset(buffer, 0x00, (FS_size_t)FS_FAT_SEC_SIZE);
  143.         s = (FS__fat_dentry_type*)buffer;
  144.         FS__CLIB_strncpy((char*)s->data, ".          ", 11);
  145.         s->data[11]     = FS_FAT_ATTR_DIRECTORY;
  146.         s->data[22]     = (unsigned char)(val_time & 0xff);   /* WrtTime */
  147.         s->data[23]     = (unsigned char)(val_time / 256);
  148.         s->data[24]     = (unsigned char)(val_date & 0xff);   /* WrtDate */
  149.         s->data[25]     = (unsigned char)(val_date / 256);
  150.         s->data[26]     = (unsigned char)(cluster & 0xff);    /* FstClusLo / FstClusHi */ 
  151.         s->data[27]     = (unsigned char)((cluster / 256) & 0xff);
  152.         s->data[20]     = (unsigned char)((cluster / 0x10000L) & 0xff);
  153.         s->data[21]     = (unsigned char)((cluster / 0x1000000L) & 0xff);
  154.         /* Make entry ".." */
  155.         s++;
  156.         FS__CLIB_strncpy((char*)s->data, "..         ", 11);
  157.         s->data[11]     = FS_FAT_ATTR_DIRECTORY;
  158.         s->data[22]     = (unsigned char)(val_time & 0xff);   /* WrtTime */
  159.         s->data[23]     = (unsigned char)(val_time / 256);
  160.         s->data[24]     = (unsigned char)(val_date & 0xff);   /* WrtDate */
  161.         s->data[25]     = (unsigned char)(val_date / 256);
  162.         s->data[26]     = (unsigned char)(DirStart & 0xff);    /* FstClusLo / FstClusHi */ 
  163.         s->data[27]     = (unsigned char)((DirStart / 256) & 0xff);
  164.         s->data[20]     = (unsigned char)((DirStart / 0x10000L) & 0xff);
  165.         s->data[21]     = (unsigned char)((DirStart / 0x1000000L) & 0xff);
  166.         dsec = FS__fat_dir_realsec(Idx, Unit, cluster, 0); /* Find 1st absolute sector of the new directory */
  167.         if (dsec == 0) {
  168.           FS__fat_free(buffer);
  169.           return -1;
  170.         }
  171.         /* Write "." & ".." entries into the new directory */
  172.         err = FS__lb_write(FS__pDevInfo[Idx].devdriver, Unit, dsec, (void*)buffer);
  173.         if (err < 0) {
  174.           FS__fat_free(buffer);
  175.           return -1;
  176.         }
  177.         /* Clear rest of the directory cluster */
  178.         FS__CLIB_memset(buffer, 0x00, (FS_size_t)FS_FAT_SEC_SIZE);
  179.         for (j = 1; j < FS__FAT_aBPBUnit[Idx][Unit].SecPerClus; j++) {
  180.           dsec = FS__fat_dir_realsec(Idx, Unit, cluster, j);
  181.           err = FS__lb_write(FS__pDevInfo[Idx].devdriver, Unit, dsec, (void*)buffer);
  182.           if (err < 0) {
  183.             FS__fat_free(buffer);
  184.             return -1;
  185.           }
  186.         }
  187.         FS__fat_free(buffer);
  188.         return 1;
  189.       }
  190.       FS__fat_free(buffer);
  191.       return -1;
  192.     }
  193.   }
  194.   FS__fat_free(buffer);
  195.   return -2;  /* Directory is full */
  196. }
  197. /*********************************************************************
  198. *
  199. *             Global functions
  200. *
  201. **********************************************************************
  202. */
  203. /*********************************************************************
  204. *
  205. *             FS__fat_opendir
  206. *
  207.   Description:
  208.   FS internal function. Open an existing directory for reading.
  209.   Parameters:
  210.   pDirName    - Directory name. 
  211.   pDir        - Pointer to a FS_DIR data structure. 
  212.   
  213.   Return value:
  214.   ==0         - Unable to open the directory.
  215.   !=0         - Address of an FS_DIR data structure.
  216. */
  217. FS_DIR *FS__fat_opendir(const char *pDirName, FS_DIR *pDir) {
  218.   FS_size_t len;
  219.   FS_u32 unit;
  220.   FS_u32 dstart;
  221.   FS_u32 dsize;
  222.   FS_i32 i;
  223.   char realname[12];
  224.   char *filename;
  225.   if (!pDir) {
  226.     return 0;  /* No valid pointer to a FS_DIR structure */
  227.   }
  228.   /* Find path on the media and return file name part of the complete path */
  229.   dsize = FS__fat_findpath(pDir->dev_index, pDirName, &filename, &unit, &dstart); 
  230.   if (dsize == 0) {
  231.     return 0;  /* Directory not found */
  232.   }
  233.   FS__lb_ioctl(FS__pDevInfo[pDir->dev_index].devdriver, unit, FS_CMD_INC_BUSYCNT, 0, (void*)0); /* Turn on busy signal */
  234.   len = FS__CLIB_strlen(filename);
  235.   if (len != 0) {
  236.     /* There is a name in the complete path (it does not end with a '') */
  237.     FS__fat_make_realname(realname, filename);  /* Convert name to FAT real name */
  238.     i =  FS__fat_find_dir(pDir->dev_index, unit, realname, dstart, dsize);  /* Search name in the directory */
  239.     if (i == 0) {
  240.       /* Directory not found */
  241.       FS__lb_ioctl(FS__pDevInfo[pDir->dev_index].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn off busy signal */
  242.       return 0;
  243.     }
  244.   }
  245.   else {
  246.     /* 
  247.        There is no name in the complete path (it does end with a ''). In that
  248.        case, FS__fat_findpath returns already start of the directory.
  249.     */
  250.     i = dstart;  /* Use 'current' path */
  251.   }
  252.   if (i) {
  253.     dsize  =  FS__fat_dir_size(pDir->dev_index, unit, i);  /* Get size of the directory */
  254.   }
  255.   if (dsize == 0) {
  256.     /* Directory not found */
  257.     FS__lb_ioctl(FS__pDevInfo[pDir->dev_index].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn off busy signal */
  258.     return 0;
  259.   }
  260.   pDir->dirid_lo  = unit;
  261.   pDir->dirid_hi  = i;
  262.   pDir->dirid_ex  = dstart;
  263.   pDir->error     = 0;
  264.   pDir->size      = dsize;
  265.   pDir->dirpos    = 0;
  266.   pDir->inuse     = 1;
  267.   return pDir;
  268. }
  269. /*********************************************************************
  270. *
  271. *             FS__fat_closedir
  272. *
  273.   Description:
  274.   FS internal function. Close a directory referred by pDir.
  275.   Parameters:
  276.   pDir        - Pointer to a FS_DIR data structure. 
  277.   
  278.   Return value:
  279.   ==0         - Directory has been closed.
  280.   ==-1        - Unable to close directory.
  281. */
  282. int FS__fat_closedir(FS_DIR *pDir) {
  283.   if (!pDir) {
  284.     return -1;  /* No valid pointer to a FS_DIR structure */
  285.   }
  286.   FS__lb_ioctl(FS__pDevInfo[pDir->dev_index].devdriver, pDir->dirid_lo, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn off busy signal */
  287.   pDir->inuse = 0;
  288.   return 0;
  289. }
  290. /*********************************************************************
  291. *
  292. *             FS__fat_readdir
  293. *
  294.   Description:
  295.   FS internal function. Read next directory entry in directory 
  296.   specified by pDir.
  297.   Parameters:
  298.   pDir        - Pointer to a FS_DIR data structure. 
  299.   
  300.   Return value:
  301.   ==0         - No more directory entries or error.
  302.   !=0         - Pointer to a directory entry.
  303. */
  304. struct FS_DIRENT *FS__fat_readdir(FS_DIR *pDir) {
  305.   FS__fat_dentry_type *s;
  306.   FS_u32 dirindex;
  307.   FS_u32 dsec;
  308.   FS_u16 bytespersec;
  309.   char *buffer;
  310.   int err;
  311.   if (!pDir) {
  312.     return 0;  /* No valid pointer to a FS_DIR structure */
  313.   }
  314.   buffer = FS__fat_malloc(FS_FAT_SEC_SIZE);
  315.   if (!buffer) {
  316.     return 0;
  317.   }
  318.   bytespersec = FS__FAT_aBPBUnit[pDir->dev_index][pDir->dirid_lo].BytesPerSec;
  319.   dirindex = pDir->dirpos / bytespersec;
  320.   while (dirindex < (FS_u32)pDir->size) {
  321.     dsec = FS__fat_dir_realsec(pDir->dev_index, pDir->dirid_lo, pDir->dirid_hi, dirindex);
  322.     if (dsec == 0) {
  323.       /* Cannot convert logical sector */
  324.       FS__fat_free(buffer);
  325.       return 0;
  326.     }
  327.     /* Read directory sector */
  328.     err = FS__lb_read(FS__pDevInfo[pDir->dev_index].devdriver, pDir->dirid_lo, dsec, (void*)buffer);
  329.     if (err < 0) {
  330.       FS__fat_free(buffer);
  331.       return 0;
  332.     }
  333.     /* Scan for valid directory entry */
  334.     s = (FS__fat_dentry_type*)&buffer[pDir->dirpos % bytespersec];
  335.     while (1) {
  336.       if (s >= (FS__fat_dentry_type*)(buffer + FS_FAT_SEC_SIZE)) {
  337.         break;  /* End of sector reached */
  338.       }
  339.       if (s->data[11] != 0x00) { /* not an empty entry */
  340.         if (s->data[0] != (unsigned char)0xe5) { /* not a deleted file */
  341.           if (s->data[11] != (FS_FAT_ATTR_READ_ONLY | FS_FAT_ATTR_HIDDEN | FS_FAT_ATTR_SYSTEM | FS_FAT_VOLUME_ID)) {
  342.             break;  /* Also not a long entry, so it is a valid entry */
  343.           }
  344.         }
  345.       }
  346.       s++;
  347.       pDir->dirpos += 32;
  348.     }
  349.     if (s < (FS__fat_dentry_type*)(buffer + FS_FAT_SEC_SIZE)) {
  350.       /* Valid entry found, copy it.*/
  351.       pDir->dirpos += 32;
  352.       FS__CLIB_memcpy(pDir->dirent.d_name, s->data, 8);
  353.       pDir->dirent.d_name[8] = '.';
  354.       FS__CLIB_memcpy(&pDir->dirent.d_name[9], &s->data[8], 3);
  355.       pDir->dirent.d_name[12] = 0;
  356.       pDir->dirent.FAT_DirAttr = s->data[11];
  357.       FS__fat_free(buffer);
  358.       return &pDir->dirent;
  359.     }
  360.     dirindex++;
  361.   }
  362.   FS__fat_free(buffer);
  363.   return 0;
  364. }
  365. /*********************************************************************
  366. *
  367. *             FS__fat_MkRmDir
  368. *
  369.   Description:
  370.   FS internal function. Create or remove a directory. If you call this 
  371.   function to remove a directory (MkDir==0), you must make sure, that 
  372.   it is already empty.
  373.   Parameters:
  374.   pDirName    - Directory name. 
  375.   Idx         - Index of device in the device information table 
  376.                 referred by FS__pDevInfo.
  377.   MkDir       - ==0 => Remove directory.
  378.                 !=0 => Create directory.
  379.   
  380.   Return value:
  381.   ==0         - Directory has been created.
  382.   ==-1        - An error has occured.
  383. */
  384. int  FS__fat_MkRmDir(const char *pDirName, int Idx, char MkDir) {
  385.   FS_size_t len;
  386.   FS_u32 dstart;
  387.   FS_u32 dsize;
  388.   FS_u32 unit;
  389.   FS_i32 i;
  390.   int lexp_a;
  391.   int lexp_b;
  392.   char realname[12];
  393.   char *filename;
  394.   if (Idx < 0) {
  395.     return -1; /* Not a valid index */
  396.   }
  397.   dsize = FS__fat_findpath(Idx, pDirName, &filename, &unit, &dstart);
  398.   if (dsize == 0) {
  399.     return -1;  /* Path not found */
  400.   }
  401.   FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_INC_BUSYCNT, 0, (void*)0); /* Turn on busy signal */
  402.   len = FS__CLIB_strlen(filename);
  403.   if (len != 0) {
  404.     FS__fat_make_realname(realname, filename);  /* Convert name to FAT real name */
  405.     i =  FS__fat_find_dir(Idx, unit, realname, dstart, dsize);
  406.     lexp_a = (i!=0) && (MkDir);  /* We want to create a direcory , but it does already exist */
  407.     lexp_b = (i==0) && (!MkDir); /* We want to remove a direcory , but it does not exist */
  408.     lexp_a = lexp_a || lexp_b;
  409.     if (lexp_a) {
  410.       /* We want to create, but dir does already exist or we want to remove, but dir is not there */
  411.       /* turn off busy signal */
  412.       FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);
  413.       return -1;
  414.     }
  415.   }
  416.   else {
  417.     FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn off busy signal */
  418.     return -1;
  419.   }
  420.   /* 
  421.       When you get here, variables have following values:
  422.        dstart="current"  
  423.        dsize="size of current"  
  424.        realname="real dir name to create" 
  425.   */
  426.   if (MkDir) {
  427.     i = _FS_fat_create_directory(Idx, unit,realname, dstart, dsize);  /* Create the directory */
  428.   }
  429.   else {
  430.     i = FS__fat_DeleteFileOrDir(Idx, unit, realname, dstart, dsize, 0);  /* Remove the directory */
  431.   }
  432.   if (i >= 0) {
  433.     /* If the operation has been successfull, flush the cache.*/
  434.     i = FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_FLUSH_CACHE, 2, (void*)0);
  435.     FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn of busy signal */
  436.     if (i < 0) {
  437.       return -1;
  438.     }
  439.     return 0;
  440.   }
  441.   FS__lb_ioctl(FS__pDevInfo[Idx].devdriver, unit, FS_CMD_DEC_BUSYCNT, 0, (void*)0);  /* Turn of busy signal */
  442.   return -1;
  443. }
  444. #endif /* FS_POSIX_DIR_SUPPORT */