| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * decompress.c |
| * |
| * Detect the decompression method based on magic number |
| */ |
| |
| #include <linux/decompress/generic.h> |
| |
| #include <linux/decompress/bunzip2.h> |
| #include <linux/decompress/unlzma.h> |
| #include <linux/decompress/unxz.h> |
| #include <linux/decompress/inflate.h> |
| #include <linux/decompress/unlzo.h> |
| #include <linux/decompress/unlz4.h> |
| #include <linux/decompress/unzstd.h> |
| |
| #include <linux/types.h> |
| #include <linux/string.h> |
| #include <linux/init.h> |
| #include <linux/printk.h> |
| |
| #ifndef CONFIG_DECOMPRESS_GZIP |
| # define gunzip NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_BZIP2 |
| # define bunzip2 NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_LZMA |
| # define unlzma NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_XZ |
| # define unxz NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_LZO |
| # define unlzo NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_LZ4 |
| # define unlz4 NULL |
| #endif |
| #ifndef CONFIG_DECOMPRESS_ZSTD |
| # define unzstd NULL |
| #endif |
| |
| struct compress_format { |
| unsigned char magic[2]; |
| const char *name; |
| decompress_fn decompressor; |
| }; |
| |
| static const struct compress_format compressed_formats[] __initconst = { |
| { .magic = {0x1f, 0x8b}, .name = "gzip", .decompressor = gunzip }, |
| { .magic = {0x1f, 0x9e}, .name = "gzip", .decompressor = gunzip }, |
| { .magic = {0x42, 0x5a}, .name = "bzip2", .decompressor = bunzip2 }, |
| { .magic = {0x5d, 0x00}, .name = "lzma", .decompressor = unlzma }, |
| { .magic = {0xfd, 0x37}, .name = "xz", .decompressor = unxz }, |
| { .magic = {0x89, 0x4c}, .name = "lzo", .decompressor = unlzo }, |
| { .magic = {0x02, 0x21}, .name = "lz4", .decompressor = unlz4 }, |
| { .magic = {0x28, 0xb5}, .name = "zstd", .decompressor = unzstd }, |
| { /* sentinel */ } |
| }; |
| |
| decompress_fn __init decompress_method(const unsigned char *inbuf, long len, |
| const char **name) |
| { |
| const struct compress_format *cf; |
| |
| if (len < 2) { |
| if (name) |
| *name = NULL; |
| return NULL; /* Need at least this much... */ |
| } |
| |
| pr_debug("Compressed data magic: %#.2x %#.2x\n", inbuf[0], inbuf[1]); |
| |
| for (cf = compressed_formats; cf->name; cf++) |
| if (!memcmp(inbuf, cf->magic, 2)) |
| break; |
| |
| if (name) |
| *name = cf->name; |
| return cf->decompressor; |
| } |