📄 dir.c
字号:
finfo.volume = finfo.i.volNumber; error = -EACCES; inode = ncp_iget(dir->i_sb, &finfo); if (inode) { ncp_new_dentry(dentry);add_entry: dentry->d_op = &ncp_dentry_operations; d_add(dentry, inode); error = 0; }finished: PPRINTK("ncp_lookup: result=%d\n", error); unlock_kernel(); return ERR_PTR(error);}/* * This code is common to create, mkdir, and mknod. */static int ncp_instantiate(struct inode *dir, struct dentry *dentry, struct ncp_entry_info *finfo){ struct inode *inode; int error = -EINVAL; finfo->ino = iunique(dir->i_sb, 2); inode = ncp_iget(dir->i_sb, finfo); if (!inode) goto out_close; d_instantiate(dentry,inode); error = 0;out: return error;out_close: PPRINTK("ncp_instantiate: %s/%s failed, closing file\n", dentry->d_parent->d_name.name, dentry->d_name.name); ncp_close_file(NCP_SERVER(dir), finfo->file_handle); goto out;}int ncp_create_new(struct inode *dir, struct dentry *dentry, int mode, dev_t rdev, __le32 attributes){ struct ncp_server *server = NCP_SERVER(dir); struct ncp_entry_info finfo; int error, result, len; int opmode; __u8 __name[NCP_MAXPATHLEN + 1]; PPRINTK("ncp_create_new: creating %s/%s, mode=%x\n", dentry->d_parent->d_name.name, dentry->d_name.name, mode); error = -EIO; lock_kernel(); if (!ncp_conn_valid(server)) goto out; ncp_age_dentry(server, dentry); len = sizeof(__name); error = ncp_io2vol(server, __name, &len, dentry->d_name.name, dentry->d_name.len, !ncp_preserve_case(dir)); if (error) goto out; error = -EACCES; if (S_ISREG(mode) && (server->m.flags & NCP_MOUNT_EXTRAS) && (mode & S_IXUGO)) attributes |= aSYSTEM | aSHARED; result = ncp_open_create_file_or_subdir(server, dir, __name, OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE, attributes, AR_READ | AR_WRITE, &finfo); opmode = O_RDWR; if (result) { result = ncp_open_create_file_or_subdir(server, dir, __name, OC_MODE_CREATE | OC_MODE_OPEN | OC_MODE_REPLACE, attributes, AR_WRITE, &finfo); if (result) { if (result == 0x87) error = -ENAMETOOLONG; DPRINTK("ncp_create: %s/%s failed\n", dentry->d_parent->d_name.name, dentry->d_name.name); goto out; } opmode = O_WRONLY; } finfo.access = opmode; if (ncp_is_nfs_extras(server, finfo.volume)) { finfo.i.nfs.mode = mode; finfo.i.nfs.rdev = new_encode_dev(rdev); if (ncp_modify_nfs_info(server, finfo.volume, finfo.i.dirEntNum, mode, new_encode_dev(rdev)) != 0) goto out; } error = ncp_instantiate(dir, dentry, &finfo);out: unlock_kernel(); return error;}static int ncp_create(struct inode *dir, struct dentry *dentry, int mode, struct nameidata *nd){ return ncp_create_new(dir, dentry, mode, 0, 0);}static int ncp_mkdir(struct inode *dir, struct dentry *dentry, int mode){ struct ncp_entry_info finfo; struct ncp_server *server = NCP_SERVER(dir); int error, len; __u8 __name[NCP_MAXPATHLEN + 1]; DPRINTK("ncp_mkdir: making %s/%s\n", dentry->d_parent->d_name.name, dentry->d_name.name); error = -EIO; lock_kernel(); if (!ncp_conn_valid(server)) goto out; ncp_age_dentry(server, dentry); len = sizeof(__name); error = ncp_io2vol(server, __name, &len, dentry->d_name.name, dentry->d_name.len, !ncp_preserve_case(dir)); if (error) goto out; error = -EACCES; if (ncp_open_create_file_or_subdir(server, dir, __name, OC_MODE_CREATE, aDIR, cpu_to_le16(0xffff), &finfo) == 0) { if (ncp_is_nfs_extras(server, finfo.volume)) { mode |= S_IFDIR; finfo.i.nfs.mode = mode; if (ncp_modify_nfs_info(server, finfo.volume, finfo.i.dirEntNum, mode, 0) != 0) goto out; } error = ncp_instantiate(dir, dentry, &finfo); }out: unlock_kernel(); return error;}static int ncp_rmdir(struct inode *dir, struct dentry *dentry){ struct ncp_server *server = NCP_SERVER(dir); int error, result, len; __u8 __name[NCP_MAXPATHLEN + 1]; DPRINTK("ncp_rmdir: removing %s/%s\n", dentry->d_parent->d_name.name, dentry->d_name.name); error = -EIO; lock_kernel(); if (!ncp_conn_valid(server)) goto out; error = -EBUSY; if (!d_unhashed(dentry)) goto out; len = sizeof(__name); error = ncp_io2vol(server, __name, &len, dentry->d_name.name, dentry->d_name.len, !ncp_preserve_case(dir)); if (error) goto out; result = ncp_del_file_or_subdir(server, dir, __name); switch (result) { case 0x00: error = 0; break; case 0x85: /* unauthorized to delete file */ case 0x8A: /* unauthorized to delete file */ error = -EACCES; break; case 0x8F: case 0x90: /* read only */ error = -EPERM; break; case 0x9F: /* in use by another client */ error = -EBUSY; break; case 0xA0: /* directory not empty */ error = -ENOTEMPTY; break; case 0xFF: /* someone deleted file */ error = -ENOENT; break; default: error = -EACCES; break; }out: unlock_kernel(); return error;}static int ncp_unlink(struct inode *dir, struct dentry *dentry){ struct inode *inode = dentry->d_inode; struct ncp_server *server; int error; lock_kernel(); server = NCP_SERVER(dir); DPRINTK("ncp_unlink: unlinking %s/%s\n", dentry->d_parent->d_name.name, dentry->d_name.name); error = -EIO; if (!ncp_conn_valid(server)) goto out; /* * Check whether to close the file ... */ if (inode) { PPRINTK("ncp_unlink: closing file\n"); ncp_make_closed(inode); } error = ncp_del_file_or_subdir2(server, dentry);#ifdef CONFIG_NCPFS_STRONG /* 9C is Invalid path.. It should be 8F, 90 - read only, but it is not :-( */ if ((error == 0x9C || error == 0x90) && server->m.flags & NCP_MOUNT_STRONG) { /* R/O */ error = ncp_force_unlink(dir, dentry); }#endif switch (error) { case 0x00: DPRINTK("ncp: removed %s/%s\n", dentry->d_parent->d_name.name, dentry->d_name.name); break; case 0x85: case 0x8A: error = -EACCES; break; case 0x8D: /* some files in use */ case 0x8E: /* all files in use */ error = -EBUSY; break; case 0x8F: /* some read only */ case 0x90: /* all read only */ case 0x9C: /* !!! returned when in-use or read-only by NW4 */ error = -EPERM; break; case 0xFF: error = -ENOENT; break; default: error = -EACCES; break; } out: unlock_kernel(); return error;}static int ncp_rename(struct inode *old_dir, struct dentry *old_dentry, struct inode *new_dir, struct dentry *new_dentry){ struct ncp_server *server = NCP_SERVER(old_dir); int error; int old_len, new_len; __u8 __old_name[NCP_MAXPATHLEN + 1], __new_name[NCP_MAXPATHLEN + 1]; DPRINTK("ncp_rename: %s/%s to %s/%s\n", old_dentry->d_parent->d_name.name, old_dentry->d_name.name, new_dentry->d_parent->d_name.name, new_dentry->d_name.name); error = -EIO; lock_kernel(); if (!ncp_conn_valid(server)) goto out; ncp_age_dentry(server, old_dentry); ncp_age_dentry(server, new_dentry); old_len = sizeof(__old_name); error = ncp_io2vol(server, __old_name, &old_len, old_dentry->d_name.name, old_dentry->d_name.len, !ncp_preserve_case(old_dir)); if (error) goto out; new_len = sizeof(__new_name); error = ncp_io2vol(server, __new_name, &new_len, new_dentry->d_name.name, new_dentry->d_name.len, !ncp_preserve_case(new_dir)); if (error) goto out; error = ncp_ren_or_mov_file_or_subdir(server, old_dir, __old_name, new_dir, __new_name);#ifdef CONFIG_NCPFS_STRONG if ((error == 0x90 || error == 0x8B || error == -EACCES) && server->m.flags & NCP_MOUNT_STRONG) { /* RO */ error = ncp_force_rename(old_dir, old_dentry, __old_name, new_dir, new_dentry, __new_name); }#endif switch (error) { case 0x00: DPRINTK("ncp renamed %s -> %s.\n", old_dentry->d_name.name,new_dentry->d_name.name); break; case 0x9E: error = -ENAMETOOLONG; break; case 0xFF: error = -ENOENT; break; default: error = -EACCES; break; }out: unlock_kernel(); return error;}static int ncp_mknod(struct inode * dir, struct dentry *dentry, int mode, dev_t rdev){ if (!new_valid_dev(rdev)) return -EINVAL; if (ncp_is_nfs_extras(NCP_SERVER(dir), NCP_FINFO(dir)->volNumber)) { DPRINTK(KERN_DEBUG "ncp_mknod: mode = 0%o\n", mode); return ncp_create_new(dir, dentry, mode, rdev, 0); } return -EPERM; /* Strange, but true */}/* The following routines are taken directly from msdos-fs *//* Linear day numbers of the respective 1sts in non-leap years. */static int day_n[] ={0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 0, 0, 0, 0};/* Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec */extern struct timezone sys_tz;static int utc2local(int time){ return time - sys_tz.tz_minuteswest * 60;}static int local2utc(int time){ return time + sys_tz.tz_minuteswest * 60;}/* Convert a MS-DOS time/date pair to a UNIX date (seconds since 1 1 70). */intncp_date_dos2unix(__le16 t, __le16 d){ unsigned short time = le16_to_cpu(t), date = le16_to_cpu(d); int month, year, secs; /* first subtract and mask after that... Otherwise, if date == 0, bad things happen */ month = ((date >> 5) - 1) & 15; year = date >> 9; secs = (time & 31) * 2 + 60 * ((time >> 5) & 63) + (time >> 11) * 3600 + 86400 * ((date & 31) - 1 + day_n[month] + (year / 4) + year * 365 - ((year & 3) == 0 && month < 2 ? 1 : 0) + 3653); /* days since 1.1.70 plus 80's leap day */ return local2utc(secs);}/* Convert linear UNIX date to a MS-DOS time/date pair. */voidncp_date_unix2dos(int unix_date, __le16 *time, __le16 *date){ int day, year, nl_day, month; unix_date = utc2local(unix_date); *time = cpu_to_le16( (unix_date % 60) / 2 + (((unix_date / 60) % 60) << 5) + (((unix_date / 3600) % 24) << 11)); day = unix_date / 86400 - 3652; year = day / 365; if ((year + 3) / 4 + 365 * year > day) year--; day -= (year + 3) / 4 + 365 * year; if (day == 59 && !(year & 3)) { nl_day = day; month = 2; } else { nl_day = (year & 3) || day <= 59 ? day : day - 1; for (month = 0; month < 12; month++) if (day_n[month] > nl_day) break; } *date = cpu_to_le16(nl_day - day_n[month - 1] + 1 + (month << 5) + (year << 9));}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -