84 SCLogError(
"Error creating thread %s: you do not have support for WinDivert "
85 "enabled; please recompile with --enable-windivert",
92 typedef struct WinDivertThreadVars_ {
93 WinDivertHandle filter_handle;
97 int64_t qpc_start_time;
98 int64_t qpc_start_count;
99 int64_t qpc_freq_usec;
103 bool offload_enabled;
106 } WinDivertThreadVars;
108 #define WINDIVERT_MAX_QUEUE 16
109 static WinDivertThreadVars g_wd_tv[WINDIVERT_MAX_QUEUE];
110 static WinDivertQueueVars g_wd_qv[WINDIVERT_MAX_QUEUE];
111 static uint16_t g_wd_num = 0;
114 void *WinDivertGetThread(
int n)
119 return (
void *)&g_wd_tv[n];
122 void *WinDivertGetQueue(
int n)
127 return (
void *)&g_wd_qv[n];
131 #define ERROR_INVALID_IMAGE_HASH 577L
132 #define ERROR_DATA_NOT_ACCEPTED 592L
138 static const char *WinDivertGetErrorString(DWORD error_code)
140 switch (error_code) {
142 case ERROR_FILE_NOT_FOUND:
143 return "The driver files WinDivert32.sys or WinDivert64.sys were "
145 case ERROR_ACCESS_DENIED:
146 return "Suricata must be run with Administrator privileges.";
147 case ERROR_INVALID_PARAMETER:
148 return "The WinDivert packet filter string is invalid.";
149 case ERROR_INVALID_IMAGE_HASH:
150 return "The WinDivert32.sys or WinDivert64.sys driver does not "
151 "have a valid digital signature, or your copy of Windows is "
152 "not up-to-date. Windows 7 and Server 2008 users need to "
153 "run Windows Update or install the following patch from "
154 "Microsoft: http://support.microsoft.com/kb/2949927";
155 case ERROR_DRIVER_BLOCKED:
156 return "This error occurs for various reasons, including: "
157 "attempting to load the 32-bit WinDivert.sys driver on a "
158 "64-bit system (or vice versa); the WinDivert.sys driver is "
159 "blocked by security software; or you are using a "
160 "virtualization environment that does not support "
162 case EPT_S_NOT_REGISTERED:
163 return "This error occurs when the Base Filtering Engine service "
164 "has been disabled.";
165 case ERROR_PROC_NOT_FOUND:
166 return "The error may occur for Windows Vista users. The "
167 "solution is to install the following patch from Microsoft: "
168 "http://support.microsoft.com/kb/2761494.";
171 case ERROR_HOST_UNREACHABLE:
172 return "This error occurs when an impostor packet (with "
173 "pAddr->Impostor set to 1) is injected and the ip.TTL or "
174 "ipv6.HopLimit field goes to zero. This is a defense of "
175 "last resort against infinite loops caused by impostor "
177 case ERROR_DATA_NOT_ACCEPTED:
178 return "This error is returned when the user application attempts "
179 "to inject a malformed packet. It may also be returned for "
180 "valid inbound packets, and the Windows TCP/IP stack "
181 "rejects the packet for some reason.";
183 return "The underlying cause of this error is unknown. However, "
184 "this error usually occurs when certain kinds of "
185 "anti-virus/firewall/security software is installed, and "
186 "the error message usually resolves once the offending "
187 "program is uninstalled. This suggests a software "
188 "compatibility problem.";
197 #define WinDivertLogError(err_code) \
199 const char *win_err_str = Win32GetErrorString((err_code), NULL); \
200 SCLogError("WinDivertOpen failed, error %" PRId32 " (0x%08" PRIx32 "): %s %s", \
201 (uint32_t)(err_code), (uint32_t)(err_code), win_err_str, \
202 WinDivertGetErrorString(err_code)); \
203 LocalFree((LPVOID)win_err_str); \
210 static void WinDivertInitQPCValues(WinDivertThreadVars *wd_tv)
213 (void)QueryPerformanceCounter((LARGE_INTEGER *)&wd_tv->qpc_start_count);
217 (void)QueryPerformanceFrequency((LARGE_INTEGER *)&wd_tv->qpc_freq_usec);
219 wd_tv->qpc_freq_usec /= 1000 * 1000;
225 static SCTime_t WinDivertTimestampToTimeStamp(WinDivertThreadVars *wd_tv, INT64 timestamp_count)
229 int64_t qpc_delta = (int64_t)timestamp_count - wd_tv->qpc_start_count;
231 wd_tv->qpc_start_time + (qpc_delta / wd_tv->qpc_freq_usec);
233 tv.tv_sec = (long)(unix_usec / (1000 * 1000));
234 tv.tv_usec = (long)(unix_usec - (int64_t)
tv.tv_sec * (1000 * 1000));
248 int WinDivertRegisterQueue(
bool forward,
char *filter_str)
253 WINDIVERT_LAYER layer =
254 forward ? WINDIVERT_LAYER_NETWORK_FORWARD : WINDIVERT_LAYER_NETWORK;
257 const char *error_str;
259 bool valid = WinDivertHelperCheckFilter(filter_str, layer, &error_str,
262 SCLogWarning(
"Invalid filter \"%s\" supplied to WinDivert: %s at position "
264 filter_str, error_str, error_pos);
271 if (g_wd_num >= WINDIVERT_MAX_QUEUE) {
272 SCLogError(
"Too many WinDivert queues specified %" PRId32
"", g_wd_num);
278 memset(&g_wd_tv, 0,
sizeof(g_wd_tv));
279 memset(&g_wd_qv, 0,
sizeof(g_wd_qv));
283 WinDivertThreadVars *wd_tv = &g_wd_tv[g_wd_num];
284 wd_tv->thread_num = g_wd_num;
287 WinDivertQueueVars *wd_qv = &g_wd_qv[g_wd_num];
288 wd_qv->queue_num = g_wd_num;
290 WinDivertInitQPCValues(wd_tv);
293 size_t filter_len = strlen(filter_str);
295 strlcpy(wd_qv->filter_str, filter_str,
sizeof(wd_qv->filter_str));
296 if (filter_len > copy_len) {
297 SCLogWarning(
"Queue length exceeds storage by %" PRId32
" bytes",
298 (int32_t)(filter_len - copy_len));
303 wd_qv->layer = layer;
319 wd_num_str[
sizeof(wd_num_str) - 1] = 0;
320 snprintf(wd_num_str,
sizeof(wd_num_str),
"%" PRId16
"", g_wd_num);
324 SCLogDebug(
"Queue %" PRId16
" registered", wd_qv->queue_num);
335 void ReceiveWinDivertThreadExitStats(
ThreadVars *,
void *);
350 static TmEcode WinDivertCloseHelper(WinDivertThreadVars *);
352 static TmEcode WinDivertCollectFilterDevices(WinDivertThreadVars *,
353 WinDivertQueueVars *);
354 static bool WinDivertIfaceMatchFilter(
const char *filter_string,
int if_index);
355 static void WinDivertDisableOffloading(WinDivertThreadVars *);
356 static void WinDivertRestoreOffloading(WinDivertThreadVars *);
362 tm_ptr->
name =
"ReceiveWinDivert";
363 tm_ptr->
ThreadInit = ReceiveWinDivertThreadInit;
374 tm_ptr->
name =
"VerdictWinDivert";
375 tm_ptr->
ThreadInit = VerdictWinDivertThreadInit;
376 tm_ptr->
Func = VerdictWinDivert;
384 tm_ptr->
name =
"DecodeWinDivert";
385 tm_ptr->
ThreadInit = DecodeWinDivertThreadInit;
386 tm_ptr->
Func = DecodeWinDivert;
398 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
399 wd_tv->slot = ((
TmSlot *)slot)->slot_next;
425 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
437 "PacketGetFromQueueOrAlloc() - failed to obtain Packet buffer");
445 bool success =
false;
447 if (wd_tv->offload_enabled) {
452 WinDivertRecv(wd_tv->filter_handle, p->
ext_pkt,
457 &p->windivert_v.addr, &pktlen);
472 SCLogInfo(
"WinDivertRecv failed: error %" PRIu32
"",
473 (uint32_t)(GetLastError()));
478 p->
ts = WinDivertTimestampToTimeStamp(wd_tv, p->windivert_v.addr.Timestamp);
479 p->windivert_v.thread_num = wd_tv->thread_num;
492 if (TmThreadsSlotProcessPkt(
tv, wd_tv->slot, p) !=
TM_ECODE_OK) {
514 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)initdata;
521 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
530 if (wd_qv->filter_handle != NULL &&
531 wd_qv->filter_handle != INVALID_HANDLE_VALUE) {
537 if (WinDivertCollectFilterDevices(wd_tv, wd_qv) ==
TM_ECODE_OK) {
538 WinDivertDisableOffloading(wd_tv);
540 SCLogWarning(
"Failed to obtain network devices for WinDivert filter");
546 wd_qv->filter_handle = WinDivertOpen(wd_qv->filter_str, wd_qv->layer,
547 wd_qv->priority, wd_qv->flags);
548 if (wd_qv->filter_handle == INVALID_HANDLE_VALUE) {
549 WinDivertLogError(GetLastError());
556 wd_tv->filter_handle = wd_qv->filter_handle;
573 static TmEcode WinDivertCollectFilterDevices(WinDivertThreadVars *wd_tv,
574 WinDivertQueueVars *wd_qv)
579 IP_ADAPTER_ADDRESSES *if_info_list;
580 DWORD err = (DWORD)Win32GetAdaptersAddresses(&if_info_list);
581 if (err != NO_ERROR) {
586 for (IP_ADAPTER_ADDRESSES *if_info = if_info_list; if_info != NULL;
587 if_info = if_info->Next) {
589 if (WinDivertIfaceMatchFilter(wd_qv->filter_str, if_info->IfIndex)) {
590 SCLogConfig(
"Found adapter %s matching WinDivert filter %s",
591 if_info->AdapterName, wd_qv->filter_str);
594 if (new_ldev == NULL) {
599 if (new_ldev->
dev == NULL) {
605 SCLogDebug(
"Adapter %s does not match WinDivert filter %s",
606 if_info->AdapterName, wd_qv->filter_str);
619 static bool WinDivertIfaceMatchFilter(
const char *filter_string,
int if_index)
623 WINDIVERT_ADDRESS if_addr = {};
624 if_addr.IfIdx = if_index;
626 uint8_t dummy[4] = {4, 4, 4, 4};
628 match = WinDivertHelperEvalFilter(filter_string, WINDIVERT_LAYER_NETWORK,
629 dummy,
sizeof(dummy), &if_addr);
631 int err = GetLastError();
633 SCLogWarning(
"Failed to evaluate filter: 0x%" PRIx32, err);
645 static void WinDivertDisableOffloading(WinDivertThreadVars *wd_tv)
652 wd_tv->offload_enabled =
true;
656 wd_tv->offload_enabled =
true;
667 static void WinDivertRestoreOffloading(WinDivertThreadVars *wd_tv)
686 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
688 SCReturnCT(WinDivertCloseHelper(wd_tv),
"TmEcode");
698 void ReceiveWinDivertThreadExitStats(
ThreadVars *
tv,
void *data)
702 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
703 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
711 SCLogInfo(
"(%s) Packets %" PRIu32
", Bytes %" PRIu64
", Errors %" PRIu32
"",
712 tv->
name, wd_qv->pkts, wd_qv->bytes, wd_qv->errs);
713 SCLogInfo(
"(%s) Verdict: Accepted %" PRIu32
", Dropped %" PRIu32
714 ", Replaced %" PRIu32
"",
715 tv->
name, wd_qv->accepted, wd_qv->dropped, wd_qv->replaced);
730 ret = WinDivertVerdictHelper(
tv, p);
744 WinDivertThreadVars *wd_tv = WinDivertGetThread(p->windivert_v.thread_num);
750 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
753 p->windivert_v.verdicted =
true;
761 if (wd_tv->filter_handle == INVALID_HANDLE_VALUE ||
762 wd_tv->filter_handle == NULL) {
769 bool finalVerdict = VerdictTunnelPacket(p);
775 if (p->
root != NULL) {
792 bool success = WinDivertSend(wd_tv->filter_handle,
GET_PKT_DATA(p),
796 WinDivertLogError(GetLastError());
818 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)initdata;
835 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
837 SCReturnCT(WinDivertCloseHelper(wd_tv),
"TmEcode");
867 SCLogDebug(
"packet unsupported by WinDivert, first byte: %02x",
907 static TmEcode WinDivertCloseHelper(WinDivertThreadVars *wd_tv)
912 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
914 SCLogDebug(
"No queue could be found for thread num %" PRId32
"",
922 if (wd_qv->filter_handle == INVALID_HANDLE_VALUE ||
923 wd_qv->filter_handle == NULL) {
927 if (!WinDivertClose(wd_qv->filter_handle)) {
928 SCLogError(
"WinDivertClose failed: error %" PRIu32
"", (uint32_t)(GetLastError()));
933 (void)WinDivertRestoreOffloading(wd_tv);
935 wd_qv->filter_handle = NULL;
949 static int SourceWinDivertTestIfaceMatchFilter(
void)
957 struct testdata tests[] = {
959 {
"ifIdx=11", 11,
true},
960 {
"ifIdx==11", 11,
true},
961 {
"ifIdx!=11", 1,
true},
962 {
"ifIdx!=11", 11,
false},
963 {
"ifIdx=3", 4,
false},
964 {
"ifIdx=11 || ifIdx=5", 5,
true},
965 {
"ifIdx=11 || ifIdx=4", 5,
false},
966 {
"ifIdx<3 || ifIdx>7", 8,
true},
967 {
"ifIdx<3 || ifIdx>7", 5,
false},
968 {
"ifIdx>3 or ifIdx<7", 5,
true},
969 {
"ifIdx>3 && ifIdx<7", 5,
true},
970 {
"ifIdx>3 && ifIdx<7", 1,
false},
971 {
"(ifIdx > 3 && ifIdx < 7) or ifIdx == 11", 11,
true}};
973 size_t count = (
sizeof(tests) /
sizeof(tests[0]));
975 for (
size_t i = 0; i < count; i++) {
976 struct testdata test = tests[i];
978 bool actual = WinDivertIfaceMatchFilter(test.filter, test.if_index);
979 if (actual != test.expected) {
980 printf(
"WinDivertIfaceMatchFilter(\"%s\", %d) == %d, expected %d\n",
981 test.filter, test.if_index, actual, test.expected);
992 void SourceWinDivertRegisterTests(
void)
996 SourceWinDivertTestIfaceMatchFilter);