Gao Xiang | 29b24f6 | 2019-07-31 23:57:31 +0800 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 2 | /* |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 3 | * Copyright (C) 2019 HUAWEI, Inc. |
Alexander A. Klimov | 592e7cd | 2020-07-13 15:09:44 +0200 | [diff] [blame] | 4 | * https://www.huawei.com/ |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 5 | */ |
| 6 | #ifndef __EROFS_FS_COMPRESS_H |
| 7 | #define __EROFS_FS_COMPRESS_H |
| 8 | |
Gao Xiang | 7fc45db | 2019-06-24 15:22:55 +0800 | [diff] [blame] | 9 | #include "internal.h" |
| 10 | |
Gao Xiang | 7fc45db | 2019-06-24 15:22:55 +0800 | [diff] [blame] | 11 | struct z_erofs_decompress_req { |
Gao Xiang | 0ffd71b | 2019-06-24 15:22:56 +0800 | [diff] [blame] | 12 | struct super_block *sb; |
Gao Xiang | 7fc45db | 2019-06-24 15:22:55 +0800 | [diff] [blame] | 13 | struct page **in, **out; |
| 14 | |
| 15 | unsigned short pageofs_out; |
| 16 | unsigned int inputsize, outputsize; |
| 17 | |
| 18 | /* indicate the algorithm will be used for decompression */ |
| 19 | unsigned int alg; |
| 20 | bool inplace_io, partial_decoding; |
| 21 | }; |
| 22 | |
Gao Xiang | 622cead | 2021-10-11 05:31:45 +0800 | [diff] [blame] | 23 | struct z_erofs_decompressor { |
| 24 | int (*decompress)(struct z_erofs_decompress_req *rq, |
Gao Xiang | eaa9172 | 2021-10-22 17:01:20 +0800 | [diff] [blame^] | 25 | struct page **pagepool); |
Gao Xiang | 622cead | 2021-10-11 05:31:45 +0800 | [diff] [blame] | 26 | char *name; |
| 27 | }; |
| 28 | |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 29 | /* some special page->private (unsigned long, see below) */ |
| 30 | #define Z_EROFS_SHORTLIVED_PAGE (-1UL << 2) |
Gao Xiang | 1825c8d | 2020-12-09 20:37:17 +0800 | [diff] [blame] | 31 | #define Z_EROFS_PREALLOCATED_PAGE (-2UL << 2) |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 32 | |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 33 | /* |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 34 | * For all pages in a pcluster, page->private should be one of |
| 35 | * Type Last 2bits page->private |
| 36 | * short-lived page 00 Z_EROFS_SHORTLIVED_PAGE |
Gao Xiang | 1825c8d | 2020-12-09 20:37:17 +0800 | [diff] [blame] | 37 | * preallocated page (tryalloc) 00 Z_EROFS_PREALLOCATED_PAGE |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 38 | * cached/managed page 00 pointer to z_erofs_pcluster |
| 39 | * online page (file-backed, 01/10/11 sub-index << 2 | count |
| 40 | * some pages can be used for inplace I/O) |
| 41 | * |
| 42 | * page->mapping should be one of |
| 43 | * Type page->mapping |
| 44 | * short-lived page NULL |
Gao Xiang | 1825c8d | 2020-12-09 20:37:17 +0800 | [diff] [blame] | 45 | * preallocated page NULL |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 46 | * cached/managed page non-NULL or NULL (invalidated/truncated page) |
| 47 | * online page non-NULL |
| 48 | * |
| 49 | * For all managed pages, PG_private should be set with 1 extra refcount, |
| 50 | * which is used for page reclaim / migration. |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 51 | */ |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 52 | |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 53 | /* |
| 54 | * short-lived pages are pages directly from buddy system with specific |
| 55 | * page->private (no need to set PagePrivate since these are non-LRU / |
| 56 | * non-movable pages and bypass reclaim / migration code). |
| 57 | */ |
| 58 | static inline bool z_erofs_is_shortlived_page(struct page *page) |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 59 | { |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 60 | if (page->private != Z_EROFS_SHORTLIVED_PAGE) |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 61 | return false; |
| 62 | |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 63 | DBG_BUGON(page->mapping); |
| 64 | return true; |
| 65 | } |
| 66 | |
Gao Xiang | eaa9172 | 2021-10-22 17:01:20 +0800 | [diff] [blame^] | 67 | static inline bool z_erofs_put_shortlivedpage(struct page **pagepool, |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 68 | struct page *page) |
| 69 | { |
| 70 | if (!z_erofs_is_shortlived_page(page)) |
| 71 | return false; |
| 72 | |
| 73 | /* short-lived pages should not be used by others at the same time */ |
| 74 | if (page_ref_count(page) > 1) { |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 75 | put_page(page); |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 76 | } else { |
| 77 | /* follow the pcluster rule above. */ |
Gao Xiang | eaa9172 | 2021-10-22 17:01:20 +0800 | [diff] [blame^] | 78 | erofs_pagepool_add(pagepool, page); |
Gao Xiang | 6aaa7b0 | 2020-12-08 17:58:32 +0800 | [diff] [blame] | 79 | } |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 80 | return true; |
| 81 | } |
| 82 | |
Gao Xiang | 622cead | 2021-10-11 05:31:45 +0800 | [diff] [blame] | 83 | #define MNGD_MAPPING(sbi) ((sbi)->managed_cache->i_mapping) |
| 84 | static inline bool erofs_page_is_managed(const struct erofs_sb_info *sbi, |
| 85 | struct page *page) |
| 86 | { |
| 87 | return page->mapping == MNGD_MAPPING(sbi); |
| 88 | } |
| 89 | |
Gao Xiang | 7fc45db | 2019-06-24 15:22:55 +0800 | [diff] [blame] | 90 | int z_erofs_decompress(struct z_erofs_decompress_req *rq, |
Gao Xiang | eaa9172 | 2021-10-22 17:01:20 +0800 | [diff] [blame^] | 91 | struct page **pagepool); |
Gao Xiang | 7fc45db | 2019-06-24 15:22:55 +0800 | [diff] [blame] | 92 | |
Gao Xiang | 622cead | 2021-10-11 05:31:45 +0800 | [diff] [blame] | 93 | /* prototypes for specific algorithms */ |
| 94 | int z_erofs_lzma_decompress(struct z_erofs_decompress_req *rq, |
Gao Xiang | eaa9172 | 2021-10-22 17:01:20 +0800 | [diff] [blame^] | 95 | struct page **pagepool); |
Gao Xiang | 2748123 | 2019-06-24 15:22:54 +0800 | [diff] [blame] | 96 | #endif |