2025-05-13 19:45:22 +03:00

210 lines
5.7 KiB
C

/*
* Process Hacker .NET Tools -
* phsvc extensions
*
* Copyright (C) 2015 wj32
*
* This file is part of Process Hacker.
*
* Process Hacker is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Process Hacker is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Process Hacker. If not, see <http://www.gnu.org/licenses/>.
*/
#include "dn.h"
#include "svcext.h"
#include "clrsup.h"
PPH_STRING CallGetRuntimeNameByAddress(
_In_ HANDLE ProcessId,
_In_ ULONG64 Address,
_Out_opt_ PULONG64 Displacement
)
{
NTSTATUS status;
PH_PLUGIN_PHSVC_CLIENT client;
DN_API_GETRUNTIMENAMEBYADDRESS in;
DN_API_GETRUNTIMENAMEBYADDRESS out;
ULONG bufferSize;
PVOID buffer;
PPH_STRING name = NULL;
if (!PhPluginQueryPhSvc(&client))
return NULL;
in.i.ProcessId = HandleToUlong(ProcessId);
in.i.Address = Address;
bufferSize = 0x1000;
if (!(buffer = client.CreateString(NULL, bufferSize, &in.i.Name)))
return NULL;
status = PhPluginCallPhSvc(PluginInstance, DnGetRuntimeNameByAddressApiNumber, &in, sizeof(in), &out, sizeof(out));
if (status == STATUS_BUFFER_OVERFLOW)
{
client.FreeHeap(buffer);
bufferSize = out.o.NameLength;
if (!(buffer = client.CreateString(NULL, bufferSize, &in.i.Name)))
return NULL;
status = PhPluginCallPhSvc(PluginInstance, DnGetRuntimeNameByAddressApiNumber, &in, sizeof(in), &out, sizeof(out));
}
if (NT_SUCCESS(status))
{
name = PhCreateStringEx(buffer, out.o.NameLength);
if (Displacement)
*Displacement = out.o.Displacement;
}
client.FreeHeap(buffer);
return name;
}
NTSTATUS DispatchGetRuntimeNameByAddress(
_In_ PPH_PLUGIN_PHSVC_REQUEST Request,
_In_ PDN_API_GETRUNTIMENAMEBYADDRESS In,
_Out_ PDN_API_GETRUNTIMENAMEBYADDRESS Out
)
{
NTSTATUS status = STATUS_SUCCESS;
PVOID nameBuffer;
PCLR_PROCESS_SUPPORT support;
PPH_STRING name;
if (!NT_SUCCESS(status = Request->ProbeBuffer(&In->i.Name, sizeof(WCHAR), FALSE, &nameBuffer)))
return status;
support = CreateClrProcessSupport(UlongToHandle(In->i.ProcessId));
if (!support)
return STATUS_UNSUCCESSFUL;
name = GetRuntimeNameByAddressClrProcess(support, In->i.Address, &Out->o.Displacement);
if (!name)
{
status = STATUS_UNSUCCESSFUL;
goto CleanupExit;
}
memcpy(nameBuffer, name->Buffer, min(name->Length, In->i.Name.Length));
Out->o.NameLength = (ULONG)name->Length;
if (In->i.Name.Length < name->Length)
status = STATUS_BUFFER_OVERFLOW;
CleanupExit:
FreeClrProcessSupport(support);
return status;
}
VOID CallPredictAddressesFromClrData(
_In_ HANDLE ProcessId,
_In_ HANDLE ThreadId,
_In_ PVOID PcAddress,
_In_ PVOID FrameAddress,
_In_ PVOID StackAddress,
_Out_ PVOID *PredictedEip,
_Out_ PVOID *PredictedEbp,
_Out_ PVOID *PredictedEsp
)
{
PH_PLUGIN_PHSVC_CLIENT client;
DN_API_PREDICTADDRESSESFROMCLRDATA in;
DN_API_PREDICTADDRESSESFROMCLRDATA out;
*PredictedEip = NULL;
*PredictedEbp = NULL;
*PredictedEsp = NULL;
if (!PhPluginQueryPhSvc(&client))
return;
in.i.ProcessId = HandleToUlong(ProcessId);
in.i.ThreadId = HandleToUlong(ThreadId);
in.i.PcAddress = PtrToUlong(PcAddress);
in.i.FrameAddress = PtrToUlong(FrameAddress);
in.i.StackAddress = PtrToUlong(StackAddress);
if (NT_SUCCESS(PhPluginCallPhSvc(PluginInstance, DnPredictAddressesFromClrDataApiNumber, &in, sizeof(in), &out, sizeof(out))))
{
*PredictedEip = UlongToPtr(out.o.PredictedEip);
*PredictedEbp = UlongToPtr(out.o.PredictedEbp);
*PredictedEsp = UlongToPtr(out.o.PredictedEsp);
}
}
NTSTATUS DispatchPredictAddressesFromClrData(
_In_ PPH_PLUGIN_PHSVC_REQUEST Request,
_In_ PDN_API_PREDICTADDRESSESFROMCLRDATA In,
_Out_ PDN_API_PREDICTADDRESSESFROMCLRDATA Out
)
{
PCLR_PROCESS_SUPPORT support;
PVOID predictedEip;
PVOID predictedEbp;
PVOID predictedEsp;
support = CreateClrProcessSupport(UlongToHandle(In->i.ProcessId));
if (!support)
return STATUS_UNSUCCESSFUL;
PredictAddressesFromClrData(
support,
UlongToHandle(In->i.ThreadId),
UlongToPtr(In->i.PcAddress),
UlongToPtr(In->i.FrameAddress),
UlongToPtr(In->i.StackAddress),
&predictedEip,
&predictedEbp,
&predictedEsp
);
FreeClrProcessSupport(support);
Out->o.PredictedEip = PtrToUlong(predictedEip);
Out->o.PredictedEbp = PtrToUlong(predictedEbp);
Out->o.PredictedEsp = PtrToUlong(predictedEsp);
return STATUS_SUCCESS;
}
VOID DispatchPhSvcRequest(
_In_ PVOID Parameter
)
{
PPH_PLUGIN_PHSVC_REQUEST request = Parameter;
PVOID inBuffer;
// InBuffer can alias OutBuffer, so make a copy of InBuffer.
inBuffer = PhAllocateCopy(request->InBuffer, request->InLength);
switch (request->SubId)
{
case DnGetRuntimeNameByAddressApiNumber:
request->ReturnStatus = DispatchGetRuntimeNameByAddress(request, inBuffer, request->OutBuffer);
break;
case DnPredictAddressesFromClrDataApiNumber:
request->ReturnStatus = DispatchPredictAddressesFromClrData(request, inBuffer, request->OutBuffer);
break;
}
PhFree(inBuffer);
}