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"
29#define TAG PROXY_TAG("channel")
32struct sChannelStateTracker
34 pServerStaticChannelContext* channel;
35 ChannelTrackerMode mode;
37 size_t currentPacketReceived;
38 size_t currentPacketSize;
39 size_t currentPacketFragments;
41 ChannelTrackerPeekFn peekFn;
47static BOOL channelTracker_resetCurrentPacket(ChannelStateTracker* tracker)
49 WINPR_ASSERT(tracker);
52 if (tracker->currentPacket)
54 const size_t cap = Stream_Capacity(tracker->currentPacket);
55 if (cap < 1ULL * 1000ULL * 1000ULL)
58 Stream_Free(tracker->currentPacket, TRUE);
62 tracker->currentPacket = Stream_New(
nullptr, 10ULL * 1024ULL);
63 if (!tracker->currentPacket)
65 Stream_ResetPosition(tracker->currentPacket);
69ChannelStateTracker* channelTracker_new(pServerStaticChannelContext* channel,
70 ChannelTrackerPeekFn fn,
void* data)
72 ChannelStateTracker* ret = calloc(1,
sizeof(ChannelStateTracker));
78 ret->channel = channel;
81 if (!channelTracker_setCustomData(ret, data))
84 if (!channelTracker_resetCurrentPacket(ret))
90 WINPR_PRAGMA_DIAG_PUSH
91 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
92 channelTracker_free(ret);
97PfChannelResult channelTracker_update(ChannelStateTracker* tracker,
const BYTE* xdata,
size_t xsize,
98 UINT32 flags,
size_t totalSize)
100 PfChannelResult result = PF_CHANNEL_RESULT_ERROR;
101 BOOL firstPacket = (flags & CHANNEL_FLAG_FIRST) != 0;
102 BOOL lastPacket = (flags & CHANNEL_FLAG_LAST) != 0;
104 WINPR_ASSERT(tracker);
106 WLog_VRB(TAG,
"channelTracker_update(%s): sz=%" PRIuz
" first=%d last=%d",
107 tracker->channel->channel_name, xsize, firstPacket, lastPacket);
108 if (flags & CHANNEL_FLAG_FIRST)
110 if (!channelTracker_resetCurrentPacket(tracker))
111 return PF_CHANNEL_RESULT_ERROR;
112 channelTracker_setCurrentPacketSize(tracker, totalSize);
113 tracker->currentPacketReceived = 0;
114 tracker->currentPacketFragments = 0;
118 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(tracker);
119 if (tracker->currentPacketReceived + xsize > currentPacketSize)
120 WLog_INFO(TAG,
"cumulated size is bigger (%" PRIuz
") than total size (%" PRIuz
")",
121 tracker->currentPacketReceived + xsize, currentPacketSize);
124 tracker->currentPacketReceived += xsize;
125 tracker->currentPacketFragments++;
127 switch (channelTracker_getMode(tracker))
129 case CHANNEL_TRACKER_PEEK:
131 wStream* currentPacket = channelTracker_getCurrentPacket(tracker);
132 if (!Stream_EnsureRemainingCapacity(currentPacket, xsize))
133 return PF_CHANNEL_RESULT_ERROR;
135 Stream_Write(currentPacket, xdata, xsize);
137 WINPR_ASSERT(tracker->peekFn);
138 result = tracker->peekFn(tracker, firstPacket, lastPacket);
141 case CHANNEL_TRACKER_PASS:
142 result = PF_CHANNEL_RESULT_PASS;
144 case CHANNEL_TRACKER_DROP:
145 result = PF_CHANNEL_RESULT_DROP;
153 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(tracker);
154 channelTracker_setMode(tracker, CHANNEL_TRACKER_PEEK);
156 if (tracker->currentPacketReceived != currentPacketSize)
157 WLog_INFO(TAG,
"cumulated size(%" PRIuz
") does not match total size (%" PRIuz
")",
158 tracker->currentPacketReceived, currentPacketSize);
164void channelTracker_free(ChannelStateTracker* t)
169 Stream_Free(t->currentPacket, TRUE);
179PfChannelResult channelTracker_flushCurrent(ChannelStateTracker* t, BOOL first, BOOL last,
182 UINT32 flags = CHANNEL_FLAG_FIRST;
184 const char* direction = toBack ?
"F->B" :
"B->F";
185 const size_t currentPacketSize = channelTracker_getCurrentPacketSize(t);
186 wStream* currentPacket = channelTracker_getCurrentPacket(t);
190 WLog_VRB(TAG,
"channelTracker_flushCurrent(%s): %s sz=%" PRIuz
" first=%d last=%d",
191 t->channel->channel_name, direction, Stream_GetPosition(currentPacket), first, last);
194 return PF_CHANNEL_RESULT_PASS;
196 proxyData* pdata = t->pdata;
197 pServerStaticChannelContext* channel = t->channel;
199 flags |= CHANNEL_FLAG_LAST;
205 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id);
206 ev.channel_name = channel->channel_name;
207 ev.data = Stream_Buffer(currentPacket);
208 ev.data_len = Stream_GetPosition(currentPacket);
210 ev.total_size = currentPacketSize;
212 pClientContext* pc = proxy_data_get_client_context(pdata);
213 if (!pc->sendChannelData)
214 return PF_CHANNEL_RESULT_ERROR;
216 return pc->sendChannelData(pc, &ev) ? PF_CHANNEL_RESULT_DROP : PF_CHANNEL_RESULT_ERROR;
219 pServerContext* ps = proxy_data_get_server_context(pdata);
220 r = ps->context.peer->SendChannelPacket(
221 ps->context.peer, WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id),
222 currentPacketSize, flags, Stream_Buffer(currentPacket), Stream_GetPosition(currentPacket));
224 return r ? PF_CHANNEL_RESULT_DROP : PF_CHANNEL_RESULT_ERROR;
228static PfChannelResult pf_channel_generic_back_data(proxyData* pdata,
229 const pServerStaticChannelContext* channel,
230 const BYTE* xdata,
size_t xsize, UINT32 flags,
236 WINPR_ASSERT(channel);
238 switch (channel->channelMode)
240 case PF_UTILS_CHANNEL_PASSTHROUGH:
241 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->back_channel_id);
242 ev.channel_name = channel->channel_name;
246 ev.total_size = totalSize;
248 if (!pf_modules_run_filter(pdata->module, FILTER_TYPE_CLIENT_PASSTHROUGH_CHANNEL_DATA,
250 return PF_CHANNEL_RESULT_DROP;
252 return PF_CHANNEL_RESULT_PASS;
254 case PF_UTILS_CHANNEL_INTERCEPT:
256 case PF_UTILS_CHANNEL_BLOCK:
258 return PF_CHANNEL_RESULT_DROP;
263static PfChannelResult pf_channel_generic_front_data(proxyData* pdata,
264 const pServerStaticChannelContext* channel,
265 const BYTE* xdata,
size_t xsize, UINT32 flags,
271 WINPR_ASSERT(channel);
273 switch (channel->channelMode)
275 case PF_UTILS_CHANNEL_PASSTHROUGH:
276 ev.channel_id = WINPR_ASSERTING_INT_CAST(UINT16, channel->front_channel_id);
277 ev.channel_name = channel->channel_name;
281 ev.total_size = totalSize;
283 if (!pf_modules_run_filter(pdata->module, FILTER_TYPE_SERVER_PASSTHROUGH_CHANNEL_DATA,
285 return PF_CHANNEL_RESULT_DROP;
287 return PF_CHANNEL_RESULT_PASS;
289 case PF_UTILS_CHANNEL_INTERCEPT:
291 case PF_UTILS_CHANNEL_BLOCK:
293 return PF_CHANNEL_RESULT_DROP;
297BOOL pf_channel_setup_generic(pServerStaticChannelContext* channel)
299 WINPR_ASSERT(channel);
300 channel->onBackData = pf_channel_generic_back_data;
301 channel->onFrontData = pf_channel_generic_front_data;
305BOOL channelTracker_setMode(ChannelStateTracker* tracker, ChannelTrackerMode mode)
307 WINPR_ASSERT(tracker);
308 tracker->mode = mode;
312ChannelTrackerMode channelTracker_getMode(ChannelStateTracker* tracker)
314 WINPR_ASSERT(tracker);
315 return tracker->mode;
318BOOL channelTracker_setPData(ChannelStateTracker* tracker, proxyData* pdata)
320 WINPR_ASSERT(tracker);
321 tracker->pdata = pdata;
325proxyData* channelTracker_getPData(ChannelStateTracker* tracker)
327 WINPR_ASSERT(tracker);
328 return tracker->pdata;
331wStream* channelTracker_getCurrentPacket(ChannelStateTracker* tracker)
333 WINPR_ASSERT(tracker);
334 return tracker->currentPacket;
337BOOL channelTracker_setCustomData(ChannelStateTracker* tracker,
void* data)
339 WINPR_ASSERT(tracker);
340 tracker->trackerData = data;
344void* channelTracker_getCustomData(ChannelStateTracker* tracker)
346 WINPR_ASSERT(tracker);
347 return tracker->trackerData;
350size_t channelTracker_getCurrentPacketSize(ChannelStateTracker* tracker)
352 WINPR_ASSERT(tracker);
353 return tracker->currentPacketSize;
356BOOL channelTracker_setCurrentPacketSize(ChannelStateTracker* tracker,
size_t size)
358 WINPR_ASSERT(tracker);
359 tracker->currentPacketSize = size;