[PATCHv2] support ZSTD compression algorithm

Mark Wielaard mark@klomp.org
Fri Dec 16 00:32:01 GMT 2022


Hi Martin,

On Tue, Nov 29, 2022 at 01:05:45PM +0100, Martin Liška wrote:
> diff --git a/libelf/elf_compress.c b/libelf/elf_compress.c
> index d7f53af2..7a6e37a4 100644
> --- a/libelf/elf_compress.c
> +++ b/libelf/elf_compress.c
> @@ -39,6 +39,10 @@
>  #include <string.h>
>  #include <zlib.h>
> +#ifdef USE_ZSTD
> +#include <zstd.h>
> +#endif
> +
>  /* Cleanup and return result.  Don't leak memory.  */
>  static void *
>  do_deflate_cleanup (void *result, z_stream *z, void *out_buf,

OK.

> @@ -54,53 +58,14 @@ do_deflate_cleanup (void *result, z_stream *z, void *out_buf,
>  #define deflate_cleanup(result, cdata) \
>      do_deflate_cleanup(result, &z, out_buf, cdata)
> -/* Given a section, uses the (in-memory) Elf_Data to extract the
> -   original data size (including the given header size) and data
> -   alignment.  Returns a buffer that has at least hsize bytes (for the
> -   caller to fill in with a header) plus zlib compressed date.  Also
> -   returns the new buffer size in new_size (hsize + compressed data
> -   size).  Returns (void *) -1 when FORCE is false and the compressed
> -   data would be bigger than the original data.  */
>  void *
>  internal_function
> -__libelf_compress (Elf_Scn *scn, size_t hsize, int ei_data,
> -		   size_t *orig_size, size_t *orig_addralign,
> -		   size_t *new_size, bool force)
> +__libelf_compress_zlib (Elf_Scn *scn, size_t hsize, int ei_data,
> +			size_t *orig_size, size_t *orig_addralign,
> +			size_t *new_size, bool force,
> +			Elf_Data *data, Elf_Data *next_data,
> +			void *out_buf, size_t out_size, size_t block)
>  {
> -  /* The compressed data is the on-disk data.  We simplify the
> -     implementation a bit by asking for the (converted) in-memory
> -     data (which might be all there is if the user created it with
> -     elf_newdata) and then convert back to raw if needed before
> -     compressing.  Should be made a bit more clever to directly
> -     use raw if that is directly available.  */
> -  Elf_Data *data = elf_getdata (scn, NULL);
> -  if (data == NULL)
> -    return NULL;
> -
> -  /* When not forced and we immediately know we would use more data by
> -     compressing, because of the header plus zlib overhead (five bytes
> -     per 16 KB block, plus a one-time overhead of six bytes for the
> -     entire stream), don't do anything.  */
> -  Elf_Data *next_data = elf_getdata (scn, data);
> -  if (next_data == NULL && !force
> -      && data->d_size <= hsize + 5 + 6)
> -    return (void *) -1;
> -
> -  *orig_addralign = data->d_align;
> -  *orig_size = data->d_size;
> -
> -  /* Guess an output block size. 1/8th of the original Elf_Data plus
> -     hsize.  Make the first chunk twice that size (25%), then increase
> -     by a block (12.5%) when necessary.  */
> -  size_t block = (data->d_size / 8) + hsize;
> -  size_t out_size = 2 * block;
> -  void *out_buf = malloc (out_size);
> -  if (out_buf == NULL)
> -    {
> -      __libelf_seterrno (ELF_E_NOMEM);
> -      return NULL;
> -    }
> -
>    /* Caller gets to fill in the header at the start.  Just skip it here.  */
>    size_t used = hsize;

OK, all this is moved lower and then calls either
__libelf_compress_zlib or __libelf_compress_zstd.

> @@ -205,9 +170,186 @@ __libelf_compress (Elf_Scn *scn, size_t hsize, int ei_data,
>    return out_buf;
>  }
> +#ifdef USE_ZSTD
> +/* Cleanup and return result.  Don't leak memory.  */
> +static void *
> +do_zstd_cleanup (void *result, ZSTD_CCtx * const cctx, void *out_buf,
> +		 Elf_Data *cdatap)
> +{
> +  ZSTD_freeCCtx (cctx);
> +  free (out_buf);
> +  if (cdatap != NULL)
> +    free (cdatap->d_buf);
> +  return result;
> +}
>
> +#define zstd_cleanup(result, cdata) \
> +    do_zstd_cleanup(result, cctx, out_buf, cdata)
> +

OK, mimics do_deflate_cleanup.

>  void *
>  internal_function
> -__libelf_decompress (void *buf_in, size_t size_in, size_t size_out)
> +__libelf_compress_zstd (Elf_Scn *scn, size_t hsize, int ei_data,
> +			size_t *orig_size, size_t *orig_addralign,
> +			size_t *new_size, bool force,
> +			Elf_Data *data, Elf_Data *next_data,
> +			void *out_buf, size_t out_size, size_t block)
> +{
> +  /* Caller gets to fill in the header at the start.  Just skip it here.  */
> +  size_t used = hsize;
> +
> +  ZSTD_CCtx* const cctx = ZSTD_createCCtx();
> +  Elf_Data cdata;
> +  cdata.d_buf = NULL;
> +
> +  /* Loop over data buffers.  */
> +  ZSTD_EndDirective mode = ZSTD_e_continue;
> +
> +  do
> +    {
> +      /* Convert to raw if different endianness.  */
> +      cdata = *data;
> +      bool convert = ei_data != MY_ELFDATA && data->d_size > 0;
> +      if (convert)
> +	{
> +	  /* Don't do this conversion in place, we might want to keep
> +	     the original data around, caller decides.  */
> +	  cdata.d_buf = malloc (data->d_size);
> +	  if (cdata.d_buf == NULL)
> +	    {
> +	      __libelf_seterrno (ELF_E_NOMEM);
> +	      return zstd_cleanup (NULL, NULL);
> +	    }
> +	  if (gelf_xlatetof (scn->elf, &cdata, data, ei_data) == NULL)
> +	    return zstd_cleanup (NULL, &cdata);
> +	}
> +
> +      ZSTD_inBuffer ib = { cdata.d_buf, cdata.d_size, 0 };
> +
> +      /* Get next buffer to see if this is the last one.  */
> +      data = next_data;
> +      if (data != NULL)
> +	{
> +	  *orig_addralign = MAX (*orig_addralign, data->d_align);
> +	  *orig_size += data->d_size;
> +	  next_data = elf_getdata (scn, data);
> +	}
> +      else
> +	mode = ZSTD_e_end;
> +
> +      /* Flush one data buffer.  */
> +      for (;;)
> +	{
> +	  ZSTD_outBuffer ob = { out_buf + used, out_size - used, 0 };
> +	  size_t ret = ZSTD_compressStream2 (cctx, &ob, &ib, mode);
> +	  if (ZSTD_isError (ret))
> +	    {
> +	      __libelf_seterrno (ELF_E_COMPRESS_ERROR);
> +	      return zstd_cleanup (NULL, convert ? &cdata : NULL);
> +	    }
> +	  used += ob.pos;
> +
> +	  /* Bail out if we are sure the user doesn't want the
> +	     compression forced and we are using more compressed data
> +	     than original data.  */
> +	  if (!force && mode == ZSTD_e_end && used >= *orig_size)
> +	    return zstd_cleanup ((void *) -1, convert ? &cdata : NULL);
> +
> +	  if (ret > 0)
> +	    {
> +	      void *bigger = realloc (out_buf, out_size + block);
> +	      if (bigger == NULL)
> +		{
> +		  __libelf_seterrno (ELF_E_NOMEM);
> +		  return zstd_cleanup (NULL, convert ? &cdata : NULL);
> +		}

OK, zstd_cleanup does also free out_buf.

> +	      out_buf = bigger;
> +	      out_size += block;
> +	    }
> +	  else
> +	    break;
> +	}
> +
> +      if (convert)
> +	{
> +	  free (cdata.d_buf);
> +	  cdata.d_buf = NULL;
> +	}
> +    }
> +  while (mode != ZSTD_e_end); /* More data blocks.  */
> +
> +  ZSTD_freeCCtx (cctx);
> +  *new_size = used;
> +  return out_buf;
> +}
> +#endif

Look good.

> +/* Given a section, uses the (in-memory) Elf_Data to extract the
> +   original data size (including the given header size) and data
> +   alignment.  Returns a buffer that has at least hsize bytes (for the
> +   caller to fill in with a header) plus zlib compressed date.  Also
> +   returns the new buffer size in new_size (hsize + compressed data
> +   size).  Returns (void *) -1 when FORCE is false and the compressed
> +   data would be bigger than the original data.  */
> +void *
> +internal_function
> +__libelf_compress (Elf_Scn *scn, size_t hsize, int ei_data,
> +		   size_t *orig_size, size_t *orig_addralign,
> +		   size_t *new_size, bool force, bool use_zstd)
> +{
> +  /* The compressed data is the on-disk data.  We simplify the
> +     implementation a bit by asking for the (converted) in-memory
> +     data (which might be all there is if the user created it with
> +     elf_newdata) and then convert back to raw if needed before
> +     compressing.  Should be made a bit more clever to directly
> +     use raw if that is directly available.  */
> +  Elf_Data *data = elf_getdata (scn, NULL);
> +  if (data == NULL)
> +    return NULL;
> +
> +  /* When not forced and we immediately know we would use more data by
> +     compressing, because of the header plus zlib overhead (five bytes
> +     per 16 KB block, plus a one-time overhead of six bytes for the
> +     entire stream), don't do anything.
> +     Size estimation for ZSTD compression would be similar.  */
> +  Elf_Data *next_data = elf_getdata (scn, data);
> +  if (next_data == NULL && !force
> +      && data->d_size <= hsize + 5 + 6)
> +    return (void *) -1;
> +
> +  *orig_addralign = data->d_align;
> +  *orig_size = data->d_size;
> +
> +  /* Guess an output block size. 1/8th of the original Elf_Data plus
> +     hsize.  Make the first chunk twice that size (25%), then increase
> +     by a block (12.5%) when necessary.  */
> +  size_t block = (data->d_size / 8) + hsize;
> +  size_t out_size = 2 * block;
> +  void *out_buf = malloc (out_size);
> +  if (out_buf == NULL)
> +    {
> +      __libelf_seterrno (ELF_E_NOMEM);
> +      return NULL;
> +    }

OK, this is all just moved from earlier with an extra use_zstd flag.

> +  if (use_zstd)
> +#ifdef USE_ZSTD
> +    return __libelf_compress_zstd (scn, hsize, ei_data, orig_size,
> +				   orig_addralign, new_size, force,
> +				   data, next_data, out_buf, out_size,
> +				   block);
> +#else
> +    return NULL;

Should this also call __libelf_seterrno (ELF_E_UNKNOWN_COMPRESSION_TYPE) ?

> +#endif
> +  else
> +    return __libelf_compress_zlib (scn, hsize, ei_data, orig_size,
> +				   orig_addralign, new_size, force,
> +				   data, next_data, out_buf, out_size,
> +				   block);
> +}
> +
> +void *
> +internal_function
> +__libelf_decompress_zlib (void *buf_in, size_t size_in, size_t size_out)
>  {
>    /* Catch highly unlikely compression ratios so we don't allocate
>       some giant amount of memory for nothing. The max compression
> @@ -218,7 +360,7 @@ __libelf_decompress (void *buf_in, size_t size_in, size_t size_out)
>        return NULL;
>      }
> -  /* Malloc might return NULL when requestion zero size.  This is highly
> +  /* Malloc might return NULL when requesting zero size.  This is highly
>       unlikely, it would only happen when the compression was forced.
>       But we do need a non-NULL buffer to return and set as result.
>       Just make sure to always allocate at least 1 byte.  */

OK, typo fix.

> @@ -260,6 +402,51 @@ __libelf_decompress (void *buf_in, size_t size_in, size_t size_out)
>    return buf_out;
>  }
> +#ifdef USE_ZSTD
> +void *
> +internal_function
> +__libelf_decompress_zstd (void *buf_in, size_t size_in, size_t size_out)
> +{
> +  /* Malloc might return NULL when requesting zero size.  This is highly
> +     unlikely, it would only happen when the compression was forced.
> +     But we do need a non-NULL buffer to return and set as result.
> +     Just make sure to always allocate at least 1 byte.  */
> +  void *buf_out = malloc (size_out ?: 1);
> +  if (unlikely (buf_out == NULL))
> +    {
> +      __libelf_seterrno (ELF_E_NOMEM);
> +      return NULL;
> +    }
> +
> +  size_t ret = ZSTD_decompress (buf_out, size_out, buf_in, size_in);
> +  if (ZSTD_isError (ret))
> +    {
> +      free (buf_out);
> +      __libelf_seterrno (ELF_E_DECOMPRESS_ERROR);
> +      return NULL;
> +    }
> +  else
> +    return buf_out;
> +}
> +#endif

OK.

> +void *
> +internal_function
> +__libelf_decompress (int chtype, void *buf_in, size_t size_in, size_t size_out)
> +{
> +  if (chtype == ELFCOMPRESS_ZLIB)
> +    return __libelf_decompress_zlib (buf_in, size_in, size_out);
> +  else
> +    {
> +#ifdef USE_ZSTD
> +    return __libelf_decompress_zstd (buf_in, size_in, size_out);
> +#else
> +    __libelf_seterrno (ELF_E_DECOMPRESS_ERROR);

Maybe ELF_E_UNKNOWN_COMPRESSION_TYPE ?

> +    return NULL;
> +#endif
> +    }
> +}
> +
>  void *
>  internal_function
>  __libelf_decompress_elf (Elf_Scn *scn, size_t *size_out, size_t *addralign)
> @@ -268,7 +455,7 @@ __libelf_decompress_elf (Elf_Scn *scn, size_t *size_out, size_t *addralign)
>    if (gelf_getchdr (scn, &chdr) == NULL)
>      return NULL;
> -  if (chdr.ch_type != ELFCOMPRESS_ZLIB)
> +  if (chdr.ch_type != ELFCOMPRESS_ZLIB && chdr.ch_type != ELFCOMPRESS_ZSTD)
>      {
>        __libelf_seterrno (ELF_E_UNKNOWN_COMPRESSION_TYPE);
>        return NULL;

Should the chdr.ch_type != ELFCOMPRESS_ZSTD be guarded by #ifdef USE_ZSTD ?

> @@ -295,7 +482,9 @@ __libelf_decompress_elf (Elf_Scn *scn, size_t *size_out, size_t *addralign)
>  		  ? sizeof (Elf32_Chdr) : sizeof (Elf64_Chdr));
>    size_t size_in = data->d_size - hsize;
>    void *buf_in = data->d_buf + hsize;
> -  void *buf_out = __libelf_decompress (buf_in, size_in, chdr.ch_size);
> +  void *buf_out
> +    = __libelf_decompress (chdr.ch_type, buf_in, size_in, chdr.ch_size);
> +
>    *size_out = chdr.ch_size;
>    *addralign = chdr.ch_addralign;
>    return buf_out;
> @@ -394,7 +583,7 @@ elf_compress (Elf_Scn *scn, int type, unsigned int flags)
>      }
>    int compressed = (sh_flags & SHF_COMPRESSED);
> -  if (type == ELFCOMPRESS_ZLIB)
> +  if (type == ELFCOMPRESS_ZLIB || type == ELFCOMPRESS_ZSTD)
>      {
>        /* Compress/Deflate.  */
>        if (compressed == 1)
> @@ -408,7 +597,8 @@ elf_compress (Elf_Scn *scn, int type, unsigned int flags)
>        size_t orig_size, orig_addralign, new_size;
>        void *out_buf = __libelf_compress (scn, hsize, elfdata,
>  					 &orig_size, &orig_addralign,
> -					 &new_size, force);
> +					 &new_size, force,
> +					 type == ELFCOMPRESS_ZSTD);
>        /* Compression would make section larger, don't change anything.  */
>        if (out_buf == (void *) -1)
> @@ -422,7 +612,7 @@ elf_compress (Elf_Scn *scn, int type, unsigned int flags)
>        if (elfclass == ELFCLASS32)
>  	{
>  	  Elf32_Chdr chdr;
> -	  chdr.ch_type = ELFCOMPRESS_ZLIB;
> +	  chdr.ch_type = type;
>  	  chdr.ch_size = orig_size;
>  	  chdr.ch_addralign = orig_addralign;
>  	  if (elfdata != MY_ELFDATA)
> @@ -436,7 +626,7 @@ elf_compress (Elf_Scn *scn, int type, unsigned int flags)
>        else
>  	{
>  	  Elf64_Chdr chdr;
> -	  chdr.ch_type = ELFCOMPRESS_ZLIB;
> +	  chdr.ch_type = type;
>  	  chdr.ch_reserved = 0;
>  	  chdr.ch_size = orig_size;
>  	  chdr.ch_addralign = sh_addralign;

OK.

> diff --git a/libelf/elf_compress_gnu.c b/libelf/elf_compress_gnu.c
> index 3d2977e7..8e20b30e 100644
> --- a/libelf/elf_compress_gnu.c
> +++ b/libelf/elf_compress_gnu.c
> @@ -103,7 +103,8 @@ elf_compress_gnu (Elf_Scn *scn, int inflate, unsigned int flags)
>        size_t orig_size, new_size, orig_addralign;
>        void *out_buf = __libelf_compress (scn, hsize, elfdata,
>  					 &orig_size, &orig_addralign,
> -					 &new_size, force);
> +					 &new_size, force,
> +					 /* use_zstd */ false);
>        /* Compression would make section larger, don't change anything.  */
>        if (out_buf == (void *) -1)

OK.

> @@ -178,7 +179,7 @@ elf_compress_gnu (Elf_Scn *scn, int inflate, unsigned int flags)
>        size_t size = gsize;
>        size_t size_in = data->d_size - hsize;
>        void *buf_in = data->d_buf + hsize;
> -      void *buf_out = __libelf_decompress (buf_in, size_in, size);
> +      void *buf_out = __libelf_decompress (ELFCOMPRESS_ZLIB, buf_in, size_in, size);
>        if (buf_out == NULL)
>  	return -1;

OK.

> diff --git a/libelf/libelfP.h b/libelf/libelfP.h
> index d88a613c..6624f38a 100644
> --- a/libelf/libelfP.h
> +++ b/libelf/libelfP.h
> @@ -574,10 +574,10 @@ extern uint32_t __libelf_crc32 (uint32_t crc, unsigned char *buf, size_t len)
>  extern void * __libelf_compress (Elf_Scn *scn, size_t hsize, int ei_data,
>  				 size_t *orig_size, size_t *orig_addralign,
> -				 size_t *size, bool force)
> +				 size_t *size, bool force, bool use_zstd)
>       internal_function;
> -extern void * __libelf_decompress (void *buf_in, size_t size_in,
> +extern void * __libelf_decompress (int chtype, void *buf_in, size_t size_in,
>  				   size_t size_out) internal_function;
>  extern void * __libelf_decompress_elf (Elf_Scn *scn,
>  				       size_t *size_out, size_t *addralign)

OK.

> diff --git a/src/elfcompress.c b/src/elfcompress.c
> index eff765e8..b0f1677c 100644
> --- a/src/elfcompress.c
> +++ b/src/elfcompress.c
> @@ -55,9 +55,10 @@ enum ch_type
>    UNSET = -1,
>    NONE,
>    ZLIB,
> +  ZSTD,
>    /* Maximal supported ch_type.  */
> -  MAXIMAL_CH_TYPE = ZLIB,
> +  MAXIMAL_CH_TYPE = ZSTD,
>    ZLIB_GNU = 1 << 16
>  };

OK.

> @@ -139,6 +140,12 @@ parse_opt (int key, char *arg __attribute__ ((unused)),
>  	type = ZLIB;
>        else if (strcmp ("zlib-gnu", arg) == 0 || strcmp ("gnu", arg) == 0)
>  	type = ZLIB_GNU;
> +      else if (strcmp ("zstd", arg) == 0)
> +#ifdef USE_ZSTD
> +	type = ZSTD;
> +#else
> +	argp_error (state, N_("ZSTD support is not enabled"));
> +#endif
>        else
>  	argp_error (state, N_("unknown compression type '%s'"), arg);
>        break;

Nice error handling.

> @@ -281,6 +288,44 @@ get_sections (unsigned int *sections, size_t shnum)
>    return s;
>  }
> +/* Return compression type of a given section SHDR.  */
> +
> +static enum ch_type
> +get_section_chtype (Elf_Scn *scn, GElf_Shdr *shdr, const char *sname,
> +		    size_t ndx)
> +{
> +  enum ch_type chtype = UNSET;
> +  if ((shdr->sh_flags & SHF_COMPRESSED) != 0)
> +    {
> +      GElf_Chdr chdr;
> +      if (gelf_getchdr (scn, &chdr) != NULL)
> +	{
> +	  chtype = (enum ch_type)chdr.ch_type;
> +	  if (chtype == NONE)
> +	    {
> +	      error (0, 0, "Compression type for section %zd"
> +		     " can't be zero ", ndx);
> +	      chtype = UNSET;
> +	    }
> +	  else if (chtype > MAXIMAL_CH_TYPE)
> +	    {
> +	      error (0, 0, "Compression type (%d) for section %zd"
> +		     " is unsupported ", chtype, ndx);
> +	      chtype = UNSET;
> +	    }
> +	}
> +      else
> +	error (0, 0, "Couldn't get chdr for section %zd", ndx);

Shouldn't this error be fatal?

> +    }
> +  /* Set ZLIB_GNU compression manually for .zdebug* sections.  */
> +  else if (startswith (sname, ".zdebug"))
> +    chtype = ZLIB_GNU;
> +  else
> +    chtype = NONE;
> +
> +  return chtype;
> +}
> +
>  static int
>  process_file (const char *fname)
>  {
> @@ -461,26 +506,29 @@ process_file (const char *fname)
>        if (section_name_matches (sname))
>  	{
> -	  if (!force && type == NONE
> -	      && (shdr->sh_flags & SHF_COMPRESSED) == 0
> -	      && !startswith (sname, ".zdebug"))
> -	    {
> -	      if (verbose > 0)
> -		printf ("[%zd] %s already decompressed\n", ndx, sname);
> -	    }
> -	  else if (!force && type == ZLIB
> -		   && (shdr->sh_flags & SHF_COMPRESSED) != 0)
> -	    {
> -	      if (verbose > 0)
> -		printf ("[%zd] %s already compressed\n", ndx, sname);
> -	    }
> -	  else if (!force && type == ZLIB_GNU
> -		   && startswith (sname, ".zdebug"))
> +	  enum ch_type schtype = get_section_chtype (scn, shdr, sname, ndx);
> +	  if (!force && verbose > 0)
>  	    {
> -	      if (verbose > 0)
> -		printf ("[%zd] %s already GNU compressed\n", ndx, sname);
> +	      /* The current compression matches the final one.  */
> +	      if (type == schtype)
> +		switch (type)
> +		  {
> +		  case NONE:
> +		    printf ("[%zd] %s already decompressed\n", ndx, sname);
> +		    break;
> +		  case ZLIB:
> +		  case ZSTD:
> +		    printf ("[%zd] %s already compressed\n", ndx, sname);
> +		    break;
> +		  case ZLIB_GNU:
> +		    printf ("[%zd] %s already GNU compressed\n", ndx, sname);
> +		    break;
> +		  default:
> +		    abort ();
> +		  }
>  	    }
> -	  else if (shdr->sh_type != SHT_NOBITS
> +
> +	  if (shdr->sh_type != SHT_NOBITS
>  	      && (shdr->sh_flags & SHF_ALLOC) == 0)
>  	    {
>  	      set_section (sections, ndx);
> @@ -692,37 +740,12 @@ process_file (const char *fname)
>  	     (de)compressed, invalidating the string pointers.  */
>  	  sname = xstrdup (sname);
> +
>  	  /* Detect source compression that is how is the section compressed
>  	     now.  */
> -	  GElf_Chdr chdr;
> -	  enum ch_type schtype = NONE;
> -	  if ((shdr->sh_flags & SHF_COMPRESSED) != 0)
> -	    {
> -	      if (gelf_getchdr (scn, &chdr) != NULL)
> -		{
> -		  schtype = (enum ch_type)chdr.ch_type;
> -		  if (schtype == NONE)
> -		    {
> -		      error (0, 0, "Compression type for section %zd"
> -			     " can't be zero ", ndx);
> -		      goto cleanup;
> -		    }
> -		  else if (schtype > MAXIMAL_CH_TYPE)
> -		    {
> -		      error (0, 0, "Compression type (%d) for section %zd"
> -			     " is unsupported ", schtype, ndx);
> -		      goto cleanup;
> -		    }
> -		}
> -	      else
> -		{
> -		  error (0, 0, "Couldn't get chdr for section %zd", ndx);
> -		  goto cleanup;
> -		}
> -	    }
> -	  /* Set ZLIB compression manually for .zdebug* sections.  */
> -	  else if (startswith (sname, ".zdebug"))
> -	    schtype = ZLIB_GNU;
> +	  enum ch_type schtype = get_section_chtype (scn, shdr, sname, ndx);
> +	  if (schtype == UNSET)
> +	    goto cleanup;
>  	  /* We might want to decompress (and rename), but not
>  	     compress during this pass since we might need the section
> @@ -754,7 +777,7 @@ process_file (const char *fname)
>  	    case ZLIB_GNU:
>  	      if (startswith (sname, ".debug"))
>  		{
> -		  if (schtype == ZLIB)
> +		  if (schtype == ZLIB || schtype == ZSTD)
>  		    {
>  		      /* First decompress to recompress GNU style.
>  			 Don't report even when verbose.  */

OK.

> @@ -818,19 +841,22 @@ process_file (const char *fname)
>  	      break;
>  	    case ZLIB:
> -	      if ((shdr->sh_flags & SHF_COMPRESSED) == 0)
> +	    case ZSTD:
> +	      if (schtype != type)
>  		{
> -		  if (schtype == ZLIB_GNU)
> +		  if (schtype != NONE)
>  		    {
> -		      /* First decompress to recompress zlib style.
> -			 Don't report even when verbose.  */
> +		      /* Decompress first.  */
>  		      if (compress_section (scn, size, sname, NULL, ndx,
>  					    schtype, NONE, false) < 0)
>  			goto cleanup;
> -		      snamebuf[0] = '.';
> -		      strcpy (&snamebuf[1], &sname[2]);
> -		      newname = snamebuf;
> +		      if (schtype == ZLIB_GNU)
> +			{
> +			  snamebuf[0] = '.';
> +			  strcpy (&snamebuf[1], &sname[2]);
> +			  newname = snamebuf;
> +			}
>  		    }
>  		  if (skip_compress_section)

OK.

> @@ -838,7 +864,7 @@ process_file (const char *fname)
>  		      if (ndx == shdrstrndx)
>  			{
>  			  shstrtab_size = size;
> -			  shstrtab_compressed = ZLIB;
> +			  shstrtab_compressed = type;
>  			  if (shstrtab_name != NULL
>  			      || shstrtab_newname != NULL)
>  			    {
> @@ -855,7 +881,7 @@ process_file (const char *fname)
>  		      else
>  			{
>  			  symtab_size = size;
> -			  symtab_compressed = ZLIB;
> +			  symtab_compressed = type;
>  			  symtab_name = xstrdup (sname);
>  			  symtab_newname = (newname == NULL
>  					    ? NULL : xstrdup (newname));

OK.

> @@ -1378,7 +1404,7 @@ main (int argc, char **argv)
>  	N_("Place (de)compressed output into FILE"),
>  	0 },
>        { "type", 't', "TYPE", 0,
> -	N_("What type of compression to apply. TYPE can be 'none' (decompress), 'zlib' (ELF ZLIB compression, the default, 'zlib-gabi' is an alias) or 'zlib-gnu' (.zdebug GNU style compression, 'gnu' is an alias)"),
> +	N_("What type of compression to apply. TYPE can be 'none' (decompress), 'zlib' (ELF ZLIB compression, the default, 'zlib-gabi' is an alias), 'zlib-gnu' (.zdebug GNU style compression, 'gnu' is an alias) or 'zstd'"),
>  	0 },
>        { "name", 'n', "SECTION", 0,
>  	N_("SECTION name to (de)compress, SECTION is an extended wildcard pattern (defaults to '.?(z)debug*')"),

I would say or 'zstd' (ELF ZSTD compression)" to match the 'zlib; type
description.

> diff --git a/src/readelf.c b/src/readelf.c
> index cc3e0229..451f8400 100644
> --- a/src/readelf.c
> +++ b/src/readelf.c
> @@ -1238,13 +1238,17 @@ get_visibility_type (int value)
>  static const char *
>  elf_ch_type_name (unsigned int code)
>  {
> -  if (code == 0)
> -    return "NONE";
> -
> -  if (code == ELFCOMPRESS_ZLIB)
> -    return "ZLIB";
> -
> -  return "UNKNOWN";
> +  switch (code)
> +    {
> +    case 0:
> +      return "NONE";
> +    case ELFCOMPRESS_ZLIB:
> +      return "ZLIB";
> +    case ELFCOMPRESS_ZSTD:
> +      return "ZSTD";
> +    default:
> +      return "UNKNOWN";
> +    }
>  }
>  /* Print the section headers.  */

OK.

> diff --git a/tests/run-compress-test.sh b/tests/run-compress-test.sh
> index a6a298f5..3f9c990e 100755
> --- a/tests/run-compress-test.sh
> +++ b/tests/run-compress-test.sh
> @@ -61,6 +61,30 @@ testrun_elfcompress_file()
>      echo "uncompress $elfcompressedfile -> $elfuncompressedfile"
>      testrun ${abs_top_builddir}/src/elfcompress -v -t none -o ${elfuncompressedfile} ${elfcompressedfile}
>      testrun ${abs_top_builddir}/src/elfcmp ${uncompressedfile} ${elfuncompressedfile}
> +
> +    outputfile="${infile}.gabi.zstd"
> +    tempfiles "$outputfile"
> +    echo "zstd compress $elfcompressedfile -> $outputfile"
> +    testrun ${abs_top_builddir}/src/elfcompress -v -t zstd -o ${outputfile} ${elfcompressedfile}
> +    testrun ${abs_top_builddir}/src/elfcmp ${uncompressedfile} ${outputfile}
> +    echo "checking compressed section header" $outputfile
> +    testrun ${abs_top_builddir}/src/readelf -Sz ${outputfile} | grep "ELF ZSTD" >/dev/null
> +
> +    zstdfile="${infile}.zstd"
> +    tempfiles "$zstdfile"
> +    echo "zstd compress $uncompressedfile -> $zstdfile"
> +    testrun ${abs_top_builddir}/src/elfcompress -v -t zstd -o ${zstdfile} ${elfuncompressedfile}
> +    testrun ${abs_top_builddir}/src/elfcmp ${uncompressedfile} ${zstdfile}
> +    echo "checking compressed section header" $zstdfile
> +    testrun ${abs_top_builddir}/src/readelf -Sz ${zstdfile} | grep "ELF ZSTD" >/dev/null
> +
> +    zstdgnufile="${infile}.zstd.gnu"
> +    tempfiles "$zstdgnufile"
> +    echo "zstd re-compress to GNU ZLIB $zstdfile -> $zstdgnufile"
> +    testrun ${abs_top_builddir}/src/elfcompress -v -t zlib-gnu -o ${zstdgnufile} ${zstdfile}
> +    testrun ${abs_top_builddir}/src/elfcmp ${uncompressedfile} ${zstdgnufile}
> +    echo "checking .zdebug section name" $zstdgnufile
> +    testrun ${abs_top_builddir}/src/readelf -S ${zstdgnufile} | grep ".zdebug" >/dev/null
>  }
>  testrun_elfcompress()

You might add these to a separate run test file or pass some ZSTD flag
through the test environment to conditionally run these new tests.

Cheers,

Mark


More information about the Elfutils-devel mailing list