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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *  dir.c
  3.  *
  4.  *  Copyright (C) 1995, 1996 by Volker Lendecke
  5.  *  Modified for big endian by J.F. Chadima and David S. Miller
  6.  *  Modified 1997 Peter Waltenberg, Bill Hawes, David Woodhouse for 2.1 dcache
  7.  *  Modified 1998, 1999 Wolfram Pienkoss for NLS
  8.  *  Modified 1999 Wolfram Pienkoss for directory caching
  9.  *
  10.  */
  11. #include <linux/config.h>
  12. #include <linux/sched.h>
  13. #include <linux/errno.h>
  14. #include <linux/stat.h>
  15. #include <linux/kernel.h>
  16. #include <linux/slab.h>
  17. #include <linux/vmalloc.h>
  18. #include <linux/mm.h>
  19. #include <asm/uaccess.h>
  20. #include <asm/byteorder.h>
  21. #include <linux/locks.h>
  22. #include <linux/smp_lock.h>
  23. #include <linux/ncp_fs.h>
  24. #include "ncplib_kernel.h"
  25. static void ncp_read_volume_list(struct file *, void *, filldir_t,
  26. struct ncp_cache_control *);
  27. static void ncp_do_readdir(struct file *, void *, filldir_t,
  28. struct ncp_cache_control *);
  29. static int ncp_readdir(struct file *, void *, filldir_t);
  30. static int ncp_create(struct inode *, struct dentry *, int);
  31. static struct dentry *ncp_lookup(struct inode *, struct dentry *);
  32. static int ncp_unlink(struct inode *, struct dentry *);
  33. static int ncp_mkdir(struct inode *, struct dentry *, int);
  34. static int ncp_rmdir(struct inode *, struct dentry *);
  35. static int ncp_rename(struct inode *, struct dentry *,
  36.          struct inode *, struct dentry *);
  37. #ifdef CONFIG_NCPFS_EXTRAS
  38. extern int ncp_symlink(struct inode *, struct dentry *, const char *);
  39. #endif
  40.       
  41. struct file_operations ncp_dir_operations =
  42. {
  43. read: generic_read_dir,
  44. readdir: ncp_readdir,
  45. ioctl: ncp_ioctl,
  46. };
  47. struct inode_operations ncp_dir_inode_operations =
  48. {
  49. create: ncp_create,
  50. lookup: ncp_lookup,
  51. unlink: ncp_unlink,
  52. #ifdef CONFIG_NCPFS_EXTRAS
  53. symlink: ncp_symlink,
  54. #endif
  55. mkdir: ncp_mkdir,
  56. rmdir: ncp_rmdir,
  57. rename: ncp_rename,
  58. setattr: ncp_notify_change,
  59. };
  60. /*
  61.  * Dentry operations routines
  62.  */
  63. static int ncp_lookup_validate(struct dentry *, int);
  64. static int ncp_hash_dentry(struct dentry *, struct qstr *);
  65. static int ncp_compare_dentry (struct dentry *, struct qstr *, struct qstr *);
  66. static int ncp_delete_dentry(struct dentry *);
  67. static struct dentry_operations ncp_dentry_operations =
  68. {
  69. d_revalidate: ncp_lookup_validate,
  70. d_hash: ncp_hash_dentry,
  71. d_compare: ncp_compare_dentry,
  72. d_delete: ncp_delete_dentry,
  73. };
  74. struct dentry_operations ncp_root_dentry_operations =
  75. {
  76. d_hash: ncp_hash_dentry,
  77. d_compare: ncp_compare_dentry,
  78. d_delete: ncp_delete_dentry,
  79. };
  80. /*
  81.  * Note: leave the hash unchanged if the directory
  82.  * is case-sensitive.
  83.  */
  84. static int 
  85. ncp_hash_dentry(struct dentry *dentry, struct qstr *this)
  86. {
  87. struct nls_table *t;
  88. unsigned long hash;
  89. int i;
  90. t = NCP_IO_TABLE(dentry);
  91. if (!ncp_case_sensitive(dentry->d_inode)) {
  92. hash = init_name_hash();
  93. for (i=0; i<this->len ; i++)
  94. hash = partial_name_hash(ncp_tolower(t, this->name[i]),
  95. hash);
  96. this->hash = end_name_hash(hash);
  97. }
  98. return 0;
  99. }
  100. static int
  101. ncp_compare_dentry(struct dentry *dentry, struct qstr *a, struct qstr *b)
  102. {
  103. if (a->len != b->len)
  104. return 1;
  105. if (ncp_case_sensitive(dentry->d_inode))
  106. return strncmp(a->name, b->name, a->len);
  107. return ncp_strnicmp(NCP_IO_TABLE(dentry), a->name, b->name, a->len);
  108. }
  109. /*
  110.  * This is the callback from dput() when d_count is going to 0.
  111.  * We use this to unhash dentries with bad inodes.
  112.  * Closing files can be safely postponed until iput() - it's done there anyway.
  113.  */
  114. static int
  115. ncp_delete_dentry(struct dentry * dentry)
  116. {
  117. struct inode *inode = dentry->d_inode;
  118. if (inode) {
  119. if (is_bad_inode(inode))
  120. return 1;
  121. } else
  122. {
  123. /* N.B. Unhash negative dentries? */
  124. }
  125. return 0;
  126. }
  127. static inline int
  128. ncp_single_volume(struct ncp_server *server)
  129. {
  130. return (server->m.mounted_vol[0] != '');
  131. }
  132. static inline int ncp_is_server_root(struct inode *inode)
  133. {
  134. return (!ncp_single_volume(NCP_SERVER(inode)) &&
  135. inode == inode->i_sb->s_root->d_inode);
  136. }
  137. /*
  138.  * This is the callback when the dcache has a lookup hit.
  139.  */
  140. #ifdef CONFIG_NCPFS_STRONG
  141. /* try to delete a readonly file (NW R bit set) */
  142. static int
  143. ncp_force_unlink(struct inode *dir, struct dentry* dentry)
  144. {
  145.         int res=0x9c,res2;
  146. struct nw_modify_dos_info info;
  147. __u32 old_nwattr;
  148. struct inode *inode;
  149. memset(&info, 0, sizeof(info));
  150.         /* remove the Read-Only flag on the NW server */
  151. inode = dentry->d_inode;
  152. old_nwattr = NCP_FINFO(inode)->nwattr;
  153. info.attributes = old_nwattr & ~(aRONLY|aDELETEINHIBIT|aRENAMEINHIBIT);
  154. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(inode), inode, NULL, DM_ATTRIBUTES, &info);
  155. if (res2)
  156. goto leave_me;
  157.         /* now try again the delete operation */
  158.         res = ncp_del_file_or_subdir2(NCP_SERVER(dir), dentry);
  159.         if (res)  /* delete failed, set R bit again */
  160.         {
  161. info.attributes = old_nwattr;
  162. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(inode), inode, NULL, DM_ATTRIBUTES, &info);
  163. if (res2)
  164.                         goto leave_me;
  165.         }
  166. leave_me:
  167.         return(res);
  168. }
  169. #endif /* CONFIG_NCPFS_STRONG */
  170. #ifdef CONFIG_NCPFS_STRONG
  171. static int
  172. ncp_force_rename(struct inode *old_dir, struct dentry* old_dentry, char *_old_name,
  173.                  struct inode *new_dir, struct dentry* new_dentry, char *_new_name)
  174. {
  175. struct nw_modify_dos_info info;
  176.         int res=0x90,res2;
  177. struct inode *old_inode = old_dentry->d_inode;
  178. __u32 old_nwattr = NCP_FINFO(old_inode)->nwattr;
  179. __u32 new_nwattr = 0; /* shut compiler warning */
  180. int old_nwattr_changed = 0;
  181. int new_nwattr_changed = 0;
  182. memset(&info, 0, sizeof(info));
  183.         /* remove the Read-Only flag on the NW server */
  184. info.attributes = old_nwattr & ~(aRONLY|aRENAMEINHIBIT|aDELETEINHIBIT);
  185. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(old_inode), old_inode, NULL, DM_ATTRIBUTES, &info);
  186. if (!res2)
  187. old_nwattr_changed = 1;
  188. if (new_dentry && new_dentry->d_inode) {
  189. new_nwattr = NCP_FINFO(new_dentry->d_inode)->nwattr;
  190. info.attributes = new_nwattr & ~(aRONLY|aRENAMEINHIBIT|aDELETEINHIBIT);
  191. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(new_dir), new_dir, _new_name, DM_ATTRIBUTES, &info);
  192. if (!res2)
  193. new_nwattr_changed = 1;
  194. }
  195.         /* now try again the rename operation */
  196. /* but only if something really happened */
  197. if (new_nwattr_changed || old_nwattr_changed) {
  198.         res = ncp_ren_or_mov_file_or_subdir(NCP_SERVER(old_dir),
  199.                                              old_dir, _old_name,
  200.                                              new_dir, _new_name);
  201. if (res)
  202. goto leave_me;
  203. /* file was successfully renamed, so:
  204.    do not set attributes on old file - it no longer exists
  205.    copy attributes from old file to new */
  206. new_nwattr_changed = old_nwattr_changed;
  207. new_nwattr = old_nwattr;
  208. old_nwattr_changed = 0;
  209. leave_me:;
  210. if (old_nwattr_changed) {
  211. info.attributes = old_nwattr;
  212. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(old_inode), old_inode, NULL, DM_ATTRIBUTES, &info);
  213. /* ignore errors */
  214. }
  215. if (new_nwattr_changed) {
  216. info.attributes = new_nwattr;
  217. res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(new_dir), new_dir, _new_name, DM_ATTRIBUTES, &info);
  218. /* ignore errors */
  219. }
  220.         return(res);
  221. }
  222. #endif /* CONFIG_NCPFS_STRONG */
  223. static int
  224. __ncp_lookup_validate(struct dentry * dentry, int flags)
  225. {
  226. struct ncp_server *server;
  227. struct inode *dir = dentry->d_parent->d_inode;
  228. struct ncp_entry_info finfo;
  229. int res, val = 0, len = dentry->d_name.len + 1;
  230. __u8 __name[len];
  231. if (!dentry->d_inode || !dir)
  232. goto finished;
  233. server = NCP_SERVER(dir);
  234. if (!ncp_conn_valid(server))
  235. goto finished;
  236. /*
  237.  * Inspired by smbfs:
  238.  * The default validation is based on dentry age:
  239.  * We set the max age at mount time.  (But each
  240.  * successful server lookup renews the timestamp.)
  241.  */
  242. val = NCP_TEST_AGE(server, dentry);
  243. if (val)
  244. goto finished;
  245. DDPRINTK("ncp_lookup_validate: %s/%s not valid, age=%ld, server lookupn",
  246. dentry->d_parent->d_name.name, dentry->d_name.name,
  247. NCP_GET_AGE(dentry));
  248. if (ncp_is_server_root(dir)) {
  249. res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  250. len-1, 1);
  251. if (!res)
  252. res = ncp_lookup_volume(server, __name, &(finfo.i));
  253. } else {
  254. res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  255. len-1, !ncp_preserve_case(dir));
  256. if (!res)
  257. res = ncp_obtain_info(server, dir, __name, &(finfo.i));
  258. }
  259. DDPRINTK("ncp_lookup_validate: looked for %s/%s, res=%dn",
  260. dentry->d_parent->d_name.name, __name, res);
  261. /*
  262.  * If we didn't find it, or if it has a different dirEntNum to
  263.  * what we remember, it's not valid any more.
  264.  */
  265. if (!res) {
  266. if (finfo.i.dirEntNum == NCP_FINFO(dentry->d_inode)->dirEntNum) {
  267. ncp_new_dentry(dentry);
  268. val=1;
  269. } else
  270. DDPRINTK("ncp_lookup_validate: found, but dirEntNum changedn");
  271. ncp_update_inode2(dentry->d_inode, &finfo);
  272. }
  273. finished:
  274. DDPRINTK("ncp_lookup_validate: result=%dn", val);
  275. return val;
  276. }
  277. static int
  278. ncp_lookup_validate(struct dentry * dentry, int flags)
  279. {
  280. int res;
  281. lock_kernel();
  282. res = __ncp_lookup_validate(dentry, flags);
  283. unlock_kernel();
  284. return res;
  285. }
  286. static struct dentry *
  287. ncp_dget_fpos(struct dentry *dentry, struct dentry *parent, unsigned long fpos)
  288. {
  289. struct dentry *dent = dentry;
  290. struct list_head *next;
  291. if (d_validate(dent, parent)) {
  292. if (dent->d_name.len <= NCP_MAXPATHLEN &&
  293.     (unsigned long)dent->d_fsdata == fpos) {
  294. if (!dent->d_inode) {
  295. dput(dent);
  296. dent = NULL;
  297. }
  298. return dent;
  299. }
  300. dput(dent);
  301. }
  302. /* If a pointer is invalid, we search the dentry. */
  303. spin_lock(&dcache_lock);
  304. next = parent->d_subdirs.next;
  305. while (next != &parent->d_subdirs) {
  306. dent = list_entry(next, struct dentry, d_child);
  307. if ((unsigned long)dent->d_fsdata == fpos) {
  308. if (dent->d_inode)
  309. dget_locked(dent);
  310. else
  311. dent = NULL;
  312. spin_unlock(&dcache_lock);
  313. goto out;
  314. }
  315. next = next->next;
  316. }
  317. spin_unlock(&dcache_lock);
  318. return NULL;
  319. out:
  320. return dent;
  321. }
  322. static time_t ncp_obtain_mtime(struct dentry *dentry)
  323. {
  324. struct inode *inode = dentry->d_inode;
  325. struct ncp_server *server = NCP_SERVER(inode);
  326. struct nw_info_struct i;
  327. if (!ncp_conn_valid(server) || ncp_is_server_root(inode))
  328. return 0;
  329. if (ncp_obtain_info(server, inode, NULL, &i))
  330. return 0;
  331. return ncp_date_dos2unix(le16_to_cpu(i.modifyTime),
  332. le16_to_cpu(i.modifyDate));
  333. }
  334. static int ncp_readdir(struct file *filp, void *dirent, filldir_t filldir)
  335. {
  336. struct dentry *dentry = filp->f_dentry;
  337. struct inode *inode = dentry->d_inode;
  338. struct page *page = NULL;
  339. struct ncp_server *server = NCP_SERVER(inode);
  340. union  ncp_dir_cache *cache = NULL;
  341. struct ncp_cache_control ctl;
  342. int result, mtime_valid = 0;
  343. time_t mtime = 0;
  344. ctl.page  = NULL;
  345. ctl.cache = NULL;
  346. DDPRINTK("ncp_readdir: reading %s/%s, pos=%dn",
  347. dentry->d_parent->d_name.name, dentry->d_name.name,
  348. (int) filp->f_pos);
  349. result = -EIO;
  350. if (!ncp_conn_valid(server))
  351. goto out;
  352. result = 0;
  353. if (filp->f_pos == 0) {
  354. if (filldir(dirent, ".", 1, 0, inode->i_ino, DT_DIR))
  355. goto out;
  356. filp->f_pos = 1;
  357. }
  358. if (filp->f_pos == 1) {
  359. if (filldir(dirent, "..", 2, 1,
  360. dentry->d_parent->d_inode->i_ino, DT_DIR))
  361. goto out;
  362. filp->f_pos = 2;
  363. }
  364. page = grab_cache_page(&inode->i_data, 0);
  365. if (!page)
  366. goto read_really;
  367. ctl.cache = cache = kmap(page);
  368. ctl.head  = cache->head;
  369. if (!Page_Uptodate(page) || !ctl.head.eof)
  370. goto init_cache;
  371. if (filp->f_pos == 2) {
  372. if (jiffies - ctl.head.time >= NCP_MAX_AGE(server))
  373. goto init_cache;
  374. mtime = ncp_obtain_mtime(dentry);
  375. mtime_valid = 1;
  376. if ((!mtime) || (mtime != ctl.head.mtime))
  377. goto init_cache;
  378. }
  379. if (filp->f_pos > ctl.head.end)
  380. goto finished;
  381. ctl.fpos = filp->f_pos + (NCP_DIRCACHE_START - 2);
  382. ctl.ofs  = ctl.fpos / NCP_DIRCACHE_SIZE;
  383. ctl.idx  = ctl.fpos % NCP_DIRCACHE_SIZE;
  384. for (;;) {
  385. if (ctl.ofs != 0) {
  386. ctl.page = find_lock_page(&inode->i_data, ctl.ofs);
  387. if (!ctl.page)
  388. goto invalid_cache;
  389. ctl.cache = kmap(ctl.page);
  390. if (!Page_Uptodate(ctl.page))
  391. goto invalid_cache;
  392. }
  393. while (ctl.idx < NCP_DIRCACHE_SIZE) {
  394. struct dentry *dent;
  395. int res;
  396. dent = ncp_dget_fpos(ctl.cache->dentry[ctl.idx],
  397. dentry, filp->f_pos);
  398. if (!dent)
  399. goto invalid_cache;
  400. res = filldir(dirent, dent->d_name.name,
  401. dent->d_name.len, filp->f_pos,
  402. dent->d_inode->i_ino, DT_UNKNOWN);
  403. dput(dent);
  404. if (res)
  405. goto finished;
  406. filp->f_pos += 1;
  407. ctl.idx += 1;
  408. if (filp->f_pos > ctl.head.end)
  409. goto finished;
  410. }
  411. if (ctl.page) {
  412. kunmap(ctl.page);
  413. SetPageUptodate(ctl.page);
  414. UnlockPage(ctl.page);
  415. page_cache_release(ctl.page);
  416. ctl.page = NULL;
  417. }
  418. ctl.idx  = 0;
  419. ctl.ofs += 1;
  420. }
  421. invalid_cache:
  422. if (ctl.page) {
  423. kunmap(ctl.page);
  424. UnlockPage(ctl.page);
  425. page_cache_release(ctl.page);
  426. ctl.page = NULL;
  427. }
  428. ctl.cache = cache;
  429. init_cache:
  430. ncp_invalidate_dircache_entries(dentry);
  431. if (!mtime_valid) {
  432. mtime = ncp_obtain_mtime(dentry);
  433. mtime_valid = 1;
  434. }
  435. ctl.head.mtime = mtime;
  436. ctl.head.time = jiffies;
  437. ctl.head.eof = 0;
  438. ctl.fpos = 2;
  439. ctl.ofs = 0;
  440. ctl.idx = NCP_DIRCACHE_START;
  441. ctl.filled = 0;
  442. ctl.valid  = 1;
  443. read_really:
  444. if (ncp_is_server_root(inode)) {
  445. ncp_read_volume_list(filp, dirent, filldir, &ctl);
  446. } else {
  447. ncp_do_readdir(filp, dirent, filldir, &ctl);
  448. }
  449. ctl.head.end = ctl.fpos - 1;
  450. ctl.head.eof = ctl.valid;
  451. finished:
  452. if (page) {
  453. cache->head = ctl.head;
  454. kunmap(page);
  455. SetPageUptodate(page);
  456. UnlockPage(page);
  457. page_cache_release(page);
  458. }
  459. if (ctl.page) {
  460. kunmap(ctl.page);
  461. SetPageUptodate(ctl.page);
  462. UnlockPage(ctl.page);
  463. page_cache_release(ctl.page);
  464. }
  465. out:
  466. return result;
  467. }
  468. static int
  469. ncp_fill_cache(struct file *filp, void *dirent, filldir_t filldir,
  470. struct ncp_cache_control *ctrl, struct ncp_entry_info *entry)
  471. {
  472. struct dentry *newdent, *dentry = filp->f_dentry;
  473. struct inode *newino, *inode = dentry->d_inode;
  474. struct ncp_cache_control ctl = *ctrl;
  475. struct qstr qname;
  476. int valid = 0;
  477. int hashed = 0;
  478. ino_t ino = 0;
  479. __u8 __name[256];
  480. qname.len = 256;
  481. if (ncp_vol2io(NCP_SERVER(inode), __name, &qname.len,
  482. entry->i.entryName, entry->i.nameLen,
  483. !ncp_preserve_entry_case(inode, entry->i.NSCreator)))
  484. return 1; /* I'm not sure */
  485. qname.name = __name;
  486. qname.hash = full_name_hash(qname.name, qname.len);
  487. if (dentry->d_op && dentry->d_op->d_hash)
  488. if (dentry->d_op->d_hash(dentry, &qname) != 0)
  489. goto end_advance;
  490. newdent = d_lookup(dentry, &qname);
  491. if (!newdent) {
  492. newdent = d_alloc(dentry, &qname);
  493. if (!newdent)
  494. goto end_advance;
  495. } else {
  496. hashed = 1;
  497. memcpy((char *) newdent->d_name.name, qname.name,
  498. newdent->d_name.len);
  499. }
  500. if (!newdent->d_inode) {
  501. entry->opened = 0;
  502. entry->ino = iunique(inode->i_sb, 2);
  503. newino = ncp_iget(inode->i_sb, entry);
  504. if (newino) {
  505. newdent->d_op = &ncp_dentry_operations;
  506. d_instantiate(newdent, newino);
  507. if (!hashed)
  508. d_rehash(newdent);
  509. }
  510. } else
  511. ncp_update_inode2(newdent->d_inode, entry);
  512. if (newdent->d_inode) {
  513. ino = newdent->d_inode->i_ino;
  514. newdent->d_fsdata = (void *) ctl.fpos;
  515. ncp_new_dentry(newdent);
  516. }
  517. if (ctl.idx >= NCP_DIRCACHE_SIZE) {
  518. if (ctl.page) {
  519. kunmap(ctl.page);
  520. SetPageUptodate(ctl.page);
  521. UnlockPage(ctl.page);
  522. page_cache_release(ctl.page);
  523. }
  524. ctl.cache = NULL;
  525. ctl.idx  -= NCP_DIRCACHE_SIZE;
  526. ctl.ofs  += 1;
  527. ctl.page  = grab_cache_page(&inode->i_data, ctl.ofs);
  528. if (ctl.page)
  529. ctl.cache = kmap(ctl.page);
  530. }
  531. if (ctl.cache) {
  532. ctl.cache->dentry[ctl.idx] = newdent;
  533. valid = 1;
  534. }
  535. dput(newdent);
  536. end_advance:
  537. if (!valid)
  538. ctl.valid = 0;
  539. if (!ctl.filled && (ctl.fpos == filp->f_pos)) {
  540. if (!ino)
  541. ino = find_inode_number(dentry, &qname);
  542. if (!ino)
  543. ino = iunique(inode->i_sb, 2);
  544. ctl.filled = filldir(dirent, qname.name, qname.len,
  545.      filp->f_pos, ino, DT_UNKNOWN);
  546. if (!ctl.filled)
  547. filp->f_pos += 1;
  548. }
  549. ctl.fpos += 1;
  550. ctl.idx  += 1;
  551. *ctrl = ctl;
  552. return (ctl.valid || !ctl.filled);
  553. }
  554. static void
  555. ncp_read_volume_list(struct file *filp, void *dirent, filldir_t filldir,
  556. struct ncp_cache_control *ctl)
  557. {
  558. struct dentry *dentry = filp->f_dentry;
  559. struct inode *inode = dentry->d_inode;
  560. struct ncp_server *server = NCP_SERVER(inode);
  561. struct ncp_volume_info info;
  562. struct ncp_entry_info entry;
  563. int i;
  564. DPRINTK("ncp_read_volume_list: pos=%ldn",
  565. (unsigned long) filp->f_pos);
  566. for (i = 0; i < NCP_NUMBER_OF_VOLUMES; i++) {
  567. if (ncp_get_volume_info_with_number(server, i, &info) != 0)
  568. return;
  569. if (!strlen(info.volume_name))
  570. continue;
  571. DPRINTK("ncp_read_volume_list: found vol: %sn",
  572. info.volume_name);
  573. if (ncp_lookup_volume(server, info.volume_name,
  574. &entry.i)) {
  575. DPRINTK("ncpfs: could not lookup vol %sn",
  576. info.volume_name);
  577. continue;
  578. }
  579. if (!ncp_fill_cache(filp, dirent, filldir, ctl, &entry))
  580. return;
  581. }
  582. }
  583. static void
  584. ncp_do_readdir(struct file *filp, void *dirent, filldir_t filldir,
  585. struct ncp_cache_control *ctl)
  586. {
  587. struct dentry *dentry = filp->f_dentry;
  588. struct inode *dir = dentry->d_inode;
  589. struct ncp_server *server = NCP_SERVER(dir);
  590. struct nw_search_sequence seq;
  591. struct ncp_entry_info entry;
  592. int err;
  593. DPRINTK("ncp_do_readdir: %s/%s, fpos=%ldn",
  594. dentry->d_parent->d_name.name, dentry->d_name.name,
  595. (unsigned long) filp->f_pos);
  596. PPRINTK("ncp_do_readdir: init %s, volnum=%d, dirent=%un",
  597. dentry->d_name.name, NCP_FINFO(dir)->volNumber,
  598. NCP_FINFO(dir)->dirEntNum);
  599. err = ncp_initialize_search(server, dir, &seq);
  600. if (err) {
  601. DPRINTK("ncp_do_readdir: init failed, err=%dn", err);
  602. return;
  603. }
  604. for (;;) {
  605. err = ncp_search_for_file_or_subdir(server, &seq, &entry.i);
  606. if (err) {
  607. DPRINTK("ncp_do_readdir: search failed, err=%dn", err);
  608. return;
  609. }
  610. if (!ncp_fill_cache(filp, dirent, filldir, ctl, &entry))
  611. return;
  612. }
  613. }
  614. int ncp_conn_logged_in(struct super_block *sb)
  615. {
  616. struct ncp_server* server = NCP_SBP(sb);
  617. struct nw_info_struct i;
  618. int result, len = strlen(server->m.mounted_vol) + 1;
  619. __u8 __name[len];
  620. if (ncp_single_volume(server)) {
  621. struct dentry* dent;
  622. result = -ENOENT;
  623. if (ncp_io2vol(server, __name, &len, server->m.mounted_vol,
  624. len-1, 1))
  625. goto out;
  626. if (ncp_lookup_volume(server, __name, &i)) {
  627. PPRINTK("ncp_conn_logged_in: %s not foundn",
  628. server->m.mounted_vol);
  629. goto out;
  630. }
  631. dent = sb->s_root;
  632. if (dent) {
  633. struct inode* ino = dent->d_inode;
  634. if (ino) {
  635. NCP_FINFO(ino)->volNumber = i.volNumber;
  636. NCP_FINFO(ino)->dirEntNum = i.dirEntNum;
  637. NCP_FINFO(ino)->DosDirNum = i.DosDirNum;
  638. } else {
  639. DPRINTK("ncpfs: sb->s_root->d_inode == NULL!n");
  640. }
  641. } else {
  642. DPRINTK("ncpfs: sb->s_root == NULL!n");
  643. }
  644. }
  645. result = 0;
  646. out:
  647. return result;
  648. }
  649. static struct dentry *ncp_lookup(struct inode *dir, struct dentry *dentry)
  650. {
  651. struct ncp_server *server = NCP_SERVER(dir);
  652. struct inode *inode = NULL;
  653. struct ncp_entry_info finfo;
  654. int error, res, len = dentry->d_name.len + 1;
  655. __u8 __name[len];
  656. error = -EIO;
  657. if (!ncp_conn_valid(server))
  658. goto finished;
  659. PPRINTK("ncp_lookup: server lookup for %s/%sn",
  660. dentry->d_parent->d_name.name, dentry->d_name.name);
  661. if (ncp_is_server_root(dir)) {
  662. res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  663. len-1, 1);
  664. if (!res)
  665. res = ncp_lookup_volume(server, __name, &(finfo.i));
  666. } else {
  667. res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  668. len-1, !ncp_preserve_case(dir));
  669. if (!res)
  670. res = ncp_obtain_info(server, dir, __name, &(finfo.i));
  671. }
  672. PPRINTK("ncp_lookup: looked for %s/%s, res=%dn",
  673. dentry->d_parent->d_name.name, __name, res);
  674. /*
  675.  * If we didn't find an entry, make a negative dentry.
  676.  */
  677. if (res)
  678. goto add_entry;
  679. /*
  680.  * Create an inode for the entry.
  681.  */
  682. finfo.opened = 0;
  683. finfo.ino = iunique(dir->i_sb, 2);
  684. error = -EACCES;
  685. inode = ncp_iget(dir->i_sb, &finfo);
  686. if (inode) {
  687. ncp_new_dentry(dentry);
  688. add_entry:
  689. dentry->d_op = &ncp_dentry_operations;
  690. d_add(dentry, inode);
  691. error = 0;
  692. }
  693. finished:
  694. PPRINTK("ncp_lookup: result=%dn", error);
  695. return ERR_PTR(error);
  696. }
  697. /*
  698.  * This code is common to create, mkdir, and mknod.
  699.  */
  700. static int ncp_instantiate(struct inode *dir, struct dentry *dentry,
  701. struct ncp_entry_info *finfo)
  702. {
  703. struct inode *inode;
  704. int error = -EINVAL;
  705. finfo->ino = iunique(dir->i_sb, 2);
  706. inode = ncp_iget(dir->i_sb, finfo);
  707. if (!inode)
  708. goto out_close;
  709. d_instantiate(dentry,inode);
  710. error = 0;
  711. out:
  712. return error;
  713. out_close:
  714. PPRINTK("ncp_instantiate: %s/%s failed, closing filen",
  715. dentry->d_parent->d_name.name, dentry->d_name.name);
  716. ncp_close_file(NCP_SERVER(dir), finfo->file_handle);
  717. goto out;
  718. }
  719. int ncp_create_new(struct inode *dir, struct dentry *dentry, int mode,
  720. int attributes)
  721. {
  722. struct ncp_server *server = NCP_SERVER(dir);
  723. struct ncp_entry_info finfo;
  724. int error, result, len = dentry->d_name.len + 1;
  725. int opmode;
  726. __u8 __name[len];
  727. PPRINTK("ncp_create_new: creating %s/%s, mode=%xn",
  728. dentry->d_parent->d_name.name, dentry->d_name.name, mode);
  729. error = -EIO;
  730. if (!ncp_conn_valid(server))
  731. goto out;
  732. ncp_age_dentry(server, dentry);
  733. error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  734. len-1, !ncp_preserve_case(dir));
  735. if (error)
  736. goto out;
  737. error = -EACCES;
  738. if (S_ISREG(mode) && 
  739.     (server->m.flags & NCP_MOUNT_EXTRAS) && 
  740.     (mode & S_IXUGO))
  741. attributes |= aSYSTEM | aSHARED;
  742. result = ncp_open_create_file_or_subdir(server, dir, __name,
  743. OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE,
  744. attributes, AR_READ | AR_WRITE, &finfo);
  745. opmode = O_RDWR;
  746. if (result) {
  747. result = ncp_open_create_file_or_subdir(server, dir, __name,
  748. OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE,
  749. attributes, AR_WRITE, &finfo);
  750. if (result) {
  751. if (result == 0x87)
  752. error = -ENAMETOOLONG;
  753. DPRINTK("ncp_create: %s/%s failedn",
  754. dentry->d_parent->d_name.name, dentry->d_name.name);
  755. goto out;
  756. }
  757. opmode = O_WRONLY;
  758. }
  759. finfo.access = opmode;
  760. error = ncp_instantiate(dir, dentry, &finfo);
  761. out:
  762. return error;
  763. }
  764. static int ncp_create(struct inode *dir, struct dentry *dentry, int mode)
  765. {
  766. return ncp_create_new(dir, dentry, mode, 0);
  767. }
  768. static int ncp_mkdir(struct inode *dir, struct dentry *dentry, int mode)
  769. {
  770. struct ncp_entry_info finfo;
  771. struct ncp_server *server = NCP_SERVER(dir);
  772. int error, len = dentry->d_name.len + 1;
  773. __u8 __name[len];
  774. DPRINTK("ncp_mkdir: making %s/%sn",
  775. dentry->d_parent->d_name.name, dentry->d_name.name);
  776. error = -EIO;
  777. if (!ncp_conn_valid(server))
  778. goto out;
  779. ncp_age_dentry(server, dentry);
  780. error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  781. len-1, !ncp_preserve_case(dir));
  782. if (error)
  783. goto out;
  784. error = -EACCES;
  785. if (ncp_open_create_file_or_subdir(server, dir, __name,
  786.    OC_MODE_CREATE, aDIR, 0xffff,
  787.    &finfo) == 0)
  788. {
  789. error = ncp_instantiate(dir, dentry, &finfo);
  790. }
  791. out:
  792. return error;
  793. }
  794. static int ncp_rmdir(struct inode *dir, struct dentry *dentry)
  795. {
  796. struct ncp_server *server = NCP_SERVER(dir);
  797. int error, result, len = dentry->d_name.len + 1;
  798. __u8 __name[len];
  799. DPRINTK("ncp_rmdir: removing %s/%sn",
  800. dentry->d_parent->d_name.name, dentry->d_name.name);
  801. error = -EIO;
  802. if (!ncp_conn_valid(server))
  803. goto out;
  804. error = -EBUSY;
  805. if (!d_unhashed(dentry))
  806. goto out;
  807. error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
  808. len-1, !ncp_preserve_case(dir));
  809. if (error)
  810. goto out;
  811. result = ncp_del_file_or_subdir(server, dir, __name);
  812. switch (result) {
  813. case 0x00:
  814. error = 0;
  815. break;
  816. case 0x85: /* unauthorized to delete file */
  817. case 0x8A: /* unauthorized to delete file */
  818. error = -EACCES;
  819. break;
  820. case 0x8F:
  821. case 0x90: /* read only */
  822. error = -EPERM;
  823. break;
  824. case 0x9F: /* in use by another client */
  825. error = -EBUSY;
  826. break;
  827. case 0xA0: /* directory not empty */
  828. error = -ENOTEMPTY;
  829. break;
  830. case 0xFF: /* someone deleted file */
  831. error = -ENOENT;
  832. break;
  833. default:
  834. error = -EACCES;
  835. break;
  836.         }
  837. out:
  838. return error;
  839. }
  840. static int ncp_unlink(struct inode *dir, struct dentry *dentry)
  841. {
  842. struct inode *inode = dentry->d_inode;
  843. struct ncp_server *server = NCP_SERVER(dir);
  844. int error;
  845. DPRINTK("ncp_unlink: unlinking %s/%sn",
  846. dentry->d_parent->d_name.name, dentry->d_name.name);
  847. error = -EIO;
  848. if (!ncp_conn_valid(server))
  849. goto out;
  850. /*
  851.  * Check whether to close the file ...
  852.  */
  853. if (inode) {
  854. PPRINTK("ncp_unlink: closing filen");
  855. ncp_make_closed(inode);
  856. }
  857. error = ncp_del_file_or_subdir2(server, dentry);
  858. #ifdef CONFIG_NCPFS_STRONG
  859. /* 9C is Invalid path.. It should be 8F, 90 - read only, but
  860.    it is not :-( */
  861. if ((error == 0x9C || error == 0x90) && server->m.flags & NCP_MOUNT_STRONG) { /* R/O */
  862. error = ncp_force_unlink(dir, dentry);
  863. }
  864. #endif
  865. switch (error) {
  866. case 0x00:
  867. DPRINTK("ncp: removed %s/%sn",
  868. dentry->d_parent->d_name.name, dentry->d_name.name);
  869. break;
  870. case 0x85:
  871. case 0x8A:
  872. error = -EACCES;
  873. break;
  874. case 0x8D: /* some files in use */
  875. case 0x8E: /* all files in use */
  876. error = -EBUSY;
  877. break;
  878. case 0x8F: /* some read only */
  879. case 0x90: /* all read only */
  880. case 0x9C: /* !!! returned when in-use or read-only by NW4 */
  881. error = -EPERM;
  882. break;
  883. case 0xFF:
  884. error = -ENOENT;
  885. break;
  886. default:
  887. error = -EACCES;
  888. break;
  889. }
  890. out:
  891. return error;
  892. }
  893. static int ncp_rename(struct inode *old_dir, struct dentry *old_dentry,
  894.       struct inode *new_dir, struct dentry *new_dentry)
  895. {
  896. struct ncp_server *server = NCP_SERVER(old_dir);
  897. int error;
  898. int old_len = old_dentry->d_name.len + 1;
  899. int new_len = new_dentry->d_name.len + 1;
  900. __u8 __old_name[old_len], __new_name[new_len];
  901. DPRINTK("ncp_rename: %s/%s to %s/%sn",
  902. old_dentry->d_parent->d_name.name, old_dentry->d_name.name,
  903. new_dentry->d_parent->d_name.name, new_dentry->d_name.name);
  904. error = -EIO;
  905. if (!ncp_conn_valid(server))
  906. goto out;
  907. ncp_age_dentry(server, old_dentry);
  908. ncp_age_dentry(server, new_dentry);
  909. error = ncp_io2vol(server, __old_name, &old_len,
  910. old_dentry->d_name.name, old_len-1,
  911. !ncp_preserve_case(old_dir));
  912. if (error)
  913. goto out;
  914. error = ncp_io2vol(server, __new_name, &new_len,
  915. new_dentry->d_name.name, new_len-1,
  916. !ncp_preserve_case(new_dir));
  917. if (error)
  918. goto out;
  919. error = ncp_ren_or_mov_file_or_subdir(server, old_dir, __old_name,
  920.       new_dir, __new_name);
  921. #ifdef CONFIG_NCPFS_STRONG
  922. if ((error == 0x90 || error == 0x8B || error == -EACCES) &&
  923. server->m.flags & NCP_MOUNT_STRONG) { /* RO */
  924. error = ncp_force_rename(old_dir, old_dentry, __old_name,
  925.  new_dir, new_dentry, __new_name);
  926. }
  927. #endif
  928. switch (error) {
  929. case 0x00:
  930.                         DPRINTK("ncp renamed %s -> %s.n",
  931.                                 old_dentry->d_name.name,new_dentry->d_name.name);
  932. break;
  933. case 0x9E:
  934. error = -ENAMETOOLONG;
  935. break;
  936. case 0xFF:
  937. error = -ENOENT;
  938. break;
  939. default:
  940. error = -EACCES;
  941. break;
  942. }
  943. out:
  944. return error;
  945. }
  946. /* The following routines are taken directly from msdos-fs */
  947. /* Linear day numbers of the respective 1sts in non-leap years. */
  948. static int day_n[] =
  949. {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 0, 0, 0, 0};
  950. /* Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec */
  951. extern struct timezone sys_tz;
  952. static int utc2local(int time)
  953. {
  954. return time - sys_tz.tz_minuteswest * 60;
  955. }
  956. static int local2utc(int time)
  957. {
  958. return time + sys_tz.tz_minuteswest * 60;
  959. }
  960. /* Convert a MS-DOS time/date pair to a UNIX date (seconds since 1 1 70). */
  961. int
  962. ncp_date_dos2unix(unsigned short time, unsigned short date)
  963. {
  964. int month, year, secs;
  965. /* first subtract and mask after that... Otherwise, if
  966.    date == 0, bad things happen */
  967. month = ((date >> 5) - 1) & 15;
  968. year = date >> 9;
  969. secs = (time & 31) * 2 + 60 * ((time >> 5) & 63) + (time >> 11) * 3600 +
  970. 86400 * ((date & 31) - 1 + day_n[month] + (year / 4) + 
  971. year * 365 - ((year & 3) == 0 && month < 2 ? 1 : 0) + 3653);
  972. /* days since 1.1.70 plus 80's leap day */
  973. return local2utc(secs);
  974. }
  975. /* Convert linear UNIX date to a MS-DOS time/date pair. */
  976. void
  977. ncp_date_unix2dos(int unix_date, unsigned short *time, unsigned short *date)
  978. {
  979. int day, year, nl_day, month;
  980. unix_date = utc2local(unix_date);
  981. *time = (unix_date % 60) / 2 + (((unix_date / 60) % 60) << 5) +
  982.     (((unix_date / 3600) % 24) << 11);
  983. day = unix_date / 86400 - 3652;
  984. year = day / 365;
  985. if ((year + 3) / 4 + 365 * year > day)
  986. year--;
  987. day -= (year + 3) / 4 + 365 * year;
  988. if (day == 59 && !(year & 3)) {
  989. nl_day = day;
  990. month = 2;
  991. } else {
  992. nl_day = (year & 3) || day <= 59 ? day : day - 1;
  993. for (month = 0; month < 12; month++)
  994. if (day_n[month] > nl_day)
  995. break;
  996. }
  997. *date = nl_day - day_n[month - 1] + 1 + (month << 5) + (year << 9);
  998. }