diff options
-rw-r--r-- | fs/squashfs/inode.c | 91 | ||||
-rw-r--r-- | fs/squashfs/namei.c | 5 | ||||
-rw-r--r-- | fs/squashfs/squashfs.h | 12 | ||||
-rw-r--r-- | fs/squashfs/squashfs_fs.h | 25 | ||||
-rw-r--r-- | fs/squashfs/super.c | 8 | ||||
-rw-r--r-- | fs/squashfs/symlink.c | 10 |
6 files changed, 135 insertions, 16 deletions
diff --git a/fs/squashfs/inode.c b/fs/squashfs/inode.c index 49daaf669e41..250701180a3b 100644 --- a/fs/squashfs/inode.c +++ b/fs/squashfs/inode.c | |||
@@ -40,6 +40,7 @@ | |||
40 | 40 | ||
41 | #include <linux/fs.h> | 41 | #include <linux/fs.h> |
42 | #include <linux/vfs.h> | 42 | #include <linux/vfs.h> |
43 | #include <linux/xattr.h> | ||
43 | 44 | ||
44 | #include "squashfs_fs.h" | 45 | #include "squashfs_fs.h" |
45 | #include "squashfs_fs_sb.h" | 46 | #include "squashfs_fs_sb.h" |
@@ -111,6 +112,7 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
111 | int err, type, offset = SQUASHFS_INODE_OFFSET(ino); | 112 | int err, type, offset = SQUASHFS_INODE_OFFSET(ino); |
112 | union squashfs_inode squashfs_ino; | 113 | union squashfs_inode squashfs_ino; |
113 | struct squashfs_base_inode *sqshb_ino = &squashfs_ino.base; | 114 | struct squashfs_base_inode *sqshb_ino = &squashfs_ino.base; |
115 | int xattr_id = SQUASHFS_INVALID_XATTR; | ||
114 | 116 | ||
115 | TRACE("Entered squashfs_read_inode\n"); | 117 | TRACE("Entered squashfs_read_inode\n"); |
116 | 118 | ||
@@ -199,8 +201,10 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
199 | frag_offset = 0; | 201 | frag_offset = 0; |
200 | } | 202 | } |
201 | 203 | ||
204 | xattr_id = le32_to_cpu(sqsh_ino->xattr); | ||
202 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); | 205 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); |
203 | inode->i_size = le64_to_cpu(sqsh_ino->file_size); | 206 | inode->i_size = le64_to_cpu(sqsh_ino->file_size); |
207 | inode->i_op = &squashfs_inode_ops; | ||
204 | inode->i_fop = &generic_ro_fops; | 208 | inode->i_fop = &generic_ro_fops; |
205 | inode->i_mode |= S_IFREG; | 209 | inode->i_mode |= S_IFREG; |
206 | inode->i_blocks = ((inode->i_size - | 210 | inode->i_blocks = ((inode->i_size - |
@@ -251,6 +255,7 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
251 | if (err < 0) | 255 | if (err < 0) |
252 | goto failed_read; | 256 | goto failed_read; |
253 | 257 | ||
258 | xattr_id = le32_to_cpu(sqsh_ino->xattr); | ||
254 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); | 259 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); |
255 | inode->i_size = le32_to_cpu(sqsh_ino->file_size); | 260 | inode->i_size = le32_to_cpu(sqsh_ino->file_size); |
256 | inode->i_op = &squashfs_dir_inode_ops; | 261 | inode->i_op = &squashfs_dir_inode_ops; |
@@ -280,21 +285,33 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
280 | 285 | ||
281 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); | 286 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); |
282 | inode->i_size = le32_to_cpu(sqsh_ino->symlink_size); | 287 | inode->i_size = le32_to_cpu(sqsh_ino->symlink_size); |
283 | inode->i_op = &page_symlink_inode_operations; | 288 | inode->i_op = &squashfs_symlink_inode_ops; |
284 | inode->i_data.a_ops = &squashfs_symlink_aops; | 289 | inode->i_data.a_ops = &squashfs_symlink_aops; |
285 | inode->i_mode |= S_IFLNK; | 290 | inode->i_mode |= S_IFLNK; |
286 | squashfs_i(inode)->start = block; | 291 | squashfs_i(inode)->start = block; |
287 | squashfs_i(inode)->offset = offset; | 292 | squashfs_i(inode)->offset = offset; |
288 | 293 | ||
294 | if (type == SQUASHFS_LSYMLINK_TYPE) { | ||
295 | __le32 xattr; | ||
296 | |||
297 | err = squashfs_read_metadata(sb, NULL, &block, | ||
298 | &offset, inode->i_size); | ||
299 | if (err < 0) | ||
300 | goto failed_read; | ||
301 | err = squashfs_read_metadata(sb, &xattr, &block, | ||
302 | &offset, sizeof(xattr)); | ||
303 | if (err < 0) | ||
304 | goto failed_read; | ||
305 | xattr_id = le32_to_cpu(xattr); | ||
306 | } | ||
307 | |||
289 | TRACE("Symbolic link inode %x:%x, start_block %llx, offset " | 308 | TRACE("Symbolic link inode %x:%x, start_block %llx, offset " |
290 | "%x\n", SQUASHFS_INODE_BLK(ino), offset, | 309 | "%x\n", SQUASHFS_INODE_BLK(ino), offset, |
291 | block, offset); | 310 | block, offset); |
292 | break; | 311 | break; |
293 | } | 312 | } |
294 | case SQUASHFS_BLKDEV_TYPE: | 313 | case SQUASHFS_BLKDEV_TYPE: |
295 | case SQUASHFS_CHRDEV_TYPE: | 314 | case SQUASHFS_CHRDEV_TYPE: { |
296 | case SQUASHFS_LBLKDEV_TYPE: | ||
297 | case SQUASHFS_LCHRDEV_TYPE: { | ||
298 | struct squashfs_dev_inode *sqsh_ino = &squashfs_ino.dev; | 315 | struct squashfs_dev_inode *sqsh_ino = &squashfs_ino.dev; |
299 | unsigned int rdev; | 316 | unsigned int rdev; |
300 | 317 | ||
@@ -315,10 +332,32 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
315 | SQUASHFS_INODE_BLK(ino), offset, rdev); | 332 | SQUASHFS_INODE_BLK(ino), offset, rdev); |
316 | break; | 333 | break; |
317 | } | 334 | } |
335 | case SQUASHFS_LBLKDEV_TYPE: | ||
336 | case SQUASHFS_LCHRDEV_TYPE: { | ||
337 | struct squashfs_ldev_inode *sqsh_ino = &squashfs_ino.ldev; | ||
338 | unsigned int rdev; | ||
339 | |||
340 | err = squashfs_read_metadata(sb, sqsh_ino, &block, &offset, | ||
341 | sizeof(*sqsh_ino)); | ||
342 | if (err < 0) | ||
343 | goto failed_read; | ||
344 | |||
345 | if (type == SQUASHFS_LCHRDEV_TYPE) | ||
346 | inode->i_mode |= S_IFCHR; | ||
347 | else | ||
348 | inode->i_mode |= S_IFBLK; | ||
349 | xattr_id = le32_to_cpu(sqsh_ino->xattr); | ||
350 | inode->i_op = &squashfs_inode_ops; | ||
351 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); | ||
352 | rdev = le32_to_cpu(sqsh_ino->rdev); | ||
353 | init_special_inode(inode, inode->i_mode, new_decode_dev(rdev)); | ||
354 | |||
355 | TRACE("Device inode %x:%x, rdev %x\n", | ||
356 | SQUASHFS_INODE_BLK(ino), offset, rdev); | ||
357 | break; | ||
358 | } | ||
318 | case SQUASHFS_FIFO_TYPE: | 359 | case SQUASHFS_FIFO_TYPE: |
319 | case SQUASHFS_SOCKET_TYPE: | 360 | case SQUASHFS_SOCKET_TYPE: { |
320 | case SQUASHFS_LFIFO_TYPE: | ||
321 | case SQUASHFS_LSOCKET_TYPE: { | ||
322 | struct squashfs_ipc_inode *sqsh_ino = &squashfs_ino.ipc; | 361 | struct squashfs_ipc_inode *sqsh_ino = &squashfs_ino.ipc; |
323 | 362 | ||
324 | err = squashfs_read_metadata(sb, sqsh_ino, &block, &offset, | 363 | err = squashfs_read_metadata(sb, sqsh_ino, &block, &offset, |
@@ -334,14 +373,52 @@ int squashfs_read_inode(struct inode *inode, long long ino) | |||
334 | init_special_inode(inode, inode->i_mode, 0); | 373 | init_special_inode(inode, inode->i_mode, 0); |
335 | break; | 374 | break; |
336 | } | 375 | } |
376 | case SQUASHFS_LFIFO_TYPE: | ||
377 | case SQUASHFS_LSOCKET_TYPE: { | ||
378 | struct squashfs_lipc_inode *sqsh_ino = &squashfs_ino.lipc; | ||
379 | |||
380 | err = squashfs_read_metadata(sb, sqsh_ino, &block, &offset, | ||
381 | sizeof(*sqsh_ino)); | ||
382 | if (err < 0) | ||
383 | goto failed_read; | ||
384 | |||
385 | if (type == SQUASHFS_LFIFO_TYPE) | ||
386 | inode->i_mode |= S_IFIFO; | ||
387 | else | ||
388 | inode->i_mode |= S_IFSOCK; | ||
389 | xattr_id = le32_to_cpu(sqsh_ino->xattr); | ||
390 | inode->i_op = &squashfs_inode_ops; | ||
391 | inode->i_nlink = le32_to_cpu(sqsh_ino->nlink); | ||
392 | init_special_inode(inode, inode->i_mode, 0); | ||
393 | break; | ||
394 | } | ||
337 | default: | 395 | default: |
338 | ERROR("Unknown inode type %d in squashfs_iget!\n", type); | 396 | ERROR("Unknown inode type %d in squashfs_iget!\n", type); |
339 | return -EINVAL; | 397 | return -EINVAL; |
340 | } | 398 | } |
341 | 399 | ||
400 | if (xattr_id != SQUASHFS_INVALID_XATTR && msblk->xattr_id_table) { | ||
401 | err = squashfs_xattr_lookup(sb, xattr_id, | ||
402 | &squashfs_i(inode)->xattr_count, | ||
403 | &squashfs_i(inode)->xattr_size, | ||
404 | &squashfs_i(inode)->xattr); | ||
405 | if (err < 0) | ||
406 | goto failed_read; | ||
407 | inode->i_blocks += ((squashfs_i(inode)->xattr_size - 1) >> 9) | ||
408 | + 1; | ||
409 | } else | ||
410 | squashfs_i(inode)->xattr_count = 0; | ||
411 | |||
342 | return 0; | 412 | return 0; |
343 | 413 | ||
344 | failed_read: | 414 | failed_read: |
345 | ERROR("Unable to read inode 0x%llx\n", ino); | 415 | ERROR("Unable to read inode 0x%llx\n", ino); |
346 | return err; | 416 | return err; |
347 | } | 417 | } |
418 | |||
419 | |||
420 | const struct inode_operations squashfs_inode_ops = { | ||
421 | .getxattr = generic_getxattr, | ||
422 | .listxattr = squashfs_listxattr | ||
423 | }; | ||
424 | |||
diff --git a/fs/squashfs/namei.c b/fs/squashfs/namei.c index 5266bd8ad932..32f5b54d1cec 100644 --- a/fs/squashfs/namei.c +++ b/fs/squashfs/namei.c | |||
@@ -57,6 +57,7 @@ | |||
57 | #include <linux/slab.h> | 57 | #include <linux/slab.h> |
58 | #include <linux/string.h> | 58 | #include <linux/string.h> |
59 | #include <linux/dcache.h> | 59 | #include <linux/dcache.h> |
60 | #include <linux/xattr.h> | ||
60 | 61 | ||
61 | #include "squashfs_fs.h" | 62 | #include "squashfs_fs.h" |
62 | #include "squashfs_fs_sb.h" | 63 | #include "squashfs_fs_sb.h" |
@@ -237,5 +238,7 @@ failed: | |||
237 | 238 | ||
238 | 239 | ||
239 | const struct inode_operations squashfs_dir_inode_ops = { | 240 | const struct inode_operations squashfs_dir_inode_ops = { |
240 | .lookup = squashfs_lookup | 241 | .lookup = squashfs_lookup, |
242 | .getxattr = generic_getxattr, | ||
243 | .listxattr = squashfs_listxattr | ||
241 | }; | 244 | }; |
diff --git a/fs/squashfs/squashfs.h b/fs/squashfs/squashfs.h index 133befe2f8b8..7d2381070581 100644 --- a/fs/squashfs/squashfs.h +++ b/fs/squashfs/squashfs.h | |||
@@ -73,6 +73,9 @@ extern struct inode *squashfs_iget(struct super_block *, long long, | |||
73 | unsigned int); | 73 | unsigned int); |
74 | extern int squashfs_read_inode(struct inode *, long long); | 74 | extern int squashfs_read_inode(struct inode *, long long); |
75 | 75 | ||
76 | /* xattr.c */ | ||
77 | extern ssize_t squashfs_listxattr(struct dentry *, char *, size_t); | ||
78 | |||
76 | /* xattr_id.c */ | 79 | /* xattr_id.c */ |
77 | extern int squashfs_xattr_lookup(struct super_block *, unsigned int, int *, | 80 | extern int squashfs_xattr_lookup(struct super_block *, unsigned int, int *, |
78 | int *, long long *); | 81 | int *, long long *); |
@@ -80,7 +83,7 @@ extern __le64 *squashfs_read_xattr_id_table(struct super_block *, u64, | |||
80 | u64 *, int *); | 83 | u64 *, int *); |
81 | 84 | ||
82 | /* | 85 | /* |
83 | * Inodes, files and decompressor operations | 86 | * Inodes, files, decompressor and xattr operations |
84 | */ | 87 | */ |
85 | 88 | ||
86 | /* dir.c */ | 89 | /* dir.c */ |
@@ -92,11 +95,18 @@ extern const struct export_operations squashfs_export_ops; | |||
92 | /* file.c */ | 95 | /* file.c */ |
93 | extern const struct address_space_operations squashfs_aops; | 96 | extern const struct address_space_operations squashfs_aops; |
94 | 97 | ||
98 | /* inode.c */ | ||
99 | extern const struct inode_operations squashfs_inode_ops; | ||
100 | |||
95 | /* namei.c */ | 101 | /* namei.c */ |
96 | extern const struct inode_operations squashfs_dir_inode_ops; | 102 | extern const struct inode_operations squashfs_dir_inode_ops; |
97 | 103 | ||
98 | /* symlink.c */ | 104 | /* symlink.c */ |
99 | extern const struct address_space_operations squashfs_symlink_aops; | 105 | extern const struct address_space_operations squashfs_symlink_aops; |
106 | extern const struct inode_operations squashfs_symlink_inode_ops; | ||
107 | |||
108 | /* xattr.c */ | ||
109 | extern struct xattr_handler *squashfs_xattr_handlers[]; | ||
100 | 110 | ||
101 | /* zlib_wrapper.c */ | 111 | /* zlib_wrapper.c */ |
102 | extern const struct squashfs_decompressor squashfs_zlib_comp_ops; | 112 | extern const struct squashfs_decompressor squashfs_zlib_comp_ops; |
diff --git a/fs/squashfs/squashfs_fs.h b/fs/squashfs/squashfs_fs.h index 6fe940cf9011..8eabb808b78d 100644 --- a/fs/squashfs/squashfs_fs.h +++ b/fs/squashfs/squashfs_fs.h | |||
@@ -287,6 +287,17 @@ struct squashfs_ipc_inode { | |||
287 | __le32 nlink; | 287 | __le32 nlink; |
288 | }; | 288 | }; |
289 | 289 | ||
290 | struct squashfs_lipc_inode { | ||
291 | __le16 inode_type; | ||
292 | __le16 mode; | ||
293 | __le16 uid; | ||
294 | __le16 guid; | ||
295 | __le32 mtime; | ||
296 | __le32 inode_number; | ||
297 | __le32 nlink; | ||
298 | __le32 xattr; | ||
299 | }; | ||
300 | |||
290 | struct squashfs_dev_inode { | 301 | struct squashfs_dev_inode { |
291 | __le16 inode_type; | 302 | __le16 inode_type; |
292 | __le16 mode; | 303 | __le16 mode; |
@@ -298,6 +309,18 @@ struct squashfs_dev_inode { | |||
298 | __le32 rdev; | 309 | __le32 rdev; |
299 | }; | 310 | }; |
300 | 311 | ||
312 | struct squashfs_ldev_inode { | ||
313 | __le16 inode_type; | ||
314 | __le16 mode; | ||
315 | __le16 uid; | ||
316 | __le16 guid; | ||
317 | __le32 mtime; | ||
318 | __le32 inode_number; | ||
319 | __le32 nlink; | ||
320 | __le32 rdev; | ||
321 | __le32 xattr; | ||
322 | }; | ||
323 | |||
301 | struct squashfs_symlink_inode { | 324 | struct squashfs_symlink_inode { |
302 | __le16 inode_type; | 325 | __le16 inode_type; |
303 | __le16 mode; | 326 | __le16 mode; |
@@ -375,12 +398,14 @@ struct squashfs_ldir_inode { | |||
375 | union squashfs_inode { | 398 | union squashfs_inode { |
376 | struct squashfs_base_inode base; | 399 | struct squashfs_base_inode base; |
377 | struct squashfs_dev_inode dev; | 400 | struct squashfs_dev_inode dev; |
401 | struct squashfs_ldev_inode ldev; | ||
378 | struct squashfs_symlink_inode symlink; | 402 | struct squashfs_symlink_inode symlink; |
379 | struct squashfs_reg_inode reg; | 403 | struct squashfs_reg_inode reg; |
380 | struct squashfs_lreg_inode lreg; | 404 | struct squashfs_lreg_inode lreg; |
381 | struct squashfs_dir_inode dir; | 405 | struct squashfs_dir_inode dir; |
382 | struct squashfs_ldir_inode ldir; | 406 | struct squashfs_ldir_inode ldir; |
383 | struct squashfs_ipc_inode ipc; | 407 | struct squashfs_ipc_inode ipc; |
408 | struct squashfs_lipc_inode lipc; | ||
384 | }; | 409 | }; |
385 | 410 | ||
386 | struct squashfs_dir_entry { | 411 | struct squashfs_dir_entry { |
diff --git a/fs/squashfs/super.c b/fs/squashfs/super.c index c4dfc393fa52..b6425ac1c2ae 100644 --- a/fs/squashfs/super.c +++ b/fs/squashfs/super.c | |||
@@ -140,13 +140,6 @@ static int squashfs_fill_super(struct super_block *sb, void *data, int silent) | |||
140 | if (msblk->decompressor == NULL) | 140 | if (msblk->decompressor == NULL) |
141 | goto failed_mount; | 141 | goto failed_mount; |
142 | 142 | ||
143 | /* | ||
144 | * Check if there's xattrs in the filesystem. These are not | ||
145 | * supported in this version, so warn that they will be ignored. | ||
146 | */ | ||
147 | if (le64_to_cpu(sblk->xattr_id_table_start) != SQUASHFS_INVALID_BLK) | ||
148 | ERROR("Xattrs in filesystem, these will be ignored\n"); | ||
149 | |||
150 | /* Check the filesystem does not extend beyond the end of the | 143 | /* Check the filesystem does not extend beyond the end of the |
151 | block device */ | 144 | block device */ |
152 | msblk->bytes_used = le64_to_cpu(sblk->bytes_used); | 145 | msblk->bytes_used = le64_to_cpu(sblk->bytes_used); |
@@ -268,6 +261,7 @@ allocate_lookup_table: | |||
268 | sb->s_export_op = &squashfs_export_ops; | 261 | sb->s_export_op = &squashfs_export_ops; |
269 | 262 | ||
270 | allocate_xattr_table: | 263 | allocate_xattr_table: |
264 | sb->s_xattr = squashfs_xattr_handlers; | ||
271 | xattr_id_table_start = le64_to_cpu(sblk->xattr_id_table_start); | 265 | xattr_id_table_start = le64_to_cpu(sblk->xattr_id_table_start); |
272 | if (xattr_id_table_start == SQUASHFS_INVALID_BLK) | 266 | if (xattr_id_table_start == SQUASHFS_INVALID_BLK) |
273 | goto allocate_root; | 267 | goto allocate_root; |
diff --git a/fs/squashfs/symlink.c b/fs/squashfs/symlink.c index 32b911f4ee39..a7ee68a8621b 100644 --- a/fs/squashfs/symlink.c +++ b/fs/squashfs/symlink.c | |||
@@ -35,6 +35,7 @@ | |||
35 | #include <linux/kernel.h> | 35 | #include <linux/kernel.h> |
36 | #include <linux/string.h> | 36 | #include <linux/string.h> |
37 | #include <linux/pagemap.h> | 37 | #include <linux/pagemap.h> |
38 | #include <linux/xattr.h> | ||
38 | 39 | ||
39 | #include "squashfs_fs.h" | 40 | #include "squashfs_fs.h" |
40 | #include "squashfs_fs_sb.h" | 41 | #include "squashfs_fs_sb.h" |
@@ -114,3 +115,12 @@ error_out: | |||
114 | const struct address_space_operations squashfs_symlink_aops = { | 115 | const struct address_space_operations squashfs_symlink_aops = { |
115 | .readpage = squashfs_symlink_readpage | 116 | .readpage = squashfs_symlink_readpage |
116 | }; | 117 | }; |
118 | |||
119 | const struct inode_operations squashfs_symlink_inode_ops = { | ||
120 | .readlink = generic_readlink, | ||
121 | .follow_link = page_follow_link_light, | ||
122 | .put_link = page_put_link, | ||
123 | .getxattr = generic_getxattr, | ||
124 | .listxattr = squashfs_listxattr | ||
125 | }; | ||
126 | |||