Commit b7fc0ff0 authored by Phillip Lougher's avatar Phillip Lougher

Squashfs: extend decompressor framework to handle compression options

Extend decompressor framework to handle compression options stored in
the filesystem.  These options can be used by the relevant decompressor
at initialisation time to over-ride defaults.

The presence of compression options in the filesystem is indicated by
the COMP_OPT filesystem flag.  If present the data is read from the
filesystem and passed to the decompressor init function.  The decompressor
init function signature has been extended to take this data.

Also update the init function signature in the glib, lzo and xz
decompressor wrappers.
Signed-off-by: default avatarPhillip Lougher <phillip@lougher.demon.co.uk>
parent 100b33c8
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/mutex.h> #include <linux/mutex.h>
#include <linux/slab.h>
#include <linux/buffer_head.h> #include <linux/buffer_head.h>
#include "squashfs_fs.h" #include "squashfs_fs.h"
...@@ -74,3 +75,36 @@ const struct squashfs_decompressor *squashfs_lookup_decompressor(int id) ...@@ -74,3 +75,36 @@ const struct squashfs_decompressor *squashfs_lookup_decompressor(int id)
return decompressor[i]; return decompressor[i];
} }
void *squashfs_decompressor_init(struct super_block *sb, unsigned short flags)
{
struct squashfs_sb_info *msblk = sb->s_fs_info;
void *strm, *buffer = NULL;
int length = 0;
/*
* Read decompressor specific options from file system if present
*/
if (SQUASHFS_COMP_OPTS(flags)) {
buffer = kmalloc(PAGE_CACHE_SIZE, GFP_KERNEL);
if (buffer == NULL)
return ERR_PTR(-ENOMEM);
length = squashfs_read_data(sb, &buffer,
sizeof(struct squashfs_super_block), 0, NULL,
PAGE_CACHE_SIZE, 1);
if (length < 0) {
strm = ERR_PTR(length);
goto finished;
}
}
strm = msblk->decompressor->init(msblk, buffer, length);
finished:
kfree(buffer);
return strm;
}
...@@ -24,7 +24,7 @@ ...@@ -24,7 +24,7 @@
*/ */
struct squashfs_decompressor { struct squashfs_decompressor {
void *(*init)(struct squashfs_sb_info *); void *(*init)(struct squashfs_sb_info *, void *, int);
void (*free)(void *); void (*free)(void *);
int (*decompress)(struct squashfs_sb_info *, void **, int (*decompress)(struct squashfs_sb_info *, void **,
struct buffer_head **, int, int, int, int, int); struct buffer_head **, int, int, int, int, int);
...@@ -33,11 +33,6 @@ struct squashfs_decompressor { ...@@ -33,11 +33,6 @@ struct squashfs_decompressor {
int supported; int supported;
}; };
static inline void *squashfs_decompressor_init(struct squashfs_sb_info *msblk)
{
return msblk->decompressor->init(msblk);
}
static inline void squashfs_decompressor_free(struct squashfs_sb_info *msblk, static inline void squashfs_decompressor_free(struct squashfs_sb_info *msblk,
void *s) void *s)
{ {
......
...@@ -37,7 +37,7 @@ struct squashfs_lzo { ...@@ -37,7 +37,7 @@ struct squashfs_lzo {
void *output; void *output;
}; };
static void *lzo_init(struct squashfs_sb_info *msblk) static void *lzo_init(struct squashfs_sb_info *msblk, void *buff, int len)
{ {
int block_size = max_t(int, msblk->block_size, SQUASHFS_METADATA_SIZE); int block_size = max_t(int, msblk->block_size, SQUASHFS_METADATA_SIZE);
...@@ -58,7 +58,7 @@ static void *lzo_init(struct squashfs_sb_info *msblk) ...@@ -58,7 +58,7 @@ static void *lzo_init(struct squashfs_sb_info *msblk)
failed: failed:
ERROR("Failed to allocate lzo workspace\n"); ERROR("Failed to allocate lzo workspace\n");
kfree(stream); kfree(stream);
return NULL; return ERR_PTR(-ENOMEM);
} }
......
...@@ -48,6 +48,7 @@ extern int squashfs_read_table(struct super_block *, void *, u64, int); ...@@ -48,6 +48,7 @@ extern int squashfs_read_table(struct super_block *, void *, u64, int);
/* decompressor.c */ /* decompressor.c */
extern const struct squashfs_decompressor *squashfs_lookup_decompressor(int); extern const struct squashfs_decompressor *squashfs_lookup_decompressor(int);
extern void *squashfs_decompressor_init(struct super_block *, unsigned short);
/* export.c */ /* export.c */
extern __le64 *squashfs_read_inode_lookup_table(struct super_block *, u64, extern __le64 *squashfs_read_inode_lookup_table(struct super_block *, u64,
......
...@@ -57,6 +57,7 @@ ...@@ -57,6 +57,7 @@
#define SQUASHFS_ALWAYS_FRAG 5 #define SQUASHFS_ALWAYS_FRAG 5
#define SQUASHFS_DUPLICATE 6 #define SQUASHFS_DUPLICATE 6
#define SQUASHFS_EXPORT 7 #define SQUASHFS_EXPORT 7
#define SQUASHFS_COMP_OPT 10
#define SQUASHFS_BIT(flag, bit) ((flag >> bit) & 1) #define SQUASHFS_BIT(flag, bit) ((flag >> bit) & 1)
...@@ -81,6 +82,9 @@ ...@@ -81,6 +82,9 @@
#define SQUASHFS_EXPORTABLE(flags) SQUASHFS_BIT(flags, \ #define SQUASHFS_EXPORTABLE(flags) SQUASHFS_BIT(flags, \
SQUASHFS_EXPORT) SQUASHFS_EXPORT)
#define SQUASHFS_COMP_OPTS(flags) SQUASHFS_BIT(flags, \
SQUASHFS_COMP_OPT)
/* Max number of types and file types */ /* Max number of types and file types */
#define SQUASHFS_DIR_TYPE 1 #define SQUASHFS_DIR_TYPE 1
#define SQUASHFS_REG_TYPE 2 #define SQUASHFS_REG_TYPE 2
......
...@@ -199,10 +199,6 @@ static int squashfs_fill_super(struct super_block *sb, void *data, int silent) ...@@ -199,10 +199,6 @@ static int squashfs_fill_super(struct super_block *sb, void *data, int silent)
err = -ENOMEM; err = -ENOMEM;
msblk->stream = squashfs_decompressor_init(msblk);
if (msblk->stream == NULL)
goto failed_mount;
msblk->block_cache = squashfs_cache_init("metadata", msblk->block_cache = squashfs_cache_init("metadata",
SQUASHFS_CACHED_BLKS, SQUASHFS_METADATA_SIZE); SQUASHFS_CACHED_BLKS, SQUASHFS_METADATA_SIZE);
if (msblk->block_cache == NULL) if (msblk->block_cache == NULL)
...@@ -215,6 +211,13 @@ static int squashfs_fill_super(struct super_block *sb, void *data, int silent) ...@@ -215,6 +211,13 @@ static int squashfs_fill_super(struct super_block *sb, void *data, int silent)
goto failed_mount; goto failed_mount;
} }
msblk->stream = squashfs_decompressor_init(sb, flags);
if (IS_ERR(msblk->stream)) {
err = PTR_ERR(msblk->stream);
msblk->stream = NULL;
goto failed_mount;
}
/* Allocate and read id index table */ /* Allocate and read id index table */
msblk->id_table = squashfs_read_id_index_table(sb, msblk->id_table = squashfs_read_id_index_table(sb,
le64_to_cpu(sblk->id_table_start), le16_to_cpu(sblk->no_ids)); le64_to_cpu(sblk->id_table_start), le16_to_cpu(sblk->no_ids));
......
...@@ -38,7 +38,8 @@ struct squashfs_xz { ...@@ -38,7 +38,8 @@ struct squashfs_xz {
struct xz_buf buf; struct xz_buf buf;
}; };
static void *squashfs_xz_init(struct squashfs_sb_info *msblk) static void *squashfs_xz_init(struct squashfs_sb_info *msblk, void *buff,
int len)
{ {
int block_size = max_t(int, msblk->block_size, SQUASHFS_METADATA_SIZE); int block_size = max_t(int, msblk->block_size, SQUASHFS_METADATA_SIZE);
...@@ -55,7 +56,7 @@ static void *squashfs_xz_init(struct squashfs_sb_info *msblk) ...@@ -55,7 +56,7 @@ static void *squashfs_xz_init(struct squashfs_sb_info *msblk)
failed: failed:
ERROR("Failed to allocate xz workspace\n"); ERROR("Failed to allocate xz workspace\n");
kfree(stream); kfree(stream);
return NULL; return ERR_PTR(-ENOMEM);
} }
......
...@@ -32,7 +32,7 @@ ...@@ -32,7 +32,7 @@
#include "squashfs.h" #include "squashfs.h"
#include "decompressor.h" #include "decompressor.h"
static void *zlib_init(struct squashfs_sb_info *dummy) static void *zlib_init(struct squashfs_sb_info *dummy, void *buff, int len)
{ {
z_stream *stream = kmalloc(sizeof(z_stream), GFP_KERNEL); z_stream *stream = kmalloc(sizeof(z_stream), GFP_KERNEL);
if (stream == NULL) if (stream == NULL)
...@@ -47,7 +47,7 @@ static void *zlib_init(struct squashfs_sb_info *dummy) ...@@ -47,7 +47,7 @@ static void *zlib_init(struct squashfs_sb_info *dummy)
failed: failed:
ERROR("Failed to allocate zlib workspace\n"); ERROR("Failed to allocate zlib workspace\n");
kfree(stream); kfree(stream);
return NULL; return ERR_PTR(-ENOMEM);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment