84 SCLogError(
"Error creating thread %s: you do not have support for WinDivert "
85 "enabled; please recompile with --enable-windivert",
94 typedef struct WinDivertThreadVars_ {
95 WinDivertHandle filter_handle;
98 int64_t qpc_start_time;
99 int64_t qpc_start_count;
100 int64_t qpc_freq_usec;
104 bool offload_enabled;
107 } WinDivertThreadVars;
109 #define WINDIVERT_MAX_QUEUE 16
110 static WinDivertThreadVars g_wd_tv[WINDIVERT_MAX_QUEUE];
111 static WinDivertQueueVars g_wd_qv[WINDIVERT_MAX_QUEUE];
112 static uint16_t g_wd_num = 0;
115 void *WinDivertGetThread(
int n)
120 return (
void *)&g_wd_tv[n];
123 void *WinDivertGetQueue(
int n)
128 return (
void *)&g_wd_qv[n];
132 #ifndef ERROR_INVALID_IMAGE_HASH
133 #define ERROR_INVALID_IMAGE_HASH 577L
135 #ifndef ERROR_DATA_NOT_ACCEPTED
136 #define ERROR_DATA_NOT_ACCEPTED 592L
143 static const char *WinDivertGetErrorString(DWORD error_code)
145 switch (error_code) {
147 case ERROR_FILE_NOT_FOUND:
148 return "The driver files WinDivert32.sys or WinDivert64.sys were "
150 case ERROR_ACCESS_DENIED:
151 return "Suricata must be run with Administrator privileges.";
152 case ERROR_INVALID_PARAMETER:
153 return "The WinDivert packet filter string is invalid.";
154 case ERROR_INVALID_IMAGE_HASH:
155 return "The WinDivert32.sys or WinDivert64.sys driver does not "
156 "have a valid digital signature, or your copy of Windows is "
157 "not up-to-date. Windows 7 and Server 2008 users need to "
158 "run Windows Update or install the following patch from "
159 "Microsoft: http://support.microsoft.com/kb/2949927";
160 case ERROR_DRIVER_BLOCKED:
161 return "This error occurs for various reasons, including: "
162 "attempting to load the 32-bit WinDivert.sys driver on a "
163 "64-bit system (or vice versa); the WinDivert.sys driver is "
164 "blocked by security software; or you are using a "
165 "virtualization environment that does not support "
167 case EPT_S_NOT_REGISTERED:
168 return "This error occurs when the Base Filtering Engine service "
169 "has been disabled.";
170 case ERROR_PROC_NOT_FOUND:
171 return "The error may occur for Windows Vista users. The "
172 "solution is to install the following patch from Microsoft: "
173 "http://support.microsoft.com/kb/2761494.";
176 case ERROR_HOST_UNREACHABLE:
177 return "This error occurs when an impostor packet (with "
178 "pAddr->Impostor set to 1) is injected and the ip.TTL or "
179 "ipv6.HopLimit field goes to zero. This is a defense of "
180 "last resort against infinite loops caused by impostor "
182 case ERROR_DATA_NOT_ACCEPTED:
183 return "This error is returned when the user application attempts "
184 "to inject a malformed packet. It may also be returned for "
185 "valid inbound packets, and the Windows TCP/IP stack "
186 "rejects the packet for some reason.";
188 return "The underlying cause of this error is unknown. However, "
189 "this error usually occurs when certain kinds of "
190 "anti-virus/firewall/security software is installed, and "
191 "the error message usually resolves once the offending "
192 "program is uninstalled. This suggests a software "
193 "compatibility problem.";
202 #define WinDivertLogError(err_code) \
204 const char *win_err_str = Win32GetErrorString((err_code), NULL); \
205 SCLogError("WinDivertOpen failed, error %" PRId32 " (0x%08" PRIx32 "): %s %s", \
206 (uint32_t)(err_code), (uint32_t)(err_code), win_err_str, \
207 WinDivertGetErrorString(err_code)); \
208 LocalFree((LPVOID)win_err_str); \
215 static void WinDivertInitQPCValues(WinDivertThreadVars *wd_tv)
218 (void)QueryPerformanceCounter((LARGE_INTEGER *)&wd_tv->qpc_start_count);
222 (void)QueryPerformanceFrequency((LARGE_INTEGER *)&wd_tv->qpc_freq_usec);
224 wd_tv->qpc_freq_usec /= 1000 * 1000;
230 static SCTime_t WinDivertTimestampToTimeStamp(WinDivertThreadVars *wd_tv, INT64 timestamp_count)
234 int64_t qpc_delta = (int64_t)timestamp_count - wd_tv->qpc_start_count;
235 int64_t unix_usec = wd_tv->qpc_start_time;
236 if (wd_tv->qpc_freq_usec) {
237 unix_usec += qpc_delta / wd_tv->qpc_freq_usec;
240 tv.tv_sec = (long)(unix_usec / (1000 * 1000));
241 tv.tv_usec = (long)(unix_usec - (int64_t)
tv.tv_sec * (1000 * 1000));
255 int WinDivertRegisterQueue(
bool forward,
char *filter_str)
260 WINDIVERT_LAYER layer =
261 forward ? WINDIVERT_LAYER_NETWORK_FORWARD : WINDIVERT_LAYER_NETWORK;
264 const char *error_str;
266 bool valid = WinDivertHelperCheckFilter(filter_str, layer, &error_str,
269 SCLogWarning(
"Invalid filter \"%s\" supplied to WinDivert: %s at position "
271 filter_str, error_str, error_pos);
278 if (g_wd_num >= WINDIVERT_MAX_QUEUE) {
279 SCLogError(
"Too many WinDivert queues specified %" PRId32
"", g_wd_num);
285 memset(&g_wd_tv, 0,
sizeof(g_wd_tv));
286 memset(&g_wd_qv, 0,
sizeof(g_wd_qv));
290 WinDivertThreadVars *wd_tv = &g_wd_tv[g_wd_num];
291 wd_tv->thread_num = g_wd_num;
294 WinDivertQueueVars *wd_qv = &g_wd_qv[g_wd_num];
295 wd_qv->queue_num = g_wd_num;
297 WinDivertInitQPCValues(wd_tv);
300 size_t filter_len = strlen(filter_str);
302 strlcpy(wd_qv->filter_str, filter_str,
sizeof(wd_qv->filter_str));
303 if (filter_len > copy_len) {
304 SCLogWarning(
"Queue length exceeds storage by %" PRId32
" bytes",
305 (int32_t)(filter_len - copy_len));
310 wd_qv->layer = layer;
326 wd_num_str[
sizeof(wd_num_str) - 1] = 0;
327 snprintf(wd_num_str,
sizeof(wd_num_str),
"%" PRId16
"", g_wd_num);
331 SCLogDebug(
"Queue %" PRId16
" registered", wd_qv->queue_num);
342 void ReceiveWinDivertThreadExitStats(
ThreadVars *,
void *);
357 static TmEcode WinDivertCloseHelper(WinDivertThreadVars *);
359 static TmEcode WinDivertCollectFilterDevices(WinDivertThreadVars *,
360 WinDivertQueueVars *);
361 static bool WinDivertIfaceMatchFilter(
const char *filter_string,
int if_index);
362 static void WinDivertDisableOffloading(WinDivertThreadVars *);
363 static void WinDivertRestoreOffloading(WinDivertThreadVars *);
369 tm_ptr->
name =
"ReceiveWinDivert";
370 tm_ptr->
ThreadInit = ReceiveWinDivertThreadInit;
381 tm_ptr->
name =
"VerdictWinDivert";
382 tm_ptr->
ThreadInit = VerdictWinDivertThreadInit;
383 tm_ptr->
Func = VerdictWinDivert;
391 tm_ptr->
name =
"DecodeWinDivert";
392 tm_ptr->
ThreadInit = DecodeWinDivertThreadInit;
393 tm_ptr->
Func = DecodeWinDivert;
405 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
406 wd_tv->slot = ((
TmSlot *)slot)->slot_next;
432 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
444 "PacketGetFromQueueOrAlloc() - failed to obtain Packet buffer");
452 bool success =
false;
454 if (wd_tv->offload_enabled) {
459 WinDivertRecv(wd_tv->filter_handle, p->
ext_pkt,
464 &p->windivert_v.addr, &pktlen);
479 SCLogInfo(
"WinDivertRecv failed: error %" PRIu32
"",
480 (uint32_t)(GetLastError()));
485 p->
ts = WinDivertTimestampToTimeStamp(wd_tv, p->windivert_v.addr.Timestamp);
486 p->windivert_v.thread_num = wd_tv->thread_num;
499 if (TmThreadsSlotProcessPkt(
tv, wd_tv->slot, p) !=
TM_ECODE_OK) {
521 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)initdata;
528 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
537 if (wd_qv->filter_handle != NULL &&
538 wd_qv->filter_handle != INVALID_HANDLE_VALUE) {
544 if (WinDivertCollectFilterDevices(wd_tv, wd_qv) ==
TM_ECODE_OK) {
545 WinDivertDisableOffloading(wd_tv);
547 SCLogWarning(
"Failed to obtain network devices for WinDivert filter");
553 wd_qv->filter_handle = WinDivertOpen(wd_qv->filter_str, wd_qv->layer,
554 wd_qv->priority, wd_qv->flags);
555 if (wd_qv->filter_handle == INVALID_HANDLE_VALUE) {
556 WinDivertLogError(GetLastError());
563 wd_tv->filter_handle = wd_qv->filter_handle;
580 static TmEcode WinDivertCollectFilterDevices(WinDivertThreadVars *wd_tv,
581 WinDivertQueueVars *wd_qv)
586 IP_ADAPTER_ADDRESSES *if_info_list;
587 DWORD err = (DWORD)Win32GetAdaptersAddresses(&if_info_list);
588 if (err != NO_ERROR) {
593 for (IP_ADAPTER_ADDRESSES *if_info = if_info_list; if_info != NULL;
594 if_info = if_info->Next) {
596 if (WinDivertIfaceMatchFilter(wd_qv->filter_str, if_info->IfIndex)) {
597 SCLogConfig(
"Found adapter %s matching WinDivert filter %s",
598 if_info->AdapterName, wd_qv->filter_str);
601 if (new_ldev == NULL) {
606 if (new_ldev->
dev == NULL) {
612 SCLogDebug(
"Adapter %s does not match WinDivert filter %s",
613 if_info->AdapterName, wd_qv->filter_str);
626 static bool WinDivertIfaceMatchFilter(
const char *filter_string,
int if_index)
630 WINDIVERT_ADDRESS if_addr = {};
631 if_addr.IfIdx = if_index;
633 uint8_t dummy[4] = {4, 4, 4, 4};
635 match = WinDivertHelperEvalFilter(filter_string, WINDIVERT_LAYER_NETWORK,
636 dummy,
sizeof(dummy), &if_addr);
638 int err = GetLastError();
640 SCLogWarning(
"Failed to evaluate filter: 0x%" PRIx32, err);
652 static void WinDivertDisableOffloading(WinDivertThreadVars *wd_tv)
659 wd_tv->offload_enabled =
true;
663 wd_tv->offload_enabled =
true;
674 static void WinDivertRestoreOffloading(WinDivertThreadVars *wd_tv)
693 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
695 SCReturnCT(WinDivertCloseHelper(wd_tv),
"TmEcode");
705 void ReceiveWinDivertThreadExitStats(
ThreadVars *
tv,
void *data)
709 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
710 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
718 SCLogInfo(
"(%s) Packets %" PRIu32
", Bytes %" PRIu64
", Errors %" PRIu32
"",
719 tv->
name, wd_qv->pkts, wd_qv->bytes, wd_qv->errs);
720 SCLogInfo(
"(%s) Verdict: Accepted %" PRIu32
", Dropped %" PRIu32
721 ", Replaced %" PRIu32
"",
722 tv->
name, wd_qv->accepted, wd_qv->dropped, wd_qv->replaced);
737 ret = WinDivertVerdictHelper(
tv, p);
751 WinDivertThreadVars *wd_tv = WinDivertGetThread(p->windivert_v.thread_num);
754 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
757 p->windivert_v.verdicted =
true;
765 if (wd_tv->filter_handle == INVALID_HANDLE_VALUE ||
766 wd_tv->filter_handle == NULL) {
772 if (PacketIsTunnel(p)) {
773 bool finalVerdict = VerdictTunnelPacket(p);
779 if (p->
root != NULL) {
796 bool success = WinDivertSend(wd_tv->filter_handle,
GET_PKT_DATA(p),
800 WinDivertLogError(GetLastError());
822 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)initdata;
836 WinDivertThreadVars *wd_tv = (WinDivertThreadVars *)data;
838 SCReturnCT(WinDivertCloseHelper(wd_tv),
"TmEcode");
868 SCLogDebug(
"packet unsupported by WinDivert, first byte: %02x",
908 static TmEcode WinDivertCloseHelper(WinDivertThreadVars *wd_tv)
913 WinDivertQueueVars *wd_qv = WinDivertGetQueue(wd_tv->thread_num);
915 SCLogDebug(
"No queue could be found for thread num %" PRId32
"",
923 if (wd_qv->filter_handle == INVALID_HANDLE_VALUE ||
924 wd_qv->filter_handle == NULL) {
928 if (!WinDivertClose(wd_qv->filter_handle)) {
929 SCLogError(
"WinDivertClose failed: error %" PRIu32
"", (uint32_t)(GetLastError()));
934 (void)WinDivertRestoreOffloading(wd_tv);
936 wd_qv->filter_handle = NULL;
950 static int SourceWinDivertTestIfaceMatchFilter(
void)
958 struct testdata tests[] = {
960 {
"ifIdx=11", 11,
true},
961 {
"ifIdx==11", 11,
true},
962 {
"ifIdx!=11", 1,
true},
963 {
"ifIdx!=11", 11,
false},
964 {
"ifIdx=3", 4,
false},
965 {
"ifIdx=11 || ifIdx=5", 5,
true},
966 {
"ifIdx=11 || ifIdx=4", 5,
false},
967 {
"ifIdx<3 || ifIdx>7", 8,
true},
968 {
"ifIdx<3 || ifIdx>7", 5,
false},
969 {
"ifIdx>3 or ifIdx<7", 5,
true},
970 {
"ifIdx>3 && ifIdx<7", 5,
true},
971 {
"ifIdx>3 && ifIdx<7", 1,
false},
972 {
"(ifIdx > 3 && ifIdx < 7) or ifIdx == 11", 11,
true}};
974 size_t count = (
sizeof(tests) /
sizeof(tests[0]));
976 for (
size_t i = 0; i < count; i++) {
977 struct testdata test = tests[i];
979 bool actual = WinDivertIfaceMatchFilter(test.filter, test.if_index);
980 if (actual != test.expected) {
981 printf(
"WinDivertIfaceMatchFilter(\"%s\", %d) == %d, expected %d\n",
982 test.filter, test.if_index, actual, test.expected);
993 void SourceWinDivertRegisterTests(
void)
997 SourceWinDivertTestIfaceMatchFilter);