18#include <winpr/assert.h>
19#include <winpr/cast.h>
21#include <freerdp/freerdp.h>
22#include <freerdp/server/proxy/proxy_log.h>
24#include "proxy_modules.h"
25#include "pf_channel.h"
27#define TAG PROXY_TAG("channel")
30struct sChannelStateTracker
32 pServerStaticChannelContext* channel;
33 ChannelTrackerMode mode;
35 size_t currentPacketReceived;
36 size_t currentPacketSize;
37 size_t currentPacketFragments;
39 ChannelTrackerPeekFn peekFn;
45static BOOL channelTracker_resetCurrentPacket(ChannelStateTracker* tracker)
47 WINPR_ASSERT(tracker);
50 if (tracker->currentPacket)
52 const size_t cap = Stream_Capacity(tracker->currentPacket);
53 if (cap < 1ULL * 1000ULL * 1000ULL)
56 Stream_Free(tracker->currentPacket, TRUE);
60 tracker->currentPacket = Stream_New(
nullptr, 10ULL * 1024ULL);
61 if (!tracker->currentPacket)
63 Stream_SetPosition(tracker->currentPacket, 0);
67ChannelStateTracker* channelTracker_new(pServerStaticChannelContext* channel,
68 ChannelTrackerPeekFn fn,
void* data)
70 ChannelStateTracker* ret = calloc(1,
sizeof(ChannelStateTracker));
76 ret->channel = channel;
79 if (!channelTracker_setCustomData(ret, data))
82 if (!channelTracker_resetCurrentPacket(ret))
88 WINPR_PRAGMA_DIAG_PUSH
89 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
90 channelTracker_free(ret);
95PfChannelResult channelTracker_update(ChannelStateTracker* tracker,
const BYTE* xdata,
size_t xsize,
96 UINT32 flags,
size_t totalSize)
98 PfChannelResult result = PF_CHANNEL_RESULT_ERROR;
99 BOOL firstPacket = (flags & CHANNEL_FLAG_FIRST) != 0;
100 BOOL lastPacket = (flags & CHANNEL_FLAG_LAST) != 0;
102 WINPR_ASSERT(tracker);
104 WLog_VRB(TAG,
"channelTracker_update(%s): sz=%" PRIuz
" first=%d last=%d",
105 tracker->channel->channel_name, xsize, firstPacket, lastPacket);
106 if (flags & CHANNEL_FLAG_FIRST)
108 if (!channelTracker_resetCurrentPacket(tracker))
109 return PF_CHANNEL_RESULT_ERROR;
110 channelTracker_setCurrentPacketSize(tracker, totalSize);
111 tracker->currentPacketReceived = 0;
112 tracker->currentPacketFragments = 0;
116 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(tracker);
117 if (tracker->currentPacketReceived + xsize > currentPacketSize)
118 WLog_INFO(TAG,
"cumulated size is bigger (%" PRIuz
") than total size (%" PRIuz
")",
119 tracker->currentPacketReceived + xsize, currentPacketSize);
122 tracker->currentPacketReceived += xsize;
123 tracker->currentPacketFragments++;
125 switch (channelTracker_getMode(tracker))
127 case CHANNEL_TRACKER_PEEK:
129 wStream* currentPacket = channelTracker_getCurrentPacket(tracker);
130 if (!Stream_EnsureRemainingCapacity(currentPacket, xsize))
131 return PF_CHANNEL_RESULT_ERROR;
133 Stream_Write(currentPacket, xdata, xsize);
135 WINPR_ASSERT(tracker->peekFn);
136 result = tracker->peekFn(tracker, firstPacket, lastPacket);
139 case CHANNEL_TRACKER_PASS:
140 result = PF_CHANNEL_RESULT_PASS;
142 case CHANNEL_TRACKER_DROP:
143 result = PF_CHANNEL_RESULT_DROP;
151 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(tracker);
152 channelTracker_setMode(tracker, CHANNEL_TRACKER_PEEK);
154 if (tracker->currentPacketReceived != currentPacketSize)
155 WLog_INFO(TAG,
"cumulated size(%" PRIuz
") does not match total size (%" PRIuz
")",
156 tracker->currentPacketReceived, currentPacketSize);
162void channelTracker_free(ChannelStateTracker* t)
167 Stream_Free(t->currentPacket, TRUE);
177PfChannelResult channelTracker_flushCurrent(ChannelStateTracker* t, BOOL first, BOOL last,
180 proxyData* pdata =
nullptr;
181 pServerContext* ps =
nullptr;
182 pServerStaticChannelContext* channel =
nullptr;
183 UINT32 flags = CHANNEL_FLAG_FIRST;
185 const char* direction = toBack ?
"F->B" :
"B->F";
186 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(t);
187 wStream* currentPacket = channelTracker_getCurrentPacket(t);
191 WLog_VRB(TAG,
"channelTracker_flushCurrent(%s): %s sz=%" PRIuz
" first=%d last=%d",
192 t->channel->channel_name, direction, Stream_GetPosition(currentPacket), first, last);
195 return PF_CHANNEL_RESULT_PASS;
198 channel = t->channel;
200 flags |= CHANNEL_FLAG_LAST;
206 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id);
207 ev.channel_name = channel->channel_name;
208 ev.data = Stream_Buffer(currentPacket);
209 ev.data_len = Stream_GetPosition(currentPacket);
211 ev.total_size = currentPacketSize;
213 if (!pdata->pc->sendChannelData)
214 return PF_CHANNEL_RESULT_ERROR;
216 return pdata->pc->sendChannelData(pdata->pc, &ev) ? PF_CHANNEL_RESULT_DROP
217 : PF_CHANNEL_RESULT_ERROR;
221 r = ps->context.peer->SendChannelPacket(
222 ps->context.peer, WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id),
223 currentPacketSize, flags, Stream_Buffer(currentPacket), Stream_GetPosition(currentPacket));
225 return r ? PF_CHANNEL_RESULT_DROP : PF_CHANNEL_RESULT_ERROR;
229static PfChannelResult pf_channel_generic_back_data(proxyData* pdata,
230 const pServerStaticChannelContext* channel,
231 const BYTE* xdata,
size_t xsize, UINT32 flags,
237 WINPR_ASSERT(channel);
239 switch (channel->channelMode)
241 case PF_UTILS_CHANNEL_PASSTHROUGH:
242 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->back_channel_id);
243 ev.channel_name = channel->channel_name;
247 ev.total_size = totalSize;
249 if (!pf_modules_run_filter(pdata->module, FILTER_TYPE_CLIENT_PASSTHROUGH_CHANNEL_DATA,
251 return PF_CHANNEL_RESULT_DROP;
253 return PF_CHANNEL_RESULT_PASS;
255 case PF_UTILS_CHANNEL_INTERCEPT:
257 case PF_UTILS_CHANNEL_BLOCK:
259 return PF_CHANNEL_RESULT_DROP;
264static PfChannelResult pf_channel_generic_front_data(proxyData* pdata,
265 const pServerStaticChannelContext* channel,
266 const BYTE* xdata,
size_t xsize, UINT32 flags,
272 WINPR_ASSERT(channel);
274 switch (channel->channelMode)
276 case PF_UTILS_CHANNEL_PASSTHROUGH:
277 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id);
278 ev.channel_name = channel->channel_name;
282 ev.total_size = totalSize;
284 if (!pf_modules_run_filter(pdata->module, FILTER_TYPE_SERVER_PASSTHROUGH_CHANNEL_DATA,
286 return PF_CHANNEL_RESULT_DROP;
288 return PF_CHANNEL_RESULT_PASS;
290 case PF_UTILS_CHANNEL_INTERCEPT:
292 case PF_UTILS_CHANNEL_BLOCK:
294 return PF_CHANNEL_RESULT_DROP;
298BOOL pf_channel_setup_generic(pServerStaticChannelContext* channel)
300 WINPR_ASSERT(channel);
301 channel->onBackData = pf_channel_generic_back_data;
302 channel->onFrontData = pf_channel_generic_front_data;
306BOOL channelTracker_setMode(ChannelStateTracker* tracker, ChannelTrackerMode mode)
308 WINPR_ASSERT(tracker);
309 tracker->mode = mode;
313ChannelTrackerMode channelTracker_getMode(ChannelStateTracker* tracker)
315 WINPR_ASSERT(tracker);
316 return tracker->mode;
319BOOL channelTracker_setPData(ChannelStateTracker* tracker, proxyData* pdata)
321 WINPR_ASSERT(tracker);
322 tracker->pdata = pdata;
326proxyData* channelTracker_getPData(ChannelStateTracker* tracker)
328 WINPR_ASSERT(tracker);
329 return tracker->pdata;
332wStream* channelTracker_getCurrentPacket(ChannelStateTracker* tracker)
334 WINPR_ASSERT(tracker);
335 return tracker->currentPacket;
338BOOL channelTracker_setCustomData(ChannelStateTracker* tracker,
void* data)
340 WINPR_ASSERT(tracker);
341 tracker->trackerData = data;
345void* channelTracker_getCustomData(ChannelStateTracker* tracker)
347 WINPR_ASSERT(tracker);
348 return tracker->trackerData;
351size_t channelTracker_getCurrentPacketSize(ChannelStateTracker* tracker)
353 WINPR_ASSERT(tracker);
354 return tracker->currentPacketSize;
357BOOL channelTracker_setCurrentPacketSize(ChannelStateTracker* tracker,
size_t size)
359 WINPR_ASSERT(tracker);
360 tracker->currentPacketSize = size;