dir.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:6k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * dir.c
  3.  *
  4.  * PURPOSE
  5.  *  Directory handling routines for the OSTA-UDF(tm) filesystem.
  6.  *
  7.  * CONTACTS
  8.  * E-mail regarding any portion of the Linux UDF file system should be
  9.  * directed to the development team mailing list (run by majordomo):
  10.  * linux_udf@hpesjro.fc.hp.com
  11.  *
  12.  * COPYRIGHT
  13.  * This file is distributed under the terms of the GNU General Public
  14.  * License (GPL). Copies of the GPL can be obtained from:
  15.  * ftp://prep.ai.mit.edu/pub/gnu/GPL
  16.  * Each contributing author retains all rights to their own work.
  17.  *
  18.  *  (C) 1998-2001 Ben Fennema
  19.  *
  20.  * HISTORY
  21.  *
  22.  *  10/05/98 dgb  Split directory operations into it's own file
  23.  *                Implemented directory reads via do_udf_readdir
  24.  *  10/06/98      Made directory operations work!
  25.  *  11/17/98      Rewrote directory to support ICBTAG_FLAG_AD_LONG
  26.  *  11/25/98 blf  Rewrote directory handling (readdir+lookup) to support reading
  27.  *                across blocks.
  28.  *  12/12/98      Split out the lookup code to namei.c. bulk of directory
  29.  *                code now in directory.c:udf_fileident_read.
  30.  */
  31. #include "udfdecl.h"
  32. #include <linux/string.h>
  33. #include <linux/errno.h>
  34. #include <linux/mm.h>
  35. #include <linux/slab.h>
  36. #include "udf_i.h"
  37. #include "udf_sb.h"
  38. /* Prototypes for file operations */
  39. static int udf_readdir(struct file *, void *, filldir_t);
  40. static int do_udf_readdir(struct inode *, struct file *, filldir_t, void *);
  41. /* readdir and lookup functions */
  42. struct file_operations udf_dir_operations = {
  43. read: generic_read_dir,
  44. readdir: udf_readdir,
  45. ioctl: udf_ioctl,
  46. fsync: udf_fsync_file,
  47. };
  48. /*
  49.  * udf_readdir
  50.  *
  51.  * PURPOSE
  52.  * Read a directory entry.
  53.  *
  54.  * DESCRIPTION
  55.  * Optional - sys_getdents() will return -ENOTDIR if this routine is not
  56.  * available.
  57.  *
  58.  * Refer to sys_getdents() in fs/readdir.c
  59.  * sys_getdents() -> .
  60.  *
  61.  * PRE-CONDITIONS
  62.  * filp Pointer to directory file.
  63.  * buf Pointer to directory entry buffer.
  64.  * filldir Pointer to filldir function.
  65.  *
  66.  * POST-CONDITIONS
  67.  * <return> >=0 on success.
  68.  *
  69.  * HISTORY
  70.  * July 1, 1997 - Andrew E. Mileski
  71.  * Written, tested, and released.
  72.  */
  73. int udf_readdir(struct file *filp, void *dirent, filldir_t filldir)
  74. {
  75. struct inode *dir = filp->f_dentry->d_inode;
  76. int result;
  77. if ( filp->f_pos == 0 ) 
  78. {
  79. if (filldir(dirent, ".", 1, filp->f_pos, dir->i_ino, DT_DIR) < 0)
  80. return 0;
  81. filp->f_pos ++;
  82. }
  83.  
  84. result = do_udf_readdir(dir, filp, filldir, dirent);
  85. UPDATE_ATIME(dir);
  86.   return result;
  87. }
  88. static int 
  89. do_udf_readdir(struct inode * dir, struct file *filp, filldir_t filldir, void *dirent)
  90. {
  91. struct udf_fileident_bh fibh;
  92. struct fileIdentDesc *fi=NULL;
  93. struct fileIdentDesc cfi;
  94. int block, iblock;
  95. loff_t nf_pos = filp->f_pos - 1;
  96. int flen;
  97. char fname[255];
  98. char *nameptr;
  99. uint16_t liu;
  100. uint8_t lfi;
  101. loff_t size = (udf_ext0_offset(dir) + dir->i_size) >> 2;
  102. struct buffer_head * bh = NULL, * tmp, * bha[16];
  103. lb_addr bloc, eloc;
  104. uint32_t extoffset, elen, offset;
  105. int i, num;
  106. unsigned int dt_type;
  107. if (nf_pos >= size)
  108. return 0;
  109. if (nf_pos == 0)
  110. nf_pos = (udf_ext0_offset(dir) >> 2);
  111. fibh.soffset = fibh.eoffset = (nf_pos & ((dir->i_sb->s_blocksize - 1) >> 2)) << 2;
  112. if (inode_bmap(dir, nf_pos >> (dir->i_sb->s_blocksize_bits - 2),
  113. &bloc, &extoffset, &eloc, &elen, &offset, &bh) == (EXT_RECORDED_ALLOCATED >> 30))
  114. {
  115. offset >>= dir->i_sb->s_blocksize_bits;
  116. block = udf_get_lb_pblock(dir->i_sb, eloc, offset);
  117. if ((++offset << dir->i_sb->s_blocksize_bits) < elen)
  118. {
  119. if (UDF_I_ALLOCTYPE(dir) == ICBTAG_FLAG_AD_SHORT)
  120. extoffset -= sizeof(short_ad);
  121. else if (UDF_I_ALLOCTYPE(dir) == ICBTAG_FLAG_AD_LONG)
  122. extoffset -= sizeof(long_ad);
  123. }
  124. else
  125. offset = 0;
  126. }
  127. else
  128. {
  129. udf_release_data(bh);
  130. return -ENOENT;
  131. }
  132. if (!(fibh.sbh = fibh.ebh = udf_tread(dir->i_sb, block)))
  133. {
  134. udf_release_data(bh);
  135. return -EIO;
  136. }
  137. if (!(offset & ((16 >> (dir->i_sb->s_blocksize_bits - 9))-1)))
  138. {
  139. i = 16 >> (dir->i_sb->s_blocksize_bits - 9);
  140. if (i+offset > (elen >> dir->i_sb->s_blocksize_bits))
  141. i = (elen >> dir->i_sb->s_blocksize_bits)-offset;
  142. for (num=0; i>0; i--)
  143. {
  144. block = udf_get_lb_pblock(dir->i_sb, eloc, offset+i);
  145. tmp = udf_tgetblk(dir->i_sb, block);
  146. if (tmp && !buffer_uptodate(tmp) && !buffer_locked(tmp))
  147. bha[num++] = tmp;
  148. else
  149. brelse(tmp);
  150. }
  151. if (num)
  152. {
  153. ll_rw_block(READA, num, bha);
  154. for (i=0; i<num; i++)
  155. brelse(bha[i]);
  156. }
  157. }
  158. while ( nf_pos < size )
  159. {
  160. filp->f_pos = nf_pos + 1;
  161. fi = udf_fileident_read(dir, &nf_pos, &fibh, &cfi, &bloc, &extoffset, &eloc, &elen, &offset, &bh);
  162. if (!fi)
  163. {
  164. if (fibh.sbh != fibh.ebh)
  165. udf_release_data(fibh.ebh);
  166. udf_release_data(fibh.sbh);
  167. udf_release_data(bh);
  168. return 0;
  169. }
  170. liu = le16_to_cpu(cfi.lengthOfImpUse);
  171. lfi = cfi.lengthFileIdent;
  172. if (fibh.sbh == fibh.ebh)
  173. nameptr = fi->fileIdent + liu;
  174. else
  175. {
  176. int poffset; /* Unpaded ending offset */
  177. poffset = fibh.soffset + sizeof(struct fileIdentDesc) + liu + lfi;
  178. if (poffset >= lfi)
  179. nameptr = (char *)(fibh.ebh->b_data + poffset - lfi);
  180. else
  181. {
  182. nameptr = fname;
  183. memcpy(nameptr, fi->fileIdent + liu, lfi - poffset);
  184. memcpy(nameptr + lfi - poffset, fibh.ebh->b_data, poffset);
  185. }
  186. }
  187. if ( (cfi.fileCharacteristics & FID_FILE_CHAR_DELETED) != 0 )
  188. {
  189. if ( !UDF_QUERY_FLAG(dir->i_sb, UDF_FLAG_UNDELETE) )
  190. continue;
  191. }
  192. if ( (cfi.fileCharacteristics & FID_FILE_CHAR_HIDDEN) != 0 )
  193. {
  194. if ( !UDF_QUERY_FLAG(dir->i_sb, UDF_FLAG_UNHIDE) )
  195. continue;
  196. }
  197. if ( cfi.fileCharacteristics & FID_FILE_CHAR_PARENT )
  198. {
  199. iblock = udf_get_lb_pblock(dir->i_sb, UDF_I_LOCATION(filp->f_dentry->d_parent->d_inode), 0);
  200. flen = 2;
  201. memcpy(fname, "..", flen);
  202. dt_type = DT_DIR;
  203. }
  204. else
  205. {
  206. iblock = udf_get_lb_pblock(dir->i_sb, lelb_to_cpu(cfi.icb.extLocation), 0);
  207. flen = udf_get_filename(dir->i_sb, nameptr, fname, lfi);
  208. dt_type = DT_UNKNOWN;
  209. }
  210. if (flen)
  211. {
  212. if (filldir(dirent, fname, flen, filp->f_pos, iblock, dt_type) < 0)
  213. {
  214. if (fibh.sbh != fibh.ebh)
  215. udf_release_data(fibh.ebh);
  216. udf_release_data(fibh.sbh);
  217. udf_release_data(bh);
  218.   return 0;
  219. }
  220. }
  221. } /* end while */
  222. filp->f_pos = nf_pos + 1;
  223. if (fibh.sbh != fibh.ebh)
  224. udf_release_data(fibh.ebh);
  225. udf_release_data(fibh.sbh);
  226. udf_release_data(bh);
  227. return 0;
  228. }