Coverage Report

Created: 2026-03-03 06:43

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/cryptsetup/lib/utils_io.c
Line
Count
Source
1
// SPDX-License-Identifier: GPL-2.0-or-later
2
/*
3
 * utils - miscellaneous I/O utilities for cryptsetup
4
 *
5
 * Copyright (C) 2004 Jana Saout <jana@saout.de>
6
 * Copyright (C) 2004-2007 Clemens Fruhwirth <clemens@endorphin.org>
7
 * Copyright (C) 2009-2025 Red Hat, Inc. All rights reserved.
8
 * Copyright (C) 2009-2025 Milan Broz
9
 */
10
11
#include <errno.h>
12
#include <limits.h>
13
#include <string.h>
14
#include <stdlib.h>
15
#include <stdint.h>
16
#include <unistd.h>
17
18
#include "utils_io.h"
19
20
/* coverity[ -taint_source : arg-1 ] */
21
static ssize_t _read_buffer(int fd, void *buf, size_t length, volatile int *quit)
22
61.9k
{
23
61.9k
  ssize_t r, read_size = 0;
24
25
61.9k
  if (fd < 0 || !buf || length > SSIZE_MAX)
26
0
    return -EINVAL;
27
28
61.9k
  do {
29
61.9k
    r = read(fd, buf, length - read_size);
30
61.9k
    if (r == -1 && errno != EINTR)
31
0
      return r;
32
61.9k
    if (r > 0) {
33
      /* coverity[overflow:FALSE] */
34
51.2k
      read_size += r;
35
51.2k
      buf = (uint8_t*)buf + r;
36
51.2k
    }
37
61.9k
    if (r == 0 || (quit && *quit))
38
10.7k
      return read_size;
39
61.9k
  } while ((size_t)read_size != length);
40
41
51.2k
  return (ssize_t)length;
42
61.9k
}
43
44
ssize_t read_buffer(int fd, void *buf, size_t length)
45
61.9k
{
46
61.9k
  return _read_buffer(fd, buf, length, NULL);
47
61.9k
}
48
49
ssize_t read_buffer_intr(int fd, void *buf, size_t length, volatile int *quit)
50
0
{
51
0
  return _read_buffer(fd, buf, length, quit);
52
0
}
53
54
static ssize_t _write_buffer(int fd, const void *buf, size_t length, volatile int *quit)
55
10.7k
{
56
10.7k
  ssize_t w, write_size = 0;
57
58
10.7k
  if (fd < 0 || !buf || !length || length > SSIZE_MAX)
59
0
    return -EINVAL;
60
61
10.7k
  do {
62
10.7k
    w = write(fd, buf, length - (size_t)write_size);
63
10.7k
    if (w < 0 && errno != EINTR)
64
0
      return w;
65
10.7k
    if (w > 0) {
66
      /* coverity[overflow:FALSE] */
67
10.7k
      write_size += w;
68
10.7k
      buf = (const uint8_t*)buf + w;
69
10.7k
    }
70
10.7k
    if (w == 0 || (quit && *quit))
71
0
      return write_size;
72
10.7k
  } while ((size_t)write_size != length);
73
74
10.7k
  return write_size;
75
10.7k
}
76
77
ssize_t write_buffer(int fd, const void *buf, size_t length)
78
10.7k
{
79
10.7k
  return _write_buffer(fd, buf, length, NULL);
80
10.7k
}
81
82
ssize_t write_buffer_intr(int fd, const void *buf, size_t length, volatile int *quit)
83
0
{
84
0
  return _write_buffer(fd, buf, length, quit);
85
0
}
86
87
ssize_t write_blockwise(int fd, size_t bsize, size_t alignment,
88
      void *orig_buf, size_t length)
89
5.50k
{
90
5.50k
  void *hangover_buf = NULL, *buf = NULL;
91
5.50k
  size_t hangover, solid;
92
5.50k
  ssize_t r, ret = -1;
93
94
5.50k
  if (fd == -1 || !orig_buf || !bsize || !alignment)
95
0
    return -1;
96
97
5.50k
  hangover = length % bsize;
98
5.50k
  solid = length - hangover;
99
100
5.50k
  if ((size_t)orig_buf & (alignment - 1)) {
101
5.49k
    if (posix_memalign(&buf, alignment, length))
102
0
      return -1;
103
5.49k
    memcpy(buf, orig_buf, length);
104
5.49k
  } else
105
3
    buf = orig_buf;
106
107
5.50k
  if (solid) {
108
5.50k
    r = write_buffer(fd, buf, solid);
109
5.50k
    if (r < 0 || r != (ssize_t)solid)
110
0
      goto out;
111
5.50k
  }
112
113
5.50k
  if (hangover) {
114
0
    if (posix_memalign(&hangover_buf, alignment, bsize))
115
0
      goto out;
116
0
    memset(hangover_buf, 0, bsize);
117
118
0
    r = read_buffer(fd, hangover_buf, bsize);
119
0
    if (r < 0)
120
0
      goto out;
121
122
0
    if (lseek(fd, -(off_t)r, SEEK_CUR) < 0)
123
0
      goto out;
124
125
0
    memcpy(hangover_buf, (char*)buf + solid, hangover);
126
127
0
    r = write_buffer(fd, hangover_buf, bsize);
128
0
    if (r < 0 || r < (ssize_t)hangover)
129
0
      goto out;
130
0
  }
131
5.50k
  ret = length;
132
5.50k
out:
133
5.50k
  free(hangover_buf);
134
5.50k
  if (buf != orig_buf)
135
5.49k
    free(buf);
136
5.50k
  return ret;
137
5.50k
}
138
139
ssize_t read_blockwise(int fd, size_t bsize, size_t alignment,
140
           void *orig_buf, size_t length)
141
50.0k
{
142
50.0k
  void *hangover_buf = NULL, *buf = NULL;
143
50.0k
  size_t hangover, solid;
144
50.0k
  ssize_t r, ret = -1;
145
146
50.0k
  if (fd == -1 || !orig_buf || !bsize || !alignment)
147
0
    return -1;
148
149
50.0k
  hangover = length % bsize;
150
50.0k
  solid = length - hangover;
151
152
50.0k
  if ((size_t)orig_buf & (alignment - 1)) {
153
49.0k
    if (posix_memalign(&buf, alignment, length))
154
0
      return -1;
155
49.0k
  } else
156
1.00k
    buf = orig_buf;
157
158
50.0k
  r = read_buffer(fd, buf, solid);
159
50.0k
  if (r < 0 || r != (ssize_t)solid)
160
0
    goto out;
161
162
50.0k
  if (hangover) {
163
11.6k
    if (posix_memalign(&hangover_buf, alignment, bsize))
164
0
      goto out;
165
11.6k
    r = read_buffer(fd, hangover_buf, bsize);
166
11.6k
    if (r <  0 || r < (ssize_t)hangover)
167
0
      goto out;
168
169
11.6k
    memcpy((char *)buf + solid, hangover_buf, hangover);
170
11.6k
  }
171
50.0k
  ret = length;
172
50.0k
out:
173
50.0k
  free(hangover_buf);
174
50.0k
  if (buf != orig_buf) {
175
49.0k
    if (ret != -1)
176
49.0k
      memcpy(orig_buf, buf, length);
177
49.0k
    free(buf);
178
49.0k
  }
179
50.0k
  return ret;
180
50.0k
}
181
182
/*
183
 * Combines llseek with blockwise write. write_blockwise can already deal with short writes
184
 * but we also need a function to deal with short writes at the start. But this information
185
 * is implicitly included in the read/write offset, which can not be set to non-aligned
186
 * boundaries. Hence, we combine llseek with write.
187
 */
188
ssize_t write_lseek_blockwise(int fd, size_t bsize, size_t alignment,
189
            void *buf, size_t length, off_t offset)
190
5.50k
{
191
5.50k
  void *frontPadBuf = NULL;
192
5.50k
  size_t frontHang, innerCount = 0;
193
5.50k
  ssize_t r, ret = -1;
194
195
5.50k
  if (fd == -1 || !buf || !bsize || !alignment)
196
0
    return -1;
197
198
5.50k
  if (offset < 0)
199
0
    offset = lseek(fd, offset, SEEK_END);
200
201
5.50k
  if (offset < 0)
202
0
    return -1;
203
204
5.50k
  frontHang = offset % bsize;
205
206
5.50k
  if (lseek(fd, offset - frontHang, SEEK_SET) < 0)
207
0
    return -1;
208
209
5.50k
  if (frontHang && length) {
210
0
    if (posix_memalign(&frontPadBuf, alignment, bsize))
211
0
      return -1;
212
213
0
    innerCount = bsize - frontHang;
214
0
    if (innerCount > length)
215
0
      innerCount = length;
216
217
0
    r = read_buffer(fd, frontPadBuf, bsize);
218
0
    if (r < 0 || r < (ssize_t)(frontHang + innerCount))
219
0
      goto out;
220
221
0
    memcpy((char*)frontPadBuf + frontHang, buf, innerCount);
222
223
0
    if (lseek(fd, offset - frontHang, SEEK_SET) < 0)
224
0
      goto out;
225
226
0
    r = write_buffer(fd, frontPadBuf, bsize);
227
0
    if (r < 0 || r != (ssize_t)bsize)
228
0
      goto out;
229
230
0
    buf = (char*)buf + innerCount;
231
0
    length -= innerCount;
232
0
  }
233
234
5.50k
  ret = length ? write_blockwise(fd, bsize, alignment, buf, length) : 0;
235
5.50k
  if (ret >= 0)
236
5.50k
    ret += innerCount;
237
5.50k
out:
238
5.50k
  free(frontPadBuf);
239
5.50k
  return ret;
240
5.50k
}
241
242
ssize_t read_lseek_blockwise(int fd, size_t bsize, size_t alignment,
243
           void *buf, size_t length, off_t offset)
244
44.8k
{
245
44.8k
  void *frontPadBuf = NULL;
246
44.8k
  size_t frontHang, innerCount = 0;
247
44.8k
  ssize_t r, ret = -1;
248
249
44.8k
  if (fd == -1 || !buf || bsize <= 0)
250
0
    return -1;
251
252
44.8k
  if (offset < 0)
253
0
    offset = lseek(fd, offset, SEEK_END);
254
255
44.8k
  if (offset < 0)
256
0
    return -1;
257
258
44.8k
  frontHang = offset % bsize;
259
260
44.8k
  if (lseek(fd, offset - frontHang, SEEK_SET) < 0)
261
0
    return -1;
262
263
44.8k
  if (frontHang && length) {
264
196
    if (posix_memalign(&frontPadBuf, alignment, bsize))
265
0
      return -1;
266
267
196
    innerCount = bsize - frontHang;
268
196
    if (innerCount > length)
269
0
      innerCount = length;
270
271
196
    r = read_buffer(fd, frontPadBuf, bsize);
272
196
    if (r < 0 || r < (ssize_t)(frontHang + innerCount))
273
0
      goto out;
274
275
196
    memcpy(buf, (char*)frontPadBuf + frontHang, innerCount);
276
277
196
    buf = (char*)buf + innerCount;
278
196
    length -= innerCount;
279
196
  }
280
281
44.8k
  ret = length ? read_blockwise(fd, bsize, alignment, buf, length) : 0;
282
44.8k
  if (ret >= 0)
283
44.8k
    ret += innerCount;
284
44.8k
out:
285
44.8k
  free(frontPadBuf);
286
44.8k
  return ret;
287
44.8k
}