Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 1 | // SPDX-License-Identifier: GPL-2.0 |
| 2 | /* |
| 3 | * Copyright (C) 2020 Bootlin |
| 4 | * |
| 5 | * Author: Joao Marcos Costa <joaomarcos.costa@bootlin.com> |
| 6 | */ |
| 7 | |
| 8 | #include <errno.h> |
| 9 | #include <stdint.h> |
| 10 | #include <stdio.h> |
| 11 | #include <stdlib.h> |
Joao Marcos Costa | 6dfed16 | 2020-08-18 17:17:24 +0200 | [diff] [blame] | 12 | |
| 13 | #if IS_ENABLED(CONFIG_LZO) |
| 14 | #include <linux/lzo.h> |
| 15 | #endif |
| 16 | |
Joao Marcos Costa | 3634b35 | 2020-07-30 15:33:50 +0200 | [diff] [blame] | 17 | #if IS_ENABLED(CONFIG_ZLIB) |
| 18 | #include <u-boot/zlib.h> |
| 19 | #endif |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 20 | |
David Oberhollenzer | fa894a3 | 2023-11-06 11:14:03 +0100 | [diff] [blame] | 21 | #if IS_ENABLED(CONFIG_LZ4) |
| 22 | #include <u-boot/lz4.h> |
| 23 | #endif |
| 24 | |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 25 | #if IS_ENABLED(CONFIG_ZSTD) |
| 26 | #include <linux/zstd.h> |
| 27 | #endif |
| 28 | |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 29 | #include "sqfs_decompressor.h" |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 30 | #include "sqfs_utils.h" |
| 31 | |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 32 | int sqfs_decompressor_init(struct squashfs_ctxt *ctxt) |
| 33 | { |
| 34 | u16 comp_type = get_unaligned_le16(&ctxt->sblk->compression); |
| 35 | |
| 36 | switch (comp_type) { |
Joao Marcos Costa | 6dfed16 | 2020-08-18 17:17:24 +0200 | [diff] [blame] | 37 | #if IS_ENABLED(CONFIG_LZO) |
| 38 | case SQFS_COMP_LZO: |
| 39 | break; |
| 40 | #endif |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 41 | #if IS_ENABLED(CONFIG_ZLIB) |
| 42 | case SQFS_COMP_ZLIB: |
| 43 | break; |
| 44 | #endif |
David Oberhollenzer | fa894a3 | 2023-11-06 11:14:03 +0100 | [diff] [blame] | 45 | #if IS_ENABLED(CONFIG_LZ4) |
| 46 | case SQFS_COMP_LZ4: |
| 47 | break; |
| 48 | #endif |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 49 | #if IS_ENABLED(CONFIG_ZSTD) |
| 50 | case SQFS_COMP_ZSTD: |
Brandon Maier | 4b9b25d | 2023-01-12 10:27:45 -0600 | [diff] [blame] | 51 | ctxt->zstd_workspace = malloc(zstd_dctx_workspace_bound()); |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 52 | if (!ctxt->zstd_workspace) |
| 53 | return -ENOMEM; |
| 54 | break; |
| 55 | #endif |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 56 | default: |
| 57 | printf("Error: unknown compression type.\n"); |
| 58 | return -EINVAL; |
| 59 | } |
| 60 | |
| 61 | return 0; |
| 62 | } |
| 63 | |
| 64 | void sqfs_decompressor_cleanup(struct squashfs_ctxt *ctxt) |
| 65 | { |
| 66 | u16 comp_type = get_unaligned_le16(&ctxt->sblk->compression); |
| 67 | |
| 68 | switch (comp_type) { |
Joao Marcos Costa | 6dfed16 | 2020-08-18 17:17:24 +0200 | [diff] [blame] | 69 | #if IS_ENABLED(CONFIG_LZO) |
| 70 | case SQFS_COMP_LZO: |
| 71 | break; |
| 72 | #endif |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 73 | #if IS_ENABLED(CONFIG_ZLIB) |
| 74 | case SQFS_COMP_ZLIB: |
| 75 | break; |
| 76 | #endif |
David Oberhollenzer | fa894a3 | 2023-11-06 11:14:03 +0100 | [diff] [blame] | 77 | #if IS_ENABLED(CONFIG_LZ4) |
| 78 | case SQFS_COMP_LZ4: |
| 79 | break; |
| 80 | #endif |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 81 | #if IS_ENABLED(CONFIG_ZSTD) |
| 82 | case SQFS_COMP_ZSTD: |
| 83 | free(ctxt->zstd_workspace); |
| 84 | break; |
| 85 | #endif |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 86 | } |
| 87 | } |
| 88 | |
Joao Marcos Costa | 3634b35 | 2020-07-30 15:33:50 +0200 | [diff] [blame] | 89 | #if IS_ENABLED(CONFIG_ZLIB) |
| 90 | static void zlib_decompression_status(int ret) |
| 91 | { |
| 92 | switch (ret) { |
| 93 | case Z_BUF_ERROR: |
| 94 | printf("Error: 'dest' buffer is not large enough.\n"); |
| 95 | break; |
| 96 | case Z_DATA_ERROR: |
| 97 | printf("Error: corrupted compressed data.\n"); |
| 98 | break; |
| 99 | case Z_MEM_ERROR: |
| 100 | printf("Error: insufficient memory.\n"); |
| 101 | break; |
| 102 | } |
| 103 | } |
| 104 | #endif |
| 105 | |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 106 | #if IS_ENABLED(CONFIG_ZSTD) |
| 107 | static int sqfs_zstd_decompress(struct squashfs_ctxt *ctxt, void *dest, |
| 108 | unsigned long dest_len, void *source, u32 src_len) |
| 109 | { |
| 110 | ZSTD_DCtx *ctx; |
| 111 | size_t wsize; |
| 112 | int ret; |
| 113 | |
Brandon Maier | 4b9b25d | 2023-01-12 10:27:45 -0600 | [diff] [blame] | 114 | wsize = zstd_dctx_workspace_bound(); |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 115 | |
Brandon Maier | 4b9b25d | 2023-01-12 10:27:45 -0600 | [diff] [blame] | 116 | ctx = zstd_init_dctx(ctxt->zstd_workspace, wsize); |
| 117 | if (!ctx) |
| 118 | return -EINVAL; |
| 119 | ret = zstd_decompress_dctx(ctx, dest, dest_len, source, src_len); |
| 120 | |
| 121 | return zstd_is_error(ret); |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 122 | } |
| 123 | #endif /* CONFIG_ZSTD */ |
| 124 | |
Joao Marcos Costa | cdc1144 | 2020-08-18 17:17:22 +0200 | [diff] [blame] | 125 | int sqfs_decompress(struct squashfs_ctxt *ctxt, void *dest, |
| 126 | unsigned long *dest_len, void *source, u32 src_len) |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 127 | { |
Joao Marcos Costa | cdc1144 | 2020-08-18 17:17:22 +0200 | [diff] [blame] | 128 | u16 comp_type = get_unaligned_le16(&ctxt->sblk->compression); |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 129 | int ret = 0; |
| 130 | |
| 131 | switch (comp_type) { |
Joao Marcos Costa | 6dfed16 | 2020-08-18 17:17:24 +0200 | [diff] [blame] | 132 | #if IS_ENABLED(CONFIG_LZO) |
| 133 | case SQFS_COMP_LZO: { |
| 134 | size_t lzo_dest_len = *dest_len; |
| 135 | ret = lzo1x_decompress_safe(source, src_len, dest, &lzo_dest_len); |
| 136 | if (ret) { |
| 137 | printf("LZO decompression failed. Error code: %d\n", ret); |
| 138 | return -EINVAL; |
| 139 | } |
| 140 | |
| 141 | break; |
| 142 | } |
| 143 | #endif |
Joao Marcos Costa | 3634b35 | 2020-07-30 15:33:50 +0200 | [diff] [blame] | 144 | #if IS_ENABLED(CONFIG_ZLIB) |
| 145 | case SQFS_COMP_ZLIB: |
Joao Marcos Costa | 10f7cf5 | 2020-08-18 17:17:21 +0200 | [diff] [blame] | 146 | ret = uncompress(dest, dest_len, source, src_len); |
Joao Marcos Costa | 3634b35 | 2020-07-30 15:33:50 +0200 | [diff] [blame] | 147 | if (ret) { |
| 148 | zlib_decompression_status(ret); |
| 149 | return -EINVAL; |
| 150 | } |
| 151 | |
| 152 | break; |
| 153 | #endif |
David Oberhollenzer | fa894a3 | 2023-11-06 11:14:03 +0100 | [diff] [blame] | 154 | #if IS_ENABLED(CONFIG_LZ4) |
| 155 | case SQFS_COMP_LZ4: |
| 156 | ret = LZ4_decompress_safe(source, dest, src_len, *dest_len); |
| 157 | if (ret < 0) { |
| 158 | printf("LZ4 decompression failed.\n"); |
| 159 | return -EINVAL; |
| 160 | } |
| 161 | |
| 162 | ret = 0; |
| 163 | break; |
| 164 | #endif |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 165 | #if IS_ENABLED(CONFIG_ZSTD) |
| 166 | case SQFS_COMP_ZSTD: |
| 167 | ret = sqfs_zstd_decompress(ctxt, dest, *dest_len, source, src_len); |
| 168 | if (ret) { |
Brandon Maier | 4b9b25d | 2023-01-12 10:27:45 -0600 | [diff] [blame] | 169 | printf("ZSTD Error code: %d\n", zstd_get_error_code(ret)); |
Joao Marcos Costa | 9c948f5 | 2020-08-18 17:17:23 +0200 | [diff] [blame] | 170 | return -EINVAL; |
| 171 | } |
| 172 | |
| 173 | break; |
| 174 | #endif |
Joao Marcos Costa | c510061 | 2020-07-30 15:33:47 +0200 | [diff] [blame] | 175 | default: |
| 176 | printf("Error: unknown compression type.\n"); |
| 177 | return -EINVAL; |
| 178 | } |
| 179 | |
| 180 | return ret; |
| 181 | } |