<p>Add a missing change:
<br>- change to generate a ctx for duplicate fragment in compression.
<br>
<br>On Thu, 1 Dec 2022 19:16:15 +0800
<br>Yue Hu <zbestahu@gmail.com> wrote:
<br>
<br>> From: Yue Hu <huyue2@coolpad.com>
<br>>
<br>> Previously, there's no fragment deduplication when this feature is
<br>> introduced. Let's support it now.
<br>>
<br>> We intend to dedupe the fragments before compression, so that duplicate
<br>> parts will not be written into packed inode.
<br>>
<br>> With this patch, for Linux 5.10.1 + 5.10.87 source code:
<br>>
<br>> [before]
<br>> 32k pcluster + T0 + lz4hc,12 + fragment 450M
<br>> 64k pcluster + T0 + lz4hc,12 + fragment 434M
<br>> 128k pcluster + T0 + lz4hc,12 + fragment 426M
<br>> 32k pcluster + T0 + lz4hc,12 + fragment + dedupe 368M
<br>> 64k pcluster + T0 + lz4hc,12 + fragment + dedupe 380M
<br>> 128k pcluster + T0 + lz4hc,12 + fragment + dedupe 395M
<br>>
<br>> [after]
<br>> 32k pcluster + T0 + lz4hc,12 + fragment 311M
<br>> 64k pcluster + T0 + lz4hc,12 + fragment 295M
<br>> 128k pcluster + T0 + lz4hc,12 + fragment 287M
<br>> 32k pcluster + T0 + lz4hc,12 + fragment + dedupe 286M
<br>> 64k pcluster + T0 + lz4hc,12 + fragment + dedupe 281M
<br>> 128k pcluster + T0 + lz4hc,12 + fragment + dedupe 278M
<br>>
<br>> Tested on SquashFS (which uses level 12 by default for lz4hc):
<br>>
<br>> 32k block + lz4hc 332M
<br>> 64k block + lz4hc 304M
<br>> 128k block + lz4hc 283M
<br>> 256k block + lz4hc 273M
<br>> 256k block + lz4hc + noI 278M
<br>>
<br>> Suggested-by: Gao Xiang <hsiangkao@linux.alibaba.com>
<br>> Signed-off-by: Yue Hu <huyue2@coolpad.com>
<br>> ---
<br>> v4:
<br>> - renaming include tofcrc/new_fragmentsize
<br>> - move fixup into ctx
<br>> - use may_fixing to check packing fragment or not
<br>> - move sb/inode flag + 64bits case from erofs_pack_fragments() to new
<br>> helper erofs_fragments_commit()
<br>> - move recompress ahead of may_inline case when compressing succeeds
<br>> - update commit message/code comments
<br>> - note that decompress will fail when enable ztailpacking at the same
<br>> time, need some time to debug
<br>>
<br>> v3:
<br>> - modify acrroding to Xiang's comments in v2
<br>> - simplify the logic in vle_compress_one
<br>> - fix the crash for 1MB pcluster
<br>>
<br>> v2: mainly improve the logic in compression
<br>>
<br>> include/erofs/fragments.h | 4 +-
<br>> lib/compress.c | 134 +++++++++++++++++++++-----
<br>> lib/fragments.c | 197 ++++++++++++++++++++++++++++++++++++--
<br>> 3 files changed, 305 insertions(+), 30 deletions(-)
<br>>
<br>> diff --git a/include/erofs/fragments.h b/include/erofs/fragments.h
<br>> index 5444384..e91fb31 100644
<br>> --- a/include/erofs/fragments.h
<br>> +++ b/include/erofs/fragments.h
<br>> @@ -15,8 +15,10 @@ extern "C"
<br>> extern const char *frags_packedname;
<br>> #define EROFS_PACKED_INODE frags_packedname
<br>>
<br>> +int z_erofs_fragments_dedupe(struct erofs_inode *inode, int fd, u32 *tofcrc_r);
<br>> int z_erofs_pack_fragments(struct erofs_inode *inode, void *data,
<br>> - unsigned int len);
<br>> + unsigned int len, u32 tofcrc);
<br>> +void z_erofs_fragments_commit(struct erofs_inode *inode);
<br>> struct erofs_inode *erofs_mkfs_build_fragments(void);
<br>> int erofs_fragments_init(void);
<br>> void erofs_fragments_exit(void);
<br>> diff --git a/lib/compress.c b/lib/compress.c
<br>> index 17b3213..afd6e8e 100644
<br>> --- a/lib/compress.c
<br>> +++ b/lib/compress.c
<br>> @@ -33,6 +33,11 @@ struct z_erofs_vle_compress_ctx {
<br>> unsigned int head, tail;
<br>> erofs_blk_t blkaddr; /* pointing to the next blkaddr */
<br>> u16 clusterofs;
<br>> +
<br>> + u32 tofcrc;
<br>> + unsigned int pclustersize;
<br>> + erofs_off_t remaining;
<br>> + bool need_fix;
<br>> };
<br>>
<br>> #define Z_EROFS_LEGACY_MAP_HEADER_SIZE \
<br>> @@ -162,10 +167,10 @@ static void z_erofs_write_indexes(struct z_erofs_vle_compress_ctx *ctx)
<br>> ctx->clusterofs = clusterofs + count;
<br>> }
<br>>
<br>> -static int z_erofs_compress_dedupe(struct erofs_inode *inode,
<br>> - struct z_erofs_vle_compress_ctx *ctx,
<br>> +static int z_erofs_compress_dedupe(struct z_erofs_vle_compress_ctx *ctx,
<br>> unsigned int *len)
<br>> {
<br>> + struct erofs_inode *inode = ctx->inode;
<br>> int ret = 0;
<br>>
<br>> do {
<br>> @@ -301,10 +306,6 @@ static void tryrecompress_trailing(void *in, unsigned int *insize,
<br>> unsigned int count;
<br>> int ret = *compressedsize;
<br>>
<br>> - /* no need to recompress */
<br>> - if (!(ret & (EROFS_BLKSIZ - 1)))
<br>> - return;
<br>> -
<br>> count = *insize;
<br>> ret = erofs_compress_destsize(&compresshandle,
<br>> in, &count, (void *)tmp,
<br>> @@ -319,30 +320,69 @@ static void tryrecompress_trailing(void *in, unsigned int *insize,
<br>> *compressedsize = ret;
<br>> }
<br>>
<br>> -static int vle_compress_one(struct z_erofs_vle_compress_ctx *ctx,
<br>> - bool final)
<br>> +static void z_erofs_fragments_dedupe_update(struct z_erofs_vle_compress_ctx *ctx,
<br>> + unsigned int *len)
<br>> +{
<br>> + struct erofs_inode *inode = ctx->inode;
<br>> + const unsigned int new_fragmentsize = ctx->remaining + *len;
<br>> +
<br>> + DBG_BUGON(!inode->fragment_size);
<br>> +
<br>> + /* try to fix it again if it gets larger (should be rare) */
<br>> + if (inode->fragment_size < new_fragmentsize) {
<br>> + ctx->pclustersize =
<br>> + roundup(new_fragmentsize - inode->fragment_size,
<br>> + EROFS_BLKSIZ);
<br>> + return;
<br>> + }
<br>> +
<br>> + inode->fragmentoff += inode->fragment_size - new_fragmentsize;
<br>> + inode->fragment_size = new_fragmentsize;
<br>> +
<br>> + erofs_dbg("Reducing fragment size to %u at %lu",
<br>> + inode->fragment_size, inode->fragmentoff);
<br>> +
<br>> + /* it's ending */
<br>> + ctx->head += new_fragmentsize;
<br>> + ctx->remaining = 0;
<br>> + *len = 0;
<br>> +}
<br>> +
<br>> +static int vle_compress_one(struct z_erofs_vle_compress_ctx *ctx)
<br>> {
<br>> static char dstbuf[EROFS_CONFIG_COMPR_MAX_SZ + EROFS_BLKSIZ];
<br>> struct erofs_inode *inode = ctx->inode;
<br>> char *const dst = dstbuf + EROFS_BLKSIZ;
<br>> struct erofs_compress *const h = &compresshandle;
<br>> unsigned int len = ctx->tail - ctx->head;
<br>> + bool final = !ctx->remaining;
<br>> int ret;
<br>>
<br>> while (len) {
<br>> - unsigned int pclustersize =
<br>> - z_erofs_get_max_pclusterblks(inode) * EROFS_BLKSIZ;
<br>> bool may_inline = (cfg.c_ztailpacking && final);
<br>> bool may_packing = (cfg.c_fragments && final &&
<br>> !erofs_is_packed_inode(inode));
<br>> + bool may_fixing = ctx->need_fix;
<br>>
<br>> - if (z_erofs_compress_dedupe(inode, ctx, &len) && !final)
<br>> + if (z_erofs_compress_dedupe(ctx, &len) && !final)
<br>> break;
<br>>
<br>> - if (len <= pclustersize) {
<br>> + if (len <= ctx->pclustersize) {
<br>> + if (may_fixing && !len) {
<br>> + may_packing = false;
<br>> + goto frag_nopacking;
<br>> + }
<br>> if (!final || !len)
<br>> break;
<br>> if (may_packing) {
<br>> + if (inode->fragment_size && !may_fixing) {
<br>> + ctx->remaining = inode->fragment_size;
<br>> + ctx->pclustersize =
<br>> + roundup(len, EROFS_BLKSIZ);
<br>> + ctx->e.length = 0;
<br>> + ctx->need_fix = true;
<br>> + break;
<br>> + }
<br>> ctx->e.length = len;
<br>> goto frag_packing;
<br>> }
<br>> @@ -353,7 +393,7 @@ static int vle_compress_one(struct z_erofs_vle_compress_ctx *ctx,
<br>> ctx->e.length = min(len,
<br>> cfg.c_max_decompressed_extent_bytes);
<br>> ret = erofs_compress_destsize(h, ctx->queue + ctx->head,
<br>> - &ctx->e.length, dst, pclustersize,
<br>> + &ctx->e.length, dst, ctx->pclustersize,
<br>> !(final && len == ctx->e.length));
<br>> if (ret <= 0) {
<br>> if (ret != -EAGAIN) {
<br>> @@ -385,15 +425,17 @@ nocompression:
<br>> ctx->e.compressedblks = 1;
<br>> ctx->e.raw = true;
<br>> } else if (may_packing && len == ctx->e.length &&
<br>> - ret < pclustersize) {
<br>> + ret < ctx->pclustersize && (!inode->fragment_size ||
<br>> + may_fixing)) {
<br>> frag_packing:
<br>> ret = z_erofs_pack_fragments(inode,
<br>> ctx->queue + ctx->head,
<br>> - len);
<br>> + len, ctx->tofcrc);
<br>> if (ret < 0)
<br>> return ret;
<br>> ctx->e.compressedblks = 0; /* indicate a fragment */
<br>> ctx->e.raw = true;
<br>> + may_fixing = false;
<br>> /* tailpcluster should be less than 1 block */
<br>> } else if (may_inline && len == ctx->e.length &&
<br>> ret < EROFS_BLKSIZ) {
<br>> @@ -415,7 +457,27 @@ frag_packing:
<br>> } else {
<br>> unsigned int tailused, padding;
<br>>
<br>> - if (may_inline && len == ctx->e.length)
<br>> + tailused = ret & (EROFS_BLKSIZ - 1);
<br>> + /*
<br>> + * If there's a space left for the last round when
<br>> + * deduping fragments, try to read fragment and
<br>> + * recompress a litte more to check whether it can be
<br>> + * filled up. Fix the fragment if succeeds. Otherwise,
<br>> + * just drop it and go to packing.
<br>> + */
<br>> + if (may_packing && len == ctx->e.length && tailused &&
<br>> + ctx->tail < sizeof(ctx->queue)) {
<br>> + DBG_BUGON(!inode->fragment_size);
<br>> +
<br>> + ctx->remaining = inode->fragment_size;
<br>> + ctx->pclustersize =
<br>> + BLK_ROUND_UP(ret) * EROFS_BLKSIZ;
<br>> + ctx->e.length = 0;
<br>> + ctx->need_fix = true;
<br>> + break;
<br>> + }
<br>> +
<br>> + if (may_inline && len == ctx->e.length && tailused)
<br>> tryrecompress_trailing(ctx->queue + ctx->head,
<br>> &ctx->e.length, dst, &ret);
<br>>
<br>> @@ -454,6 +516,21 @@ frag_packing:
<br>> ctx->head += ctx->e.length;
<br>> len -= ctx->e.length;
<br>>
<br>> + if (may_fixing)
<br>> + z_erofs_fragments_dedupe_update(ctx, &len);
<br>> +
<br>> + /* generate a ctx for duplicate fragment */
<br>> + if (final && !len && inode->fragment_size && !may_packing) {
<br>> +frag_nopacking:
<br>> + z_erofs_write_indexes(ctx);
<br>> +
<br>> + ctx->e.length = inode->fragment_size;
<br>> + ctx->e.compressedblks = 0;
<br>> + ctx->e.raw = true;
<br>> + ctx->e.partial = false;
<br>> + ctx->e.blkaddr = ctx->blkaddr;
<br>> + }
<br>> +
<br>> if (!final && ctx->head >= EROFS_CONFIG_COMPR_MAX_SZ) {
<br>> const unsigned int qh_aligned =
<br>> round_down(ctx->head, EROFS_BLKSIZ);
<br>> @@ -736,7 +813,6 @@ int erofs_write_compressed_file(struct erofs_inode *inode, int fd)
<br>> {
<br>> struct erofs_buffer_head *bh;
<br>> static struct z_erofs_vle_compress_ctx ctx;
<br>> - erofs_off_t remaining;
<br>> erofs_blk_t blkaddr, compressed_blocks;
<br>> unsigned int legacymetasize;
<br>> int ret;
<br>> @@ -775,6 +851,16 @@ int erofs_write_compressed_file(struct erofs_inode *inode, int fd)
<br>> inode->idata_size = 0;
<br>> inode->fragment_size = 0;
<br>>
<br>> + /*
<br>> + * Dedupe fragments before compression to avoid writing duplicate parts
<br>> + * into packed inode.
<br>> + */
<br>> + if (cfg.c_fragments && !erofs_is_packed_inode(inode)) {
<br>> + ret = z_erofs_fragments_dedupe(inode, fd, &ctx.tofcrc);
<br>> + if (ret < 0)
<br>> + goto err_bdrop;
<br>> + }
<br>> +
<br>> blkaddr = erofs_mapbh(bh->block); /* start_blkaddr */
<br>> ctx.inode = inode;
<br>> ctx.blkaddr = blkaddr;
<br>> @@ -782,10 +868,12 @@ int erofs_write_compressed_file(struct erofs_inode *inode, int fd)
<br>> ctx.head = ctx.tail = 0;
<br>> ctx.clusterofs = 0;
<br>> ctx.e.length = 0;
<br>> - remaining = inode->i_size;
<br>> + ctx.pclustersize = z_erofs_get_max_pclusterblks(inode) * EROFS_BLKSIZ;
<br>> + ctx.remaining = inode->i_size - inode->fragment_size;
<br>> + ctx.need_fix = false;
<br>>
<br>> - while (remaining) {
<br>> - const u64 readcount = min_t(u64, remaining,
<br>> + while (ctx.remaining) {
<br>> + const u64 readcount = min_t(u64, ctx.remaining,
<br>> sizeof(ctx.queue) - ctx.tail);
<br>>
<br>> ret = read(fd, ctx.queue + ctx.tail, readcount);
<br>> @@ -793,10 +881,10 @@ int erofs_write_compressed_file(struct erofs_inode *inode, int fd)
<br>> ret = -errno;
<br>> goto err_bdrop;
<br>> }
<br>> - remaining -= readcount;
<br>> + ctx.remaining -= readcount;
<br>> ctx.tail += readcount;
<br>>
<br>> - ret = vle_compress_one(&ctx, !remaining);
<br>> + ret = vle_compress_one(&ctx);
<br>> if (ret)
<br>> goto err_free_idata;
<br>> }
<br>> @@ -807,6 +895,8 @@ int erofs_write_compressed_file(struct erofs_inode *inode, int fd)
<br>> DBG_BUGON(compressed_blocks < !!inode->idata_size);
<br>> compressed_blocks -= !!inode->idata_size;
<br>>
<br>> + z_erofs_fragments_commit(inode);
<br>> +
<br>> z_erofs_write_indexes(&ctx);
<br>> z_erofs_write_indexes_final(&ctx);
<br>> legacymetasize = ctx.metacur - compressmeta;
<br>> diff --git a/lib/fragments.c b/lib/fragments.c
<br>> index b8c37d5..e50937c 100644
<br>> --- a/lib/fragments.c
<br>> +++ b/lib/fragments.c
<br>> @@ -10,17 +10,181 @@
<br>> #include "erofs/inode.h"
<br>> #include "erofs/compress.h"
<br>> #include "erofs/print.h"
<br>> +#include "erofs/internal.h"
<br>> #include "erofs/fragments.h"
<br>>
<br>> +struct erofs_fragment_dedupe_item {
<br>> + struct list_head list;
<br>> + unsigned int length, nr_dup;
<br>> + erofs_off_t pos;
<br>> + u8 data[];
<br>> +};
<br>> +
<br>> +#define FRAGMENT_HASHTABLE_SIZE 65536
<br>> +#define FRAGMENT_HASH(crc) (crc & (FRAGMENT_HASHTABLE_SIZE - 1))
<br>> +
<br>> +static struct list_head dupli_frags[FRAGMENT_HASHTABLE_SIZE];
<br>> +static unsigned int len_to_hash; /* the fragment length for crc32 hash */
<br>> +
<br>> static FILE *packedfile;
<br>> const char *frags_packedname = "packed_file";
<br>>
<br>> -int z_erofs_pack_fragments(struct erofs_inode *inode, void *data,
<br>> - unsigned int len)
<br>> +static int z_erofs_fragments_dedupe_find(struct erofs_inode *inode, int fd,
<br>> + u32 crc)
<br>> {
<br>> - inode->z_advise |= Z_EROFS_ADVISE_FRAGMENT_PCLUSTER;
<br>> - inode->fragmentoff = ftell(packedfile);
<br>> - inode->fragment_size = len;
<br>> + struct erofs_fragment_dedupe_item *cur, *di = NULL;
<br>> + struct list_head *head;
<br>> + u8 *data;
<br>> + unsigned int length;
<br>> + int ret;
<br>> +
<br>> + head = &dupli_frags[FRAGMENT_HASH(crc)];
<br>> + if (list_empty(head))
<br>> + return 0;
<br>> +
<br>> + /* XXX: no need to read so much for smaller? */
<br>> + if (inode->i_size < EROFS_CONFIG_COMPR_MAX_SZ)
<br>> + length = inode->i_size;
<br>> + else
<br>> + length = EROFS_CONFIG_COMPR_MAX_SZ;
<br>> +
<br>> + data = malloc(length);
<br>> + if (!data)
<br>> + return -ENOMEM;
<br>> +
<br>> + ret = lseek(fd, inode->i_size - length, SEEK_SET);
<br>> + if (ret == -1) {
<br>> + ret = -errno;
<br>> + goto out;
<br>> + }
<br>> +
<br>> + ret = read(fd, data, length);
<br>> + if (ret != length) {
<br>> + ret = -errno;
<br>> + goto out;
<br>> + }
<br>> +
<br>> + list_for_each_entry(cur, head, list) {
<br>> + unsigned int e1, e2, i = 0;
<br>> +
<br>> + DBG_BUGON(cur->length < len_to_hash + 1);
<br>> + DBG_BUGON(length < len_to_hash + 1);
<br>> +
<br>> + e1 = cur->length - len_to_hash - 1;
<br>> + e2 = length - len_to_hash - 1;
<br>> +
<br>> + if (memcmp(cur->data + e1 + 1, data + e2 + 1, len_to_hash))
<br>> + continue;
<br>> +
<br>> + while (i <= min(e1, e2) && cur->data[e1 - i] == data[e2 - i])
<br>> + i++;
<br>> +
<br>> + if (i && (!di || i + len_to_hash > di->nr_dup)) {
<br>> + cur->nr_dup = i + len_to_hash;
<br>> + di = cur;
<br>> +
<br>> + /* full match */
<br>> + if (i == min(e1, e2) + 1)
<br>> + break;
<br>> + }
<br>> + }
<br>> + if (!di)
<br>> + goto out;
<br>> +
<br>> + DBG_BUGON(di->length < di->nr_dup);
<br>> +
<br>> + inode->fragment_size = di->nr_dup;
<br>> + inode->fragmentoff = di->pos + di->length - di->nr_dup;
<br>> +
<br>> + erofs_dbg("Dedupe %u fragment data at %lu", inode->fragment_size,
<br>> + inode->fragmentoff);
<br>> +out:
<br>> + free(data);
<br>> + return ret;
<br>> +}
<br>> +
<br>> +int z_erofs_fragments_dedupe(struct erofs_inode *inode, int fd, u32 *tofcrc_r)
<br>> +{
<br>> + u8 data_to_hash[len_to_hash];
<br>> + u32 crc;
<br>> + int ret;
<br>> +
<br>> + if (inode->i_size <= len_to_hash)
<br>> + return 0;
<br>> +
<br>> + ret = lseek(fd, inode->i_size - len_to_hash, SEEK_SET);
<br>> + if (ret == -1)
<br>> + return -errno;
<br>> +
<br>> + ret = read(fd, data_to_hash, len_to_hash);
<br>> + if (ret != len_to_hash)
<br>> + return -errno;
<br>> +
<br>> + crc = erofs_crc32c(~0, data_to_hash, len_to_hash);
<br>> + *tofcrc_r = crc;
<br>> +
<br>> + ret = z_erofs_fragments_dedupe_find(inode, fd, crc);
<br>> + if (ret < 0)
<br>> + return ret;
<br>> +
<br>> + ret = lseek(fd, 0, SEEK_SET);
<br>> + if (ret == -1)
<br>> + return -errno;
<br>> + return 0;
<br>> +}
<br>> +
<br>> +static int z_erofs_fragments_dedupe_insert(void *data, unsigned int len,
<br>> + erofs_off_t pos, u32 crc)
<br>> +{
<br>> + struct erofs_fragment_dedupe_item *di;
<br>> +
<br>> + if (len <= len_to_hash)
<br>> + return 0;
<br>> +
<br>> + di = malloc(sizeof(*di) + len);
<br>> + if (!di)
<br>> + return -ENOMEM;
<br>> +
<br>> + memcpy(di->data, data, len);
<br>> + di->length = len;
<br>> + di->pos = pos;
<br>> + di->nr_dup = 0;
<br>> +
<br>> + list_add_tail(&di->list, &dupli_frags[FRAGMENT_HASH(crc)]);
<br>> + return 0;
<br>> +}
<br>> +
<br>> +static inline void z_erofs_fragments_dedupe_init(unsigned int clen)
<br>> +{
<br>> + unsigned int i;
<br>> +
<br>> + for (i = 0; i < FRAGMENT_HASHTABLE_SIZE; ++i)
<br>> + init_list_head(&dupli_frags[i]);
<br>> +
<br>> + len_to_hash = clen;
<br>> +}
<br>> +
<br>> +static void z_erofs_fragments_dedupe_exit(void)
<br>> +{
<br>> + struct erofs_fragment_dedupe_item *di, *n;
<br>> + struct list_head *head;
<br>> + unsigned int i;
<br>> +
<br>> + for (i = 0; i < FRAGMENT_HASHTABLE_SIZE; ++i) {
<br>> + head = &dupli_frags[i];
<br>> +
<br>> + if (list_empty(head))
<br>> + continue;
<br>> +
<br>> + list_for_each_entry_safe(di, n, head, list)
<br>> + free(di);
<br>> + }
<br>> +}
<br>> +
<br>> +void z_erofs_fragments_commit(struct erofs_inode *inode)
<br>> +{
<br>> + if (!inode->fragment_size)
<br>> + return;
<br>> /*
<br>> * If the packed inode is larger than 4GiB, the full fragmentoff
<br>> * will be recorded by switching to the noncompact layout anyway.
<br>> @@ -28,13 +192,28 @@ int z_erofs_pack_fragments(struct erofs_inode *inode, void *data,
<br>> if (inode->fragmentoff >> 32)
<br>> inode->datalayout = EROFS_INODE_FLAT_COMPRESSION_LEGACY;
<br>>
<br>> + inode->z_advise |= Z_EROFS_ADVISE_FRAGMENT_PCLUSTER;
<br>> + erofs_sb_set_fragments();
<br>> +}
<br>> +
<br>> +int z_erofs_pack_fragments(struct erofs_inode *inode, void *data,
<br>> + unsigned int len, u32 tofcrc)
<br>> +{
<br>> + int ret;
<br>> +
<br>> + inode->fragmentoff = ftell(packedfile);
<br>> + inode->fragment_size = len;
<br>> +
<br>> if (fwrite(data, len, 1, packedfile) != 1)
<br>> return -EIO;
<br>>
<br>> - erofs_sb_set_fragments();
<br>> -
<br>> erofs_dbg("Recording %u fragment data at %lu", inode->fragment_size,
<br>> inode->fragmentoff);
<br>> +
<br>> + ret = z_erofs_fragments_dedupe_insert(data, len, inode->fragmentoff,
<br>> + tofcrc);
<br>> + if (ret)
<br>> + return ret;
<br>> return len;
<br>> }
<br>>
<br>> @@ -50,6 +229,8 @@ void erofs_fragments_exit(void)
<br>> {
<br>> if (packedfile)
<br>> fclose(packedfile);
<br>> +
<br>> + z_erofs_fragments_dedupe_exit();
<br>> }
<br>>
<br>> int erofs_fragments_init(void)
<br>> @@ -61,5 +242,7 @@ int erofs_fragments_init(void)
<br>> #endif
<br>> if (!packedfile)
<br>> return -ENOMEM;
<br>> +
<br>> + z_erofs_fragments_dedupe_init(16);
<br>> return 0;
<br>> }</p><meta data-version="editor_version_1.2.11"/><div data-zone-id="0" data-line-index="0" data-line="true" style="white-space: pre-wrap; margin-top: 4px; margin-bottom: 4px; line-height: 1.6;">------机密:此电子邮件所包含内容为酷派机密内容,并且受到法律的保护。如果您不属于以上电子邮件的目标接收者,您不得细读,使用,传播,散布或复制该电子邮件中的任何信息。如果您已经误收此电子邮件,请您立即通知我们并删除原电子邮件。 CONFIDENTIAL: This e-mail message contains information of Coolpad that is confidential and which is subject to legal privilege.If you are not the intended recipient as indicated above,you must not peruse,use, disseminate,distribute or copy any information contained in this message.If you have received this message in error, please notify us and delete the original message immediately.
</div><div data-zone-id="0" data-line-index="1" data-line="true" style="white-space: pre-wrap; margin-top: 4px; margin-bottom: 4px; line-height: 1.6;">------ 申明:本邮件所现内容,仅作为我们之间就合作的事宜进行的交流、沟通、洽谈、商议,不作为协议或承诺,一切协议及承诺必须以书面文本盖章为准。 DECLARATION:All contents of this E-mail ,are only regarded as the cooperation we have had between the exchanges, communication, negotiation and deliberation, not as a agreement or promise. All contracts and commitments must be sealed shall prevail.
</div>