Hi Jianan,

On Wed, Jun 23, 2021 at 06:32:37PM +0800, Huang Jianan wrote:
> Hi all,
> 
> This patch works well for us.
> 
> Tested-by: Huang Jianan <huangjia...@oppo.com>

Thanks for confirmation.

> 
> Thanks,
> 
> Jianan
> 

Hi Yue,

> On 2021/6/22 11:02, Yue Hu wrote:
> > From: Yue Hu <huy...@yulong.com>
> > 
> > Android update engine will treat EROFS filesystem image as one single
> > file. Let's add block list support to optimize OTA size.
> > 
> > Change-Id: I21d6177dff0ee65d3c57023b102e991d40873f0d
> > Signed-off-by: Yue Hu <huy...@yulong.com>
> > ---
> >   include/erofs/block_list.h | 19 ++++++++++
> >   include/erofs/config.h     |  1 +
> >   lib/block_list.c           | 86 
> > ++++++++++++++++++++++++++++++++++++++++++++++
> >   lib/compress.c             |  8 +++++
> >   lib/inode.c                | 21 ++++++++++-
> >   mkfs/main.c                | 17 +++++++++
> >   6 files changed, 151 insertions(+), 1 deletion(-)
> >   create mode 100644 include/erofs/block_list.h
> >   create mode 100644 lib/block_list.c
> > 
> > diff --git a/include/erofs/block_list.h b/include/erofs/block_list.h
> > new file mode 100644
> > index 0000000..cbf1050
> > --- /dev/null
> > +++ b/include/erofs/block_list.h
> > @@ -0,0 +1,19 @@
> > +// SPDX-License-Identifier: GPL-2.0+
> > +/*
> > + * erofs-utils/include/erofs/block_list.h
> > + *
> > + * Copyright (C), 2021, Coolpad Group Limited.
> > + * Created by Yue Hu <huy...@yulong.com>
> > + */
> > +#ifndef __EROFS_BLOCK_LIST_H
> > +#define __EROFS_BLOCK_LIST_H
> > +
> > +#include "internal.h"
> > +
> > +int block_list_fopen(void);
> > +void block_list_fclose(void);
> > +void write_block_list(const char *path, erofs_blk_t blk_start,
> > +                      erofs_blk_t nblocks, bool has_tail);
> > +void write_block_list_tail_end(const char *path, erofs_blk_t nblocks,
> > +                               bool inline_data, erofs_blk_t blkaddr);

some nitpick...

How about use "erofs_droid_blocklist_" prefix here?
even though it may not be quite useful, but I'd like to avoid potential
namespace polluation...

> > +#endif
> > diff --git a/include/erofs/config.h b/include/erofs/config.h
> > index d140a73..67e7a0f 100644
> > --- a/include/erofs/config.h
> > +++ b/include/erofs/config.h
> > @@ -65,6 +65,7 @@ struct erofs_configure {
> >     char *mount_point;
> >     char *target_out_path;
> >     char *fs_config_file;
> > +   char *block_list_file;
> >   #endif
> >   };
> > diff --git a/lib/block_list.c b/lib/block_list.c
> > new file mode 100644
> > index 0000000..6ebe0f9
> > --- /dev/null
> > +++ b/lib/block_list.c
> > @@ -0,0 +1,86 @@
> > +// SPDX-License-Identifier: GPL-2.0+
> > +/*
> > + * erofs-utils/lib/block_list.c
> > + *
> > + * Copyright (C), 2021, Coolpad Group Limited.
> > + * Created by Yue Hu <huy...@yulong.com>
> > + */
> > +#ifdef WITH_ANDROID
> > +#include <stdio.h>
> > +
> > +#include "erofs/block_list.h"
> > +
> > +#define pr_fmt(fmt) "EROFS block_list: " FUNC_LINE_FMT fmt "\n"
> > +#include "erofs/print.h"
> > +
> > +static FILE *block_list_fp = NULL;
> > +
> > +int block_list_fopen(void)
> > +{
> > +   if (block_list_fp)
> > +           return 0;
> > +
> > +   block_list_fp = fopen(cfg.block_list_file, "w");
> > +
> > +   if (block_list_fp == NULL)
> > +           return -1;
> > +
> > +   return 0;
> > +}
> > +
> > +void block_list_fclose(void)
> > +{
> > +   if (block_list_fp) {
> > +           fclose(block_list_fp);
> > +           block_list_fp = NULL;
> > +   }
> > +}
> > +
> > +void write_block_list(const char *path, erofs_blk_t blk_start,
> > +                 erofs_blk_t nblocks, bool has_tail)
> > +{
> > +   const char *fspath = erofs_fspath(path);
> > +
> > +   if (!block_list_fp || !cfg.mount_point)
> > +           return;
> > +
> > +   /* only tail-end data */
> > +   if (!nblocks)
> > +           return;
> > +
> > +   fprintf(block_list_fp, "/%s", cfg.mount_point);
> > +
> > +   if (fspath[0] != '/')
> > +           fprintf(block_list_fp, "/");
> > +
> > +   if (nblocks == 1) {
> > +           fprintf(block_list_fp, "%s %u", fspath, blk_start);
> > +   } else {
> > +           fprintf(block_list_fp, "%s %u-%u", fspath, blk_start,
> > +                   blk_start + nblocks - 1);
> > +   }
> > +
> > +   if (!has_tail)
> > +           fprintf(block_list_fp, "\n");
> > +}
> > +
> > +void write_block_list_tail_end(const char *path, erofs_blk_t nblocks,
> > +                          bool inline_data, erofs_blk_t blkaddr)

This makes me wondering...

1) how about passing in struct erofs_inode *inode directly instead?
2) how about getting rid of inline_data boolean and using
blkaddr != NULL_ADDR to replace such case, e.g.

> > +{
> > +   if (!block_list_fp || !cfg.mount_point)
> > +           return;
> > +
> > +   if (!nblocks && !inline_data) {

        if (!nblocks && blkaddr != NULL_ADDR) {

> > +           erofs_dbg("%s : only tail-end non-inline data", path);
> > +           write_block_list(path, blkaddr, 1, false);
> > +           return;
> > +   }
> > +
> > +   if (nblocks) {
> > +           if (!inline_data)
> > +                   fprintf(block_list_fp, " %u", blkaddr);
> > +
> > +           fprintf(block_list_fp, "\n");
> > +   }
> > +}
> > +#endif
> > diff --git a/lib/compress.c b/lib/compress.c
> > index 2093bfd..5dec0c3 100644
> > --- a/lib/compress.c
> > +++ b/lib/compress.c
> > @@ -19,6 +19,10 @@
> >   #include "erofs/compress.h"
> >   #include "compressor.h"
> > +#ifdef WITH_ANDROID
> > +#include "erofs/block_list.h"
> > +#endif
> > +
> >   static struct erofs_compress compresshandle;
> >   static int compressionlevel;
> > @@ -553,6 +557,10 @@ int erofs_write_compressed_file(struct erofs_inode 
> > *inode)
> >                inode->i_srcpath, (unsigned long long)inode->i_size,
> >                compressed_blocks);
> > +#ifdef WITH_ANDROID
> > +   write_block_list(inode->i_srcpath, blkaddr, compressed_blocks, false);
> > +#endif
> > +
> >     /*
> >      * TODO: need to move erofs_bdrop to erofs_write_tail_end
> >      *       when both mkfs & kernel support compression inline.
> > diff --git a/lib/inode.c b/lib/inode.c
> > index 787e5b4..6be23cb 100644
> > --- a/lib/inode.c
> > +++ b/lib/inode.c
> > @@ -22,6 +22,10 @@
> >   #include "erofs/xattr.h"
> >   #include "erofs/exclude.h"
> > +#ifdef WITH_ANDROID
> > +#include "erofs/block_list.h"
> > +#endif
> > +
> >   #define S_SHIFT                 12
> >   static unsigned char erofs_ftype_by_mode[S_IFMT >> S_SHIFT] = {
> >     [S_IFREG >> S_SHIFT]  = EROFS_FT_REG_FILE,
> > @@ -369,6 +373,12 @@ static int write_uncompressed_file_from_fd(struct 
> > erofs_inode *inode, int fd)
> >                     return -EIO;
> >             }
> >     }
> > +
> > +#ifdef WITH_ANDROID
> > +   if (nblocks)
> > +           write_block_list(inode->i_srcpath, inode->u.i_blkaddr,
> > +                            nblocks, inode->idata_size ? true : false);
> > +#endif
> >     return 0;
> >   }
> > @@ -626,6 +636,7 @@ static struct erofs_bhops erofs_write_inline_bhops = {
> >   int erofs_write_tail_end(struct erofs_inode *inode)
> >   {
> >     struct erofs_buffer_head *bh, *ibh;
> > +   erofs_off_t pos;
> >     bh = inode->bh_data;
> > @@ -640,7 +651,6 @@ int erofs_write_tail_end(struct erofs_inode *inode)
> >             ibh->op = &erofs_write_inline_bhops;
> >     } else {
> >             int ret;
> > -           erofs_off_t pos;
> >             erofs_mapbh(bh->block);
> >             pos = erofs_btell(bh, true) - EROFS_BLKSIZ;
> > @@ -658,6 +668,15 @@ int erofs_write_tail_end(struct erofs_inode *inode)
> >             free(inode->idata);
> >             inode->idata = NULL;
> >     }
> > +
> > +#ifdef WITH_ANDROID
> > +   if (!S_ISDIR(inode->i_mode) && !S_ISLNK(inode->i_mode))

hmmm... that seems some hacky, but I can leave it as-is for now.
but how about moving it into erofs_droid_blocklist_write_tail_end()?

> > +           write_block_list_tail_end(inode->i_srcpath,
> > +                                     inode->i_size / EROFS_BLKSIZ,
> > +                                     inode->bh_inline ? true: false,
> > +                                     erofs_blknr(pos));

                erofs_droid_blocklist_write_tail_end(inode,
                                inode->bh_inline ? NULL_ADDR :
                                erofs_blknr(pos));

I will send a rough delta patch later, please check it out...

Thanks,
Gao Xiang

> > +#endif
> > +
> >   out:
> >     /* now bh_data can drop directly */
> >     if (bh) {
> > diff --git a/mkfs/main.c b/mkfs/main.c
> > index e476189..d5a5e07 100644
> > --- a/mkfs/main.c
> > +++ b/mkfs/main.c
> > @@ -27,6 +27,10 @@
> >   #include <uuid.h>
> >   #endif
> > +#ifdef WITH_ANDROID
> > +#include "erofs/block_list.h"
> > +#endif
> > +
> >   #define EROFS_SUPER_END (EROFS_SUPER_OFFSET + sizeof(struct 
> > erofs_super_block))
> >   static struct option long_options[] = {
> > @@ -47,6 +51,7 @@ static struct option long_options[] = {
> >     {"mount-point", required_argument, NULL, 10},
> >     {"product-out", required_argument, NULL, 11},
> >     {"fs-config-file", required_argument, NULL, 12},
> > +   {"block-list-file", required_argument, NULL, 13},
> >   #endif
> >     {0, 0, 0, 0},
> >   };
> > @@ -95,6 +100,7 @@ static void usage(void)
> >           " --mount-point=X       X=prefix of target fs path (default: /)\n"
> >           " --product-out=X       X=product_out directory\n"
> >           " --fs-config-file=X    X=fs_config file\n"
> > +         " --block-list-file=X    X=block_list file\n"
> >   #endif
> >           "\nAvailable compressors are: ", stderr);
> >     print_available_compressors(stderr, ", ");
> > @@ -293,6 +299,9 @@ static int mkfs_parse_options_cfg(int argc, char 
> > *argv[])
> >             case 12:
> >                     cfg.fs_config_file = optarg;
> >                     break;
> > +           case 13:
> > +                   cfg.block_list_file = optarg;
> > +                   break;
> >   #endif
> >             case 'C':
> >                     i = strtoull(optarg, &endptr, 0);
> > @@ -541,6 +550,11 @@ int main(int argc, char **argv)
> >             erofs_err("failed to load fs config %s", cfg.fs_config_file);
> >             return 1;
> >     }
> > +
> > +   if (cfg.block_list_file && block_list_fopen() < 0) {
> > +           erofs_err("failed to open %s", cfg.block_list_file);
> > +           return 1;
> > +   }
> >   #endif
> >     erofs_show_config();
> > @@ -607,6 +621,9 @@ int main(int argc, char **argv)
> >             err = erofs_mkfs_superblock_csum_set();
> >   exit:
> >     z_erofs_compress_exit();
> > +#ifdef WITH_ANDROID
> > +   block_list_fclose();
> > +#endif
> >     dev_close();
> >     erofs_cleanup_exclude_rules();
> >     erofs_exit_configure();

Reply via email to