Author: arty
Date: Fri Jan 16 17:44:29 2009
New Revision: 38801
URL: http://svn.reactos.org/svn/reactos?rev=38801&view=rev
Log:
Prevent accessing NULL as a PE header when we have a bad file, found with Stefan100's help
Modified:
trunk/reactos/boot/freeldr/freeldr/reactos/imageldr.c
Modified: trunk/reactos/boot/freeldr/freeldr/reactos/imageldr.c
URL: http://svn.reactos.org/svn/reactos/trunk/reactos/boot/freeldr/freeldr/react…
==============================================================================
--- trunk/reactos/boot/freeldr/freeldr/reactos/imageldr.c [iso-8859-1] (original)
+++ trunk/reactos/boot/freeldr/freeldr/reactos/imageldr.c [iso-8859-1] Fri Jan 16 17:44:29 2009
@@ -481,6 +481,11 @@
/* Get image headers */
NtHeader = RtlImageNtHeader(ReadBuffer);
+ if (!NtHeader)
+ {
+ DbgPrint("Failed to read image (bad PE signature) %s\n", Name);
+ return NULL;
+ }
/* Allocate memory for the driver */
ImageSize = NtHeader->OptionalHeader.SizeOfImage;
Author: sginsberg
Date: Fri Jan 16 10:48:17 2009
New Revision: 38794
URL: http://svn.reactos.org/svn/reactos?rev=38794&view=rev
Log:
- DbgkpPostFakeThreadMessage: Implement missing
- DbgkpSetProcessDebugObject: Fix the traversing of the debug object's event list
- Implement DbgkClearProcessDebugObject
Modified:
trunk/reactos/ntoskrnl/dbgk/dbgkobj.c
Modified: trunk/reactos/ntoskrnl/dbgk/dbgkobj.c
URL: http://svn.reactos.org/svn/reactos/trunk/reactos/ntoskrnl/dbgk/dbgkobj.c?re…
==============================================================================
--- trunk/reactos/ntoskrnl/dbgk/dbgkobj.c [iso-8859-1] (original)
+++ trunk/reactos/ntoskrnl/dbgk/dbgkobj.c [iso-8859-1] Fri Jan 16 10:48:17 2009
@@ -712,9 +712,27 @@
DebugObject);
if (!NT_SUCCESS(Status))
{
- /* We failed. FIXME: Handle this */
- DPRINT1("Unhandled Dbgk codepath!\n");
- ASSERT(FALSE);
+ /* Resume the thread if it was suspended */
+ if (Flags & DEBUG_EVENT_SUSPEND) PsResumeThread(ThisThread, NULL);
+
+ /* Check if we acquired rundown */
+ if (Flags & DEBUG_EVENT_RELEASE)
+ {
+ /* Release it */
+ ExReleaseRundownProtection(&ThisThread->RundownProtect);
+ }
+
+ /* If this was a process create, check if we got a handle */
+ if ((ApiMessage.ApiNumber == DbgKmCreateProcessApi) &&
+ (CreateProcess->FileHandle))
+ {
+ /* Close it */
+ ObCloseHandle(CreateProcess->FileHandle, KernelMode);
+ }
+
+ /* Release our reference and break out */
+ ObDereferenceObject(ThisThread);
+ break;
}
/* Check if this was the first message */
@@ -736,9 +754,10 @@
/* Check the API status */
if (!NT_SUCCESS(Status))
{
- /* We failed. FIXME: Handle this */
- DPRINT1("Unhandled Dbgk codepath!\n");
- ASSERT(FALSE);
+ /* Dereference and fail */
+ if (pFirstThread) ObDereferenceObject(pFirstThread);
+ if (pLastThread) ObDereferenceObject(pLastThread);
+ return Status;
}
/* Make sure we have a first thread */
@@ -1274,8 +1293,9 @@
NextEntry = DebugObject->EventList.Flink;
while (NextEntry != &DebugObject->EventList)
{
- /* Get the debug event */
+ /* Get the debug event and go to the next entry */
DebugEvent = CONTAINING_RECORD(NextEntry, DEBUG_EVENT, EventList);
+ NextEntry = NextEntry->Flink;
DBGKTRACE(DBGK_PROCESS_DEBUG, "DebugEvent: %p Flags: %lx TH: %p/%p\n",
DebugEvent, DebugEvent->Flags,
DebugEvent->BackoutThread, PsGetCurrentThread());
@@ -1339,9 +1359,6 @@
ExReleaseRundownProtection(&EventThread->RundownProtect);
}
}
-
- /* Go to the next entry */
- NextEntry = NextEntry->Flink;
}
/* Release the debug object */
@@ -1358,7 +1375,7 @@
{
/* Remove the event */
NextEntry = RemoveHeadList(&TempList);
- DebugEvent = CONTAINING_RECORD (NextEntry, DEBUG_EVENT, EventList);
+ DebugEvent = CONTAINING_RECORD(NextEntry, DEBUG_EVENT, EventList);
/* Wake it */
DbgkpWakeTarget(DebugEvent);
@@ -1372,12 +1389,88 @@
NTSTATUS
NTAPI
DbgkClearProcessDebugObject(IN PEPROCESS Process,
- IN PDEBUG_OBJECT SourceDebugObject)
-{
- /* FIXME: TODO */
- DBGKTRACE(DBGK_PROCESS_DEBUG, "Process: %p DebugObject: %p\n",
+ IN PDEBUG_OBJECT SourceDebugObject OPTIONAL)
+{
+ PDEBUG_OBJECT DebugObject;
+ PDEBUG_EVENT DebugEvent;
+ LIST_ENTRY TempList;
+ PLIST_ENTRY NextEntry;
+ PAGED_CODE();
+ DBGKTRACE(DBGK_OBJECT_DEBUG, "Process: %p DebugObject: %p\n",
Process, SourceDebugObject);
- return STATUS_UNSUCCESSFUL;
+
+ /* Acquire the port lock */
+ ExAcquireFastMutex(&DbgkpProcessDebugPortMutex);
+
+ /* Get the Process Debug Object */
+ DebugObject = Process->DebugPort;
+
+ /*
+ * Check if the process had an object and it matches,
+ * or if the process had an object but none was specified
+ * (in which we are called from NtTerminateProcess)
+ */
+ if ((DebugObject) &&
+ ((DebugObject == SourceDebugObject) ||
+ (SourceDebugObject == NULL)))
+ {
+ /* Clear the debug port */
+ Process->DebugPort = NULL;
+
+ /* Release the port lock and remove the PEB flag */
+ ExReleaseFastMutex(&DbgkpProcessDebugPortMutex);
+ DbgkpMarkProcessPeb(Process);
+ }
+ else
+ {
+ /* Release the port lock and fail */
+ ExReleaseFastMutex(&DbgkpProcessDebugPortMutex);
+ return STATUS_PORT_NOT_SET;
+ }
+
+ /* Initialize the temporary list */
+ InitializeListHead(&TempList);
+
+ /* Acquire the Object */
+ ExAcquireFastMutex(&DebugObject->Mutex);
+
+ /* Loop the events */
+ NextEntry = DebugObject->EventList.Flink;
+ while (NextEntry != &DebugObject->EventList)
+ {
+ /* Get the Event and go to the next entry */
+ DebugEvent = CONTAINING_RECORD(NextEntry, DEBUG_EVENT, EventList);
+ NextEntry = NextEntry->Flink;
+
+ /* Check that it belongs to the specified process */
+ if (DebugEvent->Process == Process)
+ {
+ /* Insert it into the temporary list */
+ RemoveEntryList(&DebugEvent->EventList);
+ InsertTailList(&TempList, &DebugEvent->EventList);
+ }
+ }
+
+ /* Release the Object */
+ ExReleaseFastMutex(&DebugObject->Mutex);
+
+ /* Release the initial reference */
+ ObDereferenceObject(DebugObject);
+
+ /* Loop our temporary list */
+ while (!IsListEmpty(&TempList))
+ {
+ /* Remove the event */
+ NextEntry = RemoveHeadList(&TempList);
+ DebugEvent = CONTAINING_RECORD(NextEntry, DEBUG_EVENT, EventList);
+
+ /* Wake it up */
+ DebugEvent->Status = STATUS_DEBUGGER_INACTIVE;
+ DbgkpWakeTarget(DebugEvent);
+ }
+
+ /* Return Success */
+ return STATUS_SUCCESS;
}
VOID