24#include <winpr/sysinfo.h>
25#include <winpr/path.h>
26#include <winpr/string.h>
28#include <freerdp/freerdp.h>
29#include <freerdp/streamdump.h>
30#include <freerdp/transport_io.h>
32#include "streamdump.h"
34#define TAG FREERDP_TAG("streamdump")
36struct stream_dump_context
39 size_t writeDumpOffset;
40 size_t readDumpOffset;
43 CONNECTION_STATE state;
49static UINT32 crc32b(
const BYTE* data,
size_t length)
51 UINT32 crc = 0xFFFFFFFF;
53 for (
size_t x = 0; x < length; x++)
55 const UINT32 d = data[x] & 0xFF;
57 for (
int j = 7; j >= 0; j--)
59 UINT32 mask = ~(crc & 1);
60 crc = (crc >> 1) ^ (0xEDB88320 & mask);
66#if !defined(BUILD_TESTING_INTERNAL)
69 BOOL stream_dump_read_line(FILE* fp,
wStream* s, UINT64* pts,
size_t* pOffset, UINT32* flags)
78 if (!fp || !s || !flags)
83 if (_fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *pOffset), SEEK_SET) < 0)
87 r = fread(&ts, 1,
sizeof(ts), fp);
90 r = fread(&received, 1,
sizeof(received), fp);
91 if (r !=
sizeof(received))
93 r = fread(&crc32, 1,
sizeof(crc32), fp);
94 if (r !=
sizeof(crc32))
96 r = fread(&size, 1,
sizeof(size), fp);
97 if (r !=
sizeof(size))
100 *flags = STREAM_MSG_SRV_RX;
102 *flags = STREAM_MSG_SRV_TX;
105 const size_t usize = WINPR_ASSERTING_INT_CAST(
size_t, size);
106 if (!Stream_EnsureRemainingCapacity(s, usize))
108 r = fread(Stream_Pointer(s), 1, usize, fp);
111 if (crc32 != crc32b(Stream_ConstPointer(s), usize))
113 Stream_Seek(s, usize);
118 INT64 tmp = _ftelli64(fp);
121 *pOffset = (size_t)tmp;
129 Stream_SealLength(s);
133#if !defined(BUILD_TESTING_INTERNAL)
136 BOOL stream_dump_write_line(FILE* fp, UINT32 flags,
wStream* s)
139 const UINT64 t = GetTickCount64();
140 const BYTE* data = Stream_Buffer(s);
141 const size_t usize = Stream_Length(s);
142 const uint64_t size = (uint64_t)usize;
148 const UINT32 crc32 = crc32b(data, usize);
149 const BYTE received = flags & STREAM_MSG_SRV_RX;
150 size_t r = fwrite(&t, 1,
sizeof(t), fp);
153 r = fwrite(&received, 1,
sizeof(received), fp);
154 if (r !=
sizeof(received))
156 r = fwrite(&crc32, 1,
sizeof(crc32), fp);
157 if (r !=
sizeof(crc32))
159 r = fwrite(&size, 1,
sizeof(size), fp);
160 if (r !=
sizeof(size))
162 r = fwrite(data, 1, usize, fp);
172static FILE* stream_dump_get_file(
const rdpSettings* settings,
const char* mode)
174 const char* cfolder =
nullptr;
175 char* file =
nullptr;
178 if (!settings || !mode)
183 file = GetKnownSubPath(KNOWN_PATH_TEMP,
"freerdp-transport-dump");
185 file = _strdup(cfolder);
190 fp = winpr_fopen(file, mode);
196SSIZE_T stream_dump_append(
const rdpContext* context, UINT32 flags,
wStream* s,
size_t* offset)
200 const UINT32 mask = STREAM_MSG_SRV_RX | STREAM_MSG_SRV_TX;
201 CONNECTION_STATE state = freerdp_get_state(context);
204 if (!context || !s || !offset)
207 if ((flags & STREAM_MSG_SRV_RX) && (flags & STREAM_MSG_SRV_TX))
210 if ((flags & mask) == 0)
213 if (state < context->dump->state)
216 fp = stream_dump_get_file(context->settings,
"ab");
220 r = _fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *offset), SEEK_SET);
224 if (!stream_dump_write_line(fp, flags, s))
227 const int64_t rt = _ftelli64(fp);
233 rc = WINPR_ASSERTING_INT_CAST(SSIZE_T, rt);
235 *offset = (size_t)rc;
243SSIZE_T stream_dump_get(
const rdpContext* context, UINT32* flags,
wStream* s,
size_t* offset,
250 if (!context || !s || !offset)
252 fp = stream_dump_get_file(context->settings,
"rb");
255 r = _fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *offset), SEEK_SET);
259 if (!stream_dump_read_line(fp, s, pts, offset, flags))
263 const int64_t rt = _ftelli64(fp);
266 rc = WINPR_ASSERTING_INT_CAST(SSIZE_T, rt);
275static int stream_dump_transport_write(rdpTransport* transport,
wStream* s)
278 rdpContext* ctx = transport_get_context(transport);
281 WINPR_ASSERT(ctx->dump);
284 r = stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_TX : STREAM_MSG_SRV_RX, s,
285 &ctx->dump->writeDumpOffset);
289 WINPR_ASSERT(ctx->dump->io.WritePdu);
290 return ctx->dump->io.WritePdu(transport, s);
293static int stream_dump_transport_read(rdpTransport* transport,
wStream* s)
296 rdpContext* ctx = transport_get_context(transport);
299 WINPR_ASSERT(ctx->dump);
302 WINPR_ASSERT(ctx->dump->io.ReadPdu);
303 rc = ctx->dump->io.ReadPdu(transport, s);
307 stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_RX : STREAM_MSG_SRV_TX, s,
308 &ctx->dump->readDumpOffset);
315static BOOL stream_dump_register_write_handlers(rdpContext* context)
317 rdpTransportIo dump = WINPR_C_ARRAY_INIT;
318 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
327 WINPR_ASSERT(context->dump);
328 context->dump->io.ReadPdu = dfl->ReadPdu;
329 context->dump->io.WritePdu = dfl->WritePdu;
332 dump.WritePdu = stream_dump_transport_write;
333 dump.ReadPdu = stream_dump_transport_read;
334 return freerdp_set_io_callbacks(context, &dump);
337static int stream_dump_replay_transport_write(rdpTransport* transport,
wStream* s)
339 rdpContext* ctx = transport_get_context(transport);
345 size = Stream_Length(s);
346 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay write %" PRIuz, size);
352static int stream_dump_replay_transport_read(rdpTransport* transport,
wStream* s)
354 rdpContext* ctx = transport_get_context(transport);
362 WINPR_ASSERT(ctx->dump);
365 const size_t start = Stream_GetPosition(s);
368 if (!Stream_SetPosition(s, start))
370 if (stream_dump_get(ctx, &flags, s, &ctx->dump->replayOffset, &ts) < 0)
372 }
while (flags & STREAM_MSG_SRV_RX);
374 if (!ctx->dump->nodelay)
376 if ((ctx->dump->replayTime > 0) && (ts > ctx->dump->replayTime))
377 slp = ts - ctx->dump->replayTime;
379 ctx->dump->replayTime = ts;
381 size = Stream_Length(s);
382 Stream_ResetPosition(s);
383 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay read %" PRIuz, size);
387 uint64_t duration = slp;
390 const DWORD actual = (DWORD)MIN(duration, UINT32_MAX);
393 }
while (duration > 0);
399static int stream_dump_replay_transport_tcp_connect(WINPR_ATTR_UNUSED rdpContext* context,
400 WINPR_ATTR_UNUSED rdpSettings* settings,
401 WINPR_ATTR_UNUSED
const char* hostname,
402 WINPR_ATTR_UNUSED
int port,
403 WINPR_ATTR_UNUSED DWORD timeout)
405 WINPR_ASSERT(context);
406 WINPR_ASSERT(settings);
407 WINPR_ASSERT(hostname);
413 WINPR_ATTR_UNUSED rdpTransport* transport, WINPR_ATTR_UNUSED
const char* hostname,
414 WINPR_ATTR_UNUSED
int port, WINPR_ATTR_UNUSED DWORD timeout)
416 WINPR_ASSERT(transport);
417 WINPR_ASSERT(hostname);
422static BOOL stream_dump_replay_transport_tls_connect(WINPR_ATTR_UNUSED rdpTransport* transport)
424 WINPR_ASSERT(transport);
428static BOOL stream_dump_replay_transport_accept(WINPR_ATTR_UNUSED rdpTransport* transport)
430 WINPR_ASSERT(transport);
434static BOOL stream_dump_register_read_handlers(rdpContext* context)
436 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
442 rdpTransportIo dump = *dfl;
445 WINPR_ASSERT(context->dump);
446 context->dump->nodelay =
448 context->dump->io.ReadPdu = dfl->ReadPdu;
449 context->dump->io.WritePdu = dfl->WritePdu;
452 dump.WritePdu = stream_dump_transport_write;
453 dump.ReadPdu = stream_dump_transport_read;
456 dump.WritePdu = stream_dump_replay_transport_write;
457 dump.ReadPdu = stream_dump_replay_transport_read;
458 dump.TCPConnect = stream_dump_replay_transport_tcp_connect;
459 dump.TLSAccept = stream_dump_replay_transport_accept;
460 dump.TLSConnect = stream_dump_replay_transport_tls_connect;
461 dump.ConnectLayer = stream_dump_replay_transport_connect_layer;
462 if (!freerdp_set_io_callbacks(context, &dump))
464 return freerdp_io_callback_set_event(context, TRUE);
467BOOL stream_dump_register_handlers(rdpContext* context, CONNECTION_STATE state, BOOL isServer)
469 WINPR_ASSERT(context);
470 WINPR_ASSERT(context->dump);
471 context->dump->state = state;
472 context->dump->isServer = isServer;
473 if (!stream_dump_register_write_handlers(context))
475 return stream_dump_register_read_handlers(context);
478void stream_dump_free(rdpStreamDumpContext* dump)
483rdpStreamDumpContext* stream_dump_new(
void)
485 rdpStreamDumpContext* dump = calloc(1,
sizeof(rdpStreamDumpContext));
488 dump->log = WLog_Get(TAG);
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.