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)
82 (void)_fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *pOffset), SEEK_SET);
84 r = fread(&ts, 1,
sizeof(ts), fp);
87 r = fread(&received, 1,
sizeof(received), fp);
88 if (r !=
sizeof(received))
90 r = fread(&crc32, 1,
sizeof(crc32), fp);
91 if (r !=
sizeof(crc32))
93 r = fread(&size, 1,
sizeof(size), fp);
94 if (r !=
sizeof(size))
97 *flags = STREAM_MSG_SRV_RX;
99 *flags = STREAM_MSG_SRV_TX;
102 const size_t usize = WINPR_ASSERTING_INT_CAST(
size_t, size);
103 if (!Stream_EnsureRemainingCapacity(s, usize))
105 r = fread(Stream_Pointer(s), 1, usize, fp);
108 if (crc32 != crc32b(Stream_ConstPointer(s), usize))
110 Stream_Seek(s, usize);
115 INT64 tmp = _ftelli64(fp);
118 *pOffset = (size_t)tmp;
126 Stream_SealLength(s);
130#if !defined(BUILD_TESTING_INTERNAL)
133 BOOL stream_dump_write_line(FILE* fp, UINT32 flags,
wStream* s)
136 const UINT64 t = GetTickCount64();
137 const BYTE* data = Stream_Buffer(s);
138 const size_t usize = Stream_Length(s);
139 const uint64_t size = (uint64_t)usize;
145 const UINT32 crc32 = crc32b(data, usize);
146 const BYTE received = flags & STREAM_MSG_SRV_RX;
147 size_t r = fwrite(&t, 1,
sizeof(t), fp);
150 r = fwrite(&received, 1,
sizeof(received), fp);
151 if (r !=
sizeof(received))
153 r = fwrite(&crc32, 1,
sizeof(crc32), fp);
154 if (r !=
sizeof(crc32))
156 r = fwrite(&size, 1,
sizeof(size), fp);
157 if (r !=
sizeof(size))
159 r = fwrite(data, 1, usize, fp);
169static FILE* stream_dump_get_file(
const rdpSettings* settings,
const char* mode)
171 const char* cfolder = NULL;
175 if (!settings || !mode)
180 file = GetKnownSubPath(KNOWN_PATH_TEMP,
"freerdp-transport-dump");
182 file = _strdup(cfolder);
187 fp = winpr_fopen(file, mode);
193SSIZE_T stream_dump_append(
const rdpContext* context, UINT32 flags,
wStream* s,
size_t* offset)
197 const UINT32 mask = STREAM_MSG_SRV_RX | STREAM_MSG_SRV_TX;
198 CONNECTION_STATE state = freerdp_get_state(context);
201 if (!context || !s || !offset)
204 if ((flags & STREAM_MSG_SRV_RX) && (flags & STREAM_MSG_SRV_TX))
207 if ((flags & mask) == 0)
210 if (state < context->dump->state)
213 fp = stream_dump_get_file(context->settings,
"ab");
217 r = _fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *offset), SEEK_SET);
221 if (!stream_dump_write_line(fp, flags, s))
224 const int64_t rt = _ftelli64(fp);
230 rc = WINPR_ASSERTING_INT_CAST(SSIZE_T, rt);
232 *offset = (size_t)rc;
240SSIZE_T stream_dump_get(
const rdpContext* context, UINT32* flags,
wStream* s,
size_t* offset,
247 if (!context || !s || !offset)
249 fp = stream_dump_get_file(context->settings,
"rb");
252 r = _fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *offset), SEEK_SET);
256 if (!stream_dump_read_line(fp, s, pts, offset, flags))
260 const int64_t rt = _ftelli64(fp);
263 rc = WINPR_ASSERTING_INT_CAST(SSIZE_T, rt);
272static int stream_dump_transport_write(rdpTransport* transport,
wStream* s)
275 rdpContext* ctx = transport_get_context(transport);
278 WINPR_ASSERT(ctx->dump);
281 r = stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_TX : STREAM_MSG_SRV_RX, s,
282 &ctx->dump->writeDumpOffset);
286 WINPR_ASSERT(ctx->dump->io.WritePdu);
287 return ctx->dump->io.WritePdu(transport, s);
290static int stream_dump_transport_read(rdpTransport* transport,
wStream* s)
293 rdpContext* ctx = transport_get_context(transport);
296 WINPR_ASSERT(ctx->dump);
299 WINPR_ASSERT(ctx->dump->io.ReadPdu);
300 rc = ctx->dump->io.ReadPdu(transport, s);
304 stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_RX : STREAM_MSG_SRV_TX, s,
305 &ctx->dump->readDumpOffset);
312static BOOL stream_dump_register_write_handlers(rdpContext* context)
314 rdpTransportIo dump = { 0 };
315 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
324 WINPR_ASSERT(context->dump);
325 context->dump->io.ReadPdu = dfl->ReadPdu;
326 context->dump->io.WritePdu = dfl->WritePdu;
329 dump.WritePdu = stream_dump_transport_write;
330 dump.ReadPdu = stream_dump_transport_read;
331 return freerdp_set_io_callbacks(context, &dump);
334static int stream_dump_replay_transport_write(rdpTransport* transport,
wStream* s)
336 rdpContext* ctx = transport_get_context(transport);
342 size = Stream_Length(s);
343 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay write %" PRIuz, size);
349static int stream_dump_replay_transport_read(rdpTransport* transport,
wStream* s)
351 rdpContext* ctx = transport_get_context(transport);
359 WINPR_ASSERT(ctx->dump);
362 const size_t start = Stream_GetPosition(s);
365 Stream_SetPosition(s, start);
366 if (stream_dump_get(ctx, &flags, s, &ctx->dump->replayOffset, &ts) < 0)
368 }
while (flags & STREAM_MSG_SRV_RX);
370 if (!ctx->dump->nodelay)
372 if ((ctx->dump->replayTime > 0) && (ts > ctx->dump->replayTime))
373 slp = ts - ctx->dump->replayTime;
375 ctx->dump->replayTime = ts;
377 size = Stream_Length(s);
378 Stream_SetPosition(s, 0);
379 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay read %" PRIuz, size);
383 uint64_t duration = slp;
386 const DWORD actual = (DWORD)MIN(duration, UINT32_MAX);
389 }
while (duration > 0);
395static int stream_dump_replay_transport_tcp_connect(WINPR_ATTR_UNUSED rdpContext* context,
396 WINPR_ATTR_UNUSED rdpSettings* settings,
397 WINPR_ATTR_UNUSED
const char* hostname,
398 WINPR_ATTR_UNUSED
int port,
399 WINPR_ATTR_UNUSED DWORD timeout)
401 WINPR_ASSERT(context);
402 WINPR_ASSERT(settings);
403 WINPR_ASSERT(hostname);
409 WINPR_ATTR_UNUSED rdpTransport* transport, WINPR_ATTR_UNUSED
const char* hostname,
410 WINPR_ATTR_UNUSED
int port, WINPR_ATTR_UNUSED DWORD timeout)
412 WINPR_ASSERT(transport);
413 WINPR_ASSERT(hostname);
418static BOOL stream_dump_replay_transport_tls_connect(WINPR_ATTR_UNUSED rdpTransport* transport)
420 WINPR_ASSERT(transport);
424static BOOL stream_dump_replay_transport_accept(WINPR_ATTR_UNUSED rdpTransport* transport)
426 WINPR_ASSERT(transport);
430static BOOL stream_dump_register_read_handlers(rdpContext* context)
432 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
438 rdpTransportIo dump = *dfl;
441 WINPR_ASSERT(context->dump);
442 context->dump->nodelay =
444 context->dump->io.ReadPdu = dfl->ReadPdu;
445 context->dump->io.WritePdu = dfl->WritePdu;
448 dump.WritePdu = stream_dump_transport_write;
449 dump.ReadPdu = stream_dump_transport_read;
452 dump.WritePdu = stream_dump_replay_transport_write;
453 dump.ReadPdu = stream_dump_replay_transport_read;
454 dump.TCPConnect = stream_dump_replay_transport_tcp_connect;
455 dump.TLSAccept = stream_dump_replay_transport_accept;
456 dump.TLSConnect = stream_dump_replay_transport_tls_connect;
457 dump.ConnectLayer = stream_dump_replay_transport_connect_layer;
458 if (!freerdp_set_io_callbacks(context, &dump))
460 return freerdp_io_callback_set_event(context, TRUE);
463BOOL stream_dump_register_handlers(rdpContext* context, CONNECTION_STATE state, BOOL isServer)
465 WINPR_ASSERT(context);
466 WINPR_ASSERT(context->dump);
467 context->dump->state = state;
468 context->dump->isServer = isServer;
469 if (!stream_dump_register_write_handlers(context))
471 return stream_dump_register_read_handlers(context);
474void stream_dump_free(rdpStreamDumpContext* dump)
479rdpStreamDumpContext* stream_dump_new(
void)
481 rdpStreamDumpContext* dump = calloc(1,
sizeof(rdpStreamDumpContext));
484 dump->log = WLog_Get(TAG);
FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.