mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-04 04:04:19 +00:00
Changes since last update:
- Make LZ4 global buffers configurable instead of per-CPU buffers; - Add a reserved buffer pool for LZ4 decompression for lower latencies; - Support Zstandard compression algorithm as an alternative; - Derive fsid from on-disk UUID for .statfs() if possible; - Minor cleanups. -----BEGIN PGP SIGNATURE----- iQJFBAABCgAvFiEEQ0A6bDUS9Y+83NPFUXZn5Zlu5qoFAmZBZ4QRHHhpYW5nQGtl cm5lbC5vcmcACgkQUXZn5Zlu5qpDWg/7BZMExLobfLg22Sa9jw1l5UKhQSA+Y97i 8aJIMPD7I0U1PERz5pXplDIIUWSPnVlXr6gGR43VSzpU3ud58XpQVXKkGfZFRKGg mevpLLJUHrvtTCkTeykit6m2jUzd1r24xdvgSsC0gyek0tTCXMkKA/fgtqDKud0B oQt/nIN3xDG4/vHzU1XGXHn5uHPwZAsU50mjD0jBlskl8tdCFrc3X8uV6aOZuJ3U q88F+B3Bnvby8NgxDPpeIj/7JIEtlXZLgDoOrWGRDIL1Hd31H7iPZp10eXD4yQLH kDjvGk9bFf6qIHfNtdOPtwmeQ0VRSg0avYsGo4gLPVmkh6VveNQYyvlMbCKewY4l u+DOl3XaobBUZ/efhWy5Gq9DDnl8SmgQCJoJ0ST+mo+egKhRI0Ynm2jG4u9tnB+9 1FJU6OKkHoe0OHg18nNVkMe0FbdkW2rCRpCOAavPGWD4625zmHj6P6EhAU7QE+Q+ Lz5W0N0XrXkQN+EDybRnAA0GZxHr+CDOenbdAM41yJp7fSREHxfyOZP3oIz4GP1D ru95E+XL/GWQuWXfQbDrkw/Ixyv8lmiteRSFU5Fys+W7n205+2H9exMHZrh3IgwJ HVFCfd94h9eXOkyy7A7bTkLs1SFNMbWd7hicjwDlKs2T7Xz2r7egR8r7VADU59FP +nGwOS/kNnU= =gAD0 -----END PGP SIGNATURE----- Merge tag 'erofs-for-6.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs Pull erofs updates from Gao Xiang: "The LZ4 global buffer count is now configurable instead of the previous per-CPU buffers, which is useful for bare metals with hundreds of CPUs. A reserved buffer pool for LZ4 decompression can also be enabled to minimize the tail allocation latencies under the low memory scenarios with heavy memory pressure. In addition, Zstandard algorithm is now supported as an alternative since it has been requested by users for a while. There are some random cleanups as usual. Summary: - Make LZ4 global buffers configurable instead of per-CPU buffers - Add a reserved buffer pool for LZ4 decompression for lower latencies - Support Zstandard compression algorithm as an alternative - Derive fsid from on-disk UUID for .statfs() if possible - Minor cleanups" * tag 'erofs-for-6.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs: erofs: Zstandard compression support erofs: clean up z_erofs_load_full_lcluster() erofs: derive fsid from on-disk UUID for .statfs() if possible erofs: add a reserved buffer pool for lz4 decompression erofs: do not use pagepool in z_erofs_gbuf_growsize() erofs: rename per-CPU buffers to global buffer pool and make it configurable erofs: rename utils.c to zutil.c
This commit is contained in:
commit
47e9bff7fc
@ -112,6 +112,21 @@ config EROFS_FS_ZIP_DEFLATE
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
config EROFS_FS_ZIP_ZSTD
|
||||
bool "EROFS Zstandard compressed data support"
|
||||
depends on EROFS_FS_ZIP
|
||||
select ZSTD_DECOMPRESS
|
||||
help
|
||||
Saying Y here includes support for reading EROFS file systems
|
||||
containing Zstandard compressed data. It gives better compression
|
||||
ratios than the default LZ4 format, while it costs more CPU
|
||||
overhead.
|
||||
|
||||
Zstandard support is an experimental feature for now and so most
|
||||
file systems will be readable without selecting this option.
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
config EROFS_FS_ONDEMAND
|
||||
bool "EROFS fscache-based on-demand read support"
|
||||
depends on EROFS_FS
|
||||
|
@ -1,9 +1,10 @@
|
||||
# SPDX-License-Identifier: GPL-2.0-only
|
||||
|
||||
obj-$(CONFIG_EROFS_FS) += erofs.o
|
||||
erofs-objs := super.o inode.o data.o namei.o dir.o utils.o sysfs.o
|
||||
erofs-objs := super.o inode.o data.o namei.o dir.o sysfs.o
|
||||
erofs-$(CONFIG_EROFS_FS_XATTR) += xattr.o
|
||||
erofs-$(CONFIG_EROFS_FS_ZIP) += decompressor.o zmap.o zdata.o pcpubuf.o
|
||||
erofs-$(CONFIG_EROFS_FS_ZIP) += decompressor.o zmap.o zdata.o zutil.o
|
||||
erofs-$(CONFIG_EROFS_FS_ZIP_LZMA) += decompressor_lzma.o
|
||||
erofs-$(CONFIG_EROFS_FS_ZIP_DEFLATE) += decompressor_deflate.o
|
||||
erofs-$(CONFIG_EROFS_FS_ZIP_ZSTD) += decompressor_zstd.o
|
||||
erofs-$(CONFIG_EROFS_FS_ONDEMAND) += fscache.o
|
||||
|
@ -90,8 +90,12 @@ int z_erofs_load_lzma_config(struct super_block *sb,
|
||||
struct erofs_super_block *dsb, void *data, int size);
|
||||
int z_erofs_load_deflate_config(struct super_block *sb,
|
||||
struct erofs_super_block *dsb, void *data, int size);
|
||||
int z_erofs_load_zstd_config(struct super_block *sb,
|
||||
struct erofs_super_block *dsb, void *data, int size);
|
||||
int z_erofs_lzma_decompress(struct z_erofs_decompress_req *rq,
|
||||
struct page **pagepool);
|
||||
int z_erofs_deflate_decompress(struct z_erofs_decompress_req *rq,
|
||||
struct page **pagepool);
|
||||
int z_erofs_zstd_decompress(struct z_erofs_decompress_req *rq,
|
||||
struct page **pgpl);
|
||||
#endif
|
||||
|
@ -54,7 +54,7 @@ static int z_erofs_load_lz4_config(struct super_block *sb,
|
||||
sbi->lz4.max_distance_pages = distance ?
|
||||
DIV_ROUND_UP(distance, PAGE_SIZE) + 1 :
|
||||
LZ4_MAX_DISTANCE_PAGES;
|
||||
return erofs_pcpubuf_growsize(sbi->lz4.max_pclusterblks);
|
||||
return z_erofs_gbuf_growsize(sbi->lz4.max_pclusterblks);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -111,7 +111,7 @@ static int z_erofs_lz4_prepare_dstpages(struct z_erofs_lz4_decompress_ctx *ctx,
|
||||
victim = availables[--top];
|
||||
get_page(victim);
|
||||
} else {
|
||||
victim = erofs_allocpage(pagepool, rq->gfp);
|
||||
victim = __erofs_allocpage(pagepool, rq->gfp, true);
|
||||
if (!victim)
|
||||
return -ENOMEM;
|
||||
set_page_private(victim, Z_EROFS_SHORTLIVED_PAGE);
|
||||
@ -159,7 +159,7 @@ static void *z_erofs_lz4_handle_overlap(struct z_erofs_lz4_decompress_ctx *ctx,
|
||||
docopy:
|
||||
/* Or copy compressed data which can be overlapped to per-CPU buffer */
|
||||
in = rq->in;
|
||||
src = erofs_get_pcpubuf(ctx->inpages);
|
||||
src = z_erofs_get_gbuf(ctx->inpages);
|
||||
if (!src) {
|
||||
DBG_BUGON(1);
|
||||
kunmap_local(inpage);
|
||||
@ -260,7 +260,7 @@ static int z_erofs_lz4_decompress_mem(struct z_erofs_lz4_decompress_ctx *ctx,
|
||||
} else if (maptype == 1) {
|
||||
vm_unmap_ram(src, ctx->inpages);
|
||||
} else if (maptype == 2) {
|
||||
erofs_put_pcpubuf(src);
|
||||
z_erofs_put_gbuf(src);
|
||||
} else if (maptype != 3) {
|
||||
DBG_BUGON(1);
|
||||
return -EFAULT;
|
||||
@ -399,6 +399,13 @@ const struct z_erofs_decompressor erofs_decompressors[] = {
|
||||
.name = "deflate"
|
||||
},
|
||||
#endif
|
||||
#ifdef CONFIG_EROFS_FS_ZIP_ZSTD
|
||||
[Z_EROFS_COMPRESSION_ZSTD] = {
|
||||
.config = z_erofs_load_zstd_config,
|
||||
.decompress = z_erofs_zstd_decompress,
|
||||
.name = "zstd"
|
||||
},
|
||||
#endif
|
||||
};
|
||||
|
||||
int z_erofs_parse_cfgs(struct super_block *sb, struct erofs_super_block *dsb)
|
||||
|
279
fs/erofs/decompressor_zstd.c
Normal file
279
fs/erofs/decompressor_zstd.c
Normal file
@ -0,0 +1,279 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
#include <linux/zstd.h>
|
||||
#include "compress.h"
|
||||
|
||||
struct z_erofs_zstd {
|
||||
struct z_erofs_zstd *next;
|
||||
u8 bounce[PAGE_SIZE];
|
||||
void *wksp;
|
||||
unsigned int wkspsz;
|
||||
};
|
||||
|
||||
static DEFINE_SPINLOCK(z_erofs_zstd_lock);
|
||||
static unsigned int z_erofs_zstd_max_dictsize;
|
||||
static unsigned int z_erofs_zstd_nstrms, z_erofs_zstd_avail_strms;
|
||||
static struct z_erofs_zstd *z_erofs_zstd_head;
|
||||
static DECLARE_WAIT_QUEUE_HEAD(z_erofs_zstd_wq);
|
||||
|
||||
module_param_named(zstd_streams, z_erofs_zstd_nstrms, uint, 0444);
|
||||
|
||||
static struct z_erofs_zstd *z_erofs_isolate_strms(bool all)
|
||||
{
|
||||
struct z_erofs_zstd *strm;
|
||||
|
||||
again:
|
||||
spin_lock(&z_erofs_zstd_lock);
|
||||
strm = z_erofs_zstd_head;
|
||||
if (!strm) {
|
||||
spin_unlock(&z_erofs_zstd_lock);
|
||||
wait_event(z_erofs_zstd_wq, READ_ONCE(z_erofs_zstd_head));
|
||||
goto again;
|
||||
}
|
||||
z_erofs_zstd_head = all ? NULL : strm->next;
|
||||
spin_unlock(&z_erofs_zstd_lock);
|
||||
return strm;
|
||||
}
|
||||
|
||||
void z_erofs_zstd_exit(void)
|
||||
{
|
||||
while (z_erofs_zstd_avail_strms) {
|
||||
struct z_erofs_zstd *strm, *n;
|
||||
|
||||
for (strm = z_erofs_isolate_strms(true); strm; strm = n) {
|
||||
n = strm->next;
|
||||
|
||||
kvfree(strm->wksp);
|
||||
kfree(strm);
|
||||
--z_erofs_zstd_avail_strms;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int __init z_erofs_zstd_init(void)
|
||||
{
|
||||
/* by default, use # of possible CPUs instead */
|
||||
if (!z_erofs_zstd_nstrms)
|
||||
z_erofs_zstd_nstrms = num_possible_cpus();
|
||||
|
||||
for (; z_erofs_zstd_avail_strms < z_erofs_zstd_nstrms;
|
||||
++z_erofs_zstd_avail_strms) {
|
||||
struct z_erofs_zstd *strm;
|
||||
|
||||
strm = kzalloc(sizeof(*strm), GFP_KERNEL);
|
||||
if (!strm) {
|
||||
z_erofs_zstd_exit();
|
||||
return -ENOMEM;
|
||||
}
|
||||
spin_lock(&z_erofs_zstd_lock);
|
||||
strm->next = z_erofs_zstd_head;
|
||||
z_erofs_zstd_head = strm;
|
||||
spin_unlock(&z_erofs_zstd_lock);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int z_erofs_load_zstd_config(struct super_block *sb,
|
||||
struct erofs_super_block *dsb, void *data, int size)
|
||||
{
|
||||
static DEFINE_MUTEX(zstd_resize_mutex);
|
||||
struct z_erofs_zstd_cfgs *zstd = data;
|
||||
unsigned int dict_size, wkspsz;
|
||||
struct z_erofs_zstd *strm, *head = NULL;
|
||||
void *wksp;
|
||||
|
||||
if (!zstd || size < sizeof(struct z_erofs_zstd_cfgs) || zstd->format) {
|
||||
erofs_err(sb, "unsupported zstd format, size=%u", size);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (zstd->windowlog > ilog2(Z_EROFS_ZSTD_MAX_DICT_SIZE) - 10) {
|
||||
erofs_err(sb, "unsupported zstd window log %u", zstd->windowlog);
|
||||
return -EINVAL;
|
||||
}
|
||||
dict_size = 1U << (zstd->windowlog + 10);
|
||||
|
||||
/* in case 2 z_erofs_load_zstd_config() race to avoid deadlock */
|
||||
mutex_lock(&zstd_resize_mutex);
|
||||
if (z_erofs_zstd_max_dictsize >= dict_size) {
|
||||
mutex_unlock(&zstd_resize_mutex);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* 1. collect/isolate all streams for the following check */
|
||||
while (z_erofs_zstd_avail_strms) {
|
||||
struct z_erofs_zstd *n;
|
||||
|
||||
for (strm = z_erofs_isolate_strms(true); strm; strm = n) {
|
||||
n = strm->next;
|
||||
strm->next = head;
|
||||
head = strm;
|
||||
--z_erofs_zstd_avail_strms;
|
||||
}
|
||||
}
|
||||
|
||||
/* 2. walk each isolated stream and grow max dict_size if needed */
|
||||
wkspsz = zstd_dstream_workspace_bound(dict_size);
|
||||
for (strm = head; strm; strm = strm->next) {
|
||||
wksp = kvmalloc(wkspsz, GFP_KERNEL);
|
||||
if (!wksp)
|
||||
break;
|
||||
kvfree(strm->wksp);
|
||||
strm->wksp = wksp;
|
||||
strm->wkspsz = wkspsz;
|
||||
}
|
||||
|
||||
/* 3. push back all to the global list and update max dict_size */
|
||||
spin_lock(&z_erofs_zstd_lock);
|
||||
DBG_BUGON(z_erofs_zstd_head);
|
||||
z_erofs_zstd_head = head;
|
||||
spin_unlock(&z_erofs_zstd_lock);
|
||||
z_erofs_zstd_avail_strms = z_erofs_zstd_nstrms;
|
||||
wake_up_all(&z_erofs_zstd_wq);
|
||||
if (!strm)
|
||||
z_erofs_zstd_max_dictsize = dict_size;
|
||||
mutex_unlock(&zstd_resize_mutex);
|
||||
return strm ? -ENOMEM : 0;
|
||||
}
|
||||
|
||||
int z_erofs_zstd_decompress(struct z_erofs_decompress_req *rq,
|
||||
struct page **pgpl)
|
||||
{
|
||||
const unsigned int nrpages_out =
|
||||
PAGE_ALIGN(rq->pageofs_out + rq->outputsize) >> PAGE_SHIFT;
|
||||
const unsigned int nrpages_in =
|
||||
PAGE_ALIGN(rq->inputsize) >> PAGE_SHIFT;
|
||||
zstd_dstream *stream;
|
||||
struct super_block *sb = rq->sb;
|
||||
unsigned int insz, outsz, pofs;
|
||||
struct z_erofs_zstd *strm;
|
||||
zstd_in_buffer in_buf = { NULL, 0, 0 };
|
||||
zstd_out_buffer out_buf = { NULL, 0, 0 };
|
||||
u8 *kin, *kout = NULL;
|
||||
bool bounced = false;
|
||||
int no = -1, ni = 0, j = 0, zerr, err;
|
||||
|
||||
/* 1. get the exact compressed size */
|
||||
kin = kmap_local_page(*rq->in);
|
||||
err = z_erofs_fixup_insize(rq, kin + rq->pageofs_in,
|
||||
min_t(unsigned int, rq->inputsize,
|
||||
sb->s_blocksize - rq->pageofs_in));
|
||||
if (err) {
|
||||
kunmap_local(kin);
|
||||
return err;
|
||||
}
|
||||
|
||||
/* 2. get an available ZSTD context */
|
||||
strm = z_erofs_isolate_strms(false);
|
||||
|
||||
/* 3. multi-call decompress */
|
||||
insz = rq->inputsize;
|
||||
outsz = rq->outputsize;
|
||||
stream = zstd_init_dstream(z_erofs_zstd_max_dictsize, strm->wksp, strm->wkspsz);
|
||||
if (!stream) {
|
||||
err = -EIO;
|
||||
goto failed_zinit;
|
||||
}
|
||||
|
||||
pofs = rq->pageofs_out;
|
||||
in_buf.size = min_t(u32, insz, PAGE_SIZE - rq->pageofs_in);
|
||||
insz -= in_buf.size;
|
||||
in_buf.src = kin + rq->pageofs_in;
|
||||
do {
|
||||
if (out_buf.size == out_buf.pos) {
|
||||
if (++no >= nrpages_out || !outsz) {
|
||||
erofs_err(sb, "insufficient space for decompressed data");
|
||||
err = -EFSCORRUPTED;
|
||||
break;
|
||||
}
|
||||
|
||||
if (kout)
|
||||
kunmap_local(kout);
|
||||
out_buf.size = min_t(u32, outsz, PAGE_SIZE - pofs);
|
||||
outsz -= out_buf.size;
|
||||
if (!rq->out[no]) {
|
||||
rq->out[no] = erofs_allocpage(pgpl, rq->gfp);
|
||||
if (!rq->out[no]) {
|
||||
kout = NULL;
|
||||
err = -ENOMEM;
|
||||
break;
|
||||
}
|
||||
set_page_private(rq->out[no],
|
||||
Z_EROFS_SHORTLIVED_PAGE);
|
||||
}
|
||||
kout = kmap_local_page(rq->out[no]);
|
||||
out_buf.dst = kout + pofs;
|
||||
out_buf.pos = 0;
|
||||
pofs = 0;
|
||||
}
|
||||
|
||||
if (in_buf.size == in_buf.pos && insz) {
|
||||
if (++ni >= nrpages_in) {
|
||||
erofs_err(sb, "invalid compressed data");
|
||||
err = -EFSCORRUPTED;
|
||||
break;
|
||||
}
|
||||
|
||||
if (kout) /* unlike kmap(), take care of the orders */
|
||||
kunmap_local(kout);
|
||||
kunmap_local(kin);
|
||||
in_buf.size = min_t(u32, insz, PAGE_SIZE);
|
||||
insz -= in_buf.size;
|
||||
kin = kmap_local_page(rq->in[ni]);
|
||||
in_buf.src = kin;
|
||||
in_buf.pos = 0;
|
||||
bounced = false;
|
||||
if (kout) {
|
||||
j = (u8 *)out_buf.dst - kout;
|
||||
kout = kmap_local_page(rq->out[no]);
|
||||
out_buf.dst = kout + j;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Handle overlapping: Use bounced buffer if the compressed
|
||||
* data is under processing; Or use short-lived pages from the
|
||||
* on-stack pagepool where pages share among the same request
|
||||
* and not _all_ inplace I/O pages are needed to be doubled.
|
||||
*/
|
||||
if (!bounced && rq->out[no] == rq->in[ni]) {
|
||||
memcpy(strm->bounce, in_buf.src, in_buf.size);
|
||||
in_buf.src = strm->bounce;
|
||||
bounced = true;
|
||||
}
|
||||
|
||||
for (j = ni + 1; j < nrpages_in; ++j) {
|
||||
struct page *tmppage;
|
||||
|
||||
if (rq->out[no] != rq->in[j])
|
||||
continue;
|
||||
tmppage = erofs_allocpage(pgpl, rq->gfp);
|
||||
if (!tmppage) {
|
||||
err = -ENOMEM;
|
||||
goto failed;
|
||||
}
|
||||
set_page_private(tmppage, Z_EROFS_SHORTLIVED_PAGE);
|
||||
copy_highpage(tmppage, rq->in[j]);
|
||||
rq->in[j] = tmppage;
|
||||
}
|
||||
zerr = zstd_decompress_stream(stream, &out_buf, &in_buf);
|
||||
if (zstd_is_error(zerr) || (!zerr && outsz)) {
|
||||
erofs_err(sb, "failed to decompress in[%u] out[%u]: %s",
|
||||
rq->inputsize, rq->outputsize,
|
||||
zerr ? zstd_get_error_name(zerr) : "unexpected end of stream");
|
||||
err = -EFSCORRUPTED;
|
||||
break;
|
||||
}
|
||||
} while (outsz || out_buf.pos < out_buf.size);
|
||||
failed:
|
||||
if (kout)
|
||||
kunmap_local(kout);
|
||||
failed_zinit:
|
||||
kunmap_local(kin);
|
||||
/* 4. push back ZSTD stream context to the global list */
|
||||
spin_lock(&z_erofs_zstd_lock);
|
||||
strm->next = z_erofs_zstd_head;
|
||||
z_erofs_zstd_head = strm;
|
||||
spin_unlock(&z_erofs_zstd_lock);
|
||||
wake_up(&z_erofs_zstd_wq);
|
||||
return err;
|
||||
}
|
@ -296,6 +296,7 @@ enum {
|
||||
Z_EROFS_COMPRESSION_LZ4 = 0,
|
||||
Z_EROFS_COMPRESSION_LZMA = 1,
|
||||
Z_EROFS_COMPRESSION_DEFLATE = 2,
|
||||
Z_EROFS_COMPRESSION_ZSTD = 3,
|
||||
Z_EROFS_COMPRESSION_MAX
|
||||
};
|
||||
#define Z_EROFS_ALL_COMPR_ALGS ((1 << Z_EROFS_COMPRESSION_MAX) - 1)
|
||||
@ -322,6 +323,15 @@ struct z_erofs_deflate_cfgs {
|
||||
u8 reserved[5];
|
||||
} __packed;
|
||||
|
||||
/* 6 bytes (+ length field = 8 bytes) */
|
||||
struct z_erofs_zstd_cfgs {
|
||||
u8 format;
|
||||
u8 windowlog; /* windowLog - ZSTD_WINDOWLOG_ABSOLUTEMIN(10) */
|
||||
u8 reserved[4];
|
||||
} __packed;
|
||||
|
||||
#define Z_EROFS_ZSTD_MAX_DICT_SIZE Z_EROFS_PCLUSTER_MAX_SIZE
|
||||
|
||||
/*
|
||||
* bit 0 : COMPACTED_2B indexes (0 - off; 1 - on)
|
||||
* e.g. for 4k logical cluster size, 4B if compacted 2B is off;
|
||||
@ -396,8 +406,7 @@ enum {
|
||||
Z_EROFS_LCLUSTER_TYPE_MAX
|
||||
};
|
||||
|
||||
#define Z_EROFS_LI_LCLUSTER_TYPE_BITS 2
|
||||
#define Z_EROFS_LI_LCLUSTER_TYPE_BIT 0
|
||||
#define Z_EROFS_LI_LCLUSTER_TYPE_MASK (Z_EROFS_LCLUSTER_TYPE_MAX - 1)
|
||||
|
||||
/* (noncompact only, HEAD) This pcluster refers to partial decompressed data */
|
||||
#define Z_EROFS_LI_PARTIAL_REF (1 << 15)
|
||||
@ -451,8 +460,6 @@ static inline void erofs_check_ondisk_layout_definitions(void)
|
||||
sizeof(struct z_erofs_lcluster_index));
|
||||
BUILD_BUG_ON(sizeof(struct erofs_deviceslot) != 128);
|
||||
|
||||
BUILD_BUG_ON(BIT(Z_EROFS_LI_LCLUSTER_TYPE_BITS) <
|
||||
Z_EROFS_LCLUSTER_TYPE_MAX - 1);
|
||||
/* exclude old compiler versions like gcc 7.5.0 */
|
||||
BUILD_BUG_ON(__builtin_constant_p(fmh) ?
|
||||
fmh != cpu_to_le64(1ULL << 63) : 0);
|
||||
|
@ -438,7 +438,11 @@ void erofs_unregister_sysfs(struct super_block *sb);
|
||||
int __init erofs_init_sysfs(void);
|
||||
void erofs_exit_sysfs(void);
|
||||
|
||||
struct page *erofs_allocpage(struct page **pagepool, gfp_t gfp);
|
||||
struct page *__erofs_allocpage(struct page **pagepool, gfp_t gfp, bool tryrsv);
|
||||
static inline struct page *erofs_allocpage(struct page **pagepool, gfp_t gfp)
|
||||
{
|
||||
return __erofs_allocpage(pagepool, gfp, false);
|
||||
}
|
||||
static inline void erofs_pagepool_add(struct page **pagepool, struct page *page)
|
||||
{
|
||||
set_page_private(page, (unsigned long)*pagepool);
|
||||
@ -463,11 +467,11 @@ int erofs_try_to_free_all_cached_folios(struct erofs_sb_info *sbi,
|
||||
struct erofs_workgroup *egrp);
|
||||
int z_erofs_map_blocks_iter(struct inode *inode, struct erofs_map_blocks *map,
|
||||
int flags);
|
||||
void *erofs_get_pcpubuf(unsigned int requiredpages);
|
||||
void erofs_put_pcpubuf(void *ptr);
|
||||
int erofs_pcpubuf_growsize(unsigned int nrpages);
|
||||
void __init erofs_pcpubuf_init(void);
|
||||
void erofs_pcpubuf_exit(void);
|
||||
void *z_erofs_get_gbuf(unsigned int requiredpages);
|
||||
void z_erofs_put_gbuf(void *ptr);
|
||||
int z_erofs_gbuf_growsize(unsigned int nrpages);
|
||||
int __init z_erofs_gbuf_init(void);
|
||||
void z_erofs_gbuf_exit(void);
|
||||
int erofs_init_managed_cache(struct super_block *sb);
|
||||
int z_erofs_parse_cfgs(struct super_block *sb, struct erofs_super_block *dsb);
|
||||
#else
|
||||
@ -477,8 +481,8 @@ static inline int erofs_init_shrinker(void) { return 0; }
|
||||
static inline void erofs_exit_shrinker(void) {}
|
||||
static inline int z_erofs_init_zip_subsystem(void) { return 0; }
|
||||
static inline void z_erofs_exit_zip_subsystem(void) {}
|
||||
static inline void erofs_pcpubuf_init(void) {}
|
||||
static inline void erofs_pcpubuf_exit(void) {}
|
||||
static inline int z_erofs_gbuf_init(void) { return 0; }
|
||||
static inline void z_erofs_gbuf_exit(void) {}
|
||||
static inline int erofs_init_managed_cache(struct super_block *sb) { return 0; }
|
||||
#endif /* !CONFIG_EROFS_FS_ZIP */
|
||||
|
||||
@ -498,6 +502,14 @@ static inline int z_erofs_deflate_init(void) { return 0; }
|
||||
static inline int z_erofs_deflate_exit(void) { return 0; }
|
||||
#endif /* !CONFIG_EROFS_FS_ZIP_DEFLATE */
|
||||
|
||||
#ifdef CONFIG_EROFS_FS_ZIP_ZSTD
|
||||
int __init z_erofs_zstd_init(void);
|
||||
void z_erofs_zstd_exit(void);
|
||||
#else
|
||||
static inline int z_erofs_zstd_init(void) { return 0; }
|
||||
static inline int z_erofs_zstd_exit(void) { return 0; }
|
||||
#endif /* !CONFIG_EROFS_FS_ZIP_ZSTD */
|
||||
|
||||
#ifdef CONFIG_EROFS_FS_ONDEMAND
|
||||
int erofs_fscache_register_fs(struct super_block *sb);
|
||||
void erofs_fscache_unregister_fs(struct super_block *sb);
|
||||
|
@ -1,148 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-only
|
||||
/*
|
||||
* Copyright (C) Gao Xiang <xiang@kernel.org>
|
||||
*
|
||||
* For low-latency decompression algorithms (e.g. lz4), reserve consecutive
|
||||
* per-CPU virtual memory (in pages) in advance to store such inplace I/O
|
||||
* data if inplace decompression is failed (due to unmet inplace margin for
|
||||
* example).
|
||||
*/
|
||||
#include "internal.h"
|
||||
|
||||
struct erofs_pcpubuf {
|
||||
raw_spinlock_t lock;
|
||||
void *ptr;
|
||||
struct page **pages;
|
||||
unsigned int nrpages;
|
||||
};
|
||||
|
||||
static DEFINE_PER_CPU(struct erofs_pcpubuf, erofs_pcb);
|
||||
|
||||
void *erofs_get_pcpubuf(unsigned int requiredpages)
|
||||
__acquires(pcb->lock)
|
||||
{
|
||||
struct erofs_pcpubuf *pcb = &get_cpu_var(erofs_pcb);
|
||||
|
||||
raw_spin_lock(&pcb->lock);
|
||||
/* check if the per-CPU buffer is too small */
|
||||
if (requiredpages > pcb->nrpages) {
|
||||
raw_spin_unlock(&pcb->lock);
|
||||
put_cpu_var(erofs_pcb);
|
||||
/* (for sparse checker) pretend pcb->lock is still taken */
|
||||
__acquire(pcb->lock);
|
||||
return NULL;
|
||||
}
|
||||
return pcb->ptr;
|
||||
}
|
||||
|
||||
void erofs_put_pcpubuf(void *ptr) __releases(pcb->lock)
|
||||
{
|
||||
struct erofs_pcpubuf *pcb = &per_cpu(erofs_pcb, smp_processor_id());
|
||||
|
||||
DBG_BUGON(pcb->ptr != ptr);
|
||||
raw_spin_unlock(&pcb->lock);
|
||||
put_cpu_var(erofs_pcb);
|
||||
}
|
||||
|
||||
/* the next step: support per-CPU page buffers hotplug */
|
||||
int erofs_pcpubuf_growsize(unsigned int nrpages)
|
||||
{
|
||||
static DEFINE_MUTEX(pcb_resize_mutex);
|
||||
static unsigned int pcb_nrpages;
|
||||
struct page *pagepool = NULL;
|
||||
int delta, cpu, ret, i;
|
||||
|
||||
mutex_lock(&pcb_resize_mutex);
|
||||
delta = nrpages - pcb_nrpages;
|
||||
ret = 0;
|
||||
/* avoid shrinking pcpubuf, since no idea how many fses rely on */
|
||||
if (delta <= 0)
|
||||
goto out;
|
||||
|
||||
for_each_possible_cpu(cpu) {
|
||||
struct erofs_pcpubuf *pcb = &per_cpu(erofs_pcb, cpu);
|
||||
struct page **pages, **oldpages;
|
||||
void *ptr, *old_ptr;
|
||||
|
||||
pages = kmalloc_array(nrpages, sizeof(*pages), GFP_KERNEL);
|
||||
if (!pages) {
|
||||
ret = -ENOMEM;
|
||||
break;
|
||||
}
|
||||
|
||||
for (i = 0; i < nrpages; ++i) {
|
||||
pages[i] = erofs_allocpage(&pagepool, GFP_KERNEL);
|
||||
if (!pages[i]) {
|
||||
ret = -ENOMEM;
|
||||
oldpages = pages;
|
||||
goto free_pagearray;
|
||||
}
|
||||
}
|
||||
ptr = vmap(pages, nrpages, VM_MAP, PAGE_KERNEL);
|
||||
if (!ptr) {
|
||||
ret = -ENOMEM;
|
||||
oldpages = pages;
|
||||
goto free_pagearray;
|
||||
}
|
||||
raw_spin_lock(&pcb->lock);
|
||||
old_ptr = pcb->ptr;
|
||||
pcb->ptr = ptr;
|
||||
oldpages = pcb->pages;
|
||||
pcb->pages = pages;
|
||||
i = pcb->nrpages;
|
||||
pcb->nrpages = nrpages;
|
||||
raw_spin_unlock(&pcb->lock);
|
||||
|
||||
if (!oldpages) {
|
||||
DBG_BUGON(old_ptr);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (old_ptr)
|
||||
vunmap(old_ptr);
|
||||
free_pagearray:
|
||||
while (i)
|
||||
erofs_pagepool_add(&pagepool, oldpages[--i]);
|
||||
kfree(oldpages);
|
||||
if (ret)
|
||||
break;
|
||||
}
|
||||
pcb_nrpages = nrpages;
|
||||
erofs_release_pages(&pagepool);
|
||||
out:
|
||||
mutex_unlock(&pcb_resize_mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
void __init erofs_pcpubuf_init(void)
|
||||
{
|
||||
int cpu;
|
||||
|
||||
for_each_possible_cpu(cpu) {
|
||||
struct erofs_pcpubuf *pcb = &per_cpu(erofs_pcb, cpu);
|
||||
|
||||
raw_spin_lock_init(&pcb->lock);
|
||||
}
|
||||
}
|
||||
|
||||
void erofs_pcpubuf_exit(void)
|
||||
{
|
||||
int cpu, i;
|
||||
|
||||
for_each_possible_cpu(cpu) {
|
||||
struct erofs_pcpubuf *pcb = &per_cpu(erofs_pcb, cpu);
|
||||
|
||||
if (pcb->ptr) {
|
||||
vunmap(pcb->ptr);
|
||||
pcb->ptr = NULL;
|
||||
}
|
||||
if (!pcb->pages)
|
||||
continue;
|
||||
|
||||
for (i = 0; i < pcb->nrpages; ++i)
|
||||
if (pcb->pages[i])
|
||||
put_page(pcb->pages[i]);
|
||||
kfree(pcb->pages);
|
||||
pcb->pages = NULL;
|
||||
}
|
||||
}
|
@ -859,7 +859,14 @@ static int __init erofs_module_init(void)
|
||||
if (err)
|
||||
goto deflate_err;
|
||||
|
||||
erofs_pcpubuf_init();
|
||||
err = z_erofs_zstd_init();
|
||||
if (err)
|
||||
goto zstd_err;
|
||||
|
||||
err = z_erofs_gbuf_init();
|
||||
if (err)
|
||||
goto gbuf_err;
|
||||
|
||||
err = z_erofs_init_zip_subsystem();
|
||||
if (err)
|
||||
goto zip_err;
|
||||
@ -879,6 +886,10 @@ static int __init erofs_module_init(void)
|
||||
sysfs_err:
|
||||
z_erofs_exit_zip_subsystem();
|
||||
zip_err:
|
||||
z_erofs_gbuf_exit();
|
||||
gbuf_err:
|
||||
z_erofs_zstd_exit();
|
||||
zstd_err:
|
||||
z_erofs_deflate_exit();
|
||||
deflate_err:
|
||||
z_erofs_lzma_exit();
|
||||
@ -898,33 +909,32 @@ static void __exit erofs_module_exit(void)
|
||||
|
||||
erofs_exit_sysfs();
|
||||
z_erofs_exit_zip_subsystem();
|
||||
z_erofs_zstd_exit();
|
||||
z_erofs_deflate_exit();
|
||||
z_erofs_lzma_exit();
|
||||
erofs_exit_shrinker();
|
||||
kmem_cache_destroy(erofs_inode_cachep);
|
||||
erofs_pcpubuf_exit();
|
||||
z_erofs_gbuf_exit();
|
||||
}
|
||||
|
||||
static int erofs_statfs(struct dentry *dentry, struct kstatfs *buf)
|
||||
{
|
||||
struct super_block *sb = dentry->d_sb;
|
||||
struct erofs_sb_info *sbi = EROFS_SB(sb);
|
||||
u64 id = 0;
|
||||
|
||||
if (!erofs_is_fscache_mode(sb))
|
||||
id = huge_encode_dev(sb->s_bdev->bd_dev);
|
||||
|
||||
buf->f_type = sb->s_magic;
|
||||
buf->f_bsize = sb->s_blocksize;
|
||||
buf->f_blocks = sbi->total_blocks;
|
||||
buf->f_bfree = buf->f_bavail = 0;
|
||||
|
||||
buf->f_files = ULLONG_MAX;
|
||||
buf->f_ffree = ULLONG_MAX - sbi->inos;
|
||||
|
||||
buf->f_namelen = EROFS_NAME_LEN;
|
||||
|
||||
buf->f_fsid = u64_to_fsid(id);
|
||||
if (uuid_is_null(&sb->s_uuid))
|
||||
buf->f_fsid = u64_to_fsid(erofs_is_fscache_mode(sb) ? 0 :
|
||||
huge_encode_dev(sb->s_bdev->bd_dev));
|
||||
else
|
||||
buf->f_fsid = uuid_to_fsid(sb->s_uuid.b);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -31,7 +31,7 @@ static int z_erofs_load_full_lcluster(struct z_erofs_maprecorder *m,
|
||||
vi->inode_isize + vi->xattr_isize) +
|
||||
lcn * sizeof(struct z_erofs_lcluster_index);
|
||||
struct z_erofs_lcluster_index *di;
|
||||
unsigned int advise, type;
|
||||
unsigned int advise;
|
||||
|
||||
m->kaddr = erofs_read_metabuf(&m->map->buf, inode->i_sb,
|
||||
erofs_blknr(inode->i_sb, pos), EROFS_KMAP);
|
||||
@ -43,10 +43,8 @@ static int z_erofs_load_full_lcluster(struct z_erofs_maprecorder *m,
|
||||
di = m->kaddr + erofs_blkoff(inode->i_sb, pos);
|
||||
|
||||
advise = le16_to_cpu(di->di_advise);
|
||||
type = (advise >> Z_EROFS_LI_LCLUSTER_TYPE_BIT) &
|
||||
((1 << Z_EROFS_LI_LCLUSTER_TYPE_BITS) - 1);
|
||||
switch (type) {
|
||||
case Z_EROFS_LCLUSTER_TYPE_NONHEAD:
|
||||
m->type = advise & Z_EROFS_LI_LCLUSTER_TYPE_MASK;
|
||||
if (m->type == Z_EROFS_LCLUSTER_TYPE_NONHEAD) {
|
||||
m->clusterofs = 1 << vi->z_logical_clusterbits;
|
||||
m->delta[0] = le16_to_cpu(di->di_u.delta[0]);
|
||||
if (m->delta[0] & Z_EROFS_LI_D0_CBLKCNT) {
|
||||
@ -60,24 +58,15 @@ static int z_erofs_load_full_lcluster(struct z_erofs_maprecorder *m,
|
||||
m->delta[0] = 1;
|
||||
}
|
||||
m->delta[1] = le16_to_cpu(di->di_u.delta[1]);
|
||||
break;
|
||||
case Z_EROFS_LCLUSTER_TYPE_PLAIN:
|
||||
case Z_EROFS_LCLUSTER_TYPE_HEAD1:
|
||||
case Z_EROFS_LCLUSTER_TYPE_HEAD2:
|
||||
if (advise & Z_EROFS_LI_PARTIAL_REF)
|
||||
m->partialref = true;
|
||||
} else {
|
||||
m->partialref = !!(advise & Z_EROFS_LI_PARTIAL_REF);
|
||||
m->clusterofs = le16_to_cpu(di->di_clusterofs);
|
||||
if (m->clusterofs >= 1 << vi->z_logical_clusterbits) {
|
||||
DBG_BUGON(1);
|
||||
return -EFSCORRUPTED;
|
||||
}
|
||||
m->pblk = le32_to_cpu(di->di_u.blkaddr);
|
||||
break;
|
||||
default:
|
||||
DBG_BUGON(1);
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
m->type = type;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -561,7 +550,8 @@ static int z_erofs_do_map_blocks(struct inode *inode,
|
||||
if ((flags & EROFS_GET_BLOCKS_FIEMAP) ||
|
||||
((flags & EROFS_GET_BLOCKS_READMORE) &&
|
||||
(map->m_algorithmformat == Z_EROFS_COMPRESSION_LZMA ||
|
||||
map->m_algorithmformat == Z_EROFS_COMPRESSION_DEFLATE) &&
|
||||
map->m_algorithmformat == Z_EROFS_COMPRESSION_DEFLATE ||
|
||||
map->m_algorithmformat == Z_EROFS_COMPRESSION_ZSTD) &&
|
||||
map->m_llen >= i_blocksize(inode))) {
|
||||
err = z_erofs_get_extent_decompressedlen(&m);
|
||||
if (!err)
|
||||
|
@ -5,16 +5,186 @@
|
||||
*/
|
||||
#include "internal.h"
|
||||
|
||||
struct page *erofs_allocpage(struct page **pagepool, gfp_t gfp)
|
||||
struct z_erofs_gbuf {
|
||||
spinlock_t lock;
|
||||
void *ptr;
|
||||
struct page **pages;
|
||||
unsigned int nrpages;
|
||||
};
|
||||
|
||||
static struct z_erofs_gbuf *z_erofs_gbufpool, *z_erofs_rsvbuf;
|
||||
static unsigned int z_erofs_gbuf_count, z_erofs_gbuf_nrpages,
|
||||
z_erofs_rsv_nrpages;
|
||||
|
||||
module_param_named(global_buffers, z_erofs_gbuf_count, uint, 0444);
|
||||
module_param_named(reserved_pages, z_erofs_rsv_nrpages, uint, 0444);
|
||||
|
||||
static atomic_long_t erofs_global_shrink_cnt; /* for all mounted instances */
|
||||
/* protected by 'erofs_sb_list_lock' */
|
||||
static unsigned int shrinker_run_no;
|
||||
|
||||
/* protects the mounted 'erofs_sb_list' */
|
||||
static DEFINE_SPINLOCK(erofs_sb_list_lock);
|
||||
static LIST_HEAD(erofs_sb_list);
|
||||
static struct shrinker *erofs_shrinker_info;
|
||||
|
||||
static unsigned int z_erofs_gbuf_id(void)
|
||||
{
|
||||
return raw_smp_processor_id() % z_erofs_gbuf_count;
|
||||
}
|
||||
|
||||
void *z_erofs_get_gbuf(unsigned int requiredpages)
|
||||
__acquires(gbuf->lock)
|
||||
{
|
||||
struct z_erofs_gbuf *gbuf;
|
||||
|
||||
gbuf = &z_erofs_gbufpool[z_erofs_gbuf_id()];
|
||||
spin_lock(&gbuf->lock);
|
||||
/* check if the buffer is too small */
|
||||
if (requiredpages > gbuf->nrpages) {
|
||||
spin_unlock(&gbuf->lock);
|
||||
/* (for sparse checker) pretend gbuf->lock is still taken */
|
||||
__acquire(gbuf->lock);
|
||||
return NULL;
|
||||
}
|
||||
return gbuf->ptr;
|
||||
}
|
||||
|
||||
void z_erofs_put_gbuf(void *ptr) __releases(gbuf->lock)
|
||||
{
|
||||
struct z_erofs_gbuf *gbuf;
|
||||
|
||||
gbuf = &z_erofs_gbufpool[z_erofs_gbuf_id()];
|
||||
DBG_BUGON(gbuf->ptr != ptr);
|
||||
spin_unlock(&gbuf->lock);
|
||||
}
|
||||
|
||||
int z_erofs_gbuf_growsize(unsigned int nrpages)
|
||||
{
|
||||
static DEFINE_MUTEX(gbuf_resize_mutex);
|
||||
struct page **tmp_pages = NULL;
|
||||
struct z_erofs_gbuf *gbuf;
|
||||
void *ptr, *old_ptr;
|
||||
int last, i, j;
|
||||
|
||||
mutex_lock(&gbuf_resize_mutex);
|
||||
/* avoid shrinking gbufs, since no idea how many fses rely on */
|
||||
if (nrpages <= z_erofs_gbuf_nrpages) {
|
||||
mutex_unlock(&gbuf_resize_mutex);
|
||||
return 0;
|
||||
}
|
||||
|
||||
for (i = 0; i < z_erofs_gbuf_count; ++i) {
|
||||
gbuf = &z_erofs_gbufpool[i];
|
||||
tmp_pages = kcalloc(nrpages, sizeof(*tmp_pages), GFP_KERNEL);
|
||||
if (!tmp_pages)
|
||||
goto out;
|
||||
|
||||
for (j = 0; j < gbuf->nrpages; ++j)
|
||||
tmp_pages[j] = gbuf->pages[j];
|
||||
do {
|
||||
last = j;
|
||||
j = alloc_pages_bulk_array(GFP_KERNEL, nrpages,
|
||||
tmp_pages);
|
||||
if (last == j)
|
||||
goto out;
|
||||
} while (j != nrpages);
|
||||
|
||||
ptr = vmap(tmp_pages, nrpages, VM_MAP, PAGE_KERNEL);
|
||||
if (!ptr)
|
||||
goto out;
|
||||
|
||||
spin_lock(&gbuf->lock);
|
||||
kfree(gbuf->pages);
|
||||
gbuf->pages = tmp_pages;
|
||||
old_ptr = gbuf->ptr;
|
||||
gbuf->ptr = ptr;
|
||||
gbuf->nrpages = nrpages;
|
||||
spin_unlock(&gbuf->lock);
|
||||
if (old_ptr)
|
||||
vunmap(old_ptr);
|
||||
}
|
||||
z_erofs_gbuf_nrpages = nrpages;
|
||||
out:
|
||||
if (i < z_erofs_gbuf_count && tmp_pages) {
|
||||
for (j = 0; j < nrpages; ++j)
|
||||
if (tmp_pages[j] && tmp_pages[j] != gbuf->pages[j])
|
||||
__free_page(tmp_pages[j]);
|
||||
kfree(tmp_pages);
|
||||
}
|
||||
mutex_unlock(&gbuf_resize_mutex);
|
||||
return i < z_erofs_gbuf_count ? -ENOMEM : 0;
|
||||
}
|
||||
|
||||
int __init z_erofs_gbuf_init(void)
|
||||
{
|
||||
unsigned int i, total = num_possible_cpus();
|
||||
|
||||
if (z_erofs_gbuf_count)
|
||||
total = min(z_erofs_gbuf_count, total);
|
||||
z_erofs_gbuf_count = total;
|
||||
|
||||
/* The last (special) global buffer is the reserved buffer */
|
||||
total += !!z_erofs_rsv_nrpages;
|
||||
|
||||
z_erofs_gbufpool = kcalloc(total, sizeof(*z_erofs_gbufpool),
|
||||
GFP_KERNEL);
|
||||
if (!z_erofs_gbufpool)
|
||||
return -ENOMEM;
|
||||
|
||||
if (z_erofs_rsv_nrpages) {
|
||||
z_erofs_rsvbuf = &z_erofs_gbufpool[total - 1];
|
||||
z_erofs_rsvbuf->pages = kcalloc(z_erofs_rsv_nrpages,
|
||||
sizeof(*z_erofs_rsvbuf->pages), GFP_KERNEL);
|
||||
if (!z_erofs_rsvbuf->pages) {
|
||||
z_erofs_rsvbuf = NULL;
|
||||
z_erofs_rsv_nrpages = 0;
|
||||
}
|
||||
}
|
||||
for (i = 0; i < total; ++i)
|
||||
spin_lock_init(&z_erofs_gbufpool[i].lock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void z_erofs_gbuf_exit(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < z_erofs_gbuf_count + (!!z_erofs_rsvbuf); ++i) {
|
||||
struct z_erofs_gbuf *gbuf = &z_erofs_gbufpool[i];
|
||||
|
||||
if (gbuf->ptr) {
|
||||
vunmap(gbuf->ptr);
|
||||
gbuf->ptr = NULL;
|
||||
}
|
||||
|
||||
if (!gbuf->pages)
|
||||
continue;
|
||||
|
||||
for (i = 0; i < gbuf->nrpages; ++i)
|
||||
if (gbuf->pages[i])
|
||||
put_page(gbuf->pages[i]);
|
||||
kfree(gbuf->pages);
|
||||
gbuf->pages = NULL;
|
||||
}
|
||||
kfree(z_erofs_gbufpool);
|
||||
}
|
||||
|
||||
struct page *__erofs_allocpage(struct page **pagepool, gfp_t gfp, bool tryrsv)
|
||||
{
|
||||
struct page *page = *pagepool;
|
||||
|
||||
if (page) {
|
||||
DBG_BUGON(page_ref_count(page) != 1);
|
||||
*pagepool = (struct page *)page_private(page);
|
||||
} else {
|
||||
page = alloc_page(gfp);
|
||||
} else if (tryrsv && z_erofs_rsvbuf && z_erofs_rsvbuf->nrpages) {
|
||||
spin_lock(&z_erofs_rsvbuf->lock);
|
||||
if (z_erofs_rsvbuf->nrpages)
|
||||
page = z_erofs_rsvbuf->pages[--z_erofs_rsvbuf->nrpages];
|
||||
spin_unlock(&z_erofs_rsvbuf->lock);
|
||||
}
|
||||
if (!page)
|
||||
page = alloc_page(gfp);
|
||||
DBG_BUGON(page && page_ref_count(page) != 1);
|
||||
return page;
|
||||
}
|
||||
|
||||
@ -24,14 +194,22 @@ void erofs_release_pages(struct page **pagepool)
|
||||
struct page *page = *pagepool;
|
||||
|
||||
*pagepool = (struct page *)page_private(page);
|
||||
/* try to fill reserved global pool first */
|
||||
if (z_erofs_rsvbuf && z_erofs_rsvbuf->nrpages <
|
||||
z_erofs_rsv_nrpages) {
|
||||
spin_lock(&z_erofs_rsvbuf->lock);
|
||||
if (z_erofs_rsvbuf->nrpages < z_erofs_rsv_nrpages) {
|
||||
z_erofs_rsvbuf->pages[z_erofs_rsvbuf->nrpages++]
|
||||
= page;
|
||||
spin_unlock(&z_erofs_rsvbuf->lock);
|
||||
continue;
|
||||
}
|
||||
spin_unlock(&z_erofs_rsvbuf->lock);
|
||||
}
|
||||
put_page(page);
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef CONFIG_EROFS_FS_ZIP
|
||||
/* global shrink count (for all mounted EROFS instances) */
|
||||
static atomic_long_t erofs_global_shrink_cnt;
|
||||
|
||||
static bool erofs_workgroup_get(struct erofs_workgroup *grp)
|
||||
{
|
||||
if (lockref_get_not_zero(&grp->lockref))
|
||||
@ -171,13 +349,6 @@ static unsigned long erofs_shrink_workstation(struct erofs_sb_info *sbi,
|
||||
return freed;
|
||||
}
|
||||
|
||||
/* protected by 'erofs_sb_list_lock' */
|
||||
static unsigned int shrinker_run_no;
|
||||
|
||||
/* protects the mounted 'erofs_sb_list' */
|
||||
static DEFINE_SPINLOCK(erofs_sb_list_lock);
|
||||
static LIST_HEAD(erofs_sb_list);
|
||||
|
||||
void erofs_shrinker_register(struct super_block *sb)
|
||||
{
|
||||
struct erofs_sb_info *sbi = EROFS_SB(sb);
|
||||
@ -264,8 +435,6 @@ static unsigned long erofs_shrink_scan(struct shrinker *shrink,
|
||||
return freed;
|
||||
}
|
||||
|
||||
static struct shrinker *erofs_shrinker_info;
|
||||
|
||||
int __init erofs_init_shrinker(void)
|
||||
{
|
||||
erofs_shrinker_info = shrinker_alloc(0, "erofs-shrinker");
|
||||
@ -274,9 +443,7 @@ int __init erofs_init_shrinker(void)
|
||||
|
||||
erofs_shrinker_info->count_objects = erofs_shrink_count;
|
||||
erofs_shrinker_info->scan_objects = erofs_shrink_scan;
|
||||
|
||||
shrinker_register(erofs_shrinker_info);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -284,4 +451,3 @@ void erofs_exit_shrinker(void)
|
||||
{
|
||||
shrinker_free(erofs_shrinker_info);
|
||||
}
|
||||
#endif /* !CONFIG_EROFS_FS_ZIP */
|
Loading…
Reference in New Issue
Block a user