Coverage Report

Created: 2026-05-30 06:17

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/elfutils/libelf/elf_compress_gnu.c
Line
Count
Source
1
/* Compress or decompress a section.
2
   Copyright (C) 2015 Red Hat, Inc.
3
   This file is part of elfutils.
4
5
   This file is free software; you can redistribute it and/or modify
6
   it under the terms of either
7
8
     * the GNU Lesser General Public License as published by the Free
9
       Software Foundation; either version 3 of the License, or (at
10
       your option) any later version
11
12
   or
13
14
     * the GNU General Public License as published by the Free
15
       Software Foundation; either version 2 of the License, or (at
16
       your option) any later version
17
18
   or both in parallel, as here.
19
20
   elfutils is distributed in the hope that it will be useful, but
21
   WITHOUT ANY WARRANTY; without even the implied warranty of
22
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
23
   General Public License for more details.
24
25
   You should have received copies of the GNU General Public License and
26
   the GNU Lesser General Public License along with this program.  If
27
   not, see <http://www.gnu.org/licenses/>.  */
28
29
#ifdef HAVE_CONFIG_H
30
# include <config.h>
31
#endif
32
33
#include <libelf.h>
34
#include "libelfP.h"
35
#include "common.h"
36
37
int
38
elf_compress_gnu (Elf_Scn *scn, int inflate, unsigned int flags)
39
0
{
40
0
  if (scn == NULL)
41
0
    return -1;
42
43
0
  if ((flags & ~ELF_CHF_FORCE) != 0)
44
0
    {
45
0
      __libelf_seterrno (ELF_E_INVALID_OPERAND);
46
0
      return -1;
47
0
    }
48
49
0
  bool force = (flags & ELF_CHF_FORCE) != 0;
50
51
0
  Elf *elf = scn->elf;
52
0
  GElf_Ehdr ehdr;
53
0
  if (gelf_getehdr (elf, &ehdr) == NULL)
54
0
    return -1;
55
56
0
  int elfclass = elf->class;
57
0
  int elfdata = ehdr.e_ident[EI_DATA];
58
59
0
  Elf64_Xword sh_flags;
60
0
  Elf64_Word sh_type;
61
0
  Elf64_Xword sh_addralign;
62
0
  union shdr
63
0
  {
64
0
    Elf32_Shdr *s32;
65
0
    Elf64_Shdr *s64;
66
0
  } shdr;
67
0
  if (elfclass == ELFCLASS32)
68
0
    {
69
0
      shdr.s32 = elf32_getshdr (scn);
70
0
      if (shdr.s32 == NULL)
71
0
  return -1;
72
73
0
      sh_flags = shdr.s32->sh_flags;
74
0
      sh_type = shdr.s32->sh_type;
75
0
      sh_addralign = shdr.s32->sh_addralign;
76
0
    }
77
0
  else
78
0
    {
79
0
      shdr.s64 = elf64_getshdr (scn);
80
0
      if (shdr.s64 == NULL)
81
0
  return -1;
82
83
0
      sh_flags = shdr.s64->sh_flags;
84
0
      sh_type = shdr.s64->sh_type;
85
0
      sh_addralign = shdr.s64->sh_addralign;
86
0
    }
87
88
  /* Allocated sections, or sections that are already are compressed
89
     cannot (also) be GNU compressed.  */
90
0
  if ((sh_flags & SHF_ALLOC) != 0 || (sh_flags & SHF_COMPRESSED))
91
0
    {
92
0
      __libelf_seterrno (ELF_E_INVALID_SECTION_FLAGS);
93
0
      return -1;
94
0
    }
95
96
0
  if (sh_type == SHT_NULL || sh_type == SHT_NOBITS)
97
0
    {
98
0
      __libelf_seterrno (ELF_E_INVALID_SECTION_TYPE);
99
0
      return -1;
100
0
    }
101
102
  /* For GNU compression we cannot really know whether the section is
103
     already compressed or not.  Just try and see what happens...  */
104
  // int compressed = (sh_flags & SHF_COMPRESSED);
105
0
  if (inflate == 1)
106
0
    {
107
0
      size_t hsize = 4 + 8; /* GNU "ZLIB" + 8 byte size.  */
108
0
      size_t orig_size, new_size, orig_addralign;
109
0
      void *out_buf = __libelf_compress (scn, hsize, elfdata,
110
0
           &orig_size, &orig_addralign,
111
0
           &new_size, force,
112
0
           /* use_zstd */ false);
113
114
      /* Compression would make section larger, don't change anything.  */
115
0
      if (out_buf == (void *) -1)
116
0
  return 0;
117
118
      /* Compression failed, return error.  */
119
0
      if (out_buf == NULL)
120
0
  return -1;
121
122
0
      uint64_t be64_size = htobe64 (orig_size);
123
0
      memmove (out_buf, "ZLIB", 4);
124
0
      memmove (out_buf + 4, &be64_size, sizeof (be64_size));
125
126
      /* We don't know anything about sh_entsize, sh_addralign and
127
   sh_flags won't have a SHF_COMPRESSED hint in the GNU format.
128
   Just adjust the sh_size.  */
129
0
      if (elfclass == ELFCLASS32)
130
0
    shdr.s32->sh_size = new_size;
131
0
      else
132
0
    shdr.s64->sh_size = new_size;
133
134
0
      __libelf_reset_rawdata (scn, out_buf, new_size, 1, ELF_T_BYTE);
135
136
      /* The section is now compressed, we could keep the uncompressed
137
   data around, but since that might have been multiple Elf_Data
138
   buffers let the user uncompress it explicitly again if they
139
   want it to simplify bookkeeping.  */
140
0
      scn->zdata_base = NULL;
141
142
0
      return 1;
143
0
    }
144
0
  else if (inflate == 0)
145
0
    {
146
      /* In theory the user could have constructed a compressed section
147
   by hand.  And in practice they do. For example when copying
148
   a section from one file to another using elf_newdata. So we
149
   have to use elf_getdata (not elf_rawdata).  */
150
0
      Elf_Data *data = elf_getdata (scn, NULL);
151
0
      if (data == NULL)
152
0
  return -1;
153
154
0
      size_t hsize = 4 + 8; /* GNU "ZLIB" + 8 byte size.  */
155
0
      if (data->d_size < hsize || memcmp (data->d_buf, "ZLIB", 4) != 0)
156
0
  {
157
0
          __libelf_seterrno (ELF_E_NOT_COMPRESSED);
158
0
    return -1;
159
0
  }
160
161
      /* There is a 12-byte header of "ZLIB" followed by
162
   an 8-byte big-endian size.  There is only one type and
163
   Alignment isn't preserved separately.  */
164
0
      uint64_t gsize;
165
0
      memcpy (&gsize, data->d_buf + 4, sizeof gsize);
166
0
      gsize = be64toh (gsize);
167
168
      /* One more sanity check, size should be bigger than original
169
   data size plus some overhead (4 chars ZLIB + 8 bytes size + 6
170
   bytes zlib stream overhead + 5 bytes overhead max for one 16K
171
   block) and should fit into a size_t (or in UINT32_MAX for
172
   32bit ELF).  */
173
0
      if (gsize + 4 + 8 + 6 + 5 < data->d_size
174
0
    || gsize > SIZE_MAX
175
0
    || (elfclass == ELFCLASS32 && gsize > UINT32_MAX))
176
0
  {
177
0
    __libelf_seterrno (ELF_E_NOT_COMPRESSED);
178
0
    return -1;
179
0
  }
180
181
0
      size_t size = gsize;
182
0
      size_t size_in = data->d_size - hsize;
183
0
      void *buf_in = data->d_buf + hsize;
184
0
      void *buf_out = __libelf_decompress (ELFCOMPRESS_ZLIB, buf_in, size_in, size);
185
0
      if (buf_out == NULL)
186
0
  return -1;
187
188
      /* We don't know anything about sh_entsize, sh_addralign and
189
   sh_flags won't have a SHF_COMPRESSED hint in the GNU format.
190
   Just adjust the sh_size.  */
191
0
      if (elfclass == ELFCLASS32)
192
0
  shdr.s32->sh_size = size;
193
0
      else
194
0
  shdr.s64->sh_size = size;
195
196
0
      __libelf_reset_rawdata (scn, buf_out, size, sh_addralign,
197
0
            __libelf_data_type (&ehdr, sh_type,
198
0
              sh_addralign));
199
200
0
      scn->zdata_base = buf_out;
201
202
0
      return 1;
203
0
    }
204
0
  else
205
0
    {
206
0
      __libelf_seterrno (ELF_E_UNKNOWN_COMPRESSION_TYPE);
207
0
      return -1;
208
0
    }
209
0
}