Coverage Report

Created: 2026-05-30 06:18

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
138k
{
40
138k
  if (scn == NULL)
41
0
    return -1;
42
43
138k
  if ((flags & ~ELF_CHF_FORCE) != 0)
44
0
    {
45
0
      __libelf_seterrno (ELF_E_INVALID_OPERAND);
46
0
      return -1;
47
0
    }
48
49
138k
  bool force = (flags & ELF_CHF_FORCE) != 0;
50
51
138k
  Elf *elf = scn->elf;
52
138k
  GElf_Ehdr ehdr;
53
138k
  if (gelf_getehdr (elf, &ehdr) == NULL)
54
0
    return -1;
55
56
138k
  int elfclass = elf->class;
57
138k
  int elfdata = ehdr.e_ident[EI_DATA];
58
59
138k
  Elf64_Xword sh_flags;
60
138k
  Elf64_Word sh_type;
61
138k
  Elf64_Xword sh_addralign;
62
138k
  union shdr
63
138k
  {
64
138k
    Elf32_Shdr *s32;
65
138k
    Elf64_Shdr *s64;
66
138k
  } shdr;
67
138k
  if (elfclass == ELFCLASS32)
68
123k
    {
69
123k
      shdr.s32 = elf32_getshdr (scn);
70
123k
      if (shdr.s32 == NULL)
71
0
  return -1;
72
73
123k
      sh_flags = shdr.s32->sh_flags;
74
123k
      sh_type = shdr.s32->sh_type;
75
123k
      sh_addralign = shdr.s32->sh_addralign;
76
123k
    }
77
15.1k
  else
78
15.1k
    {
79
15.1k
      shdr.s64 = elf64_getshdr (scn);
80
15.1k
      if (shdr.s64 == NULL)
81
0
  return -1;
82
83
15.1k
      sh_flags = shdr.s64->sh_flags;
84
15.1k
      sh_type = shdr.s64->sh_type;
85
15.1k
      sh_addralign = shdr.s64->sh_addralign;
86
15.1k
    }
87
88
  /* Allocated sections, or sections that are already are compressed
89
     cannot (also) be GNU compressed.  */
90
138k
  if ((sh_flags & SHF_ALLOC) != 0 || (sh_flags & SHF_COMPRESSED))
91
13.4k
    {
92
13.4k
      __libelf_seterrno (ELF_E_INVALID_SECTION_FLAGS);
93
13.4k
      return -1;
94
13.4k
    }
95
96
125k
  if (sh_type == SHT_NULL || sh_type == SHT_NOBITS)
97
53.3k
    {
98
53.3k
      __libelf_seterrno (ELF_E_INVALID_SECTION_TYPE);
99
53.3k
      return -1;
100
53.3k
    }
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
71.8k
  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
71.8k
  else if (inflate == 0)
145
71.8k
    {
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
71.8k
      Elf_Data *data = elf_getdata (scn, NULL);
151
71.8k
      if (data == NULL)
152
31.2k
  return -1;
153
154
40.6k
      size_t hsize = 4 + 8; /* GNU "ZLIB" + 8 byte size.  */
155
40.6k
      if (data->d_size < hsize || memcmp (data->d_buf, "ZLIB", 4) != 0)
156
38.8k
  {
157
38.8k
          __libelf_seterrno (ELF_E_NOT_COMPRESSED);
158
38.8k
    return -1;
159
38.8k
  }
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
1.75k
      uint64_t gsize;
165
1.75k
      memcpy (&gsize, data->d_buf + 4, sizeof gsize);
166
1.75k
      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
1.75k
      if (gsize + 4 + 8 + 6 + 5 < data->d_size
174
1.54k
    || gsize > SIZE_MAX
175
1.54k
    || (elfclass == ELFCLASS32 && gsize > UINT32_MAX))
176
752
  {
177
752
    __libelf_seterrno (ELF_E_NOT_COMPRESSED);
178
752
    return -1;
179
752
  }
180
181
1.00k
      size_t size = gsize;
182
1.00k
      size_t size_in = data->d_size - hsize;
183
1.00k
      void *buf_in = data->d_buf + hsize;
184
1.00k
      void *buf_out = __libelf_decompress (ELFCOMPRESS_ZLIB, buf_in, size_in, size);
185
1.00k
      if (buf_out == NULL)
186
574
  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
428
      if (elfclass == ELFCLASS32)
192
100
  shdr.s32->sh_size = size;
193
328
      else
194
328
  shdr.s64->sh_size = size;
195
196
428
      __libelf_reset_rawdata (scn, buf_out, size, sh_addralign,
197
428
            __libelf_data_type (&ehdr, sh_type,
198
428
              sh_addralign));
199
200
428
      scn->zdata_base = buf_out;
201
202
428
      return 1;
203
1.00k
    }
204
0
  else
205
0
    {
206
0
      __libelf_seterrno (ELF_E_UNKNOWN_COMPRESSION_TYPE);
207
0
      return -1;
208
0
    }
209
71.8k
}