diff options
Diffstat (limited to 'fs')
| -rw-r--r-- | fs/befs/linuxvfs.c | 4 | ||||
| -rw-r--r-- | fs/cifs/cifs_dfs_ref.c | 141 | ||||
| -rw-r--r-- | fs/cifs/cifsfs.c | 3 | ||||
| -rw-r--r-- | fs/cifs/connect.c | 23 | ||||
| -rw-r--r-- | fs/cifs/dns_resolve.c | 4 | ||||
| -rw-r--r-- | fs/efivarfs/file.c | 14 | ||||
| -rw-r--r-- | fs/hpfs/dir.c | 10 | ||||
| -rw-r--r-- | fs/nfs/nfs4proc.c | 2 | ||||
| -rw-r--r-- | fs/nfs/super.c | 2 | ||||
| -rw-r--r-- | fs/pnode.c | 3 | ||||
| -rw-r--r-- | fs/qnx6/dir.c | 2 | ||||
| -rw-r--r-- | fs/reiserfs/dir.c | 2 | ||||
| -rw-r--r-- | fs/reiserfs/inode.c | 9 | ||||
| -rw-r--r-- | fs/reiserfs/xattr.c | 14 | ||||
| -rw-r--r-- | fs/reiserfs/xattr_acl.c | 3 | ||||
| -rw-r--r-- | fs/xfs/xfs_attr_leaf.c | 71 | ||||
| -rw-r--r-- | fs/xfs/xfs_attr_remote.c | 408 | ||||
| -rw-r--r-- | fs/xfs/xfs_attr_remote.h | 10 | ||||
| -rw-r--r-- | fs/xfs/xfs_buf.c | 1 | ||||
| -rw-r--r-- | fs/xfs/xfs_buf_item.c | 7 | ||||
| -rw-r--r-- | fs/xfs/xfs_dfrag.c | 8 | ||||
| -rw-r--r-- | fs/xfs/xfs_dir2_format.h | 1 | ||||
| -rw-r--r-- | fs/xfs/xfs_dir2_node.c | 13 | ||||
| -rw-r--r-- | fs/xfs/xfs_fs.h | 1 | ||||
| -rw-r--r-- | fs/xfs/xfs_fsops.c | 4 | ||||
| -rw-r--r-- | fs/xfs/xfs_iops.c | 47 | ||||
| -rw-r--r-- | fs/xfs/xfs_log_recover.c | 11 | ||||
| -rw-r--r-- | fs/xfs/xfs_qm_syscalls.c | 40 | ||||
| -rw-r--r-- | fs/xfs/xfs_symlink.c | 20 |
29 files changed, 538 insertions, 340 deletions
diff --git a/fs/befs/linuxvfs.c b/fs/befs/linuxvfs.c index 8615ee89ab55..f95dddced968 100644 --- a/fs/befs/linuxvfs.c +++ b/fs/befs/linuxvfs.c | |||
| @@ -265,8 +265,8 @@ befs_readdir(struct file *filp, void *dirent, filldir_t filldir) | |||
| 265 | result = filldir(dirent, keybuf, keysize, filp->f_pos, | 265 | result = filldir(dirent, keybuf, keysize, filp->f_pos, |
| 266 | (ino_t) value, d_type); | 266 | (ino_t) value, d_type); |
| 267 | } | 267 | } |
| 268 | 268 | if (!result) | |
| 269 | filp->f_pos++; | 269 | filp->f_pos++; |
| 270 | 270 | ||
| 271 | befs_debug(sb, "<--- befs_readdir() filp->f_pos %Ld", filp->f_pos); | 271 | befs_debug(sb, "<--- befs_readdir() filp->f_pos %Ld", filp->f_pos); |
| 272 | 272 | ||
diff --git a/fs/cifs/cifs_dfs_ref.c b/fs/cifs/cifs_dfs_ref.c index 8e33ec65847b..58df174deb10 100644 --- a/fs/cifs/cifs_dfs_ref.c +++ b/fs/cifs/cifs_dfs_ref.c | |||
| @@ -18,6 +18,7 @@ | |||
| 18 | #include <linux/slab.h> | 18 | #include <linux/slab.h> |
| 19 | #include <linux/vfs.h> | 19 | #include <linux/vfs.h> |
| 20 | #include <linux/fs.h> | 20 | #include <linux/fs.h> |
| 21 | #include <linux/inet.h> | ||
| 21 | #include "cifsglob.h" | 22 | #include "cifsglob.h" |
| 22 | #include "cifsproto.h" | 23 | #include "cifsproto.h" |
| 23 | #include "cifsfs.h" | 24 | #include "cifsfs.h" |
| @@ -48,58 +49,74 @@ void cifs_dfs_release_automount_timer(void) | |||
| 48 | } | 49 | } |
| 49 | 50 | ||
| 50 | /** | 51 | /** |
| 51 | * cifs_get_share_name - extracts share name from UNC | 52 | * cifs_build_devname - build a devicename from a UNC and optional prepath |
| 52 | * @node_name: pointer to UNC string | 53 | * @nodename: pointer to UNC string |
| 54 | * @prepath: pointer to prefixpath (or NULL if there isn't one) | ||
| 53 | * | 55 | * |
| 54 | * Extracts sharename form full UNC. | 56 | * Build a new cifs devicename after chasing a DFS referral. Allocate a buffer |
| 55 | * i.e. strips from UNC trailing path that is not part of share | 57 | * big enough to hold the final thing. Copy the UNC from the nodename, and |
| 56 | * name and fixup missing '\' in the beginning of DFS node refferal | 58 | * concatenate the prepath onto the end of it if there is one. |
| 57 | * if necessary. | 59 | * |
| 58 | * Returns pointer to share name on success or ERR_PTR on error. | 60 | * Returns pointer to the built string, or a ERR_PTR. Caller is responsible |
| 59 | * Caller is responsible for freeing returned string. | 61 | * for freeing the returned string. |
| 60 | */ | 62 | */ |
| 61 | static char *cifs_get_share_name(const char *node_name) | 63 | static char * |
| 64 | cifs_build_devname(char *nodename, const char *prepath) | ||
| 62 | { | 65 | { |
| 63 | int len; | 66 | size_t pplen; |
| 64 | char *UNC; | 67 | size_t unclen; |
| 65 | char *pSep; | 68 | char *dev; |
| 66 | 69 | char *pos; | |
| 67 | len = strlen(node_name); | 70 | |
| 68 | UNC = kmalloc(len+2 /*for term null and additional \ if it's missed */, | 71 | /* skip over any preceding delimiters */ |
| 69 | GFP_KERNEL); | 72 | nodename += strspn(nodename, "\\"); |
| 70 | if (!UNC) | 73 | if (!*nodename) |
| 71 | return ERR_PTR(-ENOMEM); | 74 | return ERR_PTR(-EINVAL); |
| 72 | 75 | ||
| 73 | /* get share name and server name */ | 76 | /* get length of UNC and set pos to last char */ |
| 74 | if (node_name[1] != '\\') { | 77 | unclen = strlen(nodename); |
| 75 | UNC[0] = '\\'; | 78 | pos = nodename + unclen - 1; |
| 76 | strncpy(UNC+1, node_name, len); | ||
| 77 | len++; | ||
| 78 | UNC[len] = 0; | ||
| 79 | } else { | ||
| 80 | strncpy(UNC, node_name, len); | ||
| 81 | UNC[len] = 0; | ||
| 82 | } | ||
| 83 | 79 | ||
| 84 | /* find server name end */ | 80 | /* trim off any trailing delimiters */ |
| 85 | pSep = memchr(UNC+2, '\\', len-2); | 81 | while (*pos == '\\') { |
| 86 | if (!pSep) { | 82 | --pos; |
| 87 | cifs_dbg(VFS, "%s: no server name end in node name: %s\n", | 83 | --unclen; |
| 88 | __func__, node_name); | ||
| 89 | kfree(UNC); | ||
| 90 | return ERR_PTR(-EINVAL); | ||
| 91 | } | 84 | } |
| 92 | 85 | ||
| 93 | /* find sharename end */ | 86 | /* allocate a buffer: |
| 94 | pSep++; | 87 | * +2 for preceding "//" |
| 95 | pSep = memchr(UNC+(pSep-UNC), '\\', len-(pSep-UNC)); | 88 | * +1 for delimiter between UNC and prepath |
| 96 | if (pSep) { | 89 | * +1 for trailing NULL |
| 97 | /* trim path up to sharename end | 90 | */ |
| 98 | * now we have share name in UNC */ | 91 | pplen = prepath ? strlen(prepath) : 0; |
| 99 | *pSep = 0; | 92 | dev = kmalloc(2 + unclen + 1 + pplen + 1, GFP_KERNEL); |
| 93 | if (!dev) | ||
| 94 | return ERR_PTR(-ENOMEM); | ||
| 95 | |||
| 96 | pos = dev; | ||
| 97 | /* add the initial "//" */ | ||
| 98 | *pos = '/'; | ||
| 99 | ++pos; | ||
| 100 | *pos = '/'; | ||
| 101 | ++pos; | ||
| 102 | |||
| 103 | /* copy in the UNC portion from referral */ | ||
| 104 | memcpy(pos, nodename, unclen); | ||
| 105 | pos += unclen; | ||
| 106 | |||
| 107 | /* copy the prefixpath remainder (if there is one) */ | ||
| 108 | if (pplen) { | ||
| 109 | *pos = '/'; | ||
| 110 | ++pos; | ||
| 111 | memcpy(pos, prepath, pplen); | ||
| 112 | pos += pplen; | ||
| 100 | } | 113 | } |
| 101 | 114 | ||
| 102 | return UNC; | 115 | /* NULL terminator */ |
| 116 | *pos = '\0'; | ||
| 117 | |||
| 118 | convert_delimiter(dev, '/'); | ||
| 119 | return dev; | ||
| 103 | } | 120 | } |
| 104 | 121 | ||
| 105 | 122 | ||
| @@ -123,6 +140,7 @@ char *cifs_compose_mount_options(const char *sb_mountdata, | |||
| 123 | { | 140 | { |
| 124 | int rc; | 141 | int rc; |
| 125 | char *mountdata = NULL; | 142 | char *mountdata = NULL; |
| 143 | const char *prepath = NULL; | ||
| 126 | int md_len; | 144 | int md_len; |
| 127 | char *tkn_e; | 145 | char *tkn_e; |
| 128 | char *srvIP = NULL; | 146 | char *srvIP = NULL; |
| @@ -132,7 +150,10 @@ char *cifs_compose_mount_options(const char *sb_mountdata, | |||
| 132 | if (sb_mountdata == NULL) | 150 | if (sb_mountdata == NULL) |
| 133 | return ERR_PTR(-EINVAL); | 151 | return ERR_PTR(-EINVAL); |
| 134 | 152 | ||
| 135 | *devname = cifs_get_share_name(ref->node_name); | 153 | if (strlen(fullpath) - ref->path_consumed) |
| 154 | prepath = fullpath + ref->path_consumed; | ||
| 155 | |||
| 156 | *devname = cifs_build_devname(ref->node_name, prepath); | ||
| 136 | if (IS_ERR(*devname)) { | 157 | if (IS_ERR(*devname)) { |
| 137 | rc = PTR_ERR(*devname); | 158 | rc = PTR_ERR(*devname); |
| 138 | *devname = NULL; | 159 | *devname = NULL; |
| @@ -146,12 +167,14 @@ char *cifs_compose_mount_options(const char *sb_mountdata, | |||
| 146 | goto compose_mount_options_err; | 167 | goto compose_mount_options_err; |
| 147 | } | 168 | } |
| 148 | 169 | ||
| 149 | /* md_len = strlen(...) + 12 for 'sep+prefixpath=' | 170 | /* |
| 150 | * assuming that we have 'unc=' and 'ip=' in | 171 | * In most cases, we'll be building a shorter string than the original, |
| 151 | * the original sb_mountdata | 172 | * but we do have to assume that the address in the ip= option may be |
| 173 | * much longer than the original. Add the max length of an address | ||
| 174 | * string to the length of the original string to allow for worst case. | ||
| 152 | */ | 175 | */ |
| 153 | md_len = strlen(sb_mountdata) + rc + strlen(ref->node_name) + 12; | 176 | md_len = strlen(sb_mountdata) + INET6_ADDRSTRLEN; |
| 154 | mountdata = kzalloc(md_len+1, GFP_KERNEL); | 177 | mountdata = kzalloc(md_len + 1, GFP_KERNEL); |
| 155 | if (mountdata == NULL) { | 178 | if (mountdata == NULL) { |
| 156 | rc = -ENOMEM; | 179 | rc = -ENOMEM; |
| 157 | goto compose_mount_options_err; | 180 | goto compose_mount_options_err; |
| @@ -195,26 +218,6 @@ char *cifs_compose_mount_options(const char *sb_mountdata, | |||
| 195 | strncat(mountdata, &sep, 1); | 218 | strncat(mountdata, &sep, 1); |
| 196 | strcat(mountdata, "ip="); | 219 | strcat(mountdata, "ip="); |
| 197 | strcat(mountdata, srvIP); | 220 | strcat(mountdata, srvIP); |
| 198 | strncat(mountdata, &sep, 1); | ||
| 199 | strcat(mountdata, "unc="); | ||
| 200 | strcat(mountdata, *devname); | ||
| 201 | |||
| 202 | /* find & copy prefixpath */ | ||
| 203 | tkn_e = strchr(ref->node_name + 2, '\\'); | ||
| 204 | if (tkn_e == NULL) { | ||
| 205 | /* invalid unc, missing share name*/ | ||
| 206 | rc = -EINVAL; | ||
| 207 | goto compose_mount_options_err; | ||
| 208 | } | ||
| 209 | |||
| 210 | tkn_e = strchr(tkn_e + 1, '\\'); | ||
| 211 | if (tkn_e || (strlen(fullpath) - ref->path_consumed)) { | ||
| 212 | strncat(mountdata, &sep, 1); | ||
| 213 | strcat(mountdata, "prefixpath="); | ||
| 214 | if (tkn_e) | ||
| 215 | strcat(mountdata, tkn_e + 1); | ||
| 216 | strcat(mountdata, fullpath + ref->path_consumed); | ||
| 217 | } | ||
| 218 | 221 | ||
| 219 | /*cifs_dbg(FYI, "%s: parent mountdata: %s\n", __func__, sb_mountdata);*/ | 222 | /*cifs_dbg(FYI, "%s: parent mountdata: %s\n", __func__, sb_mountdata);*/ |
| 220 | /*cifs_dbg(FYI, "%s: submount mountdata: %s\n", __func__, mountdata );*/ | 223 | /*cifs_dbg(FYI, "%s: submount mountdata: %s\n", __func__, mountdata );*/ |
diff --git a/fs/cifs/cifsfs.c b/fs/cifs/cifsfs.c index 72e4efee1389..3752b9f6d9e4 100644 --- a/fs/cifs/cifsfs.c +++ b/fs/cifs/cifsfs.c | |||
| @@ -372,9 +372,6 @@ cifs_show_options(struct seq_file *s, struct dentry *root) | |||
| 372 | cifs_show_security(s, tcon->ses->server); | 372 | cifs_show_security(s, tcon->ses->server); |
| 373 | cifs_show_cache_flavor(s, cifs_sb); | 373 | cifs_show_cache_flavor(s, cifs_sb); |
| 374 | 374 | ||
| 375 | seq_printf(s, ",unc="); | ||
| 376 | seq_escape(s, tcon->treeName, " \t\n\\"); | ||
| 377 | |||
| 378 | if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER) | 375 | if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER) |
| 379 | seq_printf(s, ",multiuser"); | 376 | seq_printf(s, ",multiuser"); |
| 380 | else if (tcon->ses->user_name) | 377 | else if (tcon->ses->user_name) |
diff --git a/fs/cifs/connect.c b/fs/cifs/connect.c index 99eeaa17ee00..5b97e56ddbca 100644 --- a/fs/cifs/connect.c +++ b/fs/cifs/connect.c | |||
| @@ -1061,6 +1061,7 @@ static int cifs_parse_security_flavors(char *value, | |||
| 1061 | #endif | 1061 | #endif |
| 1062 | case Opt_sec_none: | 1062 | case Opt_sec_none: |
| 1063 | vol->nullauth = 1; | 1063 | vol->nullauth = 1; |
| 1064 | vol->secFlg |= CIFSSEC_MAY_NTLM; | ||
| 1064 | break; | 1065 | break; |
| 1065 | default: | 1066 | default: |
| 1066 | cifs_dbg(VFS, "bad security option: %s\n", value); | 1067 | cifs_dbg(VFS, "bad security option: %s\n", value); |
| @@ -1257,14 +1258,18 @@ cifs_parse_mount_options(const char *mountdata, const char *devname, | |||
| 1257 | vol->backupuid_specified = false; /* no backup intent for a user */ | 1258 | vol->backupuid_specified = false; /* no backup intent for a user */ |
| 1258 | vol->backupgid_specified = false; /* no backup intent for a group */ | 1259 | vol->backupgid_specified = false; /* no backup intent for a group */ |
| 1259 | 1260 | ||
| 1260 | /* | 1261 | switch (cifs_parse_devname(devname, vol)) { |
| 1261 | * For now, we ignore -EINVAL errors under the assumption that the | 1262 | case 0: |
| 1262 | * unc= and prefixpath= options will be usable. | 1263 | break; |
| 1263 | */ | 1264 | case -ENOMEM: |
| 1264 | if (cifs_parse_devname(devname, vol) == -ENOMEM) { | 1265 | cifs_dbg(VFS, "Unable to allocate memory for devname.\n"); |
| 1265 | printk(KERN_ERR "CIFS: Unable to allocate memory to parse " | 1266 | goto cifs_parse_mount_err; |
| 1266 | "device string.\n"); | 1267 | case -EINVAL: |
| 1267 | goto out_nomem; | 1268 | cifs_dbg(VFS, "Malformed UNC in devname.\n"); |
| 1269 | goto cifs_parse_mount_err; | ||
| 1270 | default: | ||
| 1271 | cifs_dbg(VFS, "Unknown error parsing devname.\n"); | ||
| 1272 | goto cifs_parse_mount_err; | ||
| 1268 | } | 1273 | } |
| 1269 | 1274 | ||
| 1270 | while ((data = strsep(&options, separator)) != NULL) { | 1275 | while ((data = strsep(&options, separator)) != NULL) { |
| @@ -1826,7 +1831,7 @@ cifs_parse_mount_options(const char *mountdata, const char *devname, | |||
| 1826 | } | 1831 | } |
| 1827 | #endif | 1832 | #endif |
| 1828 | if (!vol->UNC) { | 1833 | if (!vol->UNC) { |
| 1829 | cifs_dbg(VFS, "CIFS mount error: No usable UNC path provided in device string or in unc= option!\n"); | 1834 | cifs_dbg(VFS, "CIFS mount error: No usable UNC path provided in device string!\n"); |
| 1830 | goto cifs_parse_mount_err; | 1835 | goto cifs_parse_mount_err; |
| 1831 | } | 1836 | } |
| 1832 | 1837 | ||
diff --git a/fs/cifs/dns_resolve.c b/fs/cifs/dns_resolve.c index e7512e497611..7ede7306599f 100644 --- a/fs/cifs/dns_resolve.c +++ b/fs/cifs/dns_resolve.c | |||
| @@ -34,7 +34,7 @@ | |||
| 34 | 34 | ||
| 35 | /** | 35 | /** |
| 36 | * dns_resolve_server_name_to_ip - Resolve UNC server name to ip address. | 36 | * dns_resolve_server_name_to_ip - Resolve UNC server name to ip address. |
| 37 | * @unc: UNC path specifying the server | 37 | * @unc: UNC path specifying the server (with '/' as delimiter) |
| 38 | * @ip_addr: Where to return the IP address. | 38 | * @ip_addr: Where to return the IP address. |
| 39 | * | 39 | * |
| 40 | * The IP address will be returned in string form, and the caller is | 40 | * The IP address will be returned in string form, and the caller is |
| @@ -64,7 +64,7 @@ dns_resolve_server_name_to_ip(const char *unc, char **ip_addr) | |||
| 64 | hostname = unc + 2; | 64 | hostname = unc + 2; |
| 65 | 65 | ||
| 66 | /* Search for server name delimiter */ | 66 | /* Search for server name delimiter */ |
| 67 | sep = memchr(hostname, '\\', len); | 67 | sep = memchr(hostname, '/', len); |
| 68 | if (sep) | 68 | if (sep) |
| 69 | len = sep - hostname; | 69 | len = sep - hostname; |
| 70 | else | 70 | else |
diff --git a/fs/efivarfs/file.c b/fs/efivarfs/file.c index bfb531564319..8dd524f32284 100644 --- a/fs/efivarfs/file.c +++ b/fs/efivarfs/file.c | |||
| @@ -44,8 +44,11 @@ static ssize_t efivarfs_file_write(struct file *file, | |||
| 44 | 44 | ||
| 45 | bytes = efivar_entry_set_get_size(var, attributes, &datasize, | 45 | bytes = efivar_entry_set_get_size(var, attributes, &datasize, |
| 46 | data, &set); | 46 | data, &set); |
| 47 | if (!set && bytes) | 47 | if (!set && bytes) { |
| 48 | if (bytes == -ENOENT) | ||
| 49 | bytes = -EIO; | ||
| 48 | goto out; | 50 | goto out; |
| 51 | } | ||
| 49 | 52 | ||
| 50 | if (bytes == -ENOENT) { | 53 | if (bytes == -ENOENT) { |
| 51 | drop_nlink(inode); | 54 | drop_nlink(inode); |
| @@ -76,7 +79,14 @@ static ssize_t efivarfs_file_read(struct file *file, char __user *userbuf, | |||
| 76 | int err; | 79 | int err; |
| 77 | 80 | ||
| 78 | err = efivar_entry_size(var, &datasize); | 81 | err = efivar_entry_size(var, &datasize); |
| 79 | if (err) | 82 | |
| 83 | /* | ||
| 84 | * efivarfs represents uncommitted variables with | ||
| 85 | * zero-length files. Reading them should return EOF. | ||
| 86 | */ | ||
| 87 | if (err == -ENOENT) | ||
| 88 | return 0; | ||
| 89 | else if (err) | ||
| 80 | return err; | 90 | return err; |
| 81 | 91 | ||
| 82 | data = kmalloc(datasize + sizeof(attributes), GFP_KERNEL); | 92 | data = kmalloc(datasize + sizeof(attributes), GFP_KERNEL); |
diff --git a/fs/hpfs/dir.c b/fs/hpfs/dir.c index 546f6d39713a..834ac13c04b7 100644 --- a/fs/hpfs/dir.c +++ b/fs/hpfs/dir.c | |||
| @@ -33,25 +33,27 @@ static loff_t hpfs_dir_lseek(struct file *filp, loff_t off, int whence) | |||
| 33 | if (whence == SEEK_DATA || whence == SEEK_HOLE) | 33 | if (whence == SEEK_DATA || whence == SEEK_HOLE) |
| 34 | return -EINVAL; | 34 | return -EINVAL; |
| 35 | 35 | ||
| 36 | mutex_lock(&i->i_mutex); | ||
| 36 | hpfs_lock(s); | 37 | hpfs_lock(s); |
| 37 | 38 | ||
| 38 | /*printk("dir lseek\n");*/ | 39 | /*printk("dir lseek\n");*/ |
| 39 | if (new_off == 0 || new_off == 1 || new_off == 11 || new_off == 12 || new_off == 13) goto ok; | 40 | if (new_off == 0 || new_off == 1 || new_off == 11 || new_off == 12 || new_off == 13) goto ok; |
| 40 | mutex_lock(&i->i_mutex); | ||
| 41 | pos = ((loff_t) hpfs_de_as_down_as_possible(s, hpfs_inode->i_dno) << 4) + 1; | 41 | pos = ((loff_t) hpfs_de_as_down_as_possible(s, hpfs_inode->i_dno) << 4) + 1; |
| 42 | while (pos != new_off) { | 42 | while (pos != new_off) { |
| 43 | if (map_pos_dirent(i, &pos, &qbh)) hpfs_brelse4(&qbh); | 43 | if (map_pos_dirent(i, &pos, &qbh)) hpfs_brelse4(&qbh); |
| 44 | else goto fail; | 44 | else goto fail; |
| 45 | if (pos == 12) goto fail; | 45 | if (pos == 12) goto fail; |
| 46 | } | 46 | } |
| 47 | mutex_unlock(&i->i_mutex); | 47 | hpfs_add_pos(i, &filp->f_pos); |
| 48 | ok: | 48 | ok: |
| 49 | filp->f_pos = new_off; | ||
| 49 | hpfs_unlock(s); | 50 | hpfs_unlock(s); |
| 50 | return filp->f_pos = new_off; | ||
| 51 | fail: | ||
| 52 | mutex_unlock(&i->i_mutex); | 51 | mutex_unlock(&i->i_mutex); |
| 52 | return new_off; | ||
| 53 | fail: | ||
| 53 | /*printk("illegal lseek: %016llx\n", new_off);*/ | 54 | /*printk("illegal lseek: %016llx\n", new_off);*/ |
| 54 | hpfs_unlock(s); | 55 | hpfs_unlock(s); |
| 56 | mutex_unlock(&i->i_mutex); | ||
| 55 | return -ESPIPE; | 57 | return -ESPIPE; |
| 56 | } | 58 | } |
| 57 | 59 | ||
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c index 4e2fe714d5c2..d7ba5616989c 100644 --- a/fs/nfs/nfs4proc.c +++ b/fs/nfs/nfs4proc.c | |||
| @@ -1078,7 +1078,7 @@ static struct nfs4_state *nfs4_try_open_cached(struct nfs4_opendata *opendata) | |||
| 1078 | struct nfs4_state *state = opendata->state; | 1078 | struct nfs4_state *state = opendata->state; |
| 1079 | struct nfs_inode *nfsi = NFS_I(state->inode); | 1079 | struct nfs_inode *nfsi = NFS_I(state->inode); |
| 1080 | struct nfs_delegation *delegation; | 1080 | struct nfs_delegation *delegation; |
| 1081 | int open_mode = opendata->o_arg.open_flags & (O_EXCL|O_TRUNC); | 1081 | int open_mode = opendata->o_arg.open_flags; |
| 1082 | fmode_t fmode = opendata->o_arg.fmode; | 1082 | fmode_t fmode = opendata->o_arg.fmode; |
| 1083 | nfs4_stateid stateid; | 1083 | nfs4_stateid stateid; |
| 1084 | int ret = -EAGAIN; | 1084 | int ret = -EAGAIN; |
diff --git a/fs/nfs/super.c b/fs/nfs/super.c index a366107a7331..2d7525fbcf25 100644 --- a/fs/nfs/super.c +++ b/fs/nfs/super.c | |||
| @@ -1942,6 +1942,7 @@ static int nfs23_validate_mount_data(void *options, | |||
| 1942 | args->namlen = data->namlen; | 1942 | args->namlen = data->namlen; |
| 1943 | args->bsize = data->bsize; | 1943 | args->bsize = data->bsize; |
| 1944 | 1944 | ||
| 1945 | args->auth_flavors[0] = RPC_AUTH_UNIX; | ||
| 1945 | if (data->flags & NFS_MOUNT_SECFLAVOUR) | 1946 | if (data->flags & NFS_MOUNT_SECFLAVOUR) |
| 1946 | args->auth_flavors[0] = data->pseudoflavor; | 1947 | args->auth_flavors[0] = data->pseudoflavor; |
| 1947 | if (!args->nfs_server.hostname) | 1948 | if (!args->nfs_server.hostname) |
| @@ -2637,6 +2638,7 @@ static int nfs4_validate_mount_data(void *options, | |||
| 2637 | goto out_no_address; | 2638 | goto out_no_address; |
| 2638 | args->nfs_server.port = ntohs(((struct sockaddr_in *)sap)->sin_port); | 2639 | args->nfs_server.port = ntohs(((struct sockaddr_in *)sap)->sin_port); |
| 2639 | 2640 | ||
| 2641 | args->auth_flavors[0] = RPC_AUTH_UNIX; | ||
| 2640 | if (data->auth_flavourlen) { | 2642 | if (data->auth_flavourlen) { |
| 2641 | if (data->auth_flavourlen > 1) | 2643 | if (data->auth_flavourlen > 1) |
| 2642 | goto out_inval_auth; | 2644 | goto out_inval_auth; |
diff --git a/fs/pnode.c b/fs/pnode.c index 3d2a7141b87a..9af0df15256e 100644 --- a/fs/pnode.c +++ b/fs/pnode.c | |||
| @@ -83,7 +83,8 @@ static int do_make_slave(struct mount *mnt) | |||
| 83 | if (peer_mnt == mnt) | 83 | if (peer_mnt == mnt) |
| 84 | peer_mnt = NULL; | 84 | peer_mnt = NULL; |
| 85 | } | 85 | } |
| 86 | if (IS_MNT_SHARED(mnt) && list_empty(&mnt->mnt_share)) | 86 | if (mnt->mnt_group_id && IS_MNT_SHARED(mnt) && |
| 87 | list_empty(&mnt->mnt_share)) | ||
| 87 | mnt_release_group_id(mnt); | 88 | mnt_release_group_id(mnt); |
| 88 | 89 | ||
| 89 | list_del_init(&mnt->mnt_share); | 90 | list_del_init(&mnt->mnt_share); |
diff --git a/fs/qnx6/dir.c b/fs/qnx6/dir.c index 8798d065e400..afa6be6fc397 100644 --- a/fs/qnx6/dir.c +++ b/fs/qnx6/dir.c | |||
| @@ -120,7 +120,7 @@ static int qnx6_readdir(struct file *filp, void *dirent, filldir_t filldir) | |||
| 120 | struct inode *inode = file_inode(filp); | 120 | struct inode *inode = file_inode(filp); |
| 121 | struct super_block *s = inode->i_sb; | 121 | struct super_block *s = inode->i_sb; |
| 122 | struct qnx6_sb_info *sbi = QNX6_SB(s); | 122 | struct qnx6_sb_info *sbi = QNX6_SB(s); |
| 123 | loff_t pos = filp->f_pos & (QNX6_DIR_ENTRY_SIZE - 1); | 123 | loff_t pos = filp->f_pos & ~(QNX6_DIR_ENTRY_SIZE - 1); |
| 124 | unsigned long npages = dir_pages(inode); | 124 | unsigned long npages = dir_pages(inode); |
| 125 | unsigned long n = pos >> PAGE_CACHE_SHIFT; | 125 | unsigned long n = pos >> PAGE_CACHE_SHIFT; |
| 126 | unsigned start = (pos & ~PAGE_CACHE_MASK) / QNX6_DIR_ENTRY_SIZE; | 126 | unsigned start = (pos & ~PAGE_CACHE_MASK) / QNX6_DIR_ENTRY_SIZE; |
diff --git a/fs/reiserfs/dir.c b/fs/reiserfs/dir.c index 66c53b642a88..6c2d136561cb 100644 --- a/fs/reiserfs/dir.c +++ b/fs/reiserfs/dir.c | |||
| @@ -204,6 +204,8 @@ int reiserfs_readdir_dentry(struct dentry *dentry, void *dirent, | |||
| 204 | next_pos = deh_offset(deh) + 1; | 204 | next_pos = deh_offset(deh) + 1; |
| 205 | 205 | ||
| 206 | if (item_moved(&tmp_ih, &path_to_entry)) { | 206 | if (item_moved(&tmp_ih, &path_to_entry)) { |
| 207 | set_cpu_key_k_offset(&pos_key, | ||
| 208 | next_pos); | ||
| 207 | goto research; | 209 | goto research; |
| 208 | } | 210 | } |
| 209 | } /* for */ | 211 | } /* for */ |
diff --git a/fs/reiserfs/inode.c b/fs/reiserfs/inode.c index 77d6d47abc83..f844533792ee 100644 --- a/fs/reiserfs/inode.c +++ b/fs/reiserfs/inode.c | |||
| @@ -1811,11 +1811,16 @@ int reiserfs_new_inode(struct reiserfs_transaction_handle *th, | |||
| 1811 | TYPE_STAT_DATA, SD_SIZE, MAX_US_INT); | 1811 | TYPE_STAT_DATA, SD_SIZE, MAX_US_INT); |
| 1812 | memcpy(INODE_PKEY(inode), &(ih.ih_key), KEY_SIZE); | 1812 | memcpy(INODE_PKEY(inode), &(ih.ih_key), KEY_SIZE); |
| 1813 | args.dirid = le32_to_cpu(ih.ih_key.k_dir_id); | 1813 | args.dirid = le32_to_cpu(ih.ih_key.k_dir_id); |
| 1814 | if (insert_inode_locked4(inode, args.objectid, | 1814 | |
| 1815 | reiserfs_find_actor, &args) < 0) { | 1815 | reiserfs_write_unlock(inode->i_sb); |
| 1816 | err = insert_inode_locked4(inode, args.objectid, | ||
| 1817 | reiserfs_find_actor, &args); | ||
| 1818 | reiserfs_write_lock(inode->i_sb); | ||
| 1819 | if (err) { | ||
| 1816 | err = -EINVAL; | 1820 | err = -EINVAL; |
| 1817 | goto out_bad_inode; | 1821 | goto out_bad_inode; |
| 1818 | } | 1822 | } |
| 1823 | |||
| 1819 | if (old_format_only(sb)) | 1824 | if (old_format_only(sb)) |
| 1820 | /* not a perfect generation count, as object ids can be reused, but | 1825 | /* not a perfect generation count, as object ids can be reused, but |
| 1821 | ** this is as good as reiserfs can do right now. | 1826 | ** this is as good as reiserfs can do right now. |
diff --git a/fs/reiserfs/xattr.c b/fs/reiserfs/xattr.c index 4cce1d9552fb..821bcf70e467 100644 --- a/fs/reiserfs/xattr.c +++ b/fs/reiserfs/xattr.c | |||
| @@ -318,7 +318,19 @@ static int delete_one_xattr(struct dentry *dentry, void *data) | |||
| 318 | static int chown_one_xattr(struct dentry *dentry, void *data) | 318 | static int chown_one_xattr(struct dentry *dentry, void *data) |
| 319 | { | 319 | { |
| 320 | struct iattr *attrs = data; | 320 | struct iattr *attrs = data; |
| 321 | return reiserfs_setattr(dentry, attrs); | 321 | int ia_valid = attrs->ia_valid; |
| 322 | int err; | ||
| 323 | |||
| 324 | /* | ||
| 325 | * We only want the ownership bits. Otherwise, we'll do | ||
| 326 | * things like change a directory to a regular file if | ||
| 327 | * ATTR_MODE is set. | ||
| 328 | */ | ||
| 329 | attrs->ia_valid &= (ATTR_UID|ATTR_GID); | ||
| 330 | err = reiserfs_setattr(dentry, attrs); | ||
| 331 | attrs->ia_valid = ia_valid; | ||
| 332 | |||
| 333 | return err; | ||
| 322 | } | 334 | } |
| 323 | 335 | ||
| 324 | /* No i_mutex, but the inode is unconnected. */ | 336 | /* No i_mutex, but the inode is unconnected. */ |
diff --git a/fs/reiserfs/xattr_acl.c b/fs/reiserfs/xattr_acl.c index d7c01ef64eda..6c8767fdfc6a 100644 --- a/fs/reiserfs/xattr_acl.c +++ b/fs/reiserfs/xattr_acl.c | |||
| @@ -443,6 +443,9 @@ int reiserfs_acl_chmod(struct inode *inode) | |||
| 443 | int depth; | 443 | int depth; |
| 444 | int error; | 444 | int error; |
| 445 | 445 | ||
| 446 | if (IS_PRIVATE(inode)) | ||
| 447 | return 0; | ||
| 448 | |||
| 446 | if (S_ISLNK(inode->i_mode)) | 449 | if (S_ISLNK(inode->i_mode)) |
| 447 | return -EOPNOTSUPP; | 450 | return -EOPNOTSUPP; |
| 448 | 451 | ||
diff --git a/fs/xfs/xfs_attr_leaf.c b/fs/xfs/xfs_attr_leaf.c index 0bce1b348580..d788302e506a 100644 --- a/fs/xfs/xfs_attr_leaf.c +++ b/fs/xfs/xfs_attr_leaf.c | |||
| @@ -1412,7 +1412,7 @@ xfs_attr3_leaf_add_work( | |||
| 1412 | name_rmt->valuelen = 0; | 1412 | name_rmt->valuelen = 0; |
| 1413 | name_rmt->valueblk = 0; | 1413 | name_rmt->valueblk = 0; |
| 1414 | args->rmtblkno = 1; | 1414 | args->rmtblkno = 1; |
| 1415 | args->rmtblkcnt = XFS_B_TO_FSB(mp, args->valuelen); | 1415 | args->rmtblkcnt = xfs_attr3_rmt_blocks(mp, args->valuelen); |
| 1416 | } | 1416 | } |
| 1417 | xfs_trans_log_buf(args->trans, bp, | 1417 | xfs_trans_log_buf(args->trans, bp, |
| 1418 | XFS_DA_LOGRANGE(leaf, xfs_attr3_leaf_name(leaf, args->index), | 1418 | XFS_DA_LOGRANGE(leaf, xfs_attr3_leaf_name(leaf, args->index), |
| @@ -1445,11 +1445,12 @@ xfs_attr3_leaf_add_work( | |||
| 1445 | STATIC void | 1445 | STATIC void |
| 1446 | xfs_attr3_leaf_compact( | 1446 | xfs_attr3_leaf_compact( |
| 1447 | struct xfs_da_args *args, | 1447 | struct xfs_da_args *args, |
| 1448 | struct xfs_attr3_icleaf_hdr *ichdr_d, | 1448 | struct xfs_attr3_icleaf_hdr *ichdr_dst, |
| 1449 | struct xfs_buf *bp) | 1449 | struct xfs_buf *bp) |
| 1450 | { | 1450 | { |
| 1451 | xfs_attr_leafblock_t *leaf_s, *leaf_d; | 1451 | struct xfs_attr_leafblock *leaf_src; |
| 1452 | struct xfs_attr3_icleaf_hdr ichdr_s; | 1452 | struct xfs_attr_leafblock *leaf_dst; |
| 1453 | struct xfs_attr3_icleaf_hdr ichdr_src; | ||
| 1453 | struct xfs_trans *trans = args->trans; | 1454 | struct xfs_trans *trans = args->trans; |
| 1454 | struct xfs_mount *mp = trans->t_mountp; | 1455 | struct xfs_mount *mp = trans->t_mountp; |
| 1455 | char *tmpbuffer; | 1456 | char *tmpbuffer; |
| @@ -1457,29 +1458,38 @@ xfs_attr3_leaf_compact( | |||
| 1457 | trace_xfs_attr_leaf_compact(args); | 1458 | trace_xfs_attr_leaf_compact(args); |
| 1458 | 1459 | ||
| 1459 | tmpbuffer = kmem_alloc(XFS_LBSIZE(mp), KM_SLEEP); | 1460 | tmpbuffer = kmem_alloc(XFS_LBSIZE(mp), KM_SLEEP); |
| 1460 | ASSERT(tmpbuffer != NULL); | ||
| 1461 | memcpy(tmpbuffer, bp->b_addr, XFS_LBSIZE(mp)); | 1461 | memcpy(tmpbuffer, bp->b_addr, XFS_LBSIZE(mp)); |
| 1462 | memset(bp->b_addr, 0, XFS_LBSIZE(mp)); | 1462 | memset(bp->b_addr, 0, XFS_LBSIZE(mp)); |
| 1463 | leaf_src = (xfs_attr_leafblock_t *)tmpbuffer; | ||
| 1464 | leaf_dst = bp->b_addr; | ||
| 1463 | 1465 | ||
| 1464 | /* | 1466 | /* |
| 1465 | * Copy basic information | 1467 | * Copy the on-disk header back into the destination buffer to ensure |
| 1468 | * all the information in the header that is not part of the incore | ||
| 1469 | * header structure is preserved. | ||
| 1466 | */ | 1470 | */ |
| 1467 | leaf_s = (xfs_attr_leafblock_t *)tmpbuffer; | 1471 | memcpy(bp->b_addr, tmpbuffer, xfs_attr3_leaf_hdr_size(leaf_src)); |
| 1468 | leaf_d = bp->b_addr; | 1472 | |
| 1469 | ichdr_s = *ichdr_d; /* struct copy */ | 1473 | /* Initialise the incore headers */ |
| 1470 | ichdr_d->firstused = XFS_LBSIZE(mp); | 1474 | ichdr_src = *ichdr_dst; /* struct copy */ |
| 1471 | ichdr_d->usedbytes = 0; | 1475 | ichdr_dst->firstused = XFS_LBSIZE(mp); |
| 1472 | ichdr_d->count = 0; | 1476 | ichdr_dst->usedbytes = 0; |
| 1473 | ichdr_d->holes = 0; | 1477 | ichdr_dst->count = 0; |
| 1474 | ichdr_d->freemap[0].base = xfs_attr3_leaf_hdr_size(leaf_s); | 1478 | ichdr_dst->holes = 0; |
| 1475 | ichdr_d->freemap[0].size = ichdr_d->firstused - ichdr_d->freemap[0].base; | 1479 | ichdr_dst->freemap[0].base = xfs_attr3_leaf_hdr_size(leaf_src); |
| 1480 | ichdr_dst->freemap[0].size = ichdr_dst->firstused - | ||
| 1481 | ichdr_dst->freemap[0].base; | ||
| 1482 | |||
| 1483 | |||
| 1484 | /* write the header back to initialise the underlying buffer */ | ||
| 1485 | xfs_attr3_leaf_hdr_to_disk(leaf_dst, ichdr_dst); | ||
| 1476 | 1486 | ||
| 1477 | /* | 1487 | /* |
| 1478 | * Copy all entry's in the same (sorted) order, | 1488 | * Copy all entry's in the same (sorted) order, |
| 1479 | * but allocate name/value pairs packed and in sequence. | 1489 | * but allocate name/value pairs packed and in sequence. |
| 1480 | */ | 1490 | */ |
| 1481 | xfs_attr3_leaf_moveents(leaf_s, &ichdr_s, 0, leaf_d, ichdr_d, 0, | 1491 | xfs_attr3_leaf_moveents(leaf_src, &ichdr_src, 0, leaf_dst, ichdr_dst, 0, |
| 1482 | ichdr_s.count, mp); | 1492 | ichdr_src.count, mp); |
| 1483 | /* | 1493 | /* |
| 1484 | * this logs the entire buffer, but the caller must write the header | 1494 | * this logs the entire buffer, but the caller must write the header |
| 1485 | * back to the buffer when it is finished modifying it. | 1495 | * back to the buffer when it is finished modifying it. |
| @@ -2181,14 +2191,24 @@ xfs_attr3_leaf_unbalance( | |||
| 2181 | struct xfs_attr_leafblock *tmp_leaf; | 2191 | struct xfs_attr_leafblock *tmp_leaf; |
| 2182 | struct xfs_attr3_icleaf_hdr tmphdr; | 2192 | struct xfs_attr3_icleaf_hdr tmphdr; |
| 2183 | 2193 | ||
| 2184 | tmp_leaf = kmem_alloc(state->blocksize, KM_SLEEP); | 2194 | tmp_leaf = kmem_zalloc(state->blocksize, KM_SLEEP); |
| 2185 | memset(tmp_leaf, 0, state->blocksize); | ||
| 2186 | memset(&tmphdr, 0, sizeof(tmphdr)); | ||
| 2187 | 2195 | ||
| 2196 | /* | ||
| 2197 | * Copy the header into the temp leaf so that all the stuff | ||
| 2198 | * not in the incore header is present and gets copied back in | ||
| 2199 | * once we've moved all the entries. | ||
| 2200 | */ | ||
| 2201 | memcpy(tmp_leaf, save_leaf, xfs_attr3_leaf_hdr_size(save_leaf)); | ||
| 2202 | |||
| 2203 | memset(&tmphdr, 0, sizeof(tmphdr)); | ||
| 2188 | tmphdr.magic = savehdr.magic; | 2204 | tmphdr.magic = savehdr.magic; |
| 2189 | tmphdr.forw = savehdr.forw; | 2205 | tmphdr.forw = savehdr.forw; |
| 2190 | tmphdr.back = savehdr.back; | 2206 | tmphdr.back = savehdr.back; |
| 2191 | tmphdr.firstused = state->blocksize; | 2207 | tmphdr.firstused = state->blocksize; |
| 2208 | |||
| 2209 | /* write the header to the temp buffer to initialise it */ | ||
| 2210 | xfs_attr3_leaf_hdr_to_disk(tmp_leaf, &tmphdr); | ||
| 2211 | |||
| 2192 | if (xfs_attr3_leaf_order(save_blk->bp, &savehdr, | 2212 | if (xfs_attr3_leaf_order(save_blk->bp, &savehdr, |
| 2193 | drop_blk->bp, &drophdr)) { | 2213 | drop_blk->bp, &drophdr)) { |
| 2194 | xfs_attr3_leaf_moveents(drop_leaf, &drophdr, 0, | 2214 | xfs_attr3_leaf_moveents(drop_leaf, &drophdr, 0, |
| @@ -2334,8 +2354,9 @@ xfs_attr3_leaf_lookup_int( | |||
| 2334 | args->index = probe; | 2354 | args->index = probe; |
| 2335 | args->valuelen = be32_to_cpu(name_rmt->valuelen); | 2355 | args->valuelen = be32_to_cpu(name_rmt->valuelen); |
| 2336 | args->rmtblkno = be32_to_cpu(name_rmt->valueblk); | 2356 | args->rmtblkno = be32_to_cpu(name_rmt->valueblk); |
| 2337 | args->rmtblkcnt = XFS_B_TO_FSB(args->dp->i_mount, | 2357 | args->rmtblkcnt = xfs_attr3_rmt_blocks( |
| 2338 | args->valuelen); | 2358 | args->dp->i_mount, |
| 2359 | args->valuelen); | ||
| 2339 | return XFS_ERROR(EEXIST); | 2360 | return XFS_ERROR(EEXIST); |
| 2340 | } | 2361 | } |
| 2341 | } | 2362 | } |
| @@ -2386,7 +2407,8 @@ xfs_attr3_leaf_getvalue( | |||
| 2386 | ASSERT(memcmp(args->name, name_rmt->name, args->namelen) == 0); | 2407 | ASSERT(memcmp(args->name, name_rmt->name, args->namelen) == 0); |
| 2387 | valuelen = be32_to_cpu(name_rmt->valuelen); | 2408 | valuelen = be32_to_cpu(name_rmt->valuelen); |
| 2388 | args->rmtblkno = be32_to_cpu(name_rmt->valueblk); | 2409 | args->rmtblkno = be32_to_cpu(name_rmt->valueblk); |
| 2389 | args->rmtblkcnt = XFS_B_TO_FSB(args->dp->i_mount, valuelen); | 2410 | args->rmtblkcnt = xfs_attr3_rmt_blocks(args->dp->i_mount, |
| 2411 | valuelen); | ||
| 2390 | if (args->flags & ATTR_KERNOVAL) { | 2412 | if (args->flags & ATTR_KERNOVAL) { |
| 2391 | args->valuelen = valuelen; | 2413 | args->valuelen = valuelen; |
| 2392 | return 0; | 2414 | return 0; |
| @@ -2712,7 +2734,8 @@ xfs_attr3_leaf_list_int( | |||
| 2712 | args.valuelen = valuelen; | 2734 | args.valuelen = valuelen; |
| 2713 | args.value = kmem_alloc(valuelen, KM_SLEEP | KM_NOFS); | 2735 | args.value = kmem_alloc(valuelen, KM_SLEEP | KM_NOFS); |
| 2714 | args.rmtblkno = be32_to_cpu(name_rmt->valueblk); | 2736 | args.rmtblkno = be32_to_cpu(name_rmt->valueblk); |
| 2715 | args.rmtblkcnt = XFS_B_TO_FSB(args.dp->i_mount, valuelen); | 2737 | args.rmtblkcnt = xfs_attr3_rmt_blocks( |
| 2738 | args.dp->i_mount, valuelen); | ||
| 2716 | retval = xfs_attr_rmtval_get(&args); | 2739 | retval = xfs_attr_rmtval_get(&args); |
| 2717 | if (retval) | 2740 | if (retval) |
| 2718 | return retval; | 2741 | return retval; |
diff --git a/fs/xfs/xfs_attr_remote.c b/fs/xfs/xfs_attr_remote.c index dee84466dcc9..ef6b0c124528 100644 --- a/fs/xfs/xfs_attr_remote.c +++ b/fs/xfs/xfs_attr_remote.c | |||
| @@ -47,22 +47,55 @@ | |||
| 47 | * Each contiguous block has a header, so it is not just a simple attribute | 47 | * Each contiguous block has a header, so it is not just a simple attribute |
| 48 | * length to FSB conversion. | 48 | * length to FSB conversion. |
| 49 | */ | 49 | */ |
| 50 | static int | 50 | int |
| 51 | xfs_attr3_rmt_blocks( | 51 | xfs_attr3_rmt_blocks( |
| 52 | struct xfs_mount *mp, | 52 | struct xfs_mount *mp, |
| 53 | int attrlen) | 53 | int attrlen) |
| 54 | { | 54 | { |
| 55 | int buflen = XFS_ATTR3_RMT_BUF_SPACE(mp, | 55 | if (xfs_sb_version_hascrc(&mp->m_sb)) { |
| 56 | mp->m_sb.sb_blocksize); | 56 | int buflen = XFS_ATTR3_RMT_BUF_SPACE(mp, mp->m_sb.sb_blocksize); |
| 57 | return (attrlen + buflen - 1) / buflen; | 57 | return (attrlen + buflen - 1) / buflen; |
| 58 | } | ||
| 59 | return XFS_B_TO_FSB(mp, attrlen); | ||
| 60 | } | ||
| 61 | |||
| 62 | /* | ||
| 63 | * Checking of the remote attribute header is split into two parts. The verifier | ||
| 64 | * does CRC, location and bounds checking, the unpacking function checks the | ||
| 65 | * attribute parameters and owner. | ||
| 66 | */ | ||
| 67 | static bool | ||
| 68 | xfs_attr3_rmt_hdr_ok( | ||
| 69 | struct xfs_mount *mp, | ||
| 70 | void *ptr, | ||
| 71 | xfs_ino_t ino, | ||
| 72 | uint32_t offset, | ||
| 73 | uint32_t size, | ||
| 74 | xfs_daddr_t bno) | ||
| 75 | { | ||
| 76 | struct xfs_attr3_rmt_hdr *rmt = ptr; | ||
| 77 | |||
| 78 | if (bno != be64_to_cpu(rmt->rm_blkno)) | ||
| 79 | return false; | ||
| 80 | if (offset != be32_to_cpu(rmt->rm_offset)) | ||
| 81 | return false; | ||
| 82 | if (size != be32_to_cpu(rmt->rm_bytes)) | ||
| 83 | return false; | ||
| 84 | if (ino != be64_to_cpu(rmt->rm_owner)) | ||
| 85 | return false; | ||
| 86 | |||
| 87 | /* ok */ | ||
| 88 | return true; | ||
| 58 | } | 89 | } |
| 59 | 90 | ||
| 60 | static bool | 91 | static bool |
| 61 | xfs_attr3_rmt_verify( | 92 | xfs_attr3_rmt_verify( |
| 62 | struct xfs_buf *bp) | 93 | struct xfs_mount *mp, |
| 94 | void *ptr, | ||
| 95 | int fsbsize, | ||
| 96 | xfs_daddr_t bno) | ||
| 63 | { | 97 | { |
| 64 | struct xfs_mount *mp = bp->b_target->bt_mount; | 98 | struct xfs_attr3_rmt_hdr *rmt = ptr; |
| 65 | struct xfs_attr3_rmt_hdr *rmt = bp->b_addr; | ||
| 66 | 99 | ||
| 67 | if (!xfs_sb_version_hascrc(&mp->m_sb)) | 100 | if (!xfs_sb_version_hascrc(&mp->m_sb)) |
| 68 | return false; | 101 | return false; |
| @@ -70,7 +103,9 @@ xfs_attr3_rmt_verify( | |||
| 70 | return false; | 103 | return false; |
| 71 | if (!uuid_equal(&rmt->rm_uuid, &mp->m_sb.sb_uuid)) | 104 | if (!uuid_equal(&rmt->rm_uuid, &mp->m_sb.sb_uuid)) |
| 72 | return false; | 105 | return false; |
| 73 | if (bp->b_bn != be64_to_cpu(rmt->rm_blkno)) | 106 | if (be64_to_cpu(rmt->rm_blkno) != bno) |
| 107 | return false; | ||
| 108 | if (be32_to_cpu(rmt->rm_bytes) > fsbsize - sizeof(*rmt)) | ||
| 74 | return false; | 109 | return false; |
| 75 | if (be32_to_cpu(rmt->rm_offset) + | 110 | if (be32_to_cpu(rmt->rm_offset) + |
| 76 | be32_to_cpu(rmt->rm_bytes) >= XATTR_SIZE_MAX) | 111 | be32_to_cpu(rmt->rm_bytes) >= XATTR_SIZE_MAX) |
| @@ -86,17 +121,40 @@ xfs_attr3_rmt_read_verify( | |||
| 86 | struct xfs_buf *bp) | 121 | struct xfs_buf *bp) |
| 87 | { | 122 | { |
| 88 | struct xfs_mount *mp = bp->b_target->bt_mount; | 123 | struct xfs_mount *mp = bp->b_target->bt_mount; |
| 124 | char *ptr; | ||
| 125 | int len; | ||
| 126 | bool corrupt = false; | ||
| 127 | xfs_daddr_t bno; | ||
| 89 | 128 | ||
| 90 | /* no verification of non-crc buffers */ | 129 | /* no verification of non-crc buffers */ |
| 91 | if (!xfs_sb_version_hascrc(&mp->m_sb)) | 130 | if (!xfs_sb_version_hascrc(&mp->m_sb)) |
| 92 | return; | 131 | return; |
| 93 | 132 | ||
| 94 | if (!xfs_verify_cksum(bp->b_addr, BBTOB(bp->b_length), | 133 | ptr = bp->b_addr; |
| 95 | XFS_ATTR3_RMT_CRC_OFF) || | 134 | bno = bp->b_bn; |
| 96 | !xfs_attr3_rmt_verify(bp)) { | 135 | len = BBTOB(bp->b_length); |
| 136 | ASSERT(len >= XFS_LBSIZE(mp)); | ||
| 137 | |||
| 138 | while (len > 0) { | ||
| 139 | if (!xfs_verify_cksum(ptr, XFS_LBSIZE(mp), | ||
| 140 | XFS_ATTR3_RMT_CRC_OFF)) { | ||
| 141 | corrupt = true; | ||
| 142 | break; | ||
| 143 | } | ||
| 144 | if (!xfs_attr3_rmt_verify(mp, ptr, XFS_LBSIZE(mp), bno)) { | ||
| 145 | corrupt = true; | ||
| 146 | break; | ||
| 147 | } | ||
| 148 | len -= XFS_LBSIZE(mp); | ||
| 149 | ptr += XFS_LBSIZE(mp); | ||
| 150 | bno += mp->m_bsize; | ||
| 151 | } | ||
| 152 | |||
| 153 | if (corrupt) { | ||
| 97 | XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp, bp->b_addr); | 154 | XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp, bp->b_addr); |
| 98 | xfs_buf_ioerror(bp, EFSCORRUPTED); | 155 | xfs_buf_ioerror(bp, EFSCORRUPTED); |
| 99 | } | 156 | } else |
| 157 | ASSERT(len == 0); | ||
| 100 | } | 158 | } |
| 101 | 159 | ||
| 102 | static void | 160 | static void |
| @@ -105,23 +163,39 @@ xfs_attr3_rmt_write_verify( | |||
| 105 | { | 163 | { |
| 106 | struct xfs_mount *mp = bp->b_target->bt_mount; | 164 | struct xfs_mount *mp = bp->b_target->bt_mount; |
| 107 | struct xfs_buf_log_item *bip = bp->b_fspriv; | 165 | struct xfs_buf_log_item *bip = bp->b_fspriv; |
| 166 | char *ptr; | ||
| 167 | int len; | ||
| 168 | xfs_daddr_t bno; | ||
| 108 | 169 | ||
| 109 | /* no verification of non-crc buffers */ | 170 | /* no verification of non-crc buffers */ |
| 110 | if (!xfs_sb_version_hascrc(&mp->m_sb)) | 171 | if (!xfs_sb_version_hascrc(&mp->m_sb)) |
| 111 | return; | 172 | return; |
| 112 | 173 | ||
| 113 | if (!xfs_attr3_rmt_verify(bp)) { | 174 | ptr = bp->b_addr; |
| 114 | XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp, bp->b_addr); | 175 | bno = bp->b_bn; |
| 115 | xfs_buf_ioerror(bp, EFSCORRUPTED); | 176 | len = BBTOB(bp->b_length); |
| 116 | return; | 177 | ASSERT(len >= XFS_LBSIZE(mp)); |
| 117 | } | 178 | |
| 179 | while (len > 0) { | ||
| 180 | if (!xfs_attr3_rmt_verify(mp, ptr, XFS_LBSIZE(mp), bno)) { | ||
| 181 | XFS_CORRUPTION_ERROR(__func__, | ||
| 182 | XFS_ERRLEVEL_LOW, mp, bp->b_addr); | ||
| 183 | xfs_buf_ioerror(bp, EFSCORRUPTED); | ||
| 184 | return; | ||
| 185 | } | ||
| 186 | if (bip) { | ||
| 187 | struct xfs_attr3_rmt_hdr *rmt; | ||
| 188 | |||
| 189 | rmt = (struct xfs_attr3_rmt_hdr *)ptr; | ||
| 190 | rmt->rm_lsn = cpu_to_be64(bip->bli_item.li_lsn); | ||
| 191 | } | ||
| 192 | xfs_update_cksum(ptr, XFS_LBSIZE(mp), XFS_ATTR3_RMT_CRC_OFF); | ||
| 118 | 193 | ||
| 119 | if (bip) { | 194 | len -= XFS_LBSIZE(mp); |
| 120 | struct xfs_attr3_rmt_hdr *rmt = bp->b_addr; | 195 | ptr += XFS_LBSIZE(mp); |
| 121 | rmt->rm_lsn = cpu_to_be64(bip->bli_item.li_lsn); | 196 | bno += mp->m_bsize; |
| 122 | } | 197 | } |
| 123 | xfs_update_cksum(bp->b_addr, BBTOB(bp->b_length), | 198 | ASSERT(len == 0); |
| 124 | XFS_ATTR3_RMT_CRC_OFF); | ||
| 125 | } | 199 | } |
| 126 | 200 | ||
| 127 | const struct xfs_buf_ops xfs_attr3_rmt_buf_ops = { | 201 | const struct xfs_buf_ops xfs_attr3_rmt_buf_ops = { |
| @@ -129,15 +203,16 @@ const struct xfs_buf_ops xfs_attr3_rmt_buf_ops = { | |||
| 129 | .verify_write = xfs_attr3_rmt_write_verify, | 203 | .verify_write = xfs_attr3_rmt_write_verify, |
| 130 | }; | 204 | }; |
| 131 | 205 | ||
| 132 | static int | 206 | STATIC int |
| 133 | xfs_attr3_rmt_hdr_set( | 207 | xfs_attr3_rmt_hdr_set( |
| 134 | struct xfs_mount *mp, | 208 | struct xfs_mount *mp, |
| 209 | void *ptr, | ||
| 135 | xfs_ino_t ino, | 210 | xfs_ino_t ino, |
| 136 | uint32_t offset, | 211 | uint32_t offset, |
| 137 | uint32_t size, | 212 | uint32_t size, |
| 138 | struct xfs_buf *bp) | 213 | xfs_daddr_t bno) |
| 139 | { | 214 | { |
| 140 | struct xfs_attr3_rmt_hdr *rmt = bp->b_addr; | 215 | struct xfs_attr3_rmt_hdr *rmt = ptr; |
| 141 | 216 | ||
| 142 | if (!xfs_sb_version_hascrc(&mp->m_sb)) | 217 | if (!xfs_sb_version_hascrc(&mp->m_sb)) |
| 143 | return 0; | 218 | return 0; |
| @@ -147,36 +222,107 @@ xfs_attr3_rmt_hdr_set( | |||
| 147 | rmt->rm_bytes = cpu_to_be32(size); | 222 | rmt->rm_bytes = cpu_to_be32(size); |
| 148 | uuid_copy(&rmt->rm_uuid, &mp->m_sb.sb_uuid); | 223 | uuid_copy(&rmt->rm_uuid, &mp->m_sb.sb_uuid); |
| 149 | rmt->rm_owner = cpu_to_be64(ino); | 224 | rmt->rm_owner = cpu_to_be64(ino); |
| 150 | rmt->rm_blkno = cpu_to_be64(bp->b_bn); | 225 | rmt->rm_blkno = cpu_to_be64(bno); |
| 151 | bp->b_ops = &xfs_attr3_rmt_buf_ops; | ||
| 152 | 226 | ||
| 153 | return sizeof(struct xfs_attr3_rmt_hdr); | 227 | return sizeof(struct xfs_attr3_rmt_hdr); |
| 154 | } | 228 | } |
| 155 | 229 | ||
| 156 | /* | 230 | /* |
| 157 | * Checking of the remote attribute header is split into two parts. the verifier | 231 | * Helper functions to copy attribute data in and out of the one disk extents |
| 158 | * does CRC, location and bounds checking, the unpacking function checks the | ||
| 159 | * attribute parameters and owner. | ||
| 160 | */ | 232 | */ |
| 161 | static bool | 233 | STATIC int |
| 162 | xfs_attr3_rmt_hdr_ok( | 234 | xfs_attr_rmtval_copyout( |
| 163 | struct xfs_mount *mp, | 235 | struct xfs_mount *mp, |
| 164 | xfs_ino_t ino, | 236 | struct xfs_buf *bp, |
| 165 | uint32_t offset, | 237 | xfs_ino_t ino, |
| 166 | uint32_t size, | 238 | int *offset, |
| 167 | struct xfs_buf *bp) | 239 | int *valuelen, |
| 240 | char **dst) | ||
| 168 | { | 241 | { |
| 169 | struct xfs_attr3_rmt_hdr *rmt = bp->b_addr; | 242 | char *src = bp->b_addr; |
| 243 | xfs_daddr_t bno = bp->b_bn; | ||
| 244 | int len = BBTOB(bp->b_length); | ||
| 170 | 245 | ||
| 171 | if (offset != be32_to_cpu(rmt->rm_offset)) | 246 | ASSERT(len >= XFS_LBSIZE(mp)); |
| 172 | return false; | ||
| 173 | if (size != be32_to_cpu(rmt->rm_bytes)) | ||
| 174 | return false; | ||
| 175 | if (ino != be64_to_cpu(rmt->rm_owner)) | ||
| 176 | return false; | ||
| 177 | 247 | ||
| 178 | /* ok */ | 248 | while (len > 0 && *valuelen > 0) { |
| 179 | return true; | 249 | int hdr_size = 0; |
| 250 | int byte_cnt = XFS_ATTR3_RMT_BUF_SPACE(mp, XFS_LBSIZE(mp)); | ||
| 251 | |||
| 252 | byte_cnt = min_t(int, *valuelen, byte_cnt); | ||
| 253 | |||
| 254 | if (xfs_sb_version_hascrc(&mp->m_sb)) { | ||
| 255 | if (!xfs_attr3_rmt_hdr_ok(mp, src, ino, *offset, | ||
| 256 | byte_cnt, bno)) { | ||
| 257 | xfs_alert(mp, | ||
| 258 | "remote attribute header mismatch bno/off/len/owner (0x%llx/0x%x/Ox%x/0x%llx)", | ||
| 259 | bno, *offset, byte_cnt, ino); | ||
| 260 | return EFSCORRUPTED; | ||
| 261 | } | ||
| 262 | hdr_size = sizeof(struct xfs_attr3_rmt_hdr); | ||
| 263 | } | ||
| 264 | |||
| 265 | memcpy(*dst, src + hdr_size, byte_cnt); | ||
| 266 | |||
| 267 | /* roll buffer forwards */ | ||
| 268 | len -= XFS_LBSIZE(mp); | ||
| 269 | src += XFS_LBSIZE(mp); | ||
| 270 | bno += mp->m_bsize; | ||
| 271 | |||
| 272 | /* roll attribute data forwards */ | ||
| 273 | *valuelen -= byte_cnt; | ||
| 274 | *dst += byte_cnt; | ||
| 275 | *offset += byte_cnt; | ||
| 276 | } | ||
| 277 | return 0; | ||
| 278 | } | ||
| 279 | |||
| 280 | STATIC void | ||
| 281 | xfs_attr_rmtval_copyin( | ||
| 282 | struct xfs_mount *mp, | ||
| 283 | struct xfs_buf *bp, | ||
| 284 | xfs_ino_t ino, | ||
| 285 | int *offset, | ||
| 286 | int *valuelen, | ||
| 287 | char **src) | ||
| 288 | { | ||
| 289 | char *dst = bp->b_addr; | ||
| 290 | xfs_daddr_t bno = bp->b_bn; | ||
| 291 | int len = BBTOB(bp->b_length); | ||
| 292 | |||
| 293 | ASSERT(len >= XFS_LBSIZE(mp)); | ||
| 294 | |||
| 295 | while (len > 0 && *valuelen > 0) { | ||
| 296 | int hdr_size; | ||
| 297 | int byte_cnt = XFS_ATTR3_RMT_BUF_SPACE(mp, XFS_LBSIZE(mp)); | ||
| 298 | |||
| 299 | byte_cnt = min(*valuelen, byte_cnt); | ||
| 300 | hdr_size = xfs_attr3_rmt_hdr_set(mp, dst, ino, *offset, | ||
| 301 | byte_cnt, bno); | ||
| 302 | |||
| 303 | memcpy(dst + hdr_size, *src, byte_cnt); | ||
| 304 | |||
| 305 | /* | ||
| 306 | * If this is the last block, zero the remainder of it. | ||
| 307 | * Check that we are actually the last block, too. | ||
| 308 | */ | ||
| 309 | if (byte_cnt + hdr_size < XFS_LBSIZE(mp)) { | ||
| 310 | ASSERT(*valuelen - byte_cnt == 0); | ||
| 311 | ASSERT(len == XFS_LBSIZE(mp)); | ||
| 312 | memset(dst + hdr_size + byte_cnt, 0, | ||
| 313 | XFS_LBSIZE(mp) - hdr_size - byte_cnt); | ||
| 314 | } | ||
| 315 | |||
| 316 | /* roll buffer forwards */ | ||
| 317 | len -= XFS_LBSIZE(mp); | ||
| 318 | dst += XFS_LBSIZE(mp); | ||
| 319 | bno += mp->m_bsize; | ||
| 320 | |||
| 321 | /* roll attribute data forwards */ | ||
| 322 | *valuelen -= byte_cnt; | ||
| 323 | *src += byte_cnt; | ||
| 324 | *offset += byte_cnt; | ||
| 325 | } | ||
| 180 | } | 326 | } |
| 181 | 327 | ||
| 182 | /* | 328 | /* |
| @@ -190,13 +336,12 @@ xfs_attr_rmtval_get( | |||
| 190 | struct xfs_bmbt_irec map[ATTR_RMTVALUE_MAPSIZE]; | 336 | struct xfs_bmbt_irec map[ATTR_RMTVALUE_MAPSIZE]; |
| 191 | struct xfs_mount *mp = args->dp->i_mount; | 337 | struct xfs_mount *mp = args->dp->i_mount; |
| 192 | struct xfs_buf *bp; | 338 | struct xfs_buf *bp; |
| 193 | xfs_daddr_t dblkno; | ||
| 194 | xfs_dablk_t lblkno = args->rmtblkno; | 339 | xfs_dablk_t lblkno = args->rmtblkno; |
| 195 | void *dst = args->value; | 340 | char *dst = args->value; |
| 196 | int valuelen = args->valuelen; | 341 | int valuelen = args->valuelen; |
| 197 | int nmap; | 342 | int nmap; |
| 198 | int error; | 343 | int error; |
| 199 | int blkcnt; | 344 | int blkcnt = args->rmtblkcnt; |
| 200 | int i; | 345 | int i; |
| 201 | int offset = 0; | 346 | int offset = 0; |
| 202 | 347 | ||
| @@ -207,52 +352,36 @@ xfs_attr_rmtval_get( | |||
| 207 | while (valuelen > 0) { | 352 | while (valuelen > 0) { |
| 208 | nmap = ATTR_RMTVALUE_MAPSIZE; | 353 | nmap = ATTR_RMTVALUE_MAPSIZE; |
| 209 | error = xfs_bmapi_read(args->dp, (xfs_fileoff_t)lblkno, | 354 | error = xfs_bmapi_read(args->dp, (xfs_fileoff_t)lblkno, |
| 210 | args->rmtblkcnt, map, &nmap, | 355 | blkcnt, map, &nmap, |
| 211 | XFS_BMAPI_ATTRFORK); | 356 | XFS_BMAPI_ATTRFORK); |
| 212 | if (error) | 357 | if (error) |
| 213 | return error; | 358 | return error; |
| 214 | ASSERT(nmap >= 1); | 359 | ASSERT(nmap >= 1); |
| 215 | 360 | ||
| 216 | for (i = 0; (i < nmap) && (valuelen > 0); i++) { | 361 | for (i = 0; (i < nmap) && (valuelen > 0); i++) { |
| 217 | int byte_cnt; | 362 | xfs_daddr_t dblkno; |
| 218 | char *src; | 363 | int dblkcnt; |
| 219 | 364 | ||
| 220 | ASSERT((map[i].br_startblock != DELAYSTARTBLOCK) && | 365 | ASSERT((map[i].br_startblock != DELAYSTARTBLOCK) && |
| 221 | (map[i].br_startblock != HOLESTARTBLOCK)); | 366 | (map[i].br_startblock != HOLESTARTBLOCK)); |
| 222 | dblkno = XFS_FSB_TO_DADDR(mp, map[i].br_startblock); | 367 | dblkno = XFS_FSB_TO_DADDR(mp, map[i].br_startblock); |
| 223 | blkcnt = XFS_FSB_TO_BB(mp, map[i].br_blockcount); | 368 | dblkcnt = XFS_FSB_TO_BB(mp, map[i].br_blockcount); |
| 224 | error = xfs_trans_read_buf(mp, NULL, mp->m_ddev_targp, | 369 | error = xfs_trans_read_buf(mp, NULL, mp->m_ddev_targp, |
| 225 | dblkno, blkcnt, 0, &bp, | 370 | dblkno, dblkcnt, 0, &bp, |
| 226 | &xfs_attr3_rmt_buf_ops); | 371 | &xfs_attr3_rmt_buf_ops); |
| 227 | if (error) | 372 | if (error) |
| 228 | return error; | 373 | return error; |
| 229 | 374 | ||
| 230 | byte_cnt = min_t(int, valuelen, BBTOB(bp->b_length)); | 375 | error = xfs_attr_rmtval_copyout(mp, bp, args->dp->i_ino, |
| 231 | byte_cnt = XFS_ATTR3_RMT_BUF_SPACE(mp, byte_cnt); | 376 | &offset, &valuelen, |
| 232 | 377 | &dst); | |
| 233 | src = bp->b_addr; | ||
| 234 | if (xfs_sb_version_hascrc(&mp->m_sb)) { | ||
| 235 | if (!xfs_attr3_rmt_hdr_ok(mp, args->dp->i_ino, | ||
| 236 | offset, byte_cnt, bp)) { | ||
| 237 | xfs_alert(mp, | ||
| 238 | "remote attribute header does not match required off/len/owner (0x%x/Ox%x,0x%llx)", | ||
| 239 | offset, byte_cnt, args->dp->i_ino); | ||
| 240 | xfs_buf_relse(bp); | ||
| 241 | return EFSCORRUPTED; | ||
| 242 | |||
| 243 | } | ||
| 244 | |||
| 245 | src += sizeof(struct xfs_attr3_rmt_hdr); | ||
| 246 | } | ||
| 247 | |||
| 248 | memcpy(dst, src, byte_cnt); | ||
| 249 | xfs_buf_relse(bp); | 378 | xfs_buf_relse(bp); |
| 379 | if (error) | ||
| 380 | return error; | ||
| 250 | 381 | ||
| 251 | offset += byte_cnt; | 382 | /* roll attribute extent map forwards */ |
| 252 | dst += byte_cnt; | ||
| 253 | valuelen -= byte_cnt; | ||
| 254 | |||
| 255 | lblkno += map[i].br_blockcount; | 383 | lblkno += map[i].br_blockcount; |
| 384 | blkcnt -= map[i].br_blockcount; | ||
| 256 | } | 385 | } |
| 257 | } | 386 | } |
| 258 | ASSERT(valuelen == 0); | 387 | ASSERT(valuelen == 0); |
| @@ -270,17 +399,13 @@ xfs_attr_rmtval_set( | |||
| 270 | struct xfs_inode *dp = args->dp; | 399 | struct xfs_inode *dp = args->dp; |
| 271 | struct xfs_mount *mp = dp->i_mount; | 400 | struct xfs_mount *mp = dp->i_mount; |
| 272 | struct xfs_bmbt_irec map; | 401 | struct xfs_bmbt_irec map; |
| 273 | struct xfs_buf *bp; | ||
| 274 | xfs_daddr_t dblkno; | ||
| 275 | xfs_dablk_t lblkno; | 402 | xfs_dablk_t lblkno; |
| 276 | xfs_fileoff_t lfileoff = 0; | 403 | xfs_fileoff_t lfileoff = 0; |
| 277 | void *src = args->value; | 404 | char *src = args->value; |
| 278 | int blkcnt; | 405 | int blkcnt; |
| 279 | int valuelen; | 406 | int valuelen; |
| 280 | int nmap; | 407 | int nmap; |
| 281 | int error; | 408 | int error; |
| 282 | int hdrcnt = 0; | ||
| 283 | bool crcs = xfs_sb_version_hascrc(&mp->m_sb); | ||
| 284 | int offset = 0; | 409 | int offset = 0; |
| 285 | 410 | ||
| 286 | trace_xfs_attr_rmtval_set(args); | 411 | trace_xfs_attr_rmtval_set(args); |
| @@ -289,24 +414,14 @@ xfs_attr_rmtval_set( | |||
| 289 | * Find a "hole" in the attribute address space large enough for | 414 | * Find a "hole" in the attribute address space large enough for |
| 290 | * us to drop the new attribute's value into. Because CRC enable | 415 | * us to drop the new attribute's value into. Because CRC enable |
| 291 | * attributes have headers, we can't just do a straight byte to FSB | 416 | * attributes have headers, we can't just do a straight byte to FSB |
| 292 | * conversion. We calculate the worst case block count in this case | 417 | * conversion and have to take the header space into account. |
| 293 | * and we may not need that many, so we have to handle this when | ||
| 294 | * allocating the blocks below. | ||
| 295 | */ | 418 | */ |
| 296 | if (!crcs) | 419 | blkcnt = xfs_attr3_rmt_blocks(mp, args->valuelen); |
| 297 | blkcnt = XFS_B_TO_FSB(mp, args->valuelen); | ||
| 298 | else | ||
| 299 | blkcnt = xfs_attr3_rmt_blocks(mp, args->valuelen); | ||
| 300 | |||
| 301 | error = xfs_bmap_first_unused(args->trans, args->dp, blkcnt, &lfileoff, | 420 | error = xfs_bmap_first_unused(args->trans, args->dp, blkcnt, &lfileoff, |
| 302 | XFS_ATTR_FORK); | 421 | XFS_ATTR_FORK); |
| 303 | if (error) | 422 | if (error) |
| 304 | return error; | 423 | return error; |
| 305 | 424 | ||
| 306 | /* Start with the attribute data. We'll allocate the rest afterwards. */ | ||
| 307 | if (crcs) | ||
| 308 | blkcnt = XFS_B_TO_FSB(mp, args->valuelen); | ||
| 309 | |||
| 310 | args->rmtblkno = lblkno = (xfs_dablk_t)lfileoff; | 425 | args->rmtblkno = lblkno = (xfs_dablk_t)lfileoff; |
| 311 | args->rmtblkcnt = blkcnt; | 426 | args->rmtblkcnt = blkcnt; |
| 312 | 427 | ||
| @@ -349,26 +464,6 @@ xfs_attr_rmtval_set( | |||
| 349 | (map.br_startblock != HOLESTARTBLOCK)); | 464 | (map.br_startblock != HOLESTARTBLOCK)); |
| 350 | lblkno += map.br_blockcount; | 465 | lblkno += map.br_blockcount; |
| 351 | blkcnt -= map.br_blockcount; | 466 | blkcnt -= map.br_blockcount; |
| 352 | hdrcnt++; | ||
| 353 | |||
| 354 | /* | ||
| 355 | * If we have enough blocks for the attribute data, calculate | ||
| 356 | * how many extra blocks we need for headers. We might run | ||
| 357 | * through this multiple times in the case that the additional | ||
| 358 | * headers in the blocks needed for the data fragments spills | ||
| 359 | * into requiring more blocks. e.g. for 512 byte blocks, we'll | ||
| 360 | * spill for another block every 9 headers we require in this | ||
| 361 | * loop. | ||
| 362 | */ | ||
| 363 | if (crcs && blkcnt == 0) { | ||
| 364 | int total_len; | ||
| 365 | |||
| 366 | total_len = args->valuelen + | ||
| 367 | hdrcnt * sizeof(struct xfs_attr3_rmt_hdr); | ||
| 368 | blkcnt = XFS_B_TO_FSB(mp, total_len); | ||
| 369 | blkcnt -= args->rmtblkcnt; | ||
| 370 | args->rmtblkcnt += blkcnt; | ||
| 371 | } | ||
| 372 | 467 | ||
| 373 | /* | 468 | /* |
| 374 | * Start the next trans in the chain. | 469 | * Start the next trans in the chain. |
| @@ -385,18 +480,19 @@ xfs_attr_rmtval_set( | |||
| 385 | * the INCOMPLETE flag. | 480 | * the INCOMPLETE flag. |
| 386 | */ | 481 | */ |
| 387 | lblkno = args->rmtblkno; | 482 | lblkno = args->rmtblkno; |
| 483 | blkcnt = args->rmtblkcnt; | ||
| 388 | valuelen = args->valuelen; | 484 | valuelen = args->valuelen; |
| 389 | while (valuelen > 0) { | 485 | while (valuelen > 0) { |
| 390 | int byte_cnt; | 486 | struct xfs_buf *bp; |
| 391 | char *buf; | 487 | xfs_daddr_t dblkno; |
| 488 | int dblkcnt; | ||
| 489 | |||
| 490 | ASSERT(blkcnt > 0); | ||
| 392 | 491 | ||
| 393 | /* | ||
| 394 | * Try to remember where we decided to put the value. | ||
| 395 | */ | ||
| 396 | xfs_bmap_init(args->flist, args->firstblock); | 492 | xfs_bmap_init(args->flist, args->firstblock); |
| 397 | nmap = 1; | 493 | nmap = 1; |
| 398 | error = xfs_bmapi_read(dp, (xfs_fileoff_t)lblkno, | 494 | error = xfs_bmapi_read(dp, (xfs_fileoff_t)lblkno, |
| 399 | args->rmtblkcnt, &map, &nmap, | 495 | blkcnt, &map, &nmap, |
| 400 | XFS_BMAPI_ATTRFORK); | 496 | XFS_BMAPI_ATTRFORK); |
| 401 | if (error) | 497 | if (error) |
| 402 | return(error); | 498 | return(error); |
| @@ -405,41 +501,27 @@ xfs_attr_rmtval_set( | |||
| 405 | (map.br_startblock != HOLESTARTBLOCK)); | 501 | (map.br_startblock != HOLESTARTBLOCK)); |
| 406 | 502 | ||
| 407 | dblkno = XFS_FSB_TO_DADDR(mp, map.br_startblock), | 503 | dblkno = XFS_FSB_TO_DADDR(mp, map.br_startblock), |
| 408 | blkcnt = XFS_FSB_TO_BB(mp, map.br_blockcount); | 504 | dblkcnt = XFS_FSB_TO_BB(mp, map.br_blockcount); |
| 409 | 505 | ||
| 410 | bp = xfs_buf_get(mp->m_ddev_targp, dblkno, blkcnt, 0); | 506 | bp = xfs_buf_get(mp->m_ddev_targp, dblkno, dblkcnt, 0); |
| 411 | if (!bp) | 507 | if (!bp) |
| 412 | return ENOMEM; | 508 | return ENOMEM; |
| 413 | bp->b_ops = &xfs_attr3_rmt_buf_ops; | 509 | bp->b_ops = &xfs_attr3_rmt_buf_ops; |
| 414 | 510 | ||
| 415 | byte_cnt = BBTOB(bp->b_length); | 511 | xfs_attr_rmtval_copyin(mp, bp, args->dp->i_ino, &offset, |
| 416 | byte_cnt = XFS_ATTR3_RMT_BUF_SPACE(mp, byte_cnt); | 512 | &valuelen, &src); |
| 417 | if (valuelen < byte_cnt) | ||
| 418 | byte_cnt = valuelen; | ||
| 419 | |||
| 420 | buf = bp->b_addr; | ||
| 421 | buf += xfs_attr3_rmt_hdr_set(mp, dp->i_ino, offset, | ||
| 422 | byte_cnt, bp); | ||
| 423 | memcpy(buf, src, byte_cnt); | ||
| 424 | |||
| 425 | if (byte_cnt < BBTOB(bp->b_length)) | ||
| 426 | xfs_buf_zero(bp, byte_cnt, | ||
| 427 | BBTOB(bp->b_length) - byte_cnt); | ||
| 428 | 513 | ||
| 429 | error = xfs_bwrite(bp); /* GROT: NOTE: synchronous write */ | 514 | error = xfs_bwrite(bp); /* GROT: NOTE: synchronous write */ |
| 430 | xfs_buf_relse(bp); | 515 | xfs_buf_relse(bp); |
| 431 | if (error) | 516 | if (error) |
| 432 | return error; | 517 | return error; |
| 433 | 518 | ||
| 434 | src += byte_cnt; | ||
| 435 | valuelen -= byte_cnt; | ||
| 436 | offset += byte_cnt; | ||
| 437 | hdrcnt--; | ||
| 438 | 519 | ||
| 520 | /* roll attribute extent map forwards */ | ||
| 439 | lblkno += map.br_blockcount; | 521 | lblkno += map.br_blockcount; |
| 522 | blkcnt -= map.br_blockcount; | ||
| 440 | } | 523 | } |
| 441 | ASSERT(valuelen == 0); | 524 | ASSERT(valuelen == 0); |
| 442 | ASSERT(hdrcnt == 0); | ||
| 443 | return 0; | 525 | return 0; |
| 444 | } | 526 | } |
| 445 | 527 | ||
| @@ -448,33 +530,40 @@ xfs_attr_rmtval_set( | |||
| 448 | * out-of-line buffer that it is stored on. | 530 | * out-of-line buffer that it is stored on. |
| 449 | */ | 531 | */ |
| 450 | int | 532 | int |
| 451 | xfs_attr_rmtval_remove(xfs_da_args_t *args) | 533 | xfs_attr_rmtval_remove( |
| 534 | struct xfs_da_args *args) | ||
| 452 | { | 535 | { |
| 453 | xfs_mount_t *mp; | 536 | struct xfs_mount *mp = args->dp->i_mount; |
| 454 | xfs_bmbt_irec_t map; | 537 | xfs_dablk_t lblkno; |
| 455 | xfs_buf_t *bp; | 538 | int blkcnt; |
| 456 | xfs_daddr_t dblkno; | 539 | int error; |
| 457 | xfs_dablk_t lblkno; | 540 | int done; |
| 458 | int valuelen, blkcnt, nmap, error, done, committed; | ||
| 459 | 541 | ||
| 460 | trace_xfs_attr_rmtval_remove(args); | 542 | trace_xfs_attr_rmtval_remove(args); |
| 461 | 543 | ||
| 462 | mp = args->dp->i_mount; | ||
| 463 | |||
| 464 | /* | 544 | /* |
| 465 | * Roll through the "value", invalidating the attribute value's | 545 | * Roll through the "value", invalidating the attribute value's blocks. |
| 466 | * blocks. | 546 | * Note that args->rmtblkcnt is the minimum number of data blocks we'll |
| 547 | * see for a CRC enabled remote attribute. Each extent will have a | ||
| 548 | * header, and so we may have more blocks than we realise here. If we | ||
| 549 | * fail to map the blocks correctly, we'll have problems with the buffer | ||
| 550 | * lookups. | ||
| 467 | */ | 551 | */ |
| 468 | lblkno = args->rmtblkno; | 552 | lblkno = args->rmtblkno; |
| 469 | valuelen = args->rmtblkcnt; | 553 | blkcnt = args->rmtblkcnt; |
| 470 | while (valuelen > 0) { | 554 | while (blkcnt > 0) { |
| 555 | struct xfs_bmbt_irec map; | ||
| 556 | struct xfs_buf *bp; | ||
| 557 | xfs_daddr_t dblkno; | ||
| 558 | int dblkcnt; | ||
| 559 | int nmap; | ||
| 560 | |||
| 471 | /* | 561 | /* |
| 472 | * Try to remember where we decided to put the value. | 562 | * Try to remember where we decided to put the value. |
| 473 | */ | 563 | */ |
| 474 | nmap = 1; | 564 | nmap = 1; |
| 475 | error = xfs_bmapi_read(args->dp, (xfs_fileoff_t)lblkno, | 565 | error = xfs_bmapi_read(args->dp, (xfs_fileoff_t)lblkno, |
| 476 | args->rmtblkcnt, &map, &nmap, | 566 | blkcnt, &map, &nmap, XFS_BMAPI_ATTRFORK); |
| 477 | XFS_BMAPI_ATTRFORK); | ||
| 478 | if (error) | 567 | if (error) |
| 479 | return(error); | 568 | return(error); |
| 480 | ASSERT(nmap == 1); | 569 | ASSERT(nmap == 1); |
| @@ -482,21 +571,20 @@ xfs_attr_rmtval_remove(xfs_da_args_t *args) | |||
| 482 | (map.br_startblock != HOLESTARTBLOCK)); | 571 | (map.br_startblock != HOLESTARTBLOCK)); |
| 483 | 572 | ||
| 484 | dblkno = XFS_FSB_TO_DADDR(mp, map.br_startblock), | 573 | dblkno = XFS_FSB_TO_DADDR(mp, map.br_startblock), |
| 485 | blkcnt = XFS_FSB_TO_BB(mp, map.br_blockcount); | 574 | dblkcnt = XFS_FSB_TO_BB(mp, map.br_blockcount); |
| 486 | 575 | ||
| 487 | /* | 576 | /* |
| 488 | * If the "remote" value is in the cache, remove it. | 577 | * If the "remote" value is in the cache, remove it. |
| 489 | */ | 578 | */ |
| 490 | bp = xfs_incore(mp->m_ddev_targp, dblkno, blkcnt, XBF_TRYLOCK); | 579 | bp = xfs_incore(mp->m_ddev_targp, dblkno, dblkcnt, XBF_TRYLOCK); |
| 491 | if (bp) { | 580 | if (bp) { |
| 492 | xfs_buf_stale(bp); | 581 | xfs_buf_stale(bp); |
| 493 | xfs_buf_relse(bp); | 582 | xfs_buf_relse(bp); |
| 494 | bp = NULL; | 583 | bp = NULL; |
| 495 | } | 584 | } |
| 496 | 585 | ||
| 497 | valuelen -= map.br_blockcount; | ||
| 498 | |||
| 499 | lblkno += map.br_blockcount; | 586 | lblkno += map.br_blockcount; |
| 587 | blkcnt -= map.br_blockcount; | ||
| 500 | } | 588 | } |
| 501 | 589 | ||
| 502 | /* | 590 | /* |
| @@ -506,6 +594,8 @@ xfs_attr_rmtval_remove(xfs_da_args_t *args) | |||
| 506 | blkcnt = args->rmtblkcnt; | 594 | blkcnt = args->rmtblkcnt; |
| 507 | done = 0; | 595 | done = 0; |
| 508 | while (!done) { | 596 | while (!done) { |
| 597 | int committed; | ||
| 598 | |||
| 509 | xfs_bmap_init(args->flist, args->firstblock); | 599 | xfs_bmap_init(args->flist, args->firstblock); |
| 510 | error = xfs_bunmapi(args->trans, args->dp, lblkno, blkcnt, | 600 | error = xfs_bunmapi(args->trans, args->dp, lblkno, blkcnt, |
| 511 | XFS_BMAPI_ATTRFORK | XFS_BMAPI_METADATA, | 601 | XFS_BMAPI_ATTRFORK | XFS_BMAPI_METADATA, |
diff --git a/fs/xfs/xfs_attr_remote.h b/fs/xfs/xfs_attr_remote.h index c7cca60a062a..92a8fd7977cc 100644 --- a/fs/xfs/xfs_attr_remote.h +++ b/fs/xfs/xfs_attr_remote.h | |||
| @@ -20,6 +20,14 @@ | |||
| 20 | 20 | ||
| 21 | #define XFS_ATTR3_RMT_MAGIC 0x5841524d /* XARM */ | 21 | #define XFS_ATTR3_RMT_MAGIC 0x5841524d /* XARM */ |
| 22 | 22 | ||
| 23 | /* | ||
| 24 | * There is one of these headers per filesystem block in a remote attribute. | ||
| 25 | * This is done to ensure there is a 1:1 mapping between the attribute value | ||
| 26 | * length and the number of blocks needed to store the attribute. This makes the | ||
| 27 | * verification of a buffer a little more complex, but greatly simplifies the | ||
| 28 | * allocation, reading and writing of these attributes as we don't have to guess | ||
| 29 | * the number of blocks needed to store the attribute data. | ||
| 30 | */ | ||
| 23 | struct xfs_attr3_rmt_hdr { | 31 | struct xfs_attr3_rmt_hdr { |
| 24 | __be32 rm_magic; | 32 | __be32 rm_magic; |
| 25 | __be32 rm_offset; | 33 | __be32 rm_offset; |
| @@ -39,6 +47,8 @@ struct xfs_attr3_rmt_hdr { | |||
| 39 | 47 | ||
| 40 | extern const struct xfs_buf_ops xfs_attr3_rmt_buf_ops; | 48 | extern const struct xfs_buf_ops xfs_attr3_rmt_buf_ops; |
| 41 | 49 | ||
| 50 | int xfs_attr3_rmt_blocks(struct xfs_mount *mp, int attrlen); | ||
| 51 | |||
| 42 | int xfs_attr_rmtval_get(struct xfs_da_args *args); | 52 | int xfs_attr_rmtval_get(struct xfs_da_args *args); |
| 43 | int xfs_attr_rmtval_set(struct xfs_da_args *args); | 53 | int xfs_attr_rmtval_set(struct xfs_da_args *args); |
| 44 | int xfs_attr_rmtval_remove(struct xfs_da_args *args); | 54 | int xfs_attr_rmtval_remove(struct xfs_da_args *args); |
diff --git a/fs/xfs/xfs_buf.c b/fs/xfs/xfs_buf.c index 0d2554299688..1b2472a46e46 100644 --- a/fs/xfs/xfs_buf.c +++ b/fs/xfs/xfs_buf.c | |||
| @@ -513,6 +513,7 @@ _xfs_buf_find( | |||
| 513 | xfs_alert(btp->bt_mount, | 513 | xfs_alert(btp->bt_mount, |
| 514 | "%s: Block out of range: block 0x%llx, EOFS 0x%llx ", | 514 | "%s: Block out of range: block 0x%llx, EOFS 0x%llx ", |
| 515 | __func__, blkno, eofs); | 515 | __func__, blkno, eofs); |
| 516 | WARN_ON(1); | ||
| 516 | return NULL; | 517 | return NULL; |
| 517 | } | 518 | } |
| 518 | 519 | ||
diff --git a/fs/xfs/xfs_buf_item.c b/fs/xfs/xfs_buf_item.c index cf263476d6b4..4ec431777048 100644 --- a/fs/xfs/xfs_buf_item.c +++ b/fs/xfs/xfs_buf_item.c | |||
| @@ -262,12 +262,7 @@ xfs_buf_item_format_segment( | |||
| 262 | vecp->i_addr = xfs_buf_offset(bp, buffer_offset); | 262 | vecp->i_addr = xfs_buf_offset(bp, buffer_offset); |
| 263 | vecp->i_len = nbits * XFS_BLF_CHUNK; | 263 | vecp->i_len = nbits * XFS_BLF_CHUNK; |
| 264 | vecp->i_type = XLOG_REG_TYPE_BCHUNK; | 264 | vecp->i_type = XLOG_REG_TYPE_BCHUNK; |
| 265 | /* | 265 | nvecs++; |
| 266 | * You would think we need to bump the nvecs here too, but we do not | ||
| 267 | * this number is used by recovery, and it gets confused by the boundary | ||
| 268 | * split here | ||
| 269 | * nvecs++; | ||
| 270 | */ | ||
| 271 | vecp++; | 266 | vecp++; |
| 272 | first_bit = next_bit; | 267 | first_bit = next_bit; |
| 273 | last_bit = next_bit; | 268 | last_bit = next_bit; |
diff --git a/fs/xfs/xfs_dfrag.c b/fs/xfs/xfs_dfrag.c index f852b082a084..c407e1ccff43 100644 --- a/fs/xfs/xfs_dfrag.c +++ b/fs/xfs/xfs_dfrag.c | |||
| @@ -219,6 +219,14 @@ xfs_swap_extents( | |||
| 219 | int taforkblks = 0; | 219 | int taforkblks = 0; |
| 220 | __uint64_t tmp; | 220 | __uint64_t tmp; |
| 221 | 221 | ||
| 222 | /* | ||
| 223 | * We have no way of updating owner information in the BMBT blocks for | ||
| 224 | * each inode on CRC enabled filesystems, so to avoid corrupting the | ||
| 225 | * this metadata we simply don't allow extent swaps to occur. | ||
| 226 | */ | ||
| 227 | if (xfs_sb_version_hascrc(&mp->m_sb)) | ||
| 228 | return XFS_ERROR(EINVAL); | ||
| 229 | |||
| 222 | tempifp = kmem_alloc(sizeof(xfs_ifork_t), KM_MAYFAIL); | 230 | tempifp = kmem_alloc(sizeof(xfs_ifork_t), KM_MAYFAIL); |
| 223 | if (!tempifp) { | 231 | if (!tempifp) { |
| 224 | error = XFS_ERROR(ENOMEM); | 232 | error = XFS_ERROR(ENOMEM); |
diff --git a/fs/xfs/xfs_dir2_format.h b/fs/xfs/xfs_dir2_format.h index a3b1bd841a80..995f1f505a52 100644 --- a/fs/xfs/xfs_dir2_format.h +++ b/fs/xfs/xfs_dir2_format.h | |||
| @@ -715,6 +715,7 @@ struct xfs_dir3_free_hdr { | |||
| 715 | __be32 firstdb; /* db of first entry */ | 715 | __be32 firstdb; /* db of first entry */ |
| 716 | __be32 nvalid; /* count of valid entries */ | 716 | __be32 nvalid; /* count of valid entries */ |
| 717 | __be32 nused; /* count of used entries */ | 717 | __be32 nused; /* count of used entries */ |
| 718 | __be32 pad; /* 64 bit alignment. */ | ||
| 718 | }; | 719 | }; |
| 719 | 720 | ||
| 720 | struct xfs_dir3_free { | 721 | struct xfs_dir3_free { |
diff --git a/fs/xfs/xfs_dir2_node.c b/fs/xfs/xfs_dir2_node.c index 5246de4912d4..2226a00acd15 100644 --- a/fs/xfs/xfs_dir2_node.c +++ b/fs/xfs/xfs_dir2_node.c | |||
| @@ -263,18 +263,19 @@ xfs_dir3_free_get_buf( | |||
| 263 | * Initialize the new block to be empty, and remember | 263 | * Initialize the new block to be empty, and remember |
| 264 | * its first slot as our empty slot. | 264 | * its first slot as our empty slot. |
| 265 | */ | 265 | */ |
| 266 | hdr.magic = XFS_DIR2_FREE_MAGIC; | 266 | memset(bp->b_addr, 0, sizeof(struct xfs_dir3_free_hdr)); |
| 267 | hdr.firstdb = 0; | 267 | memset(&hdr, 0, sizeof(hdr)); |
| 268 | hdr.nused = 0; | 268 | |
| 269 | hdr.nvalid = 0; | ||
| 270 | if (xfs_sb_version_hascrc(&mp->m_sb)) { | 269 | if (xfs_sb_version_hascrc(&mp->m_sb)) { |
| 271 | struct xfs_dir3_free_hdr *hdr3 = bp->b_addr; | 270 | struct xfs_dir3_free_hdr *hdr3 = bp->b_addr; |
| 272 | 271 | ||
| 273 | hdr.magic = XFS_DIR3_FREE_MAGIC; | 272 | hdr.magic = XFS_DIR3_FREE_MAGIC; |
| 273 | |||
| 274 | hdr3->hdr.blkno = cpu_to_be64(bp->b_bn); | 274 | hdr3->hdr.blkno = cpu_to_be64(bp->b_bn); |
| 275 | hdr3->hdr.owner = cpu_to_be64(dp->i_ino); | 275 | hdr3->hdr.owner = cpu_to_be64(dp->i_ino); |
| 276 | uuid_copy(&hdr3->hdr.uuid, &mp->m_sb.sb_uuid); | 276 | uuid_copy(&hdr3->hdr.uuid, &mp->m_sb.sb_uuid); |
| 277 | } | 277 | } else |
| 278 | hdr.magic = XFS_DIR2_FREE_MAGIC; | ||
| 278 | xfs_dir3_free_hdr_to_disk(bp->b_addr, &hdr); | 279 | xfs_dir3_free_hdr_to_disk(bp->b_addr, &hdr); |
| 279 | *bpp = bp; | 280 | *bpp = bp; |
| 280 | return 0; | 281 | return 0; |
| @@ -1921,8 +1922,6 @@ xfs_dir2_node_addname_int( | |||
| 1921 | */ | 1922 | */ |
| 1922 | freehdr.firstdb = (fbno - XFS_DIR2_FREE_FIRSTDB(mp)) * | 1923 | freehdr.firstdb = (fbno - XFS_DIR2_FREE_FIRSTDB(mp)) * |
| 1923 | xfs_dir3_free_max_bests(mp); | 1924 | xfs_dir3_free_max_bests(mp); |
| 1924 | free->hdr.nvalid = 0; | ||
| 1925 | free->hdr.nused = 0; | ||
| 1926 | } else { | 1925 | } else { |
| 1927 | free = fbp->b_addr; | 1926 | free = fbp->b_addr; |
| 1928 | bests = xfs_dir3_free_bests_p(mp, free); | 1927 | bests = xfs_dir3_free_bests_p(mp, free); |
diff --git a/fs/xfs/xfs_fs.h b/fs/xfs/xfs_fs.h index 6dda3f949b04..d04695545397 100644 --- a/fs/xfs/xfs_fs.h +++ b/fs/xfs/xfs_fs.h | |||
| @@ -236,6 +236,7 @@ typedef struct xfs_fsop_resblks { | |||
| 236 | #define XFS_FSOP_GEOM_FLAGS_PROJID32 0x0800 /* 32-bit project IDs */ | 236 | #define XFS_FSOP_GEOM_FLAGS_PROJID32 0x0800 /* 32-bit project IDs */ |
| 237 | #define XFS_FSOP_GEOM_FLAGS_DIRV2CI 0x1000 /* ASCII only CI names */ | 237 | #define XFS_FSOP_GEOM_FLAGS_DIRV2CI 0x1000 /* ASCII only CI names */ |
| 238 | #define XFS_FSOP_GEOM_FLAGS_LAZYSB 0x4000 /* lazy superblock counters */ | 238 | #define XFS_FSOP_GEOM_FLAGS_LAZYSB 0x4000 /* lazy superblock counters */ |
| 239 | #define XFS_FSOP_GEOM_FLAGS_V5SB 0x8000 /* version 5 superblock */ | ||
| 239 | 240 | ||
| 240 | 241 | ||
| 241 | /* | 242 | /* |
diff --git a/fs/xfs/xfs_fsops.c b/fs/xfs/xfs_fsops.c index 87595b211da1..3c3644ea825b 100644 --- a/fs/xfs/xfs_fsops.c +++ b/fs/xfs/xfs_fsops.c | |||
| @@ -99,7 +99,9 @@ xfs_fs_geometry( | |||
| 99 | (xfs_sb_version_hasattr2(&mp->m_sb) ? | 99 | (xfs_sb_version_hasattr2(&mp->m_sb) ? |
| 100 | XFS_FSOP_GEOM_FLAGS_ATTR2 : 0) | | 100 | XFS_FSOP_GEOM_FLAGS_ATTR2 : 0) | |
| 101 | (xfs_sb_version_hasprojid32bit(&mp->m_sb) ? | 101 | (xfs_sb_version_hasprojid32bit(&mp->m_sb) ? |
| 102 | XFS_FSOP_GEOM_FLAGS_PROJID32 : 0); | 102 | XFS_FSOP_GEOM_FLAGS_PROJID32 : 0) | |
| 103 | (xfs_sb_version_hascrc(&mp->m_sb) ? | ||
| 104 | XFS_FSOP_GEOM_FLAGS_V5SB : 0); | ||
| 103 | geo->logsectsize = xfs_sb_version_hassector(&mp->m_sb) ? | 105 | geo->logsectsize = xfs_sb_version_hassector(&mp->m_sb) ? |
| 104 | mp->m_sb.sb_logsectsize : BBSIZE; | 106 | mp->m_sb.sb_logsectsize : BBSIZE; |
| 105 | geo->rtsectsize = mp->m_sb.sb_blocksize; | 107 | geo->rtsectsize = mp->m_sb.sb_blocksize; |
diff --git a/fs/xfs/xfs_iops.c b/fs/xfs/xfs_iops.c index d82efaa2ac73..ca9ecaa81112 100644 --- a/fs/xfs/xfs_iops.c +++ b/fs/xfs/xfs_iops.c | |||
| @@ -455,6 +455,28 @@ xfs_vn_getattr( | |||
| 455 | return 0; | 455 | return 0; |
| 456 | } | 456 | } |
| 457 | 457 | ||
| 458 | static void | ||
| 459 | xfs_setattr_mode( | ||
| 460 | struct xfs_trans *tp, | ||
| 461 | struct xfs_inode *ip, | ||
| 462 | struct iattr *iattr) | ||
| 463 | { | ||
| 464 | struct inode *inode = VFS_I(ip); | ||
| 465 | umode_t mode = iattr->ia_mode; | ||
| 466 | |||
| 467 | ASSERT(tp); | ||
| 468 | ASSERT(xfs_isilocked(ip, XFS_ILOCK_EXCL)); | ||
| 469 | |||
| 470 | if (!in_group_p(inode->i_gid) && !capable(CAP_FSETID)) | ||
| 471 | mode &= ~S_ISGID; | ||
| 472 | |||
| 473 | ip->i_d.di_mode &= S_IFMT; | ||
| 474 | ip->i_d.di_mode |= mode & ~S_IFMT; | ||
| 475 | |||
| 476 | inode->i_mode &= S_IFMT; | ||
| 477 | inode->i_mode |= mode & ~S_IFMT; | ||
| 478 | } | ||
| 479 | |||
| 458 | int | 480 | int |
| 459 | xfs_setattr_nonsize( | 481 | xfs_setattr_nonsize( |
| 460 | struct xfs_inode *ip, | 482 | struct xfs_inode *ip, |
| @@ -606,18 +628,8 @@ xfs_setattr_nonsize( | |||
| 606 | /* | 628 | /* |
| 607 | * Change file access modes. | 629 | * Change file access modes. |
| 608 | */ | 630 | */ |
| 609 | if (mask & ATTR_MODE) { | 631 | if (mask & ATTR_MODE) |
| 610 | umode_t mode = iattr->ia_mode; | 632 | xfs_setattr_mode(tp, ip, iattr); |
| 611 | |||
| 612 | if (!in_group_p(inode->i_gid) && !capable(CAP_FSETID)) | ||
| 613 | mode &= ~S_ISGID; | ||
| 614 | |||
| 615 | ip->i_d.di_mode &= S_IFMT; | ||
| 616 | ip->i_d.di_mode |= mode & ~S_IFMT; | ||
| 617 | |||
| 618 | inode->i_mode &= S_IFMT; | ||
| 619 | inode->i_mode |= mode & ~S_IFMT; | ||
| 620 | } | ||
| 621 | 633 | ||
| 622 | /* | 634 | /* |
| 623 | * Change file access or modified times. | 635 | * Change file access or modified times. |
| @@ -714,9 +726,8 @@ xfs_setattr_size( | |||
| 714 | return XFS_ERROR(error); | 726 | return XFS_ERROR(error); |
| 715 | 727 | ||
| 716 | ASSERT(S_ISREG(ip->i_d.di_mode)); | 728 | ASSERT(S_ISREG(ip->i_d.di_mode)); |
| 717 | ASSERT((mask & (ATTR_MODE|ATTR_UID|ATTR_GID|ATTR_ATIME|ATTR_ATIME_SET| | 729 | ASSERT((mask & (ATTR_UID|ATTR_GID|ATTR_ATIME|ATTR_ATIME_SET| |
| 718 | ATTR_MTIME_SET|ATTR_KILL_SUID|ATTR_KILL_SGID| | 730 | ATTR_MTIME_SET|ATTR_KILL_PRIV|ATTR_TIMES_SET)) == 0); |
| 719 | ATTR_KILL_PRIV|ATTR_TIMES_SET)) == 0); | ||
| 720 | 731 | ||
| 721 | if (!(flags & XFS_ATTR_NOLOCK)) { | 732 | if (!(flags & XFS_ATTR_NOLOCK)) { |
| 722 | lock_flags |= XFS_IOLOCK_EXCL; | 733 | lock_flags |= XFS_IOLOCK_EXCL; |
| @@ -860,6 +871,12 @@ xfs_setattr_size( | |||
| 860 | xfs_inode_clear_eofblocks_tag(ip); | 871 | xfs_inode_clear_eofblocks_tag(ip); |
| 861 | } | 872 | } |
| 862 | 873 | ||
| 874 | /* | ||
| 875 | * Change file access modes. | ||
| 876 | */ | ||
| 877 | if (mask & ATTR_MODE) | ||
| 878 | xfs_setattr_mode(tp, ip, iattr); | ||
| 879 | |||
| 863 | if (mask & ATTR_CTIME) { | 880 | if (mask & ATTR_CTIME) { |
| 864 | inode->i_ctime = iattr->ia_ctime; | 881 | inode->i_ctime = iattr->ia_ctime; |
| 865 | ip->i_d.di_ctime.t_sec = iattr->ia_ctime.tv_sec; | 882 | ip->i_d.di_ctime.t_sec = iattr->ia_ctime.tv_sec; |
diff --git a/fs/xfs/xfs_log_recover.c b/fs/xfs/xfs_log_recover.c index 93f03ec17eec..d9e4d3c3991a 100644 --- a/fs/xfs/xfs_log_recover.c +++ b/fs/xfs/xfs_log_recover.c | |||
| @@ -2097,6 +2097,17 @@ xlog_recover_do_reg_buffer( | |||
| 2097 | ((uint)bit << XFS_BLF_SHIFT) + (nbits << XFS_BLF_SHIFT)); | 2097 | ((uint)bit << XFS_BLF_SHIFT) + (nbits << XFS_BLF_SHIFT)); |
| 2098 | 2098 | ||
| 2099 | /* | 2099 | /* |
| 2100 | * The dirty regions logged in the buffer, even though | ||
| 2101 | * contiguous, may span multiple chunks. This is because the | ||
| 2102 | * dirty region may span a physical page boundary in a buffer | ||
| 2103 | * and hence be split into two separate vectors for writing into | ||
| 2104 | * the log. Hence we need to trim nbits back to the length of | ||
| 2105 | * the current region being copied out of the log. | ||
| 2106 | */ | ||
| 2107 | if (item->ri_buf[i].i_len < (nbits << XFS_BLF_SHIFT)) | ||
| 2108 | nbits = item->ri_buf[i].i_len >> XFS_BLF_SHIFT; | ||
| 2109 | |||
| 2110 | /* | ||
| 2100 | * Do a sanity check if this is a dquot buffer. Just checking | 2111 | * Do a sanity check if this is a dquot buffer. Just checking |
| 2101 | * the first dquot in the buffer should do. XXXThis is | 2112 | * the first dquot in the buffer should do. XXXThis is |
| 2102 | * probably a good thing to do for other buf types also. | 2113 | * probably a good thing to do for other buf types also. |
diff --git a/fs/xfs/xfs_qm_syscalls.c b/fs/xfs/xfs_qm_syscalls.c index c41190cad6e9..6cdf6ffc36a1 100644 --- a/fs/xfs/xfs_qm_syscalls.c +++ b/fs/xfs/xfs_qm_syscalls.c | |||
| @@ -489,31 +489,36 @@ xfs_qm_scall_setqlim( | |||
| 489 | if ((newlim->d_fieldmask & XFS_DQ_MASK) == 0) | 489 | if ((newlim->d_fieldmask & XFS_DQ_MASK) == 0) |
| 490 | return 0; | 490 | return 0; |
| 491 | 491 | ||
| 492 | tp = xfs_trans_alloc(mp, XFS_TRANS_QM_SETQLIM); | ||
| 493 | error = xfs_trans_reserve(tp, 0, XFS_QM_SETQLIM_LOG_RES(mp), | ||
| 494 | 0, 0, XFS_DEFAULT_LOG_COUNT); | ||
| 495 | if (error) { | ||
| 496 | xfs_trans_cancel(tp, 0); | ||
| 497 | return (error); | ||
| 498 | } | ||
| 499 | |||
| 500 | /* | 492 | /* |
| 501 | * We don't want to race with a quotaoff so take the quotaoff lock. | 493 | * We don't want to race with a quotaoff so take the quotaoff lock. |
| 502 | * (We don't hold an inode lock, so there's nothing else to stop | 494 | * We don't hold an inode lock, so there's nothing else to stop |
| 503 | * a quotaoff from happening). (XXXThis doesn't currently happen | 495 | * a quotaoff from happening. |
| 504 | * because we take the vfslock before calling xfs_qm_sysent). | ||
| 505 | */ | 496 | */ |
| 506 | mutex_lock(&q->qi_quotaofflock); | 497 | mutex_lock(&q->qi_quotaofflock); |
| 507 | 498 | ||
| 508 | /* | 499 | /* |
| 509 | * Get the dquot (locked), and join it to the transaction. | 500 | * Get the dquot (locked) before we start, as we need to do a |
| 510 | * Allocate the dquot if this doesn't exist. | 501 | * transaction to allocate it if it doesn't exist. Once we have the |
| 502 | * dquot, unlock it so we can start the next transaction safely. We hold | ||
| 503 | * a reference to the dquot, so it's safe to do this unlock/lock without | ||
| 504 | * it being reclaimed in the mean time. | ||
| 511 | */ | 505 | */ |
| 512 | if ((error = xfs_qm_dqget(mp, NULL, id, type, XFS_QMOPT_DQALLOC, &dqp))) { | 506 | error = xfs_qm_dqget(mp, NULL, id, type, XFS_QMOPT_DQALLOC, &dqp); |
| 513 | xfs_trans_cancel(tp, XFS_TRANS_ABORT); | 507 | if (error) { |
| 514 | ASSERT(error != ENOENT); | 508 | ASSERT(error != ENOENT); |
| 515 | goto out_unlock; | 509 | goto out_unlock; |
| 516 | } | 510 | } |
| 511 | xfs_dqunlock(dqp); | ||
| 512 | |||
| 513 | tp = xfs_trans_alloc(mp, XFS_TRANS_QM_SETQLIM); | ||
| 514 | error = xfs_trans_reserve(tp, 0, XFS_QM_SETQLIM_LOG_RES(mp), | ||
| 515 | 0, 0, XFS_DEFAULT_LOG_COUNT); | ||
| 516 | if (error) { | ||
| 517 | xfs_trans_cancel(tp, 0); | ||
| 518 | goto out_rele; | ||
| 519 | } | ||
| 520 | |||
| 521 | xfs_dqlock(dqp); | ||
| 517 | xfs_trans_dqjoin(tp, dqp); | 522 | xfs_trans_dqjoin(tp, dqp); |
| 518 | ddq = &dqp->q_core; | 523 | ddq = &dqp->q_core; |
| 519 | 524 | ||
| @@ -621,9 +626,10 @@ xfs_qm_scall_setqlim( | |||
| 621 | xfs_trans_log_dquot(tp, dqp); | 626 | xfs_trans_log_dquot(tp, dqp); |
| 622 | 627 | ||
| 623 | error = xfs_trans_commit(tp, 0); | 628 | error = xfs_trans_commit(tp, 0); |
| 624 | xfs_qm_dqrele(dqp); | ||
| 625 | 629 | ||
| 626 | out_unlock: | 630 | out_rele: |
| 631 | xfs_qm_dqrele(dqp); | ||
| 632 | out_unlock: | ||
| 627 | mutex_unlock(&q->qi_quotaofflock); | 633 | mutex_unlock(&q->qi_quotaofflock); |
| 628 | return error; | 634 | return error; |
| 629 | } | 635 | } |
diff --git a/fs/xfs/xfs_symlink.c b/fs/xfs/xfs_symlink.c index 5f234389327c..195a403e1522 100644 --- a/fs/xfs/xfs_symlink.c +++ b/fs/xfs/xfs_symlink.c | |||
| @@ -56,16 +56,9 @@ xfs_symlink_blocks( | |||
| 56 | struct xfs_mount *mp, | 56 | struct xfs_mount *mp, |
| 57 | int pathlen) | 57 | int pathlen) |
| 58 | { | 58 | { |
| 59 | int fsblocks = 0; | 59 | int buflen = XFS_SYMLINK_BUF_SPACE(mp, mp->m_sb.sb_blocksize); |
| 60 | int len = pathlen; | ||
| 61 | 60 | ||
| 62 | do { | 61 | return (pathlen + buflen - 1) / buflen; |
| 63 | fsblocks++; | ||
| 64 | len -= XFS_SYMLINK_BUF_SPACE(mp, mp->m_sb.sb_blocksize); | ||
| 65 | } while (len > 0); | ||
| 66 | |||
| 67 | ASSERT(fsblocks <= XFS_SYMLINK_MAPS); | ||
| 68 | return fsblocks; | ||
| 69 | } | 62 | } |
| 70 | 63 | ||
| 71 | static int | 64 | static int |
| @@ -405,7 +398,7 @@ xfs_symlink( | |||
| 405 | if (pathlen <= XFS_LITINO(mp, dp->i_d.di_version)) | 398 | if (pathlen <= XFS_LITINO(mp, dp->i_d.di_version)) |
| 406 | fs_blocks = 0; | 399 | fs_blocks = 0; |
| 407 | else | 400 | else |
| 408 | fs_blocks = XFS_B_TO_FSB(mp, pathlen); | 401 | fs_blocks = xfs_symlink_blocks(mp, pathlen); |
| 409 | resblks = XFS_SYMLINK_SPACE_RES(mp, link_name->len, fs_blocks); | 402 | resblks = XFS_SYMLINK_SPACE_RES(mp, link_name->len, fs_blocks); |
| 410 | error = xfs_trans_reserve(tp, resblks, XFS_SYMLINK_LOG_RES(mp), 0, | 403 | error = xfs_trans_reserve(tp, resblks, XFS_SYMLINK_LOG_RES(mp), 0, |
| 411 | XFS_TRANS_PERM_LOG_RES, XFS_SYMLINK_LOG_COUNT); | 404 | XFS_TRANS_PERM_LOG_RES, XFS_SYMLINK_LOG_COUNT); |
| @@ -512,7 +505,7 @@ xfs_symlink( | |||
| 512 | cur_chunk = target_path; | 505 | cur_chunk = target_path; |
| 513 | offset = 0; | 506 | offset = 0; |
| 514 | for (n = 0; n < nmaps; n++) { | 507 | for (n = 0; n < nmaps; n++) { |
| 515 | char *buf; | 508 | char *buf; |
| 516 | 509 | ||
| 517 | d = XFS_FSB_TO_DADDR(mp, mval[n].br_startblock); | 510 | d = XFS_FSB_TO_DADDR(mp, mval[n].br_startblock); |
| 518 | byte_cnt = XFS_FSB_TO_B(mp, mval[n].br_blockcount); | 511 | byte_cnt = XFS_FSB_TO_B(mp, mval[n].br_blockcount); |
| @@ -525,9 +518,7 @@ xfs_symlink( | |||
| 525 | bp->b_ops = &xfs_symlink_buf_ops; | 518 | bp->b_ops = &xfs_symlink_buf_ops; |
| 526 | 519 | ||
| 527 | byte_cnt = XFS_SYMLINK_BUF_SPACE(mp, byte_cnt); | 520 | byte_cnt = XFS_SYMLINK_BUF_SPACE(mp, byte_cnt); |
| 528 | if (pathlen < byte_cnt) { | 521 | byte_cnt = min(byte_cnt, pathlen); |
| 529 | byte_cnt = pathlen; | ||
| 530 | } | ||
| 531 | 522 | ||
| 532 | buf = bp->b_addr; | 523 | buf = bp->b_addr; |
| 533 | buf += xfs_symlink_hdr_set(mp, ip->i_ino, offset, | 524 | buf += xfs_symlink_hdr_set(mp, ip->i_ino, offset, |
| @@ -542,6 +533,7 @@ xfs_symlink( | |||
| 542 | xfs_trans_log_buf(tp, bp, 0, (buf + byte_cnt - 1) - | 533 | xfs_trans_log_buf(tp, bp, 0, (buf + byte_cnt - 1) - |
| 543 | (char *)bp->b_addr); | 534 | (char *)bp->b_addr); |
| 544 | } | 535 | } |
| 536 | ASSERT(pathlen == 0); | ||
| 545 | } | 537 | } |
| 546 | 538 | ||
| 547 | /* | 539 | /* |
