Coverage Report

Created: 2026-02-26 06:50

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/winpr/libwinpr/utils/stream.c
Line
Count
Source
1
/*
2
 * WinPR: Windows Portable Runtime
3
 * Stream Utils
4
 *
5
 * Copyright 2011 Vic Lee
6
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
7
 *
8
 * Licensed under the Apache License, Version 2.0 (the "License");
9
 * you may not use this file except in compliance with the License.
10
 * You may obtain a copy of the License at
11
 *
12
 *     http://www.apache.org/licenses/LICENSE-2.0
13
 *
14
 * Unless required by applicable law or agreed to in writing, software
15
 * distributed under the License is distributed on an "AS IS" BASIS,
16
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
 * See the License for the specific language governing permissions and
18
 * limitations under the License.
19
 */
20
21
#include <winpr/config.h>
22
23
#include <winpr/assert.h>
24
#include <winpr/crt.h>
25
#include <winpr/stream.h>
26
27
#include "stream.h"
28
#include "../log.h"
29
30
0
#define STREAM_TAG WINPR_TAG("wStream")
31
32
#define STREAM_ASSERT(cond)                                                            \
33
0
  do                                                                                 \
34
0
  {                                                                                  \
35
0
    if (!(cond))                                                                   \
36
0
    {                                                                              \
37
0
      WLog_FATAL(STREAM_TAG, "%s [%s:%s:%" PRIuz "]", #cond, __FILE__, __func__, \
38
0
                 (size_t)__LINE__);                                              \
39
0
      winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);                           \
40
0
      abort();                                                                   \
41
0
    }                                                                              \
42
0
  } while (0)
43
44
BOOL Stream_EnsureCapacity(wStream* s, size_t size)
45
0
{
46
0
  WINPR_ASSERT(s);
47
0
  if (s->capacity >= size)
48
0
    return TRUE;
49
50
0
  const size_t increment = 128ull;
51
0
  const size_t old_capacity = s->capacity;
52
0
  const size_t new_capacity = size + increment - size % increment;
53
0
  const size_t position = Stream_GetPosition(s);
54
55
0
  BYTE* new_buf = NULL;
56
0
  if (!s->isOwner)
57
0
  {
58
0
    new_buf = (BYTE*)malloc(new_capacity);
59
0
    if (!new_buf)
60
0
      return FALSE;
61
62
0
    CopyMemory(new_buf, s->buffer, s->capacity);
63
0
    s->isOwner = TRUE;
64
0
  }
65
0
  else
66
0
  {
67
0
    new_buf = (BYTE*)realloc(s->buffer, new_capacity);
68
0
    if (!new_buf)
69
0
      return FALSE;
70
0
  }
71
72
0
  s->buffer = new_buf;
73
0
  s->capacity = new_capacity;
74
0
  s->length = new_capacity;
75
0
  ZeroMemory(&s->buffer[old_capacity], s->capacity - old_capacity);
76
77
0
  return Stream_SetPosition(s, position);
78
0
}
79
80
BOOL Stream_EnsureRemainingCapacity(wStream* s, size_t size)
81
0
{
82
0
  if (Stream_GetPosition(s) + size > Stream_Capacity(s))
83
0
    return Stream_EnsureCapacity(s, Stream_Capacity(s) + size);
84
0
  return TRUE;
85
0
}
86
87
wStream* Stream_New(BYTE* buffer, size_t size)
88
0
{
89
0
  wStream* s = NULL;
90
91
0
  if (!buffer && !size)
92
0
    return NULL;
93
94
0
  s = calloc(1, sizeof(wStream));
95
0
  if (!s)
96
0
    return NULL;
97
98
0
  if (buffer)
99
0
    s->buffer = buffer;
100
0
  else
101
0
    s->buffer = (BYTE*)malloc(size);
102
103
0
  if (!s->buffer)
104
0
  {
105
0
    free(s);
106
0
    return NULL;
107
0
  }
108
109
0
  s->pointer = s->buffer;
110
0
  s->capacity = size;
111
0
  s->length = size;
112
113
0
  s->pool = NULL;
114
0
  s->count = 1;
115
0
  s->isAllocatedStream = TRUE;
116
0
  s->isOwner = TRUE;
117
0
  return s;
118
0
}
119
120
wStream* Stream_StaticConstInit(wStream* s, const BYTE* buffer, size_t size)
121
0
{
122
0
  union
123
0
  {
124
0
    BYTE* b;
125
0
    const BYTE* cb;
126
0
  } cnv;
127
128
0
  cnv.cb = buffer;
129
0
  return Stream_StaticInit(s, cnv.b, size);
130
0
}
131
132
wStream* Stream_StaticInit(wStream* s, BYTE* buffer, size_t size)
133
0
{
134
0
  const wStream empty = WINPR_C_ARRAY_INIT;
135
136
0
  WINPR_ASSERT(s);
137
0
  WINPR_ASSERT(buffer);
138
139
0
  *s = empty;
140
0
  s->buffer = s->pointer = buffer;
141
0
  s->capacity = s->length = size;
142
0
  s->pool = NULL;
143
0
  s->count = 1;
144
0
  s->isAllocatedStream = FALSE;
145
0
  s->isOwner = FALSE;
146
0
  return s;
147
0
}
148
149
void Stream_EnsureValidity(wStream* s)
150
0
{
151
0
  size_t cur = 0;
152
153
0
  STREAM_ASSERT(s);
154
0
  STREAM_ASSERT(s->pointer >= s->buffer);
155
156
0
  cur = (size_t)(s->pointer - s->buffer);
157
0
  STREAM_ASSERT(cur <= s->capacity);
158
0
  STREAM_ASSERT(s->length <= s->capacity);
159
0
}
160
161
void Stream_Free(wStream* s, BOOL bFreeBuffer)
162
0
{
163
0
  if (s)
164
0
  {
165
0
    Stream_EnsureValidity(s);
166
0
    if (bFreeBuffer && s->isOwner)
167
0
      free(s->buffer);
168
169
0
    if (s->isAllocatedStream)
170
0
      free(s);
171
0
  }
172
0
}
173
174
BOOL Stream_SetLength(wStream* _s, size_t _l)
175
0
{
176
0
  if ((_l) > Stream_Capacity(_s))
177
0
  {
178
0
    _s->length = 0;
179
0
    return FALSE;
180
0
  }
181
0
  _s->length = _l;
182
0
  return TRUE;
183
0
}
184
185
BOOL Stream_SetPosition(wStream* _s, size_t _p)
186
0
{
187
0
  if ((_p) > Stream_Capacity(_s))
188
0
  {
189
0
    _s->pointer = _s->buffer;
190
0
    return FALSE;
191
0
  }
192
0
  _s->pointer = _s->buffer + (_p);
193
0
  return TRUE;
194
0
}
195
196
void Stream_SealLength(wStream* _s)
197
0
{
198
0
  size_t cur = 0;
199
0
  WINPR_ASSERT(_s);
200
0
  WINPR_ASSERT(_s->buffer <= _s->pointer);
201
0
  cur = (size_t)(_s->pointer - _s->buffer);
202
0
  WINPR_ASSERT(cur <= _s->capacity);
203
0
  if (cur <= _s->capacity)
204
0
    _s->length = cur;
205
0
  else
206
0
  {
207
0
    WLog_FATAL(STREAM_TAG, "wStream API misuse: stream was written out of bounds");
208
0
    winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
209
0
    _s->length = 0;
210
0
  }
211
0
}
212
213
#if defined(WITH_WINPR_DEPRECATED)
214
BOOL Stream_SetPointer(wStream* _s, BYTE* _p)
215
{
216
  WINPR_ASSERT(_s);
217
  if (!_p || (_s->buffer > _p) || (_s->buffer + _s->capacity < _p))
218
  {
219
    _s->pointer = _s->buffer;
220
    return FALSE;
221
  }
222
  _s->pointer = _p;
223
  return TRUE;
224
}
225
226
BOOL Stream_SetBuffer(wStream* _s, BYTE* _b)
227
{
228
  WINPR_ASSERT(_s);
229
  WINPR_ASSERT(_b);
230
231
  _s->buffer = _b;
232
  _s->pointer = _b;
233
  return _s->buffer != NULL;
234
}
235
236
void Stream_SetCapacity(wStream* _s, size_t _c)
237
{
238
  WINPR_ASSERT(_s);
239
  _s->capacity = _c;
240
}
241
242
#endif
243
244
size_t Stream_GetRemainingCapacity(const wStream* _s)
245
0
{
246
0
  size_t cur = 0;
247
0
  WINPR_ASSERT(_s);
248
0
  WINPR_ASSERT(_s->buffer <= _s->pointer);
249
0
  cur = (size_t)(_s->pointer - _s->buffer);
250
0
  WINPR_ASSERT(cur <= _s->capacity);
251
0
  if (cur > _s->capacity)
252
0
  {
253
0
    WLog_FATAL(STREAM_TAG, "wStream API misuse: stream was written out of bounds");
254
0
    winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
255
0
    return 0;
256
0
  }
257
0
  return (_s->capacity - cur);
258
0
}
259
260
size_t Stream_GetRemainingLength(const wStream* _s)
261
0
{
262
0
  size_t cur = 0;
263
0
  WINPR_ASSERT(_s);
264
0
  WINPR_ASSERT(_s->buffer <= _s->pointer);
265
0
  WINPR_ASSERT(_s->length <= _s->capacity);
266
0
  cur = (size_t)(_s->pointer - _s->buffer);
267
0
  WINPR_ASSERT(cur <= _s->length);
268
0
  if (cur > _s->length)
269
0
  {
270
0
    WLog_FATAL(STREAM_TAG, "wStream API misuse: stream was read out of bounds");
271
0
    winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
272
0
    return 0;
273
0
  }
274
0
  return (_s->length - cur);
275
0
}
276
277
BOOL Stream_Write_UTF16_String(wStream* s, const WCHAR* src, size_t length)
278
0
{
279
0
  WINPR_ASSERT(s);
280
0
  WINPR_ASSERT(src || (length == 0));
281
0
  if (!s || !src)
282
0
    return FALSE;
283
284
0
  if (!Stream_CheckAndLogRequiredCapacityOfSize(STREAM_TAG, (s), length, sizeof(WCHAR)))
285
0
    return FALSE;
286
287
0
  for (size_t x = 0; x < length; x++)
288
0
    Stream_Write_UINT16(s, src[x]);
289
290
0
  return TRUE;
291
0
}
292
293
BOOL Stream_Read_UTF16_String(wStream* s, WCHAR* dst, size_t length)
294
0
{
295
0
  WINPR_ASSERT(s);
296
0
  WINPR_ASSERT(dst);
297
298
0
  if (!Stream_CheckAndLogRequiredLengthOfSize(STREAM_TAG, s, length, sizeof(WCHAR)))
299
0
    return FALSE;
300
301
0
  for (size_t x = 0; x < length; x++)
302
0
    Stream_Read_UINT16(s, dst[x]);
303
304
0
  return TRUE;
305
0
}
306
307
BOOL Stream_CheckAndLogRequiredCapacityEx(const char* tag, DWORD level, wStream* s, size_t nmemb,
308
                                          size_t size, const char* fmt, ...)
309
0
{
310
0
  WINPR_ASSERT(size != 0);
311
0
  const size_t actual = Stream_GetRemainingCapacity(s) / size;
312
313
0
  if (actual < nmemb)
314
0
  {
315
0
    va_list args = WINPR_C_ARRAY_INIT;
316
317
0
    va_start(args, fmt);
318
0
    Stream_CheckAndLogRequiredCapacityExVa(tag, level, s, nmemb, size, fmt, args);
319
0
    va_end(args);
320
321
0
    return FALSE;
322
0
  }
323
0
  return TRUE;
324
0
}
325
326
BOOL Stream_CheckAndLogRequiredCapacityExVa(const char* tag, DWORD level, wStream* s, size_t nmemb,
327
                                            size_t size, const char* fmt, va_list args)
328
0
{
329
0
  WINPR_ASSERT(size != 0);
330
0
  const size_t actual = Stream_GetRemainingCapacity(s) / size;
331
332
0
  if (actual < nmemb)
333
0
    return Stream_CheckAndLogRequiredCapacityWLogExVa(WLog_Get(tag), level, s, nmemb, size, fmt,
334
0
                                                      args);
335
0
  return TRUE;
336
0
}
337
338
WINPR_ATTR_FORMAT_ARG(6, 0)
339
BOOL Stream_CheckAndLogRequiredCapacityWLogExVa(wLog* log, DWORD level, wStream* s, size_t nmemb,
340
                                                size_t size, WINPR_FORMAT_ARG const char* fmt,
341
                                                va_list args)
342
0
{
343
344
0
  WINPR_ASSERT(size != 0);
345
0
  const size_t actual = Stream_GetRemainingCapacity(s) / size;
346
347
0
  if (actual < nmemb)
348
0
  {
349
0
    char prefix[1024] = WINPR_C_ARRAY_INIT;
350
351
0
    (void)vsnprintf(prefix, sizeof(prefix), fmt, args);
352
353
0
    WLog_Print(log, level,
354
0
               "[%s] invalid remaining capacity, got %" PRIuz ", require at least %" PRIuz
355
0
               " [element size=%" PRIuz "]",
356
0
               prefix, actual, nmemb, size);
357
0
    winpr_log_backtrace_ex(log, level, 20);
358
0
    return FALSE;
359
0
  }
360
0
  return TRUE;
361
0
}
362
363
WINPR_ATTR_FORMAT_ARG(6, 7)
364
BOOL Stream_CheckAndLogRequiredCapacityWLogEx(wLog* log, DWORD level, wStream* s, size_t nmemb,
365
                                              size_t size, WINPR_FORMAT_ARG const char* fmt, ...)
366
0
{
367
368
0
  WINPR_ASSERT(size != 0);
369
0
  const size_t actual = Stream_GetRemainingCapacity(s) / size;
370
371
0
  if (actual < nmemb)
372
0
  {
373
0
    va_list args = WINPR_C_ARRAY_INIT;
374
375
0
    va_start(args, fmt);
376
0
    Stream_CheckAndLogRequiredCapacityWLogExVa(log, level, s, nmemb, size, fmt, args);
377
0
    va_end(args);
378
379
0
    return FALSE;
380
0
  }
381
0
  return TRUE;
382
0
}
383
384
WINPR_ATTR_FORMAT_ARG(6, 7)
385
BOOL Stream_CheckAndLogRequiredLengthEx(const char* tag, DWORD level, wStream* s, size_t nmemb,
386
                                        size_t size, WINPR_FORMAT_ARG const char* fmt, ...)
387
0
{
388
0
  WINPR_ASSERT(size > 0);
389
0
  const size_t actual = Stream_GetRemainingLength(s) / size;
390
391
0
  if (actual < nmemb)
392
0
  {
393
0
    va_list args = WINPR_C_ARRAY_INIT;
394
395
0
    va_start(args, fmt);
396
0
    Stream_CheckAndLogRequiredLengthExVa(tag, level, s, nmemb, size, fmt, args);
397
0
    va_end(args);
398
399
0
    return FALSE;
400
0
  }
401
0
  return TRUE;
402
0
}
403
404
BOOL Stream_CheckAndLogRequiredLengthExVa(const char* tag, DWORD level, wStream* s, size_t nmemb,
405
                                          size_t size, const char* fmt, va_list args)
406
0
{
407
0
  WINPR_ASSERT(size > 0);
408
0
  const size_t actual = Stream_GetRemainingLength(s) / size;
409
410
0
  if (actual < nmemb)
411
0
    return Stream_CheckAndLogRequiredLengthWLogExVa(WLog_Get(tag), level, s, nmemb, size, fmt,
412
0
                                                    args);
413
0
  return TRUE;
414
0
}
415
416
BOOL Stream_CheckAndLogRequiredLengthWLogEx(wLog* log, DWORD level, wStream* s, size_t nmemb,
417
                                            size_t size, const char* fmt, ...)
418
0
{
419
0
  WINPR_ASSERT(size > 0);
420
0
  const size_t actual = Stream_GetRemainingLength(s) / size;
421
422
0
  if (actual < nmemb)
423
0
  {
424
0
    va_list args = WINPR_C_ARRAY_INIT;
425
426
0
    va_start(args, fmt);
427
0
    Stream_CheckAndLogRequiredLengthWLogExVa(log, level, s, nmemb, size, fmt, args);
428
0
    va_end(args);
429
430
0
    return FALSE;
431
0
  }
432
0
  return TRUE;
433
0
}
434
435
WINPR_ATTR_FORMAT_ARG(6, 0)
436
BOOL Stream_CheckAndLogRequiredLengthWLogExVa(wLog* log, DWORD level, wStream* s, size_t nmemb,
437
                                              size_t size, WINPR_FORMAT_ARG const char* fmt,
438
                                              va_list args)
439
0
{
440
0
  WINPR_ASSERT(size > 0);
441
0
  const size_t actual = Stream_GetRemainingLength(s) / size;
442
443
0
  if (actual < nmemb)
444
0
  {
445
0
    char prefix[1024] = WINPR_C_ARRAY_INIT;
446
447
0
    (void)vsnprintf(prefix, sizeof(prefix), fmt, args);
448
449
0
    WLog_Print(log, level,
450
0
               "[%s] invalid length, got %" PRIuz ", require at least %" PRIuz
451
0
               " [element size=%" PRIuz "]",
452
0
               prefix, actual, nmemb, size);
453
0
    winpr_log_backtrace_ex(log, level, 20);
454
0
    return FALSE;
455
0
  }
456
0
  return TRUE;
457
0
}
458
459
SSIZE_T Stream_Write_UTF16_String_From_UTF8(wStream* s, size_t wcharLength, const char* src,
460
                                            size_t length, BOOL fill)
461
0
{
462
0
  SSIZE_T rc = 0;
463
0
  WCHAR* str = Stream_PointerAs(s, WCHAR);
464
465
0
  if (length != 0)
466
0
  {
467
0
    if (!Stream_CheckAndLogRequiredCapacityOfSize(STREAM_TAG, s, wcharLength, sizeof(WCHAR)))
468
0
      return -1;
469
470
0
    rc = ConvertUtf8NToWChar(src, length, str, wcharLength);
471
0
    if (rc < 0)
472
0
      return -1;
473
474
0
    Stream_Seek(s, (size_t)rc * sizeof(WCHAR));
475
0
  }
476
477
0
  if (fill)
478
0
    Stream_Zero(s, (wcharLength - (size_t)rc) * sizeof(WCHAR));
479
0
  return rc;
480
0
}
481
482
char* Stream_Read_UTF16_String_As_UTF8(wStream* s, size_t wcharLength, size_t* pUtfCharLength)
483
0
{
484
0
  const WCHAR* str = Stream_ConstPointer(s);
485
0
  if (wcharLength > SIZE_MAX / sizeof(WCHAR))
486
0
    return NULL;
487
488
0
  if (!Stream_CheckAndLogRequiredLength(STREAM_TAG, s, wcharLength * sizeof(WCHAR)))
489
0
    return NULL;
490
491
0
  Stream_Seek(s, wcharLength * sizeof(WCHAR));
492
0
  return ConvertWCharNToUtf8Alloc(str, wcharLength, pUtfCharLength);
493
0
}
494
495
SSIZE_T Stream_Read_UTF16_String_As_UTF8_Buffer(wStream* s, size_t wcharLength, char* utfBuffer,
496
                                                size_t utfBufferCharLength)
497
0
{
498
0
  const WCHAR* ptr = Stream_ConstPointer(s);
499
0
  if (wcharLength > SIZE_MAX / sizeof(WCHAR))
500
0
    return -1;
501
502
0
  if (!Stream_CheckAndLogRequiredLength(STREAM_TAG, s, wcharLength * sizeof(WCHAR)))
503
0
    return -1;
504
505
0
  Stream_Seek(s, wcharLength * sizeof(WCHAR));
506
0
  return ConvertWCharNToUtf8(ptr, wcharLength, utfBuffer, utfBufferCharLength);
507
0
}
508
509
BOOL Stream_SafeSeekEx(wStream* s, size_t size, const char* file, size_t line, const char* fkt)
510
0
{
511
0
  if (!Stream_CheckAndLogRequiredLengthEx(STREAM_TAG, WLOG_WARN, s, size, 1, "%s(%s:%" PRIuz ")",
512
0
                                          fkt, file, line))
513
0
    return FALSE;
514
515
0
  Stream_Seek(s, size);
516
0
  return TRUE;
517
0
}