|
|
/*++
Copyright (c) 2001 Microsoft Corporation
Module Name:
file.c
Abstract:
This module contains kd host machine file I/O support.
Author:
Drew Bliss (drewb) 21-Feb-2001
Revision History:
--*/
#include "kdp.h"
#pragma hdrstop
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGEKD, KdCreateRemoteFile)
#pragma alloc_text(PAGEKD, KdReadRemoteFile)
#pragma alloc_text(PAGEKD, KdWriteRemoteFile)
#pragma alloc_text(PAGEKD, KdCloseRemoteFile)
#pragma alloc_text(PAGEKD, KdPullRemoteFile)
#pragma alloc_text(PAGEKD, KdPushRemoteFile)
#endif
NTSTATUS KdCreateRemoteFile( OUT PHANDLE Handle, OUT PULONG64 Length, OPTIONAL IN PUNICODE_STRING FileName, IN ACCESS_MASK DesiredAccess, IN ULONG FileAttributes, IN ULONG ShareAccess, IN ULONG CreateDisposition, IN ULONG CreateOptions ) { BOOLEAN Enable; DBGKD_FILE_IO Irp; ULONG Index;
if (FileName->Length > PACKET_MAX_SIZE - sizeof(Irp)) { return STATUS_INVALID_PARAMETER; } if (KdDebuggerNotPresent != FALSE) { return STATUS_DEBUGGER_INACTIVE; } Enable = KdEnterDebugger(NULL, NULL);
//
// Look for an open slot.
//
for (Index = 0; Index < KD_MAX_REMOTE_FILES; Index++) { if (KdpRemoteFiles[Index].RemoteHandle == 0) { break; } }
if (Index >= KD_MAX_REMOTE_FILES) { Irp.Status = STATUS_NO_MEMORY; goto Exit; }
Irp.ApiNumber = DbgKdCreateFileApi; Irp.u.CreateFile.DesiredAccess = DesiredAccess; Irp.u.CreateFile.FileAttributes = FileAttributes; Irp.u.CreateFile.ShareAccess = ShareAccess; Irp.u.CreateFile.CreateDisposition = CreateDisposition; Irp.u.CreateFile.CreateOptions = CreateOptions;
for (;;) { STRING MessageData; STRING MessageHeader; ULONG ReturnCode; ULONG Length;
MessageHeader.Length = sizeof(Irp); MessageHeader.MaximumLength = sizeof(Irp); MessageHeader.Buffer = (PCHAR)&Irp;
// Copy the filename to the message buffer
// so that a terminator can be added.
KdpCopyFromPtr(KdpMessageBuffer, FileName->Buffer, FileName->Length, &Length); MessageData.Length = (USHORT)Length + sizeof(WCHAR); MessageData.Buffer = KdpMessageBuffer; *(PWCHAR)&MessageData.Buffer[MessageData.Length - sizeof(WCHAR)] = UNICODE_NULL; //
// Send packet to the kernel debugger on the host machine.
//
KdSendPacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &KdpContext);
if (KdDebuggerNotPresent != FALSE) { Irp.Status = STATUS_DEBUGGER_INACTIVE; break; } //
// Receive packet from the kernel debugger on the host machine.
//
MessageData.MaximumLength = KDP_MESSAGE_BUFFER_SIZE; MessageData.Buffer = KdpMessageBuffer;
do { ReturnCode = KdReceivePacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &Length, &KdpContext); } while (ReturnCode == KDP_PACKET_TIMEOUT);
if (ReturnCode == KDP_PACKET_RECEIVED) { break; } } if (NT_SUCCESS(Irp.Status)) { KdpRemoteFiles[Index].RemoteHandle = Irp.u.CreateFile.Handle; // Add one so that zero is reserved for invalid-handle.
*Handle = UlongToHandle(Index + 1); if (ARGUMENT_PRESENT(Length)) { *Length = Irp.u.CreateFile.Length; } } Exit: KdExitDebugger(Enable); return Irp.Status; }
NTSTATUS KdReadRemoteFile( IN HANDLE Handle, IN ULONG64 Offset, OUT PVOID Buffer, IN ULONG Length, OUT PULONG Completed ) { BOOLEAN Enable; DBGKD_FILE_IO Irp; ULONG Index; ULONG _Completed = 0;
Index = HandleToUlong(Handle) - 1; if (Index >= KD_MAX_REMOTE_FILES) { return STATUS_INVALID_PARAMETER; } Enable = KdEnterDebugger(NULL, NULL);
if (KdpRemoteFiles[Index].RemoteHandle == 0) { Irp.Status = STATUS_INVALID_PARAMETER; goto Exit; }
Irp.ApiNumber = DbgKdReadFileApi; Irp.Status = STATUS_SUCCESS; Irp.u.ReadFile.Handle = KdpRemoteFiles[Index].RemoteHandle; Irp.u.ReadFile.Offset = Offset;
while (Length > 0) { STRING MessageData; STRING MessageHeader; ULONG ReturnCode; ULONG RecvLength;
if (Length > PACKET_MAX_SIZE - sizeof(Irp)) { Irp.u.ReadFile.Length = PACKET_MAX_SIZE - sizeof(Irp); } else { Irp.u.ReadFile.Length = Length; } MessageHeader.Length = sizeof(Irp); MessageHeader.MaximumLength = sizeof(Irp); MessageHeader.Buffer = (PCHAR)&Irp; //
// Send packet to the kernel debugger on the host machine.
//
KdSendPacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, NULL, &KdpContext);
//
// Receive packet from the kernel debugger on the host machine.
//
MessageData.MaximumLength = (USHORT)Irp.u.ReadFile.Length; MessageData.Buffer = Buffer;
do { ReturnCode = KdReceivePacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &RecvLength, &KdpContext); } while (ReturnCode == KDP_PACKET_TIMEOUT);
if (ReturnCode == KDP_PACKET_RECEIVED) { if (!NT_SUCCESS(Irp.Status)) { break; }
_Completed += RecvLength; Buffer = (PVOID)((PUCHAR)Buffer + RecvLength); Irp.u.ReadFile.Offset += RecvLength; Length -= RecvLength; } } *Completed = _Completed; Exit: KdExitDebugger(Enable); return Irp.Status; }
NTSTATUS KdWriteRemoteFile( IN HANDLE Handle, IN ULONG64 Offset, IN PVOID Buffer, IN ULONG Length, OUT PULONG Completed ) { BOOLEAN Enable; DBGKD_FILE_IO Irp; ULONG Index; ULONG _Completed = 0;
Index = HandleToUlong(Handle) - 1; if (Index >= KD_MAX_REMOTE_FILES) { return STATUS_INVALID_PARAMETER; } Enable = KdEnterDebugger(NULL, NULL);
if (KdpRemoteFiles[Index].RemoteHandle == 0) { Irp.Status = STATUS_INVALID_PARAMETER; goto Exit; }
Irp.ApiNumber = DbgKdWriteFileApi; Irp.Status = STATUS_SUCCESS; Irp.u.WriteFile.Handle = KdpRemoteFiles[Index].RemoteHandle; Irp.u.WriteFile.Offset = Offset;
while (Length > 0) { STRING MessageData; STRING MessageHeader; ULONG ReturnCode; ULONG RecvLength;
if (Length > PACKET_MAX_SIZE - sizeof(Irp)) { Irp.u.WriteFile.Length = PACKET_MAX_SIZE - sizeof(Irp); } else { Irp.u.WriteFile.Length = Length; } MessageHeader.Length = sizeof(Irp); MessageHeader.MaximumLength = sizeof(Irp); MessageHeader.Buffer = (PCHAR)&Irp; MessageData.Length = (USHORT)Irp.u.WriteFile.Length; MessageData.Buffer = Buffer;
//
// Send packet to the kernel debugger on the host machine.
//
KdSendPacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &KdpContext);
//
// Receive packet from the kernel debugger on the host machine.
//
MessageData.MaximumLength = KDP_MESSAGE_BUFFER_SIZE; MessageData.Buffer = KdpMessageBuffer;
do { ReturnCode = KdReceivePacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &RecvLength, &KdpContext); } while (ReturnCode == KDP_PACKET_TIMEOUT);
if (ReturnCode == KDP_PACKET_RECEIVED) { if (!NT_SUCCESS(Irp.Status)) { break; }
_Completed += Irp.u.WriteFile.Length; Buffer = (PVOID)((PUCHAR)Buffer + Irp.u.WriteFile.Length); Irp.u.WriteFile.Offset += Irp.u.WriteFile.Length; Length -= Irp.u.WriteFile.Length; } } *Completed = _Completed; Exit: KdExitDebugger(Enable); return Irp.Status; }
NTSTATUS KdCloseRemoteFile( IN HANDLE Handle ) { BOOLEAN Enable; DBGKD_FILE_IO Irp; ULONG Index;
Index = HandleToUlong(Handle) - 1; if (Index >= KD_MAX_REMOTE_FILES) { return STATUS_INVALID_PARAMETER; } Enable = KdEnterDebugger(NULL, NULL);
if (KdpRemoteFiles[Index].RemoteHandle == 0) { Irp.Status = STATUS_INVALID_PARAMETER; goto Exit; }
Irp.ApiNumber = DbgKdCloseFileApi; Irp.u.CloseFile.Handle = KdpRemoteFiles[Index].RemoteHandle;
for (;;) { STRING MessageData; STRING MessageHeader; ULONG ReturnCode; ULONG RecvLength;
MessageHeader.Length = sizeof(Irp); MessageHeader.MaximumLength = sizeof(Irp); MessageHeader.Buffer = (PCHAR)&Irp; //
// Send packet to the kernel debugger on the host machine.
//
KdSendPacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, NULL, &KdpContext);
//
// Receive packet from the kernel debugger on the host machine.
//
MessageData.MaximumLength = KDP_MESSAGE_BUFFER_SIZE; MessageData.Buffer = KdpMessageBuffer;
do { ReturnCode = KdReceivePacket(PACKET_TYPE_KD_FILE_IO, &MessageHeader, &MessageData, &RecvLength, &KdpContext); } while (ReturnCode == KDP_PACKET_TIMEOUT);
if (ReturnCode == KDP_PACKET_RECEIVED) { break; } } if (NT_SUCCESS(Irp.Status)) { KdpRemoteFiles[Index].RemoteHandle = 0; } Exit: KdExitDebugger(Enable); return Irp.Status; }
#define TRANSFER_LENGTH 8192
NTSTATUS KdPullRemoteFile( IN PUNICODE_STRING FileName, IN ULONG FileAttributes, IN ULONG CreateDisposition, IN ULONG CreateOptions ) { NTSTATUS Status; PVOID Buffer = NULL; ULONG64 Length; HANDLE RemoteHandle = NULL; HANDLE LocalHandle = NULL; OBJECT_ATTRIBUTES ObjectAttributes; IO_STATUS_BLOCK IoStatus; LARGE_INTEGER LargeInt; ULONG64 Offset;
// Allocate a buffer for data transfers.
Buffer = ExAllocatePoolWithTag(NonPagedPool, TRANSFER_LENGTH, 'oIdK'); if (Buffer == NULL) { return STATUS_NO_MEMORY; }
// Open the remote file for reading.
Status = KdCreateRemoteFile(&RemoteHandle, &Length, FileName, FILE_GENERIC_READ, FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ, FILE_OPEN, 0); if (!NT_SUCCESS(Status)) { goto Exit; }
// Open the local file for writing.
LargeInt.QuadPart = Length; InitializeObjectAttributes(&ObjectAttributes, FileName, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, NULL, NULL); Status = ZwCreateFile(&LocalHandle, FILE_GENERIC_WRITE, &ObjectAttributes, &IoStatus, &LargeInt, FileAttributes, 0, CreateDisposition, CreateOptions, NULL, 0); if (!NT_SUCCESS(Status)) { goto Exit; }
// Copy the file contents.
Offset = 0; while (Length > 0) { ULONG ReqLength, ReqCompleted;
if (Length > TRANSFER_LENGTH) { ReqLength = TRANSFER_LENGTH; } else { ReqLength = (ULONG)Length; } Status = KdReadRemoteFile(RemoteHandle, Offset, Buffer, ReqLength, &ReqCompleted); if (!NT_SUCCESS(Status) || ReqCompleted == 0) { break; }
LargeInt.QuadPart = Offset; Status = ZwWriteFile(LocalHandle, NULL, NULL, NULL, &IoStatus, Buffer, ReqCompleted, &LargeInt, NULL); if (!NT_SUCCESS(Status)) { break; } if (IoStatus.Information < ReqCompleted) { Status = STATUS_UNSUCCESSFUL; break; }
Offset += IoStatus.Information; Length -= IoStatus.Information; } Exit: if (RemoteHandle != NULL) { KdCloseRemoteFile(RemoteHandle); } if (LocalHandle != NULL) { ZwClose(LocalHandle); } if (Buffer != NULL) { ExFreePool(Buffer); } return Status; }
NTSTATUS KdPushRemoteFile( IN PUNICODE_STRING FileName, IN ULONG FileAttributes, IN ULONG CreateDisposition, IN ULONG CreateOptions ) { NTSTATUS Status; PVOID Buffer = NULL; ULONG64 Length; HANDLE RemoteHandle = NULL; HANDLE LocalHandle = NULL; OBJECT_ATTRIBUTES ObjectAttributes; IO_STATUS_BLOCK IoStatus; LARGE_INTEGER LargeInt; ULONG64 Offset; FILE_END_OF_FILE_INFORMATION EndOfFile;
// Allocate a buffer for data transfers.
Buffer = ExAllocatePoolWithTag(NonPagedPool, TRANSFER_LENGTH, 'oIdK'); if (Buffer == NULL) { return STATUS_NO_MEMORY; }
// Open the remote file for writing.
Status = KdCreateRemoteFile(&RemoteHandle, &Length, FileName, FILE_GENERIC_WRITE, FileAttributes, 0, CreateDisposition, CreateOptions); if (!NT_SUCCESS(Status)) { goto Exit; }
// Open the local file for reading.
InitializeObjectAttributes(&ObjectAttributes, FileName, OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE, NULL, NULL); Status = ZwOpenFile(&LocalHandle, FILE_GENERIC_READ, &ObjectAttributes, &IoStatus, FILE_SHARE_READ, 0); if (!NT_SUCCESS(Status)) { goto Exit; }
Status = NtQueryInformationFile(LocalHandle, &IoStatus, &EndOfFile, sizeof(EndOfFile), FileEndOfFileInformation); if (!NT_SUCCESS(Status)) { goto Exit; }
// Copy the file contents.
Offset = 0; Length = EndOfFile.EndOfFile.QuadPart; while (Length > 0) { ULONG ReqLength, ReqCompleted;
if (Length > TRANSFER_LENGTH) { ReqLength = TRANSFER_LENGTH; } else { ReqLength = (ULONG)Length; } LargeInt.QuadPart = Offset; Status = ZwReadFile(LocalHandle, NULL, NULL, NULL, &IoStatus, Buffer, ReqLength, &LargeInt, NULL); if (!NT_SUCCESS(Status) || IoStatus.Information == 0) { break; }
Status = KdWriteRemoteFile(RemoteHandle, Offset, Buffer, (ULONG)IoStatus.Information, &ReqCompleted); if (!NT_SUCCESS(Status)) { break; } if (ReqCompleted < IoStatus.Information) { Status = STATUS_UNSUCCESSFUL; break; }
Offset += ReqCompleted; Length -= ReqCompleted; } Exit: if (RemoteHandle != NULL) { KdCloseRemoteFile(RemoteHandle); } if (LocalHandle != NULL) { ZwClose(LocalHandle); } if (Buffer != NULL) { ExFreePool(Buffer); } return Status; }
|