Commit de8999dc authored by Urban Widmark's avatar Urban Widmark Committed by Linus Torvalds

[PATCH] smbfs LFS

This patch adds LFS and moves some smb operations into per-protocol level
structs. It wants the nls patch to applied already.
parent 9ef6e588
ChangeLog for smbfs. ChangeLog for smbfs.
2001-08-23 Jochen Dolze <dolze@epcnet.de>
* proc.c: Correct rsize/wsize computation for readX/writeX
2001-0?-?? Urban Widmark <urban@teststation.com>
* *.c: Add LFS
* *.c: Move to a "driver" style handling of different servertypes.
(Not all operations are done this way. yet.)
2001-12-31 Ren Scharfe <l.s.r@web.de> 2001-12-31 Ren Scharfe <l.s.r@web.de>
* inode.c: added smb_show_options to show mount options in /proc/mounts * inode.c: added smb_show_options to show mount options in /proc/mounts
......
...@@ -187,7 +187,7 @@ smb_readdir(struct file *filp, void *dirent, filldir_t filldir) ...@@ -187,7 +187,7 @@ smb_readdir(struct file *filp, void *dirent, filldir_t filldir)
ctl.filled = 0; ctl.filled = 0;
ctl.valid = 1; ctl.valid = 1;
read_really: read_really:
result = smb_proc_readdir(filp, dirent, filldir, &ctl); result = server->ops->readdir(filp, dirent, filldir, &ctl);
if (ctl.idx == -1) if (ctl.idx == -1)
goto invalid_cache; /* retry */ goto invalid_cache; /* retry */
ctl.head.end = ctl.fpos - 1; ctl.head.end = ctl.fpos - 1;
......
...@@ -55,12 +55,13 @@ static int ...@@ -55,12 +55,13 @@ static int
smb_readpage_sync(struct dentry *dentry, struct page *page) smb_readpage_sync(struct dentry *dentry, struct page *page)
{ {
char *buffer = kmap(page); char *buffer = kmap(page);
unsigned long offset = page->index << PAGE_CACHE_SHIFT; loff_t offset = (loff_t)page->index << PAGE_CACHE_SHIFT;
int rsize = smb_get_rsize(server_from_dentry(dentry)); struct smb_sb_info *server = server_from_dentry(dentry);
unsigned int rsize = smb_get_rsize(server);
int count = PAGE_SIZE; int count = PAGE_SIZE;
int result; int result;
VERBOSE("file %s/%s, count=%d@%ld, rsize=%d\n", VERBOSE("file %s/%s, count=%d@%Ld, rsize=%d\n",
DENTRY_PATH(dentry), count, offset, rsize); DENTRY_PATH(dentry), count, offset, rsize);
result = smb_open(dentry, SMB_O_RDONLY); result = smb_open(dentry, SMB_O_RDONLY);
...@@ -74,7 +75,7 @@ smb_readpage_sync(struct dentry *dentry, struct page *page) ...@@ -74,7 +75,7 @@ smb_readpage_sync(struct dentry *dentry, struct page *page)
if (count < rsize) if (count < rsize)
rsize = count; rsize = count;
result = smb_proc_read(dentry->d_inode, offset, rsize, buffer); result = server->ops->read(dentry->d_inode,offset,rsize,buffer);
if (result < 0) if (result < 0)
goto io_error; goto io_error;
...@@ -118,21 +119,23 @@ smb_readpage(struct file *file, struct page *page) ...@@ -118,21 +119,23 @@ smb_readpage(struct file *file, struct page *page)
*/ */
static int static int
smb_writepage_sync(struct inode *inode, struct page *page, smb_writepage_sync(struct inode *inode, struct page *page,
unsigned long offset, unsigned int count) unsigned long pageoffset, unsigned int count)
{ {
char *buffer = kmap(page) + offset; loff_t offset;
int wsize = smb_get_wsize(server_from_inode(inode)); char *buffer = kmap(page) + pageoffset;
struct smb_sb_info *server = server_from_inode(inode);
unsigned int wsize = smb_get_wsize(server);
int result, written = 0; int result, written = 0;
offset += page->index << PAGE_CACHE_SHIFT; offset = ((loff_t)page->index << PAGE_CACHE_SHIFT) + pageoffset;
VERBOSE("file ino=%ld, fileid=%d, count=%d@%ld, wsize=%d\n", VERBOSE("file ino=%ld, fileid=%d, count=%d@%Ld, wsize=%d\n",
inode->i_ino, SMB_I(inode)->fileid, count, offset, wsize); inode->i_ino, inode->u.smbfs_i.fileid, count, offset, wsize);
do { do {
if (count < wsize) if (count < wsize)
wsize = count; wsize = count;
result = smb_proc_write(inode, offset, wsize, buffer); result = server->ops->write(inode, offset, wsize, buffer);
if (result < 0) { if (result < 0) {
PARANOIA("failed write, wsize=%d, result=%d\n", PARANOIA("failed write, wsize=%d, result=%d\n",
wsize, result); wsize, result);
......
...@@ -445,8 +445,7 @@ smb_put_super(struct super_block *sb) ...@@ -445,8 +445,7 @@ smb_put_super(struct super_block *sb)
if (server->conn_pid) if (server->conn_pid)
kill_proc(server->conn_pid, SIGTERM, 1); kill_proc(server->conn_pid, SIGTERM, 1);
smb_kfree(server->mnt); smb_kfree(server->ops);
smb_kfree(server->temp_buf);
if (server->packet) if (server->packet)
smb_vfree(server->packet); smb_vfree(server->packet);
...@@ -468,6 +467,7 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent) ...@@ -468,6 +467,7 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent)
struct inode *root_inode; struct inode *root_inode;
struct smb_fattr root; struct smb_fattr root;
int ver; int ver;
void *mem;
if (!raw_data) if (!raw_data)
goto out_no_data; goto out_no_data;
...@@ -494,22 +494,29 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent) ...@@ -494,22 +494,29 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent)
if (!server->packet) if (!server->packet)
goto out_no_mem; goto out_no_mem;
/* Allocate the global temp buffer */ /* Allocate the global temp buffer and some superblock helper structs */
server->temp_buf = smb_kmalloc(2*SMB_MAXPATHLEN+20, GFP_KERNEL); VERBOSE("alloc chunk = %d\n", sizeof(struct smb_ops) +
if (!server->temp_buf) sizeof(struct smb_mount_data_kernel) +
2*SMB_MAXPATHLEN + 20);
mem = smb_kmalloc(sizeof(struct smb_ops) +
sizeof(struct smb_mount_data_kernel) +
2*SMB_MAXPATHLEN + 20, GFP_KERNEL);
if (!mem)
goto out_no_temp; goto out_no_temp;
server->ops = mem;
server->mnt = mem + sizeof(struct smb_ops);
server->name_buf = mem + sizeof(struct smb_ops) +
sizeof(struct smb_mount_data_kernel);
server->temp_buf = mem + sizeof(struct smb_ops) +
sizeof(struct smb_mount_data_kernel) +
SMB_MAXPATHLEN + 1;
/* Setup NLS stuff */ /* Setup NLS stuff */
server->remote_nls = NULL; server->remote_nls = NULL;
server->local_nls = NULL; server->local_nls = NULL;
server->name_buf = server->temp_buf + SMB_MAXPATHLEN + 20;
/* Allocate the mount data structure */ mnt = server->mnt;
/* FIXME: merge this with the other malloc and get a whole page? */
mnt = smb_kmalloc(sizeof(struct smb_mount_data_kernel), GFP_KERNEL);
if (!mnt)
goto out_no_mount;
server->mnt = mnt;
memset(mnt, 0, sizeof(struct smb_mount_data_kernel)); memset(mnt, 0, sizeof(struct smb_mount_data_kernel));
strncpy(mnt->codepage.local_name, CONFIG_NLS_DEFAULT, strncpy(mnt->codepage.local_name, CONFIG_NLS_DEFAULT,
...@@ -565,9 +572,7 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent) ...@@ -565,9 +572,7 @@ int smb_fill_super(struct super_block *sb, void *raw_data, int silent)
out_no_root: out_no_root:
iput(root_inode); iput(root_inode);
out_bad_option: out_bad_option:
smb_kfree(server->mnt); smb_kfree(mem);
out_no_mount:
smb_kfree(server->temp_buf);
out_no_temp: out_no_temp:
smb_vfree(server->packet); smb_vfree(server->packet);
out_no_mem: out_no_mem:
...@@ -630,8 +635,7 @@ smb_notify_change(struct dentry *dentry, struct iattr *attr) ...@@ -630,8 +635,7 @@ smb_notify_change(struct dentry *dentry, struct iattr *attr)
error = smb_open(dentry, O_WRONLY); error = smb_open(dentry, O_WRONLY);
if (error) if (error)
goto out; goto out;
error = smb_proc_trunc(server, SMB_I(inode)->fileid, error = server->ops->truncate(inode, attr->ia_size);
attr->ia_size);
if (error) if (error)
goto out; goto out;
error = vmtruncate(inode, attr->ia_size); error = vmtruncate(inode, attr->ia_size);
......
...@@ -48,16 +48,29 @@ ...@@ -48,16 +48,29 @@
#define SMB_ST_BLKSIZE (PAGE_SIZE) #define SMB_ST_BLKSIZE (PAGE_SIZE)
#define SMB_ST_BLKSHIFT (PAGE_SHIFT) #define SMB_ST_BLKSHIFT (PAGE_SHIFT)
static struct smb_ops smb_ops_core;
static struct smb_ops smb_ops_os2;
static struct smb_ops smb_ops_win95;
static struct smb_ops smb_ops_winNT;
static void
smb_init_dirent(struct smb_sb_info *server, struct smb_fattr *fattr);
static void
smb_finish_dirent(struct smb_sb_info *server, struct smb_fattr *fattr);
static int static int
smb_proc_setattr_ext(struct smb_sb_info *, struct inode *, smb_proc_getattr_core(struct smb_sb_info *server, struct dentry *dir,
struct smb_fattr *); struct smb_fattr *fattr);
static int
smb_proc_getattr_ff(struct smb_sb_info *server, struct dentry *dentry,
struct smb_fattr *fattr);
static int static int
smb_proc_setattr_core(struct smb_sb_info *server, struct dentry *dentry, smb_proc_setattr_core(struct smb_sb_info *server, struct dentry *dentry,
__u16 attr); u16 attr);
static int static int
smb_proc_do_getattr(struct smb_sb_info *server, struct dentry *dir, smb_proc_setattr_ext(struct smb_sb_info *server,
struct smb_fattr *fattr); struct inode *inode, struct smb_fattr *fattr);
static void
install_ops(struct smb_ops *dst, struct smb_ops *src);
static void static void
...@@ -206,9 +219,9 @@ int smb_setcodepage(struct smb_sb_info *server, struct smb_nls_codepage *cp) ...@@ -206,9 +219,9 @@ int smb_setcodepage(struct smb_sb_info *server, struct smb_nls_codepage *cp)
out: out:
if (server->local_nls != NULL && server->remote_nls != NULL) if (server->local_nls != NULL && server->remote_nls != NULL)
server->convert = convert_cp; server->ops->convert = convert_cp;
else else
server->convert = convert_memcpy; server->ops->convert = convert_memcpy;
smb_unlock_server(server); smb_unlock_server(server);
return n; return n;
...@@ -274,7 +287,7 @@ static int smb_build_path(struct smb_sb_info *server, char * buf, int maxlen, ...@@ -274,7 +287,7 @@ static int smb_build_path(struct smb_sb_info *server, char * buf, int maxlen,
return -ENAMETOOLONG; return -ENAMETOOLONG;
} }
len = server->convert(path, maxlen-2, len = server->ops->convert(path, maxlen-2,
entry->d_name.name, entry->d_name.len, entry->d_name.name, entry->d_name.len,
server->local_nls, server->remote_nls); server->local_nls, server->remote_nls);
if (len < 0) { if (len < 0) {
...@@ -299,7 +312,7 @@ static int smb_build_path(struct smb_sb_info *server, char * buf, int maxlen, ...@@ -299,7 +312,7 @@ static int smb_build_path(struct smb_sb_info *server, char * buf, int maxlen,
if (maxlen < 3) if (maxlen < 3)
return -ENAMETOOLONG; return -ENAMETOOLONG;
*path++ = '\\'; *path++ = '\\';
len = server->convert(path, maxlen-2, len = server->ops->convert(path, maxlen-2,
name->name, name->len, name->name, name->len,
server->local_nls, server->remote_nls); server->local_nls, server->remote_nls);
if (len < 0) if (len < 0)
...@@ -527,7 +540,8 @@ smb_get_xmitsize(struct smb_sb_info *server, int overhead) ...@@ -527,7 +540,8 @@ smb_get_xmitsize(struct smb_sb_info *server, int overhead)
int int
smb_get_rsize(struct smb_sb_info *server) smb_get_rsize(struct smb_sb_info *server)
{ {
int overhead = SMB_HEADER_LEN + 5 * sizeof(__u16) + 2 + 1 + 2; /* readX has 12 parameters, read has 5 */
int overhead = SMB_HEADER_LEN + 12 * sizeof(__u16) + 2 + 1 + 2;
int size = smb_get_xmitsize(server, overhead); int size = smb_get_xmitsize(server, overhead);
VERBOSE("packet=%d, xmit=%d, size=%d\n", VERBOSE("packet=%d, xmit=%d, size=%d\n",
...@@ -542,7 +556,8 @@ smb_get_rsize(struct smb_sb_info *server) ...@@ -542,7 +556,8 @@ smb_get_rsize(struct smb_sb_info *server)
int int
smb_get_wsize(struct smb_sb_info *server) smb_get_wsize(struct smb_sb_info *server)
{ {
int overhead = SMB_HEADER_LEN + 5 * sizeof(__u16) + 2 + 1 + 2; /* writeX has 14 parameters, write has 5 */
int overhead = SMB_HEADER_LEN + 14 * sizeof(__u16) + 2 + 1 + 2;
int size = smb_get_xmitsize(server, overhead); int size = smb_get_xmitsize(server, overhead);
VERBOSE("packet=%d, xmit=%d, size=%d\n", VERBOSE("packet=%d, xmit=%d, size=%d\n",
...@@ -858,12 +873,53 @@ smb_newconn(struct smb_sb_info *server, struct smb_conn_opt *opt) ...@@ -858,12 +873,53 @@ smb_newconn(struct smb_sb_info *server, struct smb_conn_opt *opt)
/* now that we have an established connection we can detect the server /* now that we have an established connection we can detect the server
type and enable bug workarounds */ type and enable bug workarounds */
if (server->opt.protocol == SMB_PROTOCOL_NT1 && if (server->opt.protocol < SMB_PROTOCOL_LANMAN2)
(server->opt.max_xmit < 0x1000) && install_ops(server->ops, &smb_ops_core);
!(server->opt.capabilities & SMB_CAP_NT_SMBS)) { else if (server->opt.protocol == SMB_PROTOCOL_LANMAN2)
install_ops(server->ops, &smb_ops_os2);
else if (server->opt.protocol == SMB_PROTOCOL_NT1 &&
(server->opt.max_xmit < 0x1000) &&
!(server->opt.capabilities & SMB_CAP_NT_SMBS)) {
/* FIXME: can we kill the WIN95 flag now? */
server->mnt->flags |= SMB_MOUNT_WIN95; server->mnt->flags |= SMB_MOUNT_WIN95;
VERBOSE("smb_newconn: detected WIN95 server\n"); VERBOSE("detected WIN95 server\n");
install_ops(server->ops, &smb_ops_win95);
} else {
/*
* Samba has max_xmit 65535
* NT4spX has max_xmit 4536 (or something like that)
* win2k has ...
*/
VERBOSE("detected NT1 (Samba, NT4/5) server\n");
install_ops(server->ops, &smb_ops_winNT);
}
/* FIXME: the win9x code wants to modify these ... (seek/trunc bug) */
if (server->mnt->flags & SMB_MOUNT_OLDATTR) {
server->ops->getattr = smb_proc_getattr_core;
} else if (server->mnt->flags & SMB_MOUNT_DIRATTR) {
server->ops->getattr = smb_proc_getattr_ff;
}
/* Decode server capabilities */
if (server->opt.capabilities & SMB_CAP_LARGE_FILES) {
/* Should be ok to set this now, as no one can access the
mount until the connection has been established. */
SB_of(server)->s_maxbytes = ~0ULL >> 1;
VERBOSE("LFS enabled\n");
} }
#if 0
/* flags we will test for other patches ... */
if (server->opt.capabilities & SMB_CAP_UNICODE) {
VERBOSE("Unicode enabled\n");
}
if (server->opt.capabilities & SMB_CAP_LARGE_READX) {
VERBOSE("Large reads enabled\n");
}
if (server->opt.capabilities & SMB_CAP_LARGE_WRITEX) {
VERBOSE("Large writes enabled\n");
}
#endif
VERBOSE("protocol=%d, max_xmit=%d, pid=%d capabilities=0x%x\n", VERBOSE("protocol=%d, max_xmit=%d, pid=%d capabilities=0x%x\n",
server->opt.protocol, server->opt.max_xmit, server->conn_pid, server->opt.protocol, server->opt.max_xmit, server->conn_pid,
...@@ -1203,8 +1259,8 @@ smb_close_fileid(struct dentry *dentry, __u16 fileid) ...@@ -1203,8 +1259,8 @@ smb_close_fileid(struct dentry *dentry, __u16 fileid)
/* In smb_proc_read and smb_proc_write we do not retry, because the /* In smb_proc_read and smb_proc_write we do not retry, because the
file-id would not be valid after a reconnection. */ file-id would not be valid after a reconnection. */
int static int
smb_proc_read(struct inode *inode, off_t offset, int count, char *data) smb_proc_read(struct inode *inode, loff_t offset, int count, char *data)
{ {
struct smb_sb_info *server = server_from_inode(inode); struct smb_sb_info *server = server_from_inode(inode);
__u16 returned_count, data_len; __u16 returned_count, data_len;
...@@ -1252,15 +1308,15 @@ smb_proc_read(struct inode *inode, off_t offset, int count, char *data) ...@@ -1252,15 +1308,15 @@ smb_proc_read(struct inode *inode, off_t offset, int count, char *data)
return result; return result;
} }
int static int
smb_proc_write(struct inode *inode, off_t offset, int count, const char *data) smb_proc_write(struct inode *inode, loff_t offset, int count, const char *data)
{ {
struct smb_sb_info *server = server_from_inode(inode); struct smb_sb_info *server = server_from_inode(inode);
int result; int result;
__u8 *p; __u8 *p;
__u16 fileid = SMB_I(inode)->fileid; __u16 fileid = SMB_I(inode)->fileid;
VERBOSE("ino=%ld, fileid=%d, count=%d@%ld, packet_size=%d\n", VERBOSE("ino=%ld, fileid=%d, count=%d@%Ld, packet_size=%d\n",
inode->i_ino, SMB_I(inode)->fileid, count, offset, inode->i_ino, SMB_I(inode)->fileid, count, offset,
server->packet_size); server->packet_size);
...@@ -1283,6 +1339,94 @@ smb_proc_write(struct inode *inode, off_t offset, int count, const char *data) ...@@ -1283,6 +1339,94 @@ smb_proc_write(struct inode *inode, off_t offset, int count, const char *data)
return result; return result;
} }
/*
* In smb_proc_readX and smb_proc_writeX we do not retry, because the
* file-id would not be valid after a reconnection.
*/
static int
smb_proc_readX(struct inode *inode, loff_t offset, int count, char *data)
{
struct smb_sb_info *server = server_from_inode(inode);
u16 data_len, data_off;
unsigned char *buf;
int result;
smb_lock_server(server);
smb_setup_header(server, SMBreadX, 12, 0);
buf = server->packet;
WSET(buf, smb_vwv0, 0x00ff);
WSET(buf, smb_vwv1, 0);
WSET(buf, smb_vwv2, SMB_I(inode)->fileid);
DSET(buf, smb_vwv3, (u32)offset); /* low 32 bits */
WSET(buf, smb_vwv5, count);
WSET(buf, smb_vwv6, 0);
DSET(buf, smb_vwv7, 0);
WSET(buf, smb_vwv9, 0);
DSET(buf, smb_vwv10, (u32)(offset >> 32)); /* high 32 bits */
WSET(buf, smb_vwv11, 0);
result = smb_request_ok(server, SMBreadX, 12, -1);
if (result < 0)
goto out;
data_len = WVAL(server->packet, smb_vwv5);
data_off = WVAL(server->packet, smb_vwv6);
buf = smb_base(server->packet) + data_off;
/* we can NOT simply trust the info given by the server ... */
if (data_len > server->packet_size - (buf - server->packet)) {
printk(KERN_ERR "smb_proc_read: invalid data length!! "
"%d > %d - (%p - %p)\n",
data_len, server->packet_size, buf, server->packet);
result = -EIO;
goto out;
}
memcpy(data, buf, data_len);
result = data_len;
out:
smb_unlock_server(server);
VERBOSE("ino=%ld, fileid=%d, count=%d, result=%d\n",
inode->i_ino, SMB_I(inode)->fileid, count, result);
return result;
}
static int
smb_proc_writeX(struct inode *inode, loff_t offset, int count, const char *data)
{
struct smb_sb_info *server = server_from_inode(inode);
int result;
u8 *p;
VERBOSE("ino=%ld, fileid=%d, count=%d@%Ld, packet_size=%d\n",
inode->i_ino, SMB_I(inode)->fileid, count, offset,
server->packet_size);
smb_lock_server(server);
p = smb_setup_header(server, SMBwriteX, 14, count + 2);
WSET(server->packet, smb_vwv0, 0x00ff);
WSET(server->packet, smb_vwv1, 0);
WSET(server->packet, smb_vwv2, SMB_I(inode)->fileid);
DSET(server->packet, smb_vwv3, (u32)offset); /* low 32 bits */
DSET(server->packet, smb_vwv5, 0);
WSET(server->packet, smb_vwv7, 0); /* write mode */
WSET(server->packet, smb_vwv8, 0);
WSET(server->packet, smb_vwv9, 0);
WSET(server->packet, smb_vwv10, count); /* data length */
WSET(server->packet, smb_vwv11, p + 2 - smb_base(server->packet));
DSET(server->packet, smb_vwv12, (u32)(offset >> 32));
*p++ = 0; /* FIXME: pad to short or long ... change +2 above also */
*p++ = 0;
memcpy(p, data, count);
result = smb_request_ok(server, SMBwriteX, 6, 0);
if (result >= 0)
result = WVAL(server->packet, smb_vwv2);
smb_unlock_server(server);
return result;
}
int int
smb_proc_create(struct dentry *dentry, __u16 attr, time_t ctime, __u16 *fileid) smb_proc_create(struct dentry *dentry, __u16 attr, time_t ctime, __u16 *fileid)
{ {
...@@ -1402,7 +1546,9 @@ smb_set_rw(struct dentry *dentry,struct smb_sb_info *server) ...@@ -1402,7 +1546,9 @@ smb_set_rw(struct dentry *dentry,struct smb_sb_info *server)
struct smb_fattr fattr; struct smb_fattr fattr;
/* first get current attribute */ /* first get current attribute */
result = smb_proc_do_getattr(server, dentry, &fattr); smb_init_dirent(server, &fattr);
result = server->ops->getattr(server, dentry, &fattr);
smb_finish_dirent(server, &fattr);
if (result < 0) if (result < 0)
return result; return result;
...@@ -1477,38 +1623,72 @@ smb_proc_flush(struct smb_sb_info *server, __u16 fileid) ...@@ -1477,38 +1623,72 @@ smb_proc_flush(struct smb_sb_info *server, __u16 fileid)
return smb_request_ok(server, SMBflush, 0, 0); return smb_request_ok(server, SMBflush, 0, 0);
} }
int static int
smb_proc_trunc(struct smb_sb_info *server, __u16 fid, __u32 length) smb_proc_trunc32(struct inode *inode, loff_t length)
{ {
char *p; /*
* Writing 0bytes is old-SMB magic for truncating files.
* MAX_NON_LFS should prevent this from being called with a too
* large offset.
*/
return smb_proc_write(inode, length, 0, NULL);
}
static int
smb_proc_trunc64(struct inode *inode, loff_t length)
{
struct smb_sb_info *server = server_from_inode(inode);
int command;
int result; int result;
char *param = server->temp_buf;
char data[8];
unsigned char *resp_data = NULL;
unsigned char *resp_param = NULL;
int resp_data_len = 0;
int resp_param_len = 0;
smb_lock_server(server); smb_lock_server(server);
command = TRANSACT2_SETFILEINFO;
/* FIXME: must we also set allocation size? winNT seems to do that */
retry: retry:
p = smb_setup_header(server, SMBwrite, 5, 3); WSET(param, 0, SMB_I(inode)->fileid);
WSET(server->packet, smb_vwv0, fid); WSET(param, 2, SMB_SET_FILE_END_OF_FILE_INFO);
WSET(server->packet, smb_vwv1, 0); WSET(param, 4, 0);
DSET(server->packet, smb_vwv2, length); LSET(data, 0, length);
WSET(server->packet, smb_vwv4, 0); result = smb_trans2_request(server, command,
*p++ = 1; 8, data, 6, param,
WSET(p, 0, 0); &resp_data_len, &resp_data,
&resp_param_len, &resp_param);
if ((result = smb_request_ok(server, SMBwrite, 1, 0)) < 0) { if (result < 0) {
if (smb_retry(server)) if (smb_retry(server))
goto retry; goto retry;
goto out; goto out;
} }
result = 0;
if (server->rcls != 0)
result = smb_errno(server);
out:
smb_unlock_server(server);
return result;
}
static int
smb_proc_trunc95(struct inode *inode, loff_t length)
{
struct smb_sb_info *server = server_from_inode(inode);
int result = smb_proc_trunc32(inode, length);
/* /*
* win9x doesn't appear to update the size immediately. * win9x doesn't appear to update the size immediately.
* It will return the old file size after the truncate, * It will return the old file size after the truncate,
* confusing smbfs. * confusing smbfs. So we force an update.
* NT and Samba return the new value immediately. *
* FIXME: is this still necessary?
*/ */
if (server->mnt->flags & SMB_MOUNT_WIN95) smb_lock_server(server);
smb_proc_flush(server, fid); smb_proc_flush(server, SMB_I(inode)->fileid);
out:
smb_unlock_server(server); smb_unlock_server(server);
return result; return result;
} }
...@@ -1608,9 +1788,9 @@ smb_decode_short_dirent(struct smb_sb_info *server, char *p, ...@@ -1608,9 +1788,9 @@ smb_decode_short_dirent(struct smb_sb_info *server, char *p,
#endif #endif
qname->len = 0; qname->len = 0;
len = server->convert(server->name_buf, SMB_MAXNAMELEN, len = server->ops->convert(server->name_buf, SMB_MAXNAMELEN,
qname->name, len, qname->name, len,
server->remote_nls, server->local_nls); server->remote_nls, server->local_nls);
if (len > 0) { if (len > 0) {
qname->len = len; qname->len = len;
qname->name = server->name_buf; qname->name = server->name_buf;
...@@ -1817,7 +1997,7 @@ smb_decode_long_dirent(struct smb_sb_info *server, char *p, int level, ...@@ -1817,7 +1997,7 @@ smb_decode_long_dirent(struct smb_sb_info *server, char *p, int level,
fattr->f_atime = smb_ntutc2unixutc(LVAL(p, 16)); fattr->f_atime = smb_ntutc2unixutc(LVAL(p, 16));
fattr->f_mtime = smb_ntutc2unixutc(LVAL(p, 24)); fattr->f_mtime = smb_ntutc2unixutc(LVAL(p, 24));
/* change time (32) */ /* change time (32) */
fattr->f_size = DVAL(p, 40); fattr->f_size = LVAL(p, 40);
/* alloc size (48) */ /* alloc size (48) */
fattr->attr = DVAL(p, 56); fattr->attr = DVAL(p, 56);
...@@ -1848,9 +2028,9 @@ smb_decode_long_dirent(struct smb_sb_info *server, char *p, int level, ...@@ -1848,9 +2028,9 @@ smb_decode_long_dirent(struct smb_sb_info *server, char *p, int level,
#endif #endif
qname->len = 0; qname->len = 0;
n = server->convert(server->name_buf, SMB_MAXNAMELEN, n = server->ops->convert(server->name_buf, SMB_MAXNAMELEN,
qname->name, len, qname->name, len,
server->remote_nls, server->local_nls); server->remote_nls, server->local_nls);
if (n > 0) { if (n > 0) {
qname->len = n; qname->len = n;
qname->name = server->name_buf; qname->name = server->name_buf;
...@@ -2095,18 +2275,6 @@ smb_proc_readdir_long(struct file *filp, void *dirent, filldir_t filldir, ...@@ -2095,18 +2275,6 @@ smb_proc_readdir_long(struct file *filp, void *dirent, filldir_t filldir,
return result; return result;
} }
int
smb_proc_readdir(struct file *filp, void *dirent, filldir_t filldir,
struct smb_cache_control *ctl)
{
struct smb_sb_info *server = server_from_dentry(filp->f_dentry);
if (server->opt.protocol >= SMB_PROTOCOL_LANMAN2)
return smb_proc_readdir_long(filp, dirent, filldir, ctl);
else
return smb_proc_readdir_short(filp, dirent, filldir, ctl);
}
/* /*
* This version uses the trans2 TRANSACT2_FINDFIRST message * This version uses the trans2 TRANSACT2_FINDFIRST message
* to get the attribute data. * to get the attribute data.
...@@ -2239,19 +2407,15 @@ smb_proc_getattr_core(struct smb_sb_info *server, struct dentry *dir, ...@@ -2239,19 +2407,15 @@ smb_proc_getattr_core(struct smb_sb_info *server, struct dentry *dir,
*/ */
static int static int
smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir, smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir,
struct smb_fattr *attr) int *lrdata, unsigned char **rdata,
int *lrparam, unsigned char **rparam,
int infolevel)
{ {
char *p, *param = server->temp_buf; char *p, *param = server->temp_buf;
__u16 date, time;
int off_date = 0, off_time = 2;
unsigned char *resp_data = NULL;
unsigned char *resp_param = NULL;
int resp_data_len = 0;
int resp_param_len = 0;
int result; int result;
retry: retry:
WSET(param, 0, 1); /* Info level SMB_INFO_STANDARD */ WSET(param, 0, infolevel);
DSET(param, 2, 0); DSET(param, 2, 0);
result = smb_encode_path(server, param+6, SMB_MAXPATHLEN+1, dir, NULL); result = smb_encode_path(server, param+6, SMB_MAXPATHLEN+1, dir, NULL);
if (result < 0) if (result < 0)
...@@ -2260,8 +2424,8 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir, ...@@ -2260,8 +2424,8 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir,
result = smb_trans2_request(server, TRANSACT2_QPATHINFO, result = smb_trans2_request(server, TRANSACT2_QPATHINFO,
0, NULL, p - param, param, 0, NULL, p - param, param,
&resp_data_len, &resp_data, lrdata, rdata,
&resp_param_len, &resp_param); lrparam, rparam);
if (result < 0) if (result < 0)
{ {
if (smb_retry(server)) if (smb_retry(server))
...@@ -2276,13 +2440,36 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir, ...@@ -2276,13 +2440,36 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir,
goto out; goto out;
} }
result = -ENOENT; result = -ENOENT;
if (resp_data_len < 22) if (*lrdata < 22)
{ {
PARANOIA("not enough data for %s, len=%d\n", PARANOIA("not enough data for %s, len=%d\n",
&param[6], resp_data_len); &param[6], *lrdata);
goto out; goto out;
} }
result = 0;
out:
return result;
}
static int
smb_proc_getattr_trans2_std(struct smb_sb_info *server, struct dentry *dir,
struct smb_fattr *attr)
{
u16 date, time;
int off_date = 0, off_time = 2;
unsigned char *resp_data = NULL;
unsigned char *resp_param = NULL;
int resp_data_len = 0;
int resp_param_len = 0;
int result = smb_proc_getattr_trans2(server, dir,
&resp_data_len, &resp_data,
&resp_param_len, &resp_param,
SMB_INFO_STANDARD);
if (result < 0)
goto out;
/* /*
* Kludge alert: Win 95 swaps the date and time field, * Kludge alert: Win 95 swaps the date and time field,
* contrary to the CIFS docs and Win NT practice. * contrary to the CIFS docs and Win NT practice.
...@@ -2308,37 +2495,51 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir, ...@@ -2308,37 +2495,51 @@ smb_proc_getattr_trans2(struct smb_sb_info *server, struct dentry *dir,
#endif #endif
attr->f_size = DVAL(resp_data, 12); attr->f_size = DVAL(resp_data, 12);
attr->attr = WVAL(resp_data, 20); attr->attr = WVAL(resp_data, 20);
result = 0;
out: out:
return result; return result;
} }
/*
* Note: called with the server locked
*/
static int static int
smb_proc_do_getattr(struct smb_sb_info *server, struct dentry *dir, smb_proc_getattr_trans2_all(struct smb_sb_info *server, struct dentry *dir,
struct smb_fattr *fattr) struct smb_fattr *attr)
{ {
int result; unsigned char *resp_data = NULL;
struct inode *inode = dir->d_inode; unsigned char *resp_param = NULL;
int resp_data_len = 0;
int resp_param_len = 0;
int result = smb_proc_getattr_trans2(server, dir,
&resp_data_len, &resp_data,
&resp_param_len, &resp_param,
SMB_QUERY_FILE_ALL_INFO);
if (result < 0)
goto out;
attr->f_ctime = smb_ntutc2unixutc(LVAL(resp_data, 0));
attr->f_atime = smb_ntutc2unixutc(LVAL(resp_data, 8));
attr->f_mtime = smb_ntutc2unixutc(LVAL(resp_data, 16));
/* change (24) */
attr->attr = WVAL(resp_data, 32);
/* pad? (34) */
/* allocated size (40) */
attr->f_size = LVAL(resp_data, 48);
smb_init_dirent(server, fattr); out:
return result;
}
/* static int
* Select whether to use core or trans2 getattr. smb_proc_getattr_95(struct smb_sb_info *server, struct dentry *dir,
* Win 95 appears to break with the trans2 getattr. struct smb_fattr *attr)
*/ {
if (server->opt.protocol < SMB_PROTOCOL_LANMAN2 || struct inode *inode = dir->d_inode;
(server->mnt->flags & (SMB_MOUNT_OLDATTR|SMB_MOUNT_WIN95)) ) { int result;
result = smb_proc_getattr_core(server, dir, fattr);
} else { /* FIXME: why not use the "all" version? */
if (server->mnt->flags & SMB_MOUNT_DIRATTR) result = smb_proc_getattr_trans2_std(server, dir, attr);
result = smb_proc_getattr_ff(server, dir, fattr); if (result < 0)
else goto out;
result = smb_proc_getattr_trans2(server, dir, fattr);
}
/* /*
* None of the getattr versions here can make win9x return the right * None of the getattr versions here can make win9x return the right
...@@ -2346,16 +2547,14 @@ smb_proc_do_getattr(struct smb_sb_info *server, struct dentry *dir, ...@@ -2346,16 +2547,14 @@ smb_proc_do_getattr(struct smb_sb_info *server, struct dentry *dir,
* A seek-to-end does return the right size, but we only need to do * A seek-to-end does return the right size, but we only need to do
* that on files we have written. * that on files we have written.
*/ */
if (server->mnt->flags & SMB_MOUNT_WIN95 && if (inode && SMB_I(inode)->flags & SMB_F_LOCALWRITE &&
inode &&
SMB_I(inode)->flags & SMB_F_LOCALWRITE &&
smb_is_open(inode)) smb_is_open(inode))
{ {
__u16 fileid = SMB_I(inode)->fileid; __u16 fileid = SMB_I(inode)->fileid;
fattr->f_size = smb_proc_seek(server, fileid, 2, 0); attr->f_size = smb_proc_seek(server, fileid, 2, 0);
} }
smb_finish_dirent(server, fattr); out:
return result; return result;
} }
...@@ -2366,7 +2565,11 @@ smb_proc_getattr(struct dentry *dir, struct smb_fattr *fattr) ...@@ -2366,7 +2565,11 @@ smb_proc_getattr(struct dentry *dir, struct smb_fattr *fattr)
int result; int result;
smb_lock_server(server); smb_lock_server(server);
result = smb_proc_do_getattr(server, dir, fattr);
smb_init_dirent(server, fattr);
result = server->ops->getattr(server, dir, fattr);
smb_finish_dirent(server, fattr);
smb_unlock_server(server); smb_unlock_server(server);
return result; return result;
} }
...@@ -2628,3 +2831,50 @@ smb_proc_dskattr(struct super_block *sb, struct statfs *attr) ...@@ -2628,3 +2831,50 @@ smb_proc_dskattr(struct super_block *sb, struct statfs *attr)
smb_unlock_server(server); smb_unlock_server(server);
return result; return result;
} }
static void
install_ops(struct smb_ops *dst, struct smb_ops *src)
{
memcpy(dst, src, sizeof(void *) * SMB_OPS_NUM_STATIC);
}
/* < LANMAN2 */
static struct smb_ops smb_ops_core =
{
read: smb_proc_read,
write: smb_proc_write,
readdir: smb_proc_readdir_short,
getattr: smb_proc_getattr_core,
truncate: smb_proc_trunc32,
};
/* LANMAN2, OS/2, others? */
static struct smb_ops smb_ops_os2 =
{
read: smb_proc_read,
write: smb_proc_write,
readdir: smb_proc_readdir_long,
getattr: smb_proc_getattr_trans2_std,
truncate: smb_proc_trunc32,
};
/* Win95, and possibly some NetApp versions too */
static struct smb_ops smb_ops_win95 =
{
read: smb_proc_read, /* does not support 12word readX */
write: smb_proc_write,
readdir: smb_proc_readdir_long,
getattr: smb_proc_getattr_95,
truncate: smb_proc_trunc95,
};
/* Samba, NT4 and NT5 */
static struct smb_ops smb_ops_winNT =
{
read: smb_proc_readX,
write: smb_proc_writeX,
readdir: smb_proc_readdir_long,
getattr: smb_proc_getattr_trans2_all,
truncate: smb_proc_trunc64,
};
/* /*
* Autogenerated with cproto on: Tue Oct 2 20:40:54 CEST 2001 * Autogenerated with cproto on: Thu Nov 22 21:18:04 CET 2001
*/ */
/* proc.c */ /* proc.c */
...@@ -14,17 +14,13 @@ extern __u8 *smb_setup_header(struct smb_sb_info *server, __u8 command, __u16 wc ...@@ -14,17 +14,13 @@ extern __u8 *smb_setup_header(struct smb_sb_info *server, __u8 command, __u16 wc
extern int smb_open(struct dentry *dentry, int wish); extern int smb_open(struct dentry *dentry, int wish);
extern int smb_close(struct inode *ino); extern int smb_close(struct inode *ino);
extern int smb_close_fileid(struct dentry *dentry, __u16 fileid); extern int smb_close_fileid(struct dentry *dentry, __u16 fileid);
extern int smb_proc_read(struct inode *inode, off_t offset, int count, char *data);
extern int smb_proc_write(struct inode *inode, off_t offset, int count, const char *data);
extern int smb_proc_create(struct dentry *dentry, __u16 attr, time_t ctime, __u16 *fileid); extern int smb_proc_create(struct dentry *dentry, __u16 attr, time_t ctime, __u16 *fileid);
extern int smb_proc_mv(struct dentry *old_dentry, struct dentry *new_dentry); extern int smb_proc_mv(struct dentry *old_dentry, struct dentry *new_dentry);
extern int smb_proc_mkdir(struct dentry *dentry); extern int smb_proc_mkdir(struct dentry *dentry);
extern int smb_proc_rmdir(struct dentry *dentry); extern int smb_proc_rmdir(struct dentry *dentry);
extern int smb_proc_unlink(struct dentry *dentry); extern int smb_proc_unlink(struct dentry *dentry);
extern int smb_proc_flush(struct smb_sb_info *server, __u16 fileid); extern int smb_proc_flush(struct smb_sb_info *server, __u16 fileid);
extern int smb_proc_trunc(struct smb_sb_info *server, __u16 fid, __u32 length);
extern void smb_init_root_dirent(struct smb_sb_info *server, struct smb_fattr *fattr); extern void smb_init_root_dirent(struct smb_sb_info *server, struct smb_fattr *fattr);
extern int smb_proc_readdir(struct file *filp, void *dirent, filldir_t filldir, struct smb_cache_control *ctl);
extern int smb_proc_getattr(struct dentry *dir, struct smb_fattr *fattr); extern int smb_proc_getattr(struct dentry *dir, struct smb_fattr *fattr);
extern int smb_proc_setattr(struct dentry *dir, struct smb_fattr *fattr); extern int smb_proc_setattr(struct dentry *dir, struct smb_fattr *fattr);
extern int smb_proc_settime(struct dentry *dentry, struct smb_fattr *fattr); extern int smb_proc_settime(struct dentry *dentry, struct smb_fattr *fattr);
......
...@@ -86,7 +86,7 @@ struct smb_fattr { ...@@ -86,7 +86,7 @@ struct smb_fattr {
uid_t f_uid; uid_t f_uid;
gid_t f_gid; gid_t f_gid;
kdev_t f_rdev; kdev_t f_rdev;
off_t f_size; loff_t f_size;
time_t f_atime; time_t f_atime;
time_t f_mtime; time_t f_mtime;
time_t f_ctime; time_t f_ctime;
......
...@@ -117,6 +117,7 @@ smb_kfree(void *obj) ...@@ -117,6 +117,7 @@ smb_kfree(void *obj)
#define SMB_CAP_NT_FIND 0x0200 #define SMB_CAP_NT_FIND 0x0200
#define SMB_CAP_DFS 0x1000 #define SMB_CAP_DFS 0x1000
#define SMB_CAP_LARGE_READX 0x4000 #define SMB_CAP_LARGE_READX 0x4000
#define SMB_CAP_LARGE_WRITEX 0x8000
/* /*
...@@ -159,6 +160,30 @@ struct smb_cache_control { ...@@ -159,6 +160,30 @@ struct smb_cache_control {
int filled, valid, idx; int filled, valid, idx;
}; };
#define SMB_OPS_NUM_STATIC 5
struct smb_ops {
int (*read)(struct inode *inode, loff_t offset, int count,
char *data);
int (*write)(struct inode *inode, loff_t offset, int count, const
char *data);
int (*readdir)(struct file *filp, void *dirent, filldir_t filldir,
struct smb_cache_control *ctl);
int (*getattr)(struct smb_sb_info *server, struct dentry *dir,
struct smb_fattr *fattr);
/* int (*setattr)(...); */ /* setattr is really icky! */
int (*truncate)(struct inode *inode, loff_t length);
/* --- --- --- end of "static" entries --- --- --- */
int (*convert)(char *output, int olen,
const char *input, int ilen,
struct nls_table *nls_from,
struct nls_table *nls_to);
};
static inline int static inline int
smb_is_open(struct inode *i) smb_is_open(struct inode *i)
{ {
......
...@@ -54,8 +54,7 @@ struct smb_sb_info { ...@@ -54,8 +54,7 @@ struct smb_sb_info {
to put it on the stack. This points to temp_buf space. */ to put it on the stack. This points to temp_buf space. */
char *name_buf; char *name_buf;
int (*convert)(char *, int, const char *, int, struct smb_ops *ops;
struct nls_table *, struct nls_table *);
}; };
......
...@@ -281,4 +281,33 @@ ...@@ -281,4 +281,33 @@
#define TRANSACT2_FINDNOTIFYNEXT 12 #define TRANSACT2_FINDNOTIFYNEXT 12
#define TRANSACT2_MKDIR 13 #define TRANSACT2_MKDIR 13
/* Information Levels - Shared? */
#define SMB_INFO_STANDARD 1
#define SMB_INFO_QUERY_EA_SIZE 2
#define SMB_INFO_QUERY_EAS_FROM_LIST 3
#define SMB_INFO_QUERY_ALL_EAS 4
#define SMB_INFO_IS_NAME_VALID 6
/* Information Levels - TRANSACT2_FINDFIRST */
#define SMB_FIND_FILE_DIRECTORY_INFO 0x101
#define SMB_FIND_FILE_FULL_DIRECTORY_INFO 0x102
#define SMB_FIND_FILE_NAMES_INFO 0x103
#define SMB_FIND_FILE_BOTH_DIRECTORY_INFO 0x104
/* Information Levels - TRANSACT2_QPATHINFO */
#define SMB_QUERY_FILE_BASIC_INFO 0x101
#define SMB_QUERY_FILE_STANDARD_INFO 0x102
#define SMB_QUERY_FILE_EA_INFO 0x103
#define SMB_QUERY_FILE_NAME_INFO 0x104
#define SMB_QUERY_FILE_ALL_INFO 0x107
#define SMB_QUERY_FILE_ALT_NAME_INFO 0x108
#define SMB_QUERY_FILE_STREAM_INFO 0x109
#define SMB_QUERY_FILE_COMPRESSION_INFO 0x10b
/* Information Levels - TRANSACT2_SETFILEINFO */
#define SMB_SET_FILE_BASIC_INFO 0x101
#define SMB_SET_FILE_DISPOSITION_INFO 0x102
#define SMB_SET_FILE_ALLOCATION_INFO 0x103
#define SMB_SET_FILE_END_OF_FILE_INFO 0x104
#endif /* _SMBNO_H_ */ #endif /* _SMBNO_H_ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment