21#include <winpr/config.h>
23#include <winpr/assert.h>
25#include <winpr/stream.h>
30#define STREAM_TAG WINPR_TAG("wStream")
32#define STREAM_ASSERT(cond) \
37 WLog_FATAL(STREAM_TAG, "%s [%s:%s:%" PRIuz "]", #cond, __FILE__, __func__, \
39 winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20); \
44BOOL Stream_EnsureCapacity(
wStream* s,
size_t size)
47 if (s->capacity >= size)
50 const size_t increment = 128ull;
51 const size_t old_capacity = s->capacity;
52 const size_t new_capacity = size + increment - size % increment;
53 const size_t position = Stream_GetPosition(s);
55 BYTE* new_buf =
nullptr;
58 new_buf = (BYTE*)malloc(new_capacity);
62 CopyMemory(new_buf, s->buffer, s->capacity);
67 new_buf = (BYTE*)realloc(s->buffer, new_capacity);
73 s->capacity = new_capacity;
74 s->length = new_capacity;
75 ZeroMemory(&s->buffer[old_capacity], s->capacity - old_capacity);
77 return Stream_SetPosition(s, position);
80BOOL Stream_EnsureRemainingCapacity(
wStream* s,
size_t size)
82 if (Stream_GetPosition(s) + size > Stream_Capacity(s))
83 return Stream_EnsureCapacity(s, Stream_Capacity(s) + size);
87wStream* Stream_New(BYTE* buffer,
size_t size)
101 s->buffer = (BYTE*)malloc(size);
109 s->pointer = s->buffer;
115 s->isAllocatedStream = TRUE;
120wStream* Stream_StaticConstInit(
wStream* s,
const BYTE* buffer,
size_t size)
129 return Stream_StaticInit(s, cnv.b, size);
134 const wStream empty = WINPR_C_ARRAY_INIT;
137 WINPR_ASSERT(buffer);
140 s->buffer = s->pointer = buffer;
141 s->capacity = s->length = size;
144 s->isAllocatedStream = FALSE;
149void Stream_EnsureValidity(
wStream* s)
154 STREAM_ASSERT(s->pointer >= s->buffer);
156 cur = (size_t)(s->pointer - s->buffer);
157 STREAM_ASSERT(cur <= s->capacity);
158 STREAM_ASSERT(s->length <= s->capacity);
161void Stream_Free(
wStream* s, BOOL bFreeBuffer)
165 Stream_EnsureValidity(s);
166 if (bFreeBuffer && s->isOwner)
169 if (s->isAllocatedStream)
174BOOL Stream_SetLength(
wStream* _s,
size_t _l)
176 if ((_l) > Stream_Capacity(_s))
185BOOL Stream_SetPosition(
wStream* _s,
size_t _p)
187 if ((_p) > Stream_Capacity(_s))
189 _s->pointer = _s->buffer;
192 _s->pointer = _s->buffer + (_p);
196void Stream_SealLength(
wStream* _s)
200 WINPR_ASSERT(_s->buffer <= _s->pointer);
201 cur = (size_t)(_s->pointer - _s->buffer);
202 WINPR_ASSERT(cur <= _s->capacity);
203 if (cur <= _s->capacity)
207 WLog_FATAL(STREAM_TAG,
"wStream API misuse: stream was written out of bounds");
208 winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
213#if defined(WITH_WINPR_DEPRECATED)
214BOOL Stream_SetPointer(
wStream* _s, BYTE* _p)
217 if (!_p || (_s->buffer > _p) || (_s->buffer + _s->capacity < _p))
219 _s->pointer = _s->buffer;
226BOOL Stream_SetBuffer(
wStream* _s, BYTE* _b)
233 return _s->buffer !=
nullptr;
236void Stream_SetCapacity(
wStream* _s,
size_t _c)
244size_t Stream_GetRemainingCapacity(
const wStream* _s)
248 WINPR_ASSERT(_s->buffer <= _s->pointer);
249 cur = (size_t)(_s->pointer - _s->buffer);
250 WINPR_ASSERT(cur <= _s->capacity);
251 if (cur > _s->capacity)
253 WLog_FATAL(STREAM_TAG,
"wStream API misuse: stream was written out of bounds");
254 winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
257 return (_s->capacity - cur);
260size_t Stream_GetRemainingLength(
const wStream* _s)
264 WINPR_ASSERT(_s->buffer <= _s->pointer);
265 WINPR_ASSERT(_s->length <= _s->capacity);
266 cur = (size_t)(_s->pointer - _s->buffer);
267 WINPR_ASSERT(cur <= _s->length);
268 if (cur > _s->length)
270 WLog_FATAL(STREAM_TAG,
"wStream API misuse: stream was read out of bounds");
271 winpr_log_backtrace(STREAM_TAG, WLOG_FATAL, 20);
274 return (_s->length - cur);
277BOOL Stream_Write_UTF16_String(
wStream* s,
const WCHAR* src,
size_t length)
280 WINPR_ASSERT(src || (length == 0));
284 if (!Stream_CheckAndLogRequiredCapacityOfSize(STREAM_TAG, (s), length,
sizeof(WCHAR)))
287 for (
size_t x = 0; x < length; x++)
288 Stream_Write_UINT16(s, src[x]);
293BOOL Stream_Read_UTF16_String(
wStream* s, WCHAR* dst,
size_t length)
298 if (!Stream_CheckAndLogRequiredLengthOfSize(STREAM_TAG, s, length,
sizeof(WCHAR)))
301 for (
size_t x = 0; x < length; x++)
302 Stream_Read_UINT16(s, dst[x]);
307BOOL Stream_CheckAndLogRequiredCapacityEx(
const char* tag, DWORD level,
wStream* s,
size_t nmemb,
308 size_t size,
const char* fmt, ...)
310 WINPR_ASSERT(size != 0);
311 const size_t actual = Stream_GetRemainingCapacity(s) / size;
315 va_list args = WINPR_C_ARRAY_INIT;
319 Stream_CheckAndLogRequiredCapacityExVa(tag, level, s, nmemb, size, fmt, args);
326BOOL Stream_CheckAndLogRequiredCapacityExVa(
const char* tag, DWORD level,
wStream* s,
size_t nmemb,
327 size_t size,
const char* fmt, va_list args)
329 WINPR_ASSERT(size != 0);
330 const size_t actual = Stream_GetRemainingCapacity(s) / size;
333 return Stream_CheckAndLogRequiredCapacityWLogExVa(WLog_Get(tag), level, s, nmemb, size, fmt,
338WINPR_ATTR_FORMAT_ARG(6, 0)
339BOOL Stream_CheckAndLogRequiredCapacityWLogExVa(wLog* log, DWORD level,
wStream* s,
size_t nmemb,
340 size_t size, WINPR_FORMAT_ARG const
char* fmt,
344 WINPR_ASSERT(size != 0);
345 const size_t actual = Stream_GetRemainingCapacity(s) / size;
349 char prefix[1024] = WINPR_C_ARRAY_INIT;
351 (void)vsnprintf(prefix,
sizeof(prefix), fmt, args);
353 WLog_Print(log, level,
354 "[%s] invalid remaining capacity, got %" PRIuz
", require at least %" PRIuz
355 " [element size=%" PRIuz
"]",
356 prefix, actual, nmemb, size);
357 winpr_log_backtrace_ex(log, level, 20);
363WINPR_ATTR_FORMAT_ARG(6, 7)
364BOOL Stream_CheckAndLogRequiredCapacityWLogEx(wLog* log, DWORD level,
wStream* s,
size_t nmemb,
365 size_t size, WINPR_FORMAT_ARG const
char* fmt, ...)
368 WINPR_ASSERT(size != 0);
369 const size_t actual = Stream_GetRemainingCapacity(s) / size;
373 va_list args = WINPR_C_ARRAY_INIT;
377 Stream_CheckAndLogRequiredCapacityWLogExVa(log, level, s, nmemb, size, fmt, args);
384WINPR_ATTR_FORMAT_ARG(6, 7)
385BOOL Stream_CheckAndLogRequiredLengthEx(const
char* tag, DWORD level,
wStream* s,
size_t nmemb,
386 size_t size, WINPR_FORMAT_ARG const
char* fmt, ...)
388 WINPR_ASSERT(size > 0);
389 const size_t actual = Stream_GetRemainingLength(s) / size;
393 va_list args = WINPR_C_ARRAY_INIT;
396 const BOOL rc = Stream_CheckAndLogRequiredLengthExVa(tag, level, s, nmemb, size, fmt, args);
403BOOL Stream_CheckAndLogRequiredLengthExVa(
const char* tag, DWORD level,
wStream* s,
size_t nmemb,
404 size_t size,
const char* fmt, va_list args)
406 WINPR_ASSERT(size > 0);
407 const size_t actual = Stream_GetRemainingLength(s) / size;
410 return Stream_CheckAndLogRequiredLengthWLogExVa(WLog_Get(tag), level, s, nmemb, size, fmt,
415BOOL Stream_CheckAndLogRequiredLengthWLogEx(wLog* log, DWORD level,
wStream* s,
size_t nmemb,
416 size_t size,
const char* fmt, ...)
418 WINPR_ASSERT(size > 0);
419 const size_t actual = Stream_GetRemainingLength(s) / size;
423 va_list args = WINPR_C_ARRAY_INIT;
427 Stream_CheckAndLogRequiredLengthWLogExVa(log, level, s, nmemb, size, fmt, args);
434WINPR_ATTR_FORMAT_ARG(6, 0)
435BOOL Stream_CheckAndLogRequiredLengthWLogExVa(wLog* log, DWORD level,
wStream* s,
size_t nmemb,
436 size_t size, WINPR_FORMAT_ARG const
char* fmt,
439 WINPR_ASSERT(size > 0);
440 const size_t actual = Stream_GetRemainingLength(s) / size;
444 char prefix[1024] = WINPR_C_ARRAY_INIT;
446 (void)vsnprintf(prefix,
sizeof(prefix), fmt, args);
448 WLog_Print(log, level,
449 "[%s] invalid length, got %" PRIuz
", require at least %" PRIuz
450 " [element size=%" PRIuz
"]",
451 prefix, actual, nmemb, size);
452 winpr_log_backtrace_ex(log, level, 20);
458SSIZE_T Stream_Write_UTF16_String_From_UTF8(
wStream* s,
size_t wcharLength,
const char* src,
459 size_t length, BOOL fill)
462 WCHAR* str = Stream_PointerAs(s, WCHAR);
466 if (!Stream_CheckAndLogRequiredCapacityOfSize(STREAM_TAG, s, wcharLength,
sizeof(WCHAR)))
469 rc = ConvertUtf8NToWChar(src, length, str, wcharLength);
473 Stream_Seek(s, (
size_t)rc *
sizeof(WCHAR));
477 Stream_Zero(s, (wcharLength - (
size_t)rc) *
sizeof(WCHAR));
481char* Stream_Read_UTF16_String_As_UTF8(
wStream* s,
size_t wcharLength,
size_t* pUtfCharLength)
483 const WCHAR* str = Stream_ConstPointer(s);
484 if (wcharLength > SIZE_MAX /
sizeof(WCHAR))
487 if (!Stream_CheckAndLogRequiredLength(STREAM_TAG, s, wcharLength *
sizeof(WCHAR)))
490 Stream_Seek(s, wcharLength *
sizeof(WCHAR));
491 return ConvertWCharNToUtf8Alloc(str, wcharLength, pUtfCharLength);
494SSIZE_T Stream_Read_UTF16_String_As_UTF8_Buffer(
wStream* s,
size_t wcharLength,
char* utfBuffer,
495 size_t utfBufferCharLength)
497 const WCHAR* ptr = Stream_ConstPointer(s);
498 if (wcharLength > SIZE_MAX /
sizeof(WCHAR))
501 if (!Stream_CheckAndLogRequiredLength(STREAM_TAG, s, wcharLength *
sizeof(WCHAR)))
504 Stream_Seek(s, wcharLength *
sizeof(WCHAR));
505 return ConvertWCharNToUtf8(ptr, wcharLength, utfBuffer, utfBufferCharLength);
508BOOL Stream_SafeSeekEx(
wStream* s,
size_t size,
const char* file,
size_t line,
const char* fkt)
510 if (!Stream_CheckAndLogRequiredLengthEx(STREAM_TAG, WLOG_WARN, s, size, 1,
"%s(%s:%" PRIuz
")",
514 Stream_Seek(s, size);