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)
70 stream_dump_read_line(FILE* fp,
wStream* s, UINT64* pts,
size_t* pOffset, UINT32* flags)
79 if (!fp || !s || !flags)
83 (void)_fseeki64(fp, WINPR_ASSERTING_INT_CAST(int64_t, *pOffset), SEEK_SET);
85 r = fread(&ts, 1,
sizeof(ts), fp);
88 r = fread(&received, 1,
sizeof(received), fp);
89 if (r !=
sizeof(received))
91 r = fread(&crc32, 1,
sizeof(crc32), fp);
92 if (r !=
sizeof(crc32))
94 r = fread(&size, 1,
sizeof(size), fp);
95 if (r !=
sizeof(size))
98 *flags = STREAM_MSG_SRV_RX;
100 *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);
114 INT64 tmp = _ftelli64(fp);
117 *pOffset = (size_t)tmp;
125 Stream_SealLength(s);
129#if !defined(BUILD_TESTING_INTERNAL)
133 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))
259 const int64_t rt = _ftelli64(fp);
262 rc = WINPR_ASSERTING_INT_CAST(SSIZE_T, rt);
269static int stream_dump_transport_write(rdpTransport* transport,
wStream* s)
272 rdpContext* ctx = transport_get_context(transport);
275 WINPR_ASSERT(ctx->dump);
278 r = stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_TX : STREAM_MSG_SRV_RX, s,
279 &ctx->dump->writeDumpOffset);
283 WINPR_ASSERT(ctx->dump->io.WritePdu);
284 return ctx->dump->io.WritePdu(transport, s);
287static int stream_dump_transport_read(rdpTransport* transport,
wStream* s)
290 rdpContext* ctx = transport_get_context(transport);
293 WINPR_ASSERT(ctx->dump);
296 WINPR_ASSERT(ctx->dump->io.ReadPdu);
297 rc = ctx->dump->io.ReadPdu(transport, s);
301 stream_dump_append(ctx, ctx->dump->isServer ? STREAM_MSG_SRV_RX : STREAM_MSG_SRV_TX, s,
302 &ctx->dump->readDumpOffset);
309static BOOL stream_dump_register_write_handlers(rdpContext* context)
311 rdpTransportIo dump = { 0 };
312 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
321 WINPR_ASSERT(context->dump);
322 context->dump->io.ReadPdu = dfl->ReadPdu;
323 context->dump->io.WritePdu = dfl->WritePdu;
326 dump.WritePdu = stream_dump_transport_write;
327 dump.ReadPdu = stream_dump_transport_read;
328 return freerdp_set_io_callbacks(context, &dump);
331static int stream_dump_replay_transport_write(rdpTransport* transport,
wStream* s)
333 rdpContext* ctx = transport_get_context(transport);
339 size = Stream_Length(s);
340 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay write %" PRIuz, size);
346static int stream_dump_replay_transport_read(rdpTransport* transport,
wStream* s)
348 rdpContext* ctx = transport_get_context(transport);
356 WINPR_ASSERT(ctx->dump);
359 const size_t start = Stream_GetPosition(s);
362 Stream_SetPosition(s, start);
363 if (stream_dump_get(ctx, &flags, s, &ctx->dump->replayOffset, &ts) < 0)
365 }
while (flags & STREAM_MSG_SRV_RX);
367 if (!ctx->dump->nodelay)
369 if ((ctx->dump->replayTime > 0) && (ts > ctx->dump->replayTime))
370 slp = ts - ctx->dump->replayTime;
372 ctx->dump->replayTime = ts;
374 size = Stream_Length(s);
375 Stream_SetPosition(s, 0);
376 WLog_Print(ctx->dump->log, WLOG_TRACE,
"replay read %" PRIuz, size);
380 uint64_t duration = slp;
383 const DWORD actual = (DWORD)MIN(duration, UINT32_MAX);
386 }
while (duration > 0);
392static int stream_dump_replay_transport_tcp_connect(WINPR_ATTR_UNUSED rdpContext* context,
393 WINPR_ATTR_UNUSED rdpSettings* settings,
394 WINPR_ATTR_UNUSED
const char* hostname,
395 WINPR_ATTR_UNUSED
int port,
396 WINPR_ATTR_UNUSED DWORD timeout)
398 WINPR_ASSERT(context);
399 WINPR_ASSERT(settings);
400 WINPR_ASSERT(hostname);
406 WINPR_ATTR_UNUSED rdpTransport* transport, WINPR_ATTR_UNUSED
const char* hostname,
407 WINPR_ATTR_UNUSED
int port, WINPR_ATTR_UNUSED DWORD timeout)
409 WINPR_ASSERT(transport);
410 WINPR_ASSERT(hostname);
415static BOOL stream_dump_replay_transport_tls_connect(WINPR_ATTR_UNUSED rdpTransport* transport)
417 WINPR_ASSERT(transport);
421static BOOL stream_dump_replay_transport_accept(WINPR_ATTR_UNUSED rdpTransport* transport)
423 WINPR_ASSERT(transport);
427static BOOL stream_dump_register_read_handlers(rdpContext* context)
429 const rdpTransportIo* dfl = freerdp_get_io_callbacks(context);
435 rdpTransportIo dump = *dfl;
438 WINPR_ASSERT(context->dump);
439 context->dump->nodelay =
441 context->dump->io.ReadPdu = dfl->ReadPdu;
442 context->dump->io.WritePdu = dfl->WritePdu;
445 dump.WritePdu = stream_dump_transport_write;
446 dump.ReadPdu = stream_dump_transport_read;
449 dump.WritePdu = stream_dump_replay_transport_write;
450 dump.ReadPdu = stream_dump_replay_transport_read;
451 dump.TCPConnect = stream_dump_replay_transport_tcp_connect;
452 dump.TLSAccept = stream_dump_replay_transport_accept;
453 dump.TLSConnect = stream_dump_replay_transport_tls_connect;
454 dump.ConnectLayer = stream_dump_replay_transport_connect_layer;
455 if (!freerdp_set_io_callbacks(context, &dump))
457 return freerdp_io_callback_set_event(context, TRUE);
460BOOL stream_dump_register_handlers(rdpContext* context, CONNECTION_STATE state, BOOL isServer)
462 WINPR_ASSERT(context);
463 WINPR_ASSERT(context->dump);
464 context->dump->state = state;
465 context->dump->isServer = isServer;
466 if (!stream_dump_register_write_handlers(context))
468 return stream_dump_register_read_handlers(context);
471void stream_dump_free(rdpStreamDumpContext* dump)
476rdpStreamDumpContext* stream_dump_new(
void)
478 rdpStreamDumpContext* dump = calloc(1,
sizeof(rdpStreamDumpContext));
481 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.