blob: b8b5b30d53f0c03a8c11c580bf29470225246293 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
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 * Modified 2000 Ben Harris, University of Cambridge for NFS NS meta-info
10 *
11 */
12
Linus Torvalds1da177e2005-04-16 15:20:36 -070013
14#include <linux/time.h>
15#include <linux/errno.h>
16#include <linux/stat.h>
17#include <linux/kernel.h>
18#include <linux/slab.h>
19#include <linux/vmalloc.h>
20#include <linux/mm.h>
21#include <asm/uaccess.h>
22#include <asm/byteorder.h>
23#include <linux/smp_lock.h>
24
25#include <linux/ncp_fs.h>
26
27#include "ncplib_kernel.h"
28
29static void ncp_read_volume_list(struct file *, void *, filldir_t,
30 struct ncp_cache_control *);
31static void ncp_do_readdir(struct file *, void *, filldir_t,
32 struct ncp_cache_control *);
33
34static int ncp_readdir(struct file *, void *, filldir_t);
35
36static int ncp_create(struct inode *, struct dentry *, int, struct nameidata *);
37static struct dentry *ncp_lookup(struct inode *, struct dentry *, struct nameidata *);
38static int ncp_unlink(struct inode *, struct dentry *);
39static int ncp_mkdir(struct inode *, struct dentry *, int);
40static int ncp_rmdir(struct inode *, struct dentry *);
41static int ncp_rename(struct inode *, struct dentry *,
42 struct inode *, struct dentry *);
43static int ncp_mknod(struct inode * dir, struct dentry *dentry,
44 int mode, dev_t rdev);
45#if defined(CONFIG_NCPFS_EXTRAS) || defined(CONFIG_NCPFS_NFS_NS)
46extern int ncp_symlink(struct inode *, struct dentry *, const char *);
47#else
48#define ncp_symlink NULL
49#endif
50
Arjan van de Ven4b6f5d22006-03-28 01:56:42 -080051const struct file_operations ncp_dir_operations =
Linus Torvalds1da177e2005-04-16 15:20:36 -070052{
53 .read = generic_read_dir,
54 .readdir = ncp_readdir,
55 .ioctl = ncp_ioctl,
Petr Vandrovec54f67f62006-09-30 23:27:55 -070056#ifdef CONFIG_COMPAT
57 .compat_ioctl = ncp_compat_ioctl,
58#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070059};
60
Arjan van de Ven92e1d5b2007-02-12 00:55:39 -080061const struct inode_operations ncp_dir_inode_operations =
Linus Torvalds1da177e2005-04-16 15:20:36 -070062{
63 .create = ncp_create,
64 .lookup = ncp_lookup,
65 .unlink = ncp_unlink,
66 .symlink = ncp_symlink,
67 .mkdir = ncp_mkdir,
68 .rmdir = ncp_rmdir,
69 .mknod = ncp_mknod,
70 .rename = ncp_rename,
71 .setattr = ncp_notify_change,
72};
73
74/*
75 * Dentry operations routines
76 */
77static int ncp_lookup_validate(struct dentry *, struct nameidata *);
78static int ncp_hash_dentry(struct dentry *, struct qstr *);
79static int ncp_compare_dentry (struct dentry *, struct qstr *, struct qstr *);
80static int ncp_delete_dentry(struct dentry *);
81
Al Viroe16404e2009-02-20 05:55:13 +000082static const struct dentry_operations ncp_dentry_operations =
Linus Torvalds1da177e2005-04-16 15:20:36 -070083{
84 .d_revalidate = ncp_lookup_validate,
85 .d_hash = ncp_hash_dentry,
86 .d_compare = ncp_compare_dentry,
87 .d_delete = ncp_delete_dentry,
88};
89
Al Viroe16404e2009-02-20 05:55:13 +000090const struct dentry_operations ncp_root_dentry_operations =
Linus Torvalds1da177e2005-04-16 15:20:36 -070091{
92 .d_hash = ncp_hash_dentry,
93 .d_compare = ncp_compare_dentry,
94 .d_delete = ncp_delete_dentry,
95};
96
97
98/*
99 * Note: leave the hash unchanged if the directory
100 * is case-sensitive.
101 */
102static int
103ncp_hash_dentry(struct dentry *dentry, struct qstr *this)
104{
105 struct nls_table *t;
106 unsigned long hash;
107 int i;
108
109 t = NCP_IO_TABLE(dentry);
110
111 if (!ncp_case_sensitive(dentry->d_inode)) {
112 hash = init_name_hash();
113 for (i=0; i<this->len ; i++)
114 hash = partial_name_hash(ncp_tolower(t, this->name[i]),
115 hash);
116 this->hash = end_name_hash(hash);
117 }
118 return 0;
119}
120
121static int
122ncp_compare_dentry(struct dentry *dentry, struct qstr *a, struct qstr *b)
123{
124 if (a->len != b->len)
125 return 1;
126
127 if (ncp_case_sensitive(dentry->d_inode))
128 return strncmp(a->name, b->name, a->len);
129
130 return ncp_strnicmp(NCP_IO_TABLE(dentry), a->name, b->name, a->len);
131}
132
133/*
134 * This is the callback from dput() when d_count is going to 0.
135 * We use this to unhash dentries with bad inodes.
136 * Closing files can be safely postponed until iput() - it's done there anyway.
137 */
138static int
139ncp_delete_dentry(struct dentry * dentry)
140{
141 struct inode *inode = dentry->d_inode;
142
143 if (inode) {
144 if (is_bad_inode(inode))
145 return 1;
146 } else
147 {
148 /* N.B. Unhash negative dentries? */
149 }
150 return 0;
151}
152
153static inline int
154ncp_single_volume(struct ncp_server *server)
155{
156 return (server->m.mounted_vol[0] != '\0');
157}
158
159static inline int ncp_is_server_root(struct inode *inode)
160{
161 return (!ncp_single_volume(NCP_SERVER(inode)) &&
162 inode == inode->i_sb->s_root->d_inode);
163}
164
165
166/*
167 * This is the callback when the dcache has a lookup hit.
168 */
169
170
171#ifdef CONFIG_NCPFS_STRONG
172/* try to delete a readonly file (NW R bit set) */
173
174static int
175ncp_force_unlink(struct inode *dir, struct dentry* dentry)
176{
177 int res=0x9c,res2;
178 struct nw_modify_dos_info info;
179 __le32 old_nwattr;
180 struct inode *inode;
181
182 memset(&info, 0, sizeof(info));
183
184 /* remove the Read-Only flag on the NW server */
185 inode = dentry->d_inode;
186
187 old_nwattr = NCP_FINFO(inode)->nwattr;
188 info.attributes = old_nwattr & ~(aRONLY|aDELETEINHIBIT|aRENAMEINHIBIT);
189 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(inode), inode, NULL, DM_ATTRIBUTES, &info);
190 if (res2)
191 goto leave_me;
192
193 /* now try again the delete operation */
194 res = ncp_del_file_or_subdir2(NCP_SERVER(dir), dentry);
195
196 if (res) /* delete failed, set R bit again */
197 {
198 info.attributes = old_nwattr;
199 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(inode), inode, NULL, DM_ATTRIBUTES, &info);
200 if (res2)
201 goto leave_me;
202 }
203leave_me:
204 return(res);
205}
206#endif /* CONFIG_NCPFS_STRONG */
207
208#ifdef CONFIG_NCPFS_STRONG
209static int
210ncp_force_rename(struct inode *old_dir, struct dentry* old_dentry, char *_old_name,
211 struct inode *new_dir, struct dentry* new_dentry, char *_new_name)
212{
213 struct nw_modify_dos_info info;
214 int res=0x90,res2;
215 struct inode *old_inode = old_dentry->d_inode;
216 __le32 old_nwattr = NCP_FINFO(old_inode)->nwattr;
217 __le32 new_nwattr = 0; /* shut compiler warning */
218 int old_nwattr_changed = 0;
219 int new_nwattr_changed = 0;
220
221 memset(&info, 0, sizeof(info));
222
223 /* remove the Read-Only flag on the NW server */
224
225 info.attributes = old_nwattr & ~(aRONLY|aRENAMEINHIBIT|aDELETEINHIBIT);
226 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(old_inode), old_inode, NULL, DM_ATTRIBUTES, &info);
227 if (!res2)
228 old_nwattr_changed = 1;
229 if (new_dentry && new_dentry->d_inode) {
230 new_nwattr = NCP_FINFO(new_dentry->d_inode)->nwattr;
231 info.attributes = new_nwattr & ~(aRONLY|aRENAMEINHIBIT|aDELETEINHIBIT);
232 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(new_dir), new_dir, _new_name, DM_ATTRIBUTES, &info);
233 if (!res2)
234 new_nwattr_changed = 1;
235 }
236 /* now try again the rename operation */
237 /* but only if something really happened */
238 if (new_nwattr_changed || old_nwattr_changed) {
239 res = ncp_ren_or_mov_file_or_subdir(NCP_SERVER(old_dir),
240 old_dir, _old_name,
241 new_dir, _new_name);
242 }
243 if (res)
244 goto leave_me;
245 /* file was successfully renamed, so:
246 do not set attributes on old file - it no longer exists
247 copy attributes from old file to new */
248 new_nwattr_changed = old_nwattr_changed;
249 new_nwattr = old_nwattr;
250 old_nwattr_changed = 0;
251
252leave_me:;
253 if (old_nwattr_changed) {
254 info.attributes = old_nwattr;
255 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(old_inode), old_inode, NULL, DM_ATTRIBUTES, &info);
256 /* ignore errors */
257 }
258 if (new_nwattr_changed) {
259 info.attributes = new_nwattr;
260 res2 = ncp_modify_file_or_subdir_dos_info_path(NCP_SERVER(new_dir), new_dir, _new_name, DM_ATTRIBUTES, &info);
261 /* ignore errors */
262 }
263 return(res);
264}
265#endif /* CONFIG_NCPFS_STRONG */
266
267
268static int
Al Viro58ec42b2008-07-23 14:45:55 -0400269__ncp_lookup_validate(struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700270{
271 struct ncp_server *server;
272 struct dentry *parent;
273 struct inode *dir;
274 struct ncp_entry_info finfo;
275 int res, val = 0, len;
276 __u8 __name[NCP_MAXPATHLEN + 1];
277
278 parent = dget_parent(dentry);
279 dir = parent->d_inode;
280
281 if (!dentry->d_inode)
282 goto finished;
283
284 server = NCP_SERVER(dir);
285
286 if (!ncp_conn_valid(server))
287 goto finished;
288
289 /*
290 * Inspired by smbfs:
291 * The default validation is based on dentry age:
292 * We set the max age at mount time. (But each
293 * successful server lookup renews the timestamp.)
294 */
295 val = NCP_TEST_AGE(server, dentry);
296 if (val)
297 goto finished;
298
299 DDPRINTK("ncp_lookup_validate: %s/%s not valid, age=%ld, server lookup\n",
300 dentry->d_parent->d_name.name, dentry->d_name.name,
301 NCP_GET_AGE(dentry));
302
303 len = sizeof(__name);
304 if (ncp_is_server_root(dir)) {
305 res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
306 dentry->d_name.len, 1);
307 if (!res)
308 res = ncp_lookup_volume(server, __name, &(finfo.i));
309 } else {
310 res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
311 dentry->d_name.len, !ncp_preserve_case(dir));
312 if (!res)
313 res = ncp_obtain_info(server, dir, __name, &(finfo.i));
314 }
315 finfo.volume = finfo.i.volNumber;
316 DDPRINTK("ncp_lookup_validate: looked for %s/%s, res=%d\n",
317 dentry->d_parent->d_name.name, __name, res);
318 /*
319 * If we didn't find it, or if it has a different dirEntNum to
320 * what we remember, it's not valid any more.
321 */
322 if (!res) {
323 if (finfo.i.dirEntNum == NCP_FINFO(dentry->d_inode)->dirEntNum) {
324 ncp_new_dentry(dentry);
325 val=1;
326 } else
327 DDPRINTK("ncp_lookup_validate: found, but dirEntNum changed\n");
328
329 ncp_update_inode2(dentry->d_inode, &finfo);
330 }
331
332finished:
333 DDPRINTK("ncp_lookup_validate: result=%d\n", val);
334 dput(parent);
335 return val;
336}
337
338static int
339ncp_lookup_validate(struct dentry * dentry, struct nameidata *nd)
340{
341 int res;
342 lock_kernel();
Al Viro58ec42b2008-07-23 14:45:55 -0400343 res = __ncp_lookup_validate(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700344 unlock_kernel();
345 return res;
346}
347
348static struct dentry *
349ncp_dget_fpos(struct dentry *dentry, struct dentry *parent, unsigned long fpos)
350{
351 struct dentry *dent = dentry;
352 struct list_head *next;
353
354 if (d_validate(dent, parent)) {
355 if (dent->d_name.len <= NCP_MAXPATHLEN &&
356 (unsigned long)dent->d_fsdata == fpos) {
357 if (!dent->d_inode) {
358 dput(dent);
359 dent = NULL;
360 }
361 return dent;
362 }
363 dput(dent);
364 }
365
366 /* If a pointer is invalid, we search the dentry. */
367 spin_lock(&dcache_lock);
368 next = parent->d_subdirs.next;
369 while (next != &parent->d_subdirs) {
Eric Dumazet5160ee62006-01-08 01:03:32 -0800370 dent = list_entry(next, struct dentry, d_u.d_child);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371 if ((unsigned long)dent->d_fsdata == fpos) {
372 if (dent->d_inode)
373 dget_locked(dent);
374 else
375 dent = NULL;
376 spin_unlock(&dcache_lock);
377 goto out;
378 }
379 next = next->next;
380 }
381 spin_unlock(&dcache_lock);
382 return NULL;
383
384out:
385 return dent;
386}
387
388static time_t ncp_obtain_mtime(struct dentry *dentry)
389{
390 struct inode *inode = dentry->d_inode;
391 struct ncp_server *server = NCP_SERVER(inode);
392 struct nw_info_struct i;
393
394 if (!ncp_conn_valid(server) || ncp_is_server_root(inode))
395 return 0;
396
397 if (ncp_obtain_info(server, inode, NULL, &i))
398 return 0;
399
400 return ncp_date_dos2unix(i.modifyTime, i.modifyDate);
401}
402
403static int ncp_readdir(struct file *filp, void *dirent, filldir_t filldir)
404{
Josef Sipek92e5bae2006-12-08 02:37:22 -0800405 struct dentry *dentry = filp->f_path.dentry;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700406 struct inode *inode = dentry->d_inode;
407 struct page *page = NULL;
408 struct ncp_server *server = NCP_SERVER(inode);
409 union ncp_dir_cache *cache = NULL;
410 struct ncp_cache_control ctl;
411 int result, mtime_valid = 0;
412 time_t mtime = 0;
413
414 lock_kernel();
415
416 ctl.page = NULL;
417 ctl.cache = NULL;
418
419 DDPRINTK("ncp_readdir: reading %s/%s, pos=%d\n",
420 dentry->d_parent->d_name.name, dentry->d_name.name,
421 (int) filp->f_pos);
422
423 result = -EIO;
424 if (!ncp_conn_valid(server))
425 goto out;
426
427 result = 0;
428 if (filp->f_pos == 0) {
429 if (filldir(dirent, ".", 1, 0, inode->i_ino, DT_DIR))
430 goto out;
431 filp->f_pos = 1;
432 }
433 if (filp->f_pos == 1) {
434 if (filldir(dirent, "..", 2, 1, parent_ino(dentry), DT_DIR))
435 goto out;
436 filp->f_pos = 2;
437 }
438
439 page = grab_cache_page(&inode->i_data, 0);
440 if (!page)
441 goto read_really;
442
443 ctl.cache = cache = kmap(page);
444 ctl.head = cache->head;
445
446 if (!PageUptodate(page) || !ctl.head.eof)
447 goto init_cache;
448
449 if (filp->f_pos == 2) {
450 if (jiffies - ctl.head.time >= NCP_MAX_AGE(server))
451 goto init_cache;
452
453 mtime = ncp_obtain_mtime(dentry);
454 mtime_valid = 1;
455 if ((!mtime) || (mtime != ctl.head.mtime))
456 goto init_cache;
457 }
458
459 if (filp->f_pos > ctl.head.end)
460 goto finished;
461
462 ctl.fpos = filp->f_pos + (NCP_DIRCACHE_START - 2);
463 ctl.ofs = ctl.fpos / NCP_DIRCACHE_SIZE;
464 ctl.idx = ctl.fpos % NCP_DIRCACHE_SIZE;
465
466 for (;;) {
467 if (ctl.ofs != 0) {
468 ctl.page = find_lock_page(&inode->i_data, ctl.ofs);
469 if (!ctl.page)
470 goto invalid_cache;
471 ctl.cache = kmap(ctl.page);
472 if (!PageUptodate(ctl.page))
473 goto invalid_cache;
474 }
475 while (ctl.idx < NCP_DIRCACHE_SIZE) {
476 struct dentry *dent;
477 int res;
478
479 dent = ncp_dget_fpos(ctl.cache->dentry[ctl.idx],
480 dentry, filp->f_pos);
481 if (!dent)
482 goto invalid_cache;
483 res = filldir(dirent, dent->d_name.name,
484 dent->d_name.len, filp->f_pos,
485 dent->d_inode->i_ino, DT_UNKNOWN);
486 dput(dent);
487 if (res)
488 goto finished;
489 filp->f_pos += 1;
490 ctl.idx += 1;
491 if (filp->f_pos > ctl.head.end)
492 goto finished;
493 }
494 if (ctl.page) {
495 kunmap(ctl.page);
496 SetPageUptodate(ctl.page);
497 unlock_page(ctl.page);
498 page_cache_release(ctl.page);
499 ctl.page = NULL;
500 }
501 ctl.idx = 0;
502 ctl.ofs += 1;
503 }
504invalid_cache:
505 if (ctl.page) {
506 kunmap(ctl.page);
507 unlock_page(ctl.page);
508 page_cache_release(ctl.page);
509 ctl.page = NULL;
510 }
511 ctl.cache = cache;
512init_cache:
513 ncp_invalidate_dircache_entries(dentry);
514 if (!mtime_valid) {
515 mtime = ncp_obtain_mtime(dentry);
516 mtime_valid = 1;
517 }
518 ctl.head.mtime = mtime;
519 ctl.head.time = jiffies;
520 ctl.head.eof = 0;
521 ctl.fpos = 2;
522 ctl.ofs = 0;
523 ctl.idx = NCP_DIRCACHE_START;
524 ctl.filled = 0;
525 ctl.valid = 1;
526read_really:
527 if (ncp_is_server_root(inode)) {
528 ncp_read_volume_list(filp, dirent, filldir, &ctl);
529 } else {
530 ncp_do_readdir(filp, dirent, filldir, &ctl);
531 }
532 ctl.head.end = ctl.fpos - 1;
533 ctl.head.eof = ctl.valid;
534finished:
535 if (page) {
536 cache->head = ctl.head;
537 kunmap(page);
538 SetPageUptodate(page);
539 unlock_page(page);
540 page_cache_release(page);
541 }
542 if (ctl.page) {
543 kunmap(ctl.page);
544 SetPageUptodate(ctl.page);
545 unlock_page(ctl.page);
546 page_cache_release(ctl.page);
547 }
548out:
549 unlock_kernel();
550 return result;
551}
552
553static int
554ncp_fill_cache(struct file *filp, void *dirent, filldir_t filldir,
555 struct ncp_cache_control *ctrl, struct ncp_entry_info *entry)
556{
Josef Sipek92e5bae2006-12-08 02:37:22 -0800557 struct dentry *newdent, *dentry = filp->f_path.dentry;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558 struct inode *newino, *inode = dentry->d_inode;
559 struct ncp_cache_control ctl = *ctrl;
560 struct qstr qname;
561 int valid = 0;
562 int hashed = 0;
563 ino_t ino = 0;
564 __u8 __name[NCP_MAXPATHLEN + 1];
565
566 qname.len = sizeof(__name);
567 if (ncp_vol2io(NCP_SERVER(inode), __name, &qname.len,
568 entry->i.entryName, entry->i.nameLen,
569 !ncp_preserve_entry_case(inode, entry->i.NSCreator)))
570 return 1; /* I'm not sure */
571
572 qname.name = __name;
573 qname.hash = full_name_hash(qname.name, qname.len);
574
575 if (dentry->d_op && dentry->d_op->d_hash)
576 if (dentry->d_op->d_hash(dentry, &qname) != 0)
577 goto end_advance;
578
579 newdent = d_lookup(dentry, &qname);
580
581 if (!newdent) {
582 newdent = d_alloc(dentry, &qname);
583 if (!newdent)
584 goto end_advance;
585 } else {
586 hashed = 1;
587 memcpy((char *) newdent->d_name.name, qname.name,
588 newdent->d_name.len);
589 }
590
591 if (!newdent->d_inode) {
592 entry->opened = 0;
593 entry->ino = iunique(inode->i_sb, 2);
594 newino = ncp_iget(inode->i_sb, entry);
595 if (newino) {
596 newdent->d_op = &ncp_dentry_operations;
597 d_instantiate(newdent, newino);
598 if (!hashed)
599 d_rehash(newdent);
600 }
601 } else
602 ncp_update_inode2(newdent->d_inode, entry);
603
604 if (newdent->d_inode) {
605 ino = newdent->d_inode->i_ino;
606 newdent->d_fsdata = (void *) ctl.fpos;
607 ncp_new_dentry(newdent);
608 }
609
610 if (ctl.idx >= NCP_DIRCACHE_SIZE) {
611 if (ctl.page) {
612 kunmap(ctl.page);
613 SetPageUptodate(ctl.page);
614 unlock_page(ctl.page);
615 page_cache_release(ctl.page);
616 }
617 ctl.cache = NULL;
618 ctl.idx -= NCP_DIRCACHE_SIZE;
619 ctl.ofs += 1;
620 ctl.page = grab_cache_page(&inode->i_data, ctl.ofs);
621 if (ctl.page)
622 ctl.cache = kmap(ctl.page);
623 }
624 if (ctl.cache) {
625 ctl.cache->dentry[ctl.idx] = newdent;
626 valid = 1;
627 }
628 dput(newdent);
629end_advance:
630 if (!valid)
631 ctl.valid = 0;
632 if (!ctl.filled && (ctl.fpos == filp->f_pos)) {
633 if (!ino)
634 ino = find_inode_number(dentry, &qname);
635 if (!ino)
636 ino = iunique(inode->i_sb, 2);
637 ctl.filled = filldir(dirent, qname.name, qname.len,
638 filp->f_pos, ino, DT_UNKNOWN);
639 if (!ctl.filled)
640 filp->f_pos += 1;
641 }
642 ctl.fpos += 1;
643 ctl.idx += 1;
644 *ctrl = ctl;
645 return (ctl.valid || !ctl.filled);
646}
647
648static void
649ncp_read_volume_list(struct file *filp, void *dirent, filldir_t filldir,
650 struct ncp_cache_control *ctl)
651{
Josef Sipek92e5bae2006-12-08 02:37:22 -0800652 struct dentry *dentry = filp->f_path.dentry;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700653 struct inode *inode = dentry->d_inode;
654 struct ncp_server *server = NCP_SERVER(inode);
655 struct ncp_volume_info info;
656 struct ncp_entry_info entry;
657 int i;
658
659 DPRINTK("ncp_read_volume_list: pos=%ld\n",
660 (unsigned long) filp->f_pos);
661
662 for (i = 0; i < NCP_NUMBER_OF_VOLUMES; i++) {
663
664 if (ncp_get_volume_info_with_number(server, i, &info) != 0)
665 return;
666 if (!strlen(info.volume_name))
667 continue;
668
669 DPRINTK("ncp_read_volume_list: found vol: %s\n",
670 info.volume_name);
671
672 if (ncp_lookup_volume(server, info.volume_name,
673 &entry.i)) {
674 DPRINTK("ncpfs: could not lookup vol %s\n",
675 info.volume_name);
676 continue;
677 }
678 entry.volume = entry.i.volNumber;
679 if (!ncp_fill_cache(filp, dirent, filldir, ctl, &entry))
680 return;
681 }
682}
683
684static void
685ncp_do_readdir(struct file *filp, void *dirent, filldir_t filldir,
686 struct ncp_cache_control *ctl)
687{
Josef Sipek92e5bae2006-12-08 02:37:22 -0800688 struct dentry *dentry = filp->f_path.dentry;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700689 struct inode *dir = dentry->d_inode;
690 struct ncp_server *server = NCP_SERVER(dir);
691 struct nw_search_sequence seq;
692 struct ncp_entry_info entry;
693 int err;
694 void* buf;
695 int more;
696 size_t bufsize;
697
698 DPRINTK("ncp_do_readdir: %s/%s, fpos=%ld\n",
699 dentry->d_parent->d_name.name, dentry->d_name.name,
700 (unsigned long) filp->f_pos);
701 PPRINTK("ncp_do_readdir: init %s, volnum=%d, dirent=%u\n",
702 dentry->d_name.name, NCP_FINFO(dir)->volNumber,
703 NCP_FINFO(dir)->dirEntNum);
704
705 err = ncp_initialize_search(server, dir, &seq);
706 if (err) {
707 DPRINTK("ncp_do_readdir: init failed, err=%d\n", err);
708 return;
709 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700710 /* We MUST NOT use server->buffer_size handshaked with server if we are
711 using UDP, as for UDP server uses max. buffer size determined by
712 MTU, and for TCP server uses hardwired value 65KB (== 66560 bytes).
713 So we use 128KB, just to be sure, as there is no way how to know
714 this value in advance. */
715 bufsize = 131072;
716 buf = vmalloc(bufsize);
717 if (!buf)
718 return;
719 do {
720 int cnt;
721 char* rpl;
722 size_t rpls;
723
724 err = ncp_search_for_fileset(server, &seq, &more, &cnt, buf, bufsize, &rpl, &rpls);
725 if (err) /* Error */
726 break;
727 if (!cnt) /* prevent endless loop */
728 break;
729 while (cnt--) {
730 size_t onerpl;
731
732 if (rpls < offsetof(struct nw_info_struct, entryName))
733 break; /* short packet */
734 ncp_extract_file_info(rpl, &entry.i);
735 onerpl = offsetof(struct nw_info_struct, entryName) + entry.i.nameLen;
736 if (rpls < onerpl)
737 break; /* short packet */
738 (void)ncp_obtain_nfs_info(server, &entry.i);
739 rpl += onerpl;
740 rpls -= onerpl;
741 entry.volume = entry.i.volNumber;
742 if (!ncp_fill_cache(filp, dirent, filldir, ctl, &entry))
743 break;
744 }
745 } while (more);
746 vfree(buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700747 return;
748}
749
750int ncp_conn_logged_in(struct super_block *sb)
751{
752 struct ncp_server* server = NCP_SBP(sb);
753 int result;
754
755 if (ncp_single_volume(server)) {
756 int len;
757 struct dentry* dent;
758 __u32 volNumber;
759 __le32 dirEntNum;
760 __le32 DosDirNum;
761 __u8 __name[NCP_MAXPATHLEN + 1];
762
763 len = sizeof(__name);
764 result = ncp_io2vol(server, __name, &len, server->m.mounted_vol,
765 strlen(server->m.mounted_vol), 1);
766 if (result)
767 goto out;
768 result = -ENOENT;
769 if (ncp_get_volume_root(server, __name, &volNumber, &dirEntNum, &DosDirNum)) {
770 PPRINTK("ncp_conn_logged_in: %s not found\n",
771 server->m.mounted_vol);
772 goto out;
773 }
774 dent = sb->s_root;
775 if (dent) {
776 struct inode* ino = dent->d_inode;
777 if (ino) {
778 NCP_FINFO(ino)->volNumber = volNumber;
779 NCP_FINFO(ino)->dirEntNum = dirEntNum;
780 NCP_FINFO(ino)->DosDirNum = DosDirNum;
781 } else {
782 DPRINTK("ncpfs: sb->s_root->d_inode == NULL!\n");
783 }
784 } else {
785 DPRINTK("ncpfs: sb->s_root == NULL!\n");
786 }
787 }
788 result = 0;
789
790out:
791 return result;
792}
793
794static struct dentry *ncp_lookup(struct inode *dir, struct dentry *dentry, struct nameidata *nd)
795{
796 struct ncp_server *server = NCP_SERVER(dir);
797 struct inode *inode = NULL;
798 struct ncp_entry_info finfo;
799 int error, res, len;
800 __u8 __name[NCP_MAXPATHLEN + 1];
801
802 lock_kernel();
803 error = -EIO;
804 if (!ncp_conn_valid(server))
805 goto finished;
806
807 PPRINTK("ncp_lookup: server lookup for %s/%s\n",
808 dentry->d_parent->d_name.name, dentry->d_name.name);
809
810 len = sizeof(__name);
811 if (ncp_is_server_root(dir)) {
812 res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
813 dentry->d_name.len, 1);
814 if (!res)
815 res = ncp_lookup_volume(server, __name, &(finfo.i));
816 } else {
817 res = ncp_io2vol(server, __name, &len, dentry->d_name.name,
818 dentry->d_name.len, !ncp_preserve_case(dir));
819 if (!res)
820 res = ncp_obtain_info(server, dir, __name, &(finfo.i));
821 }
822 PPRINTK("ncp_lookup: looked for %s/%s, res=%d\n",
823 dentry->d_parent->d_name.name, __name, res);
824 /*
825 * If we didn't find an entry, make a negative dentry.
826 */
827 if (res)
828 goto add_entry;
829
830 /*
831 * Create an inode for the entry.
832 */
833 finfo.opened = 0;
834 finfo.ino = iunique(dir->i_sb, 2);
835 finfo.volume = finfo.i.volNumber;
836 error = -EACCES;
837 inode = ncp_iget(dir->i_sb, &finfo);
838
839 if (inode) {
840 ncp_new_dentry(dentry);
841add_entry:
842 dentry->d_op = &ncp_dentry_operations;
843 d_add(dentry, inode);
844 error = 0;
845 }
846
847finished:
848 PPRINTK("ncp_lookup: result=%d\n", error);
849 unlock_kernel();
850 return ERR_PTR(error);
851}
852
853/*
854 * This code is common to create, mkdir, and mknod.
855 */
856static int ncp_instantiate(struct inode *dir, struct dentry *dentry,
857 struct ncp_entry_info *finfo)
858{
859 struct inode *inode;
860 int error = -EINVAL;
861
862 finfo->ino = iunique(dir->i_sb, 2);
863 inode = ncp_iget(dir->i_sb, finfo);
864 if (!inode)
865 goto out_close;
866 d_instantiate(dentry,inode);
867 error = 0;
868out:
869 return error;
870
871out_close:
872 PPRINTK("ncp_instantiate: %s/%s failed, closing file\n",
873 dentry->d_parent->d_name.name, dentry->d_name.name);
874 ncp_close_file(NCP_SERVER(dir), finfo->file_handle);
875 goto out;
876}
877
878int ncp_create_new(struct inode *dir, struct dentry *dentry, int mode,
879 dev_t rdev, __le32 attributes)
880{
881 struct ncp_server *server = NCP_SERVER(dir);
882 struct ncp_entry_info finfo;
883 int error, result, len;
884 int opmode;
885 __u8 __name[NCP_MAXPATHLEN + 1];
886
887 PPRINTK("ncp_create_new: creating %s/%s, mode=%x\n",
888 dentry->d_parent->d_name.name, dentry->d_name.name, mode);
889
890 error = -EIO;
891 lock_kernel();
892 if (!ncp_conn_valid(server))
893 goto out;
894
895 ncp_age_dentry(server, dentry);
896 len = sizeof(__name);
897 error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
898 dentry->d_name.len, !ncp_preserve_case(dir));
899 if (error)
900 goto out;
901
902 error = -EACCES;
903
904 if (S_ISREG(mode) &&
905 (server->m.flags & NCP_MOUNT_EXTRAS) &&
906 (mode & S_IXUGO))
907 attributes |= aSYSTEM | aSHARED;
908
909 result = ncp_open_create_file_or_subdir(server, dir, __name,
910 OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE,
911 attributes, AR_READ | AR_WRITE, &finfo);
912 opmode = O_RDWR;
913 if (result) {
914 result = ncp_open_create_file_or_subdir(server, dir, __name,
915 OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE,
916 attributes, AR_WRITE, &finfo);
917 if (result) {
918 if (result == 0x87)
919 error = -ENAMETOOLONG;
920 DPRINTK("ncp_create: %s/%s failed\n",
921 dentry->d_parent->d_name.name, dentry->d_name.name);
922 goto out;
923 }
924 opmode = O_WRONLY;
925 }
926 finfo.access = opmode;
927 if (ncp_is_nfs_extras(server, finfo.volume)) {
928 finfo.i.nfs.mode = mode;
929 finfo.i.nfs.rdev = new_encode_dev(rdev);
930 if (ncp_modify_nfs_info(server, finfo.volume,
931 finfo.i.dirEntNum,
932 mode, new_encode_dev(rdev)) != 0)
933 goto out;
934 }
935
936 error = ncp_instantiate(dir, dentry, &finfo);
937out:
938 unlock_kernel();
939 return error;
940}
941
942static int ncp_create(struct inode *dir, struct dentry *dentry, int mode,
943 struct nameidata *nd)
944{
945 return ncp_create_new(dir, dentry, mode, 0, 0);
946}
947
948static int ncp_mkdir(struct inode *dir, struct dentry *dentry, int mode)
949{
950 struct ncp_entry_info finfo;
951 struct ncp_server *server = NCP_SERVER(dir);
952 int error, len;
953 __u8 __name[NCP_MAXPATHLEN + 1];
954
955 DPRINTK("ncp_mkdir: making %s/%s\n",
956 dentry->d_parent->d_name.name, dentry->d_name.name);
957
958 error = -EIO;
959 lock_kernel();
960 if (!ncp_conn_valid(server))
961 goto out;
962
963 ncp_age_dentry(server, dentry);
964 len = sizeof(__name);
965 error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
966 dentry->d_name.len, !ncp_preserve_case(dir));
967 if (error)
968 goto out;
969
970 error = -EACCES;
971 if (ncp_open_create_file_or_subdir(server, dir, __name,
972 OC_MODE_CREATE, aDIR,
973 cpu_to_le16(0xffff),
974 &finfo) == 0)
975 {
976 if (ncp_is_nfs_extras(server, finfo.volume)) {
977 mode |= S_IFDIR;
978 finfo.i.nfs.mode = mode;
979 if (ncp_modify_nfs_info(server,
980 finfo.volume,
981 finfo.i.dirEntNum,
982 mode, 0) != 0)
983 goto out;
984 }
985 error = ncp_instantiate(dir, dentry, &finfo);
986 }
987out:
988 unlock_kernel();
989 return error;
990}
991
992static int ncp_rmdir(struct inode *dir, struct dentry *dentry)
993{
994 struct ncp_server *server = NCP_SERVER(dir);
995 int error, result, len;
996 __u8 __name[NCP_MAXPATHLEN + 1];
997
998 DPRINTK("ncp_rmdir: removing %s/%s\n",
999 dentry->d_parent->d_name.name, dentry->d_name.name);
1000
1001 error = -EIO;
1002 lock_kernel();
1003 if (!ncp_conn_valid(server))
1004 goto out;
1005
1006 error = -EBUSY;
1007 if (!d_unhashed(dentry))
1008 goto out;
1009
1010 len = sizeof(__name);
1011 error = ncp_io2vol(server, __name, &len, dentry->d_name.name,
1012 dentry->d_name.len, !ncp_preserve_case(dir));
1013 if (error)
1014 goto out;
1015
1016 result = ncp_del_file_or_subdir(server, dir, __name);
1017 switch (result) {
1018 case 0x00:
1019 error = 0;
1020 break;
1021 case 0x85: /* unauthorized to delete file */
1022 case 0x8A: /* unauthorized to delete file */
1023 error = -EACCES;
1024 break;
1025 case 0x8F:
1026 case 0x90: /* read only */
1027 error = -EPERM;
1028 break;
1029 case 0x9F: /* in use by another client */
1030 error = -EBUSY;
1031 break;
1032 case 0xA0: /* directory not empty */
1033 error = -ENOTEMPTY;
1034 break;
1035 case 0xFF: /* someone deleted file */
1036 error = -ENOENT;
1037 break;
1038 default:
1039 error = -EACCES;
1040 break;
1041 }
1042out:
1043 unlock_kernel();
1044 return error;
1045}
1046
1047static int ncp_unlink(struct inode *dir, struct dentry *dentry)
1048{
1049 struct inode *inode = dentry->d_inode;
1050 struct ncp_server *server;
1051 int error;
1052
1053 lock_kernel();
1054 server = NCP_SERVER(dir);
1055 DPRINTK("ncp_unlink: unlinking %s/%s\n",
1056 dentry->d_parent->d_name.name, dentry->d_name.name);
1057
1058 error = -EIO;
1059 if (!ncp_conn_valid(server))
1060 goto out;
1061
1062 /*
1063 * Check whether to close the file ...
1064 */
1065 if (inode) {
1066 PPRINTK("ncp_unlink: closing file\n");
1067 ncp_make_closed(inode);
1068 }
1069
1070 error = ncp_del_file_or_subdir2(server, dentry);
1071#ifdef CONFIG_NCPFS_STRONG
1072 /* 9C is Invalid path.. It should be 8F, 90 - read only, but
1073 it is not :-( */
1074 if ((error == 0x9C || error == 0x90) && server->m.flags & NCP_MOUNT_STRONG) { /* R/O */
1075 error = ncp_force_unlink(dir, dentry);
1076 }
1077#endif
1078 switch (error) {
1079 case 0x00:
1080 DPRINTK("ncp: removed %s/%s\n",
1081 dentry->d_parent->d_name.name, dentry->d_name.name);
1082 break;
1083 case 0x85:
1084 case 0x8A:
1085 error = -EACCES;
1086 break;
1087 case 0x8D: /* some files in use */
1088 case 0x8E: /* all files in use */
1089 error = -EBUSY;
1090 break;
1091 case 0x8F: /* some read only */
1092 case 0x90: /* all read only */
1093 case 0x9C: /* !!! returned when in-use or read-only by NW4 */
1094 error = -EPERM;
1095 break;
1096 case 0xFF:
1097 error = -ENOENT;
1098 break;
1099 default:
1100 error = -EACCES;
1101 break;
1102 }
1103
1104out:
1105 unlock_kernel();
1106 return error;
1107}
1108
1109static int ncp_rename(struct inode *old_dir, struct dentry *old_dentry,
1110 struct inode *new_dir, struct dentry *new_dentry)
1111{
1112 struct ncp_server *server = NCP_SERVER(old_dir);
1113 int error;
1114 int old_len, new_len;
1115 __u8 __old_name[NCP_MAXPATHLEN + 1], __new_name[NCP_MAXPATHLEN + 1];
1116
1117 DPRINTK("ncp_rename: %s/%s to %s/%s\n",
1118 old_dentry->d_parent->d_name.name, old_dentry->d_name.name,
1119 new_dentry->d_parent->d_name.name, new_dentry->d_name.name);
1120
1121 error = -EIO;
1122 lock_kernel();
1123 if (!ncp_conn_valid(server))
1124 goto out;
1125
1126 ncp_age_dentry(server, old_dentry);
1127 ncp_age_dentry(server, new_dentry);
1128
1129 old_len = sizeof(__old_name);
1130 error = ncp_io2vol(server, __old_name, &old_len,
1131 old_dentry->d_name.name, old_dentry->d_name.len,
1132 !ncp_preserve_case(old_dir));
1133 if (error)
1134 goto out;
1135
1136 new_len = sizeof(__new_name);
1137 error = ncp_io2vol(server, __new_name, &new_len,
1138 new_dentry->d_name.name, new_dentry->d_name.len,
1139 !ncp_preserve_case(new_dir));
1140 if (error)
1141 goto out;
1142
1143 error = ncp_ren_or_mov_file_or_subdir(server, old_dir, __old_name,
1144 new_dir, __new_name);
1145#ifdef CONFIG_NCPFS_STRONG
1146 if ((error == 0x90 || error == 0x8B || error == -EACCES) &&
1147 server->m.flags & NCP_MOUNT_STRONG) { /* RO */
1148 error = ncp_force_rename(old_dir, old_dentry, __old_name,
1149 new_dir, new_dentry, __new_name);
1150 }
1151#endif
1152 switch (error) {
1153 case 0x00:
1154 DPRINTK("ncp renamed %s -> %s.\n",
1155 old_dentry->d_name.name,new_dentry->d_name.name);
1156 break;
1157 case 0x9E:
1158 error = -ENAMETOOLONG;
1159 break;
1160 case 0xFF:
1161 error = -ENOENT;
1162 break;
1163 default:
1164 error = -EACCES;
1165 break;
1166 }
1167out:
1168 unlock_kernel();
1169 return error;
1170}
1171
1172static int ncp_mknod(struct inode * dir, struct dentry *dentry,
1173 int mode, dev_t rdev)
1174{
1175 if (!new_valid_dev(rdev))
1176 return -EINVAL;
1177 if (ncp_is_nfs_extras(NCP_SERVER(dir), NCP_FINFO(dir)->volNumber)) {
1178 DPRINTK(KERN_DEBUG "ncp_mknod: mode = 0%o\n", mode);
1179 return ncp_create_new(dir, dentry, mode, rdev, 0);
1180 }
1181 return -EPERM; /* Strange, but true */
1182}
1183
1184/* The following routines are taken directly from msdos-fs */
1185
1186/* Linear day numbers of the respective 1sts in non-leap years. */
1187
1188static int day_n[] =
1189{0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 0, 0, 0, 0};
1190/* Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec */
1191
1192
1193extern struct timezone sys_tz;
1194
1195static int utc2local(int time)
1196{
1197 return time - sys_tz.tz_minuteswest * 60;
1198}
1199
1200static int local2utc(int time)
1201{
1202 return time + sys_tz.tz_minuteswest * 60;
1203}
1204
1205/* Convert a MS-DOS time/date pair to a UNIX date (seconds since 1 1 70). */
1206int
1207ncp_date_dos2unix(__le16 t, __le16 d)
1208{
1209 unsigned short time = le16_to_cpu(t), date = le16_to_cpu(d);
1210 int month, year, secs;
1211
1212 /* first subtract and mask after that... Otherwise, if
1213 date == 0, bad things happen */
1214 month = ((date >> 5) - 1) & 15;
1215 year = date >> 9;
1216 secs = (time & 31) * 2 + 60 * ((time >> 5) & 63) + (time >> 11) * 3600 +
1217 86400 * ((date & 31) - 1 + day_n[month] + (year / 4) +
1218 year * 365 - ((year & 3) == 0 && month < 2 ? 1 : 0) + 3653);
1219 /* days since 1.1.70 plus 80's leap day */
1220 return local2utc(secs);
1221}
1222
1223
1224/* Convert linear UNIX date to a MS-DOS time/date pair. */
1225void
1226ncp_date_unix2dos(int unix_date, __le16 *time, __le16 *date)
1227{
1228 int day, year, nl_day, month;
1229
1230 unix_date = utc2local(unix_date);
1231 *time = cpu_to_le16(
1232 (unix_date % 60) / 2 + (((unix_date / 60) % 60) << 5) +
1233 (((unix_date / 3600) % 24) << 11));
1234 day = unix_date / 86400 - 3652;
1235 year = day / 365;
1236 if ((year + 3) / 4 + 365 * year > day)
1237 year--;
1238 day -= (year + 3) / 4 + 365 * year;
1239 if (day == 59 && !(year & 3)) {
1240 nl_day = day;
1241 month = 2;
1242 } else {
1243 nl_day = (year & 3) || day <= 59 ? day : day - 1;
Roel Kluinc5df5912009-09-22 16:45:54 -07001244 for (month = 1; month < 12; month++)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001245 if (day_n[month] > nl_day)
1246 break;
1247 }
1248 *date = cpu_to_le16(nl_day - day_n[month - 1] + 1 + (month << 5) + (year << 9));
1249}