-
Notifications
You must be signed in to change notification settings - Fork 163
/
Copy pathattach-process.ps1
143 lines (113 loc) · 5.2 KB
/
attach-process.ps1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
<#
.PARAMETER service_name
Service to restart (optional)
.PARAMETER path
Path to executable to debug (optional)
.PARAMETER process_name
Process name to debug (required)
.PARAMETER commands
String of windbg commands to be run at startup; separate more than one command with semi-colons (optional)
.EXAMPLE
C:\PS> .\attach-process.ps1 -service-name fastbackserver -process-name fastbackserver -commands 'bp fastbackserver!recvfrom'
Restart the fastback server service and then attach to the fastback server process. Addtionally, set a breakpoint as an initial command.
#>
[CmdletBinding()]
param (
[Parameter()]
[ValidateNotNullOrEmpty()]
[string]
$commands
)
DynamicParam {
# Set the dynamic parameters' name
$svc_param = 'service-name'
# Create the dictionary
$RuntimeParameterDictionary = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary
# Create the collection of attributes
$AttributeCollection = New-Object System.Collections.ObjectModel.Collection[System.Attribute]
# Create and set the parameters' attributes
$ParameterAttribute = New-Object System.Management.Automation.ParameterAttribute
# Add the attributes to the attributes collection
$AttributeCollection.Add($ParameterAttribute)
# Generate and set the ValidateSet
$svc_set = Get-Service | select -ExpandProperty Name
$ValidateSetAttribute = New-Object System.Management.Automation.ValidateSetAttribute($svc_set)
# Add the ValidateSet to the attributes collection
$AttributeCollection.Add($ValidateSetAttribute)
# Create and return the dynamic parameter
$RuntimeParameter = New-Object System.Management.Automation.RuntimeDefinedParameter($svc_param, [string], $AttributeCollection)
$RuntimeParameterDictionary.Add($svc_param, $RuntimeParameter)
# repeat the process for the next dynamic param
$ps_param = 'process-name'
$ps_attrs = New-Object System.Collections.ObjectModel.Collection[System.Attribute]
$ps_paramattr = New-Object System.Management.Automation.ParameterAttribute
$ps_attrs.Add($ps_paramattr)
$ps_rtp = New-Object System.Management.Automation.RuntimeDefinedParameter($ps_param, [string], $ps_attrs)
$RuntimeParameterDictionary.add($ps_param, $ps_rtp)
# adding path name argument
$name_param = 'path'
$name_attrs = New-Object System.Collections.ObjectModel.Collection[System.Attribute]
$name_paramattr = New-Object System.Management.Automation.ParameterAttribute
$name_attrs.Add($name_paramattr)
$name_rtp = New-Object System.Management.Automation.RuntimeDefinedParameter($name_param, [string], $name_attrs)
$RuntimeParameterDictionary.add($name_param, $name_rtp)
return $RuntimeParameterDictionary
}
begin {
$service_name = $PsBoundParameters[$svc_param]
$path = $PsBoundParameters[$name_param]
if ($service_name -and $path) {
Write-Error "Cannot specify -service-name and -path arguments together." -ErrorAction Stop
}
if ($path) {
$path_validate = Test-Path $path
if ($path_validate -eq $false ) {
Write-Error "Supplied -path $path argument does not exist" -ErrorAction Stop
}
Write-Host "[+] Starting $path"
$pathproc = Start-Process -FilePath $path -PassThru
}
if ($service_name) {
$svc = get-service -name $service_name
if ($svc.status -ne 'Running') {
Write-Host "[+] Starting $service_name"
start-service -name $service_name
}
}
$process_name = $PsBoundParameters[$ps_param]
}
process {
$process = Get-Process $process_name
if (-not $process) {
Write-Host "[-] Killing $pathproc"
stop-process $pathproc
Write-Error "Supplied -process-name $process_name not found" -ErrorAction Stop
}
$cmd_args = "-WF c:\windbg_custom.wew -p $($process.id)"
if ($commands) {
$cmd_args += " -c '$commands'"
} else {
$cmd_args += " -g"
}
write-host "[+] Attaching to $process_name"
$winDBGPath = ""
switch (([IntPtr]::Size).ToString()) {
"4" { $winDBGPath = "C:\Program Files\Windows Kits\10\Debuggers\x86\windbg.exe" }
"8" { $winDBGPath = "C:\Program Files (x86)\Windows Kits\10\Debuggers\x86\windbg.exe" }
Default {
Write-Error "Could not determine architecture! Proceeding as x86!"
$winDBGPath = "C:\Program Files\Windows Kits\10\Debuggers\x86\windbg.exe"
}
}
start-process -wait -filepath $winDBGPath -verb RunAs -argumentlist $cmd_args
if ($service_name) {
Do {
# restart the service once we detach from windbg
restart-service -name $service_name -force -erroraction silentlycontinue
$svc = get-service -name $service_name
If ($svc.status -ne 'Running') { Write-Host "Waiting for service $service_name to start" ; Start-Sleep -Milliseconds 250 }
Else { Write-Host "[+] $service_name has been restarted"}
}
Until ($svc.status -eq 'Running')
}
}