2007-01-10 04:27:40 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Kernel
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
2000-06-04 17:27:39 +00:00
|
|
|
* FILE: ntoskrnl/lpc/reply.c
|
2007-01-10 04:27:40 +00:00
|
|
|
* PURPOSE: Local Procedure Call: Receive (Replies)
|
|
|
|
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
|
2000-06-04 17:27:39 +00:00
|
|
|
*/
|
|
|
|
|
2000-12-28 03:38:08 +00:00
|
|
|
/* INCLUDES ******************************************************************/
|
2000-06-04 17:27:39 +00:00
|
|
|
|
2004-08-15 16:39:12 +00:00
|
|
|
#include <ntoskrnl.h>
|
2000-06-04 17:27:39 +00:00
|
|
|
#define NDEBUG
|
2007-01-10 06:09:57 +00:00
|
|
|
#include <debug.h>
|
2000-06-04 17:27:39 +00:00
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/* PRIVATE FUNCTIONS *********************************************************/
|
2001-03-07 16:48:45 +00:00
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
LpcpFreeDataInfoMessage(IN PLPCP_PORT_OBJECT Port,
|
|
|
|
IN ULONG MessageId,
|
2007-01-21 17:21:42 +00:00
|
|
|
IN ULONG CallbackId,
|
|
|
|
IN CLIENT_ID ClientId)
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
|
|
|
PLPCP_MESSAGE Message;
|
|
|
|
PLIST_ENTRY ListHead, NextEntry;
|
|
|
|
|
|
|
|
/* Check if the port we want is the connection port */
|
|
|
|
if ((Port->Flags & LPCP_PORT_TYPE_MASK) > LPCP_UNCONNECTED_PORT)
|
|
|
|
{
|
|
|
|
/* Use it */
|
|
|
|
Port = Port->ConnectionPort;
|
2007-01-21 17:21:42 +00:00
|
|
|
if (!Port) return;
|
2007-01-10 04:27:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Loop the list */
|
|
|
|
ListHead = &Port->LpcDataInfoChainHead;
|
|
|
|
NextEntry = ListHead->Flink;
|
|
|
|
while (ListHead != NextEntry)
|
|
|
|
{
|
|
|
|
/* Get the message */
|
|
|
|
Message = CONTAINING_RECORD(NextEntry, LPCP_MESSAGE, Entry);
|
|
|
|
|
|
|
|
/* Make sure it matches */
|
|
|
|
if ((Message->Request.MessageId == MessageId) &&
|
2007-01-21 17:21:42 +00:00
|
|
|
(Message->Request.ClientId.UniqueThread == ClientId.UniqueThread) &&
|
|
|
|
(Message->Request.ClientId.UniqueProcess == ClientId.UniqueProcess))
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
|
|
|
/* Unlink and free it */
|
|
|
|
RemoveEntryList(&Message->Entry);
|
|
|
|
InitializeListHead(&Message->Entry);
|
2007-01-21 17:21:42 +00:00
|
|
|
LpcpFreeToPortZone(Message, 1);
|
2007-01-10 04:27:40 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Go to the next entry */
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
|
|
}
|
|
|
|
}
|
2000-06-04 17:27:39 +00:00
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
LpcpSaveDataInfoMessage(IN PLPCP_PORT_OBJECT Port,
|
2007-01-18 09:44:49 +00:00
|
|
|
IN PLPCP_MESSAGE Message,
|
2007-01-21 17:21:42 +00:00
|
|
|
IN ULONG LockHeld)
|
2000-06-04 17:27:39 +00:00
|
|
|
{
|
2007-01-10 04:27:40 +00:00
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
/* Acquire the lock */
|
2007-01-21 17:21:42 +00:00
|
|
|
if (!LockHeld) KeAcquireGuardedMutex(&LpcpLock);
|
2007-01-10 04:27:40 +00:00
|
|
|
|
|
|
|
/* Check if the port we want is the connection port */
|
|
|
|
if ((Port->Flags & LPCP_PORT_TYPE_MASK) > LPCP_UNCONNECTED_PORT)
|
|
|
|
{
|
|
|
|
/* Use it */
|
|
|
|
Port = Port->ConnectionPort;
|
2007-01-21 17:21:42 +00:00
|
|
|
if (!Port)
|
|
|
|
{
|
|
|
|
/* Release the lock and return */
|
|
|
|
if (!LockHeld) KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
return;
|
|
|
|
}
|
2007-01-10 04:27:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Link the message */
|
|
|
|
InsertTailList(&Port->LpcDataInfoChainHead, &Message->Entry);
|
|
|
|
|
|
|
|
/* Release the lock */
|
2007-01-21 17:21:42 +00:00
|
|
|
if (!LockHeld) KeReleaseGuardedMutex(&LpcpLock);
|
2000-06-04 17:27:39 +00:00
|
|
|
}
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
LpcpMoveMessage(IN PPORT_MESSAGE Destination,
|
|
|
|
IN PPORT_MESSAGE Origin,
|
|
|
|
IN PVOID Data,
|
|
|
|
IN ULONG MessageType,
|
|
|
|
IN PCLIENT_ID ClientId)
|
|
|
|
{
|
|
|
|
/* Set the Message size */
|
|
|
|
LPCTRACE((LPC_REPLY_DEBUG | LPC_SEND_DEBUG),
|
|
|
|
"Destination/Origin: %p/%p. Data: %p. Length: %lx\n",
|
|
|
|
Destination,
|
|
|
|
Origin,
|
|
|
|
Data,
|
|
|
|
Origin->u1.Length);
|
|
|
|
Destination->u1.Length = Origin->u1.Length;
|
|
|
|
|
|
|
|
/* Set the Message Type */
|
|
|
|
Destination->u2.s2.Type = !MessageType ?
|
|
|
|
Origin->u2.s2.Type : MessageType & 0xFFFF;
|
|
|
|
|
|
|
|
/* Check if we have a Client ID */
|
|
|
|
if (ClientId)
|
|
|
|
{
|
|
|
|
/* Set the Client ID */
|
|
|
|
Destination->ClientId.UniqueProcess = ClientId->UniqueProcess;
|
|
|
|
Destination->ClientId.UniqueThread = ClientId->UniqueThread;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Otherwise, copy it */
|
|
|
|
Destination->ClientId.UniqueProcess = Origin->ClientId.UniqueProcess;
|
|
|
|
Destination->ClientId.UniqueThread = Origin->ClientId.UniqueThread;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the MessageId and ClientViewSize */
|
|
|
|
Destination->MessageId = Origin->MessageId;
|
|
|
|
Destination->ClientViewSize = Origin->ClientViewSize;
|
|
|
|
|
|
|
|
/* Copy the Message Data */
|
2007-01-21 17:21:42 +00:00
|
|
|
RtlCopyMemory(Destination + 1,
|
2007-01-10 04:27:40 +00:00
|
|
|
Data,
|
|
|
|
((Destination->u1.Length & 0xFFFF) + 3) &~3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PUBLIC FUNCTIONS **********************************************************/
|
2000-06-04 17:27:39 +00:00
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/*
|
2009-10-19 16:37:12 +00:00
|
|
|
* @implemented
|
2000-06-04 17:27:39 +00:00
|
|
|
*/
|
2007-01-10 04:27:40 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
NtReplyPort(IN HANDLE PortHandle,
|
2009-10-19 15:49:29 +00:00
|
|
|
IN PPORT_MESSAGE ReplyMessage)
|
2000-06-04 17:27:39 +00:00
|
|
|
{
|
2009-10-19 15:49:29 +00:00
|
|
|
PLPCP_PORT_OBJECT Port, ConnectionPort = NULL;
|
|
|
|
KPROCESSOR_MODE PreviousMode = KeGetPreviousMode();
|
|
|
|
NTSTATUS Status;
|
|
|
|
PLPCP_MESSAGE Message;
|
|
|
|
PETHREAD Thread = PsGetCurrentThread(), WakeupThread;
|
|
|
|
//PORT_MESSAGE CapturedReplyMessage;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
LPCTRACE(LPC_REPLY_DEBUG,
|
|
|
|
"Handle: %lx. Message: %p.\n",
|
|
|
|
PortHandle,
|
|
|
|
ReplyMessage);
|
|
|
|
|
|
|
|
if (KeGetPreviousMode() == UserMode)
|
|
|
|
{
|
|
|
|
_SEH2_TRY
|
|
|
|
{
|
|
|
|
if (ReplyMessage != NULL)
|
|
|
|
{
|
|
|
|
ProbeForRead(ReplyMessage, sizeof(PORT_MESSAGE), sizeof(ULONG));
|
|
|
|
/*RtlCopyMemory(&CapturedReplyMessage, ReplyMessage, sizeof(PORT_MESSAGE));
|
|
|
|
ReplyMessage = &CapturedReplyMessage;*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
|
|
|
{
|
|
|
|
DPRINT1("SEH crash [1]\n");
|
|
|
|
DbgBreakPoint();
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
|
|
|
}
|
|
|
|
_SEH2_END;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate its length */
|
|
|
|
if (((ULONG)ReplyMessage->u1.s1.DataLength + sizeof(PORT_MESSAGE)) >
|
|
|
|
(ULONG)ReplyMessage->u1.s1.TotalLength)
|
|
|
|
{
|
|
|
|
/* Fail */
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure it has a valid ID */
|
|
|
|
if (!ReplyMessage->MessageId) return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
/* Get the Port object */
|
|
|
|
Status = ObReferenceObjectByHandle(PortHandle,
|
|
|
|
0,
|
|
|
|
LpcPortObjectType,
|
|
|
|
PreviousMode,
|
|
|
|
(PVOID*)&Port,
|
|
|
|
NULL);
|
|
|
|
if (!NT_SUCCESS(Status)) return Status;
|
|
|
|
|
|
|
|
/* Validate its length in respect to the port object */
|
|
|
|
if (((ULONG)ReplyMessage->u1.s1.TotalLength > Port->MaxMessageLength) ||
|
|
|
|
((ULONG)ReplyMessage->u1.s1.TotalLength <=
|
|
|
|
(ULONG)ReplyMessage->u1.s1.DataLength))
|
|
|
|
{
|
|
|
|
/* Too large, fail */
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_PORT_MESSAGE_TOO_LONG;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the ETHREAD corresponding to it */
|
|
|
|
Status = PsLookupProcessThreadByCid(&ReplyMessage->ClientId,
|
|
|
|
NULL,
|
|
|
|
&WakeupThread);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* No thread found, fail */
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate a message from the port zone */
|
|
|
|
Message = LpcpAllocateFromPortZone();
|
|
|
|
if (!Message)
|
|
|
|
{
|
|
|
|
/* Fail if we couldn't allocate a message */
|
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Keep the lock acquired */
|
|
|
|
KeAcquireGuardedMutex(&LpcpLock);
|
|
|
|
|
|
|
|
/* Make sure this is the reply the thread is waiting for */
|
|
|
|
if ((WakeupThread->LpcReplyMessageId != ReplyMessage->MessageId) ||
|
|
|
|
((LpcpGetMessageFromThread(WakeupThread)) &&
|
|
|
|
(LpcpGetMessageType(&LpcpGetMessageFromThread(WakeupThread)->
|
|
|
|
Request) != LPC_REQUEST)))
|
|
|
|
{
|
|
|
|
/* It isn't, fail */
|
|
|
|
LpcpFreeToPortZone(Message, 3);
|
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_REPLY_MESSAGE_MISMATCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the message */
|
|
|
|
_SEH2_TRY
|
|
|
|
{
|
|
|
|
LpcpMoveMessage(&Message->Request,
|
|
|
|
ReplyMessage,
|
|
|
|
ReplyMessage + 1,
|
|
|
|
LPC_REPLY,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
|
|
{
|
|
|
|
/* Fail */
|
|
|
|
LpcpFreeToPortZone(Message, 3);
|
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
|
|
|
}
|
|
|
|
_SEH2_END;
|
|
|
|
|
|
|
|
/* Reference the thread while we use it */
|
|
|
|
ObReferenceObject(WakeupThread);
|
|
|
|
Message->RepliedToThread = WakeupThread;
|
|
|
|
|
|
|
|
/* Set this as the reply message */
|
|
|
|
WakeupThread->LpcReplyMessageId = 0;
|
|
|
|
WakeupThread->LpcReplyMessage = (PVOID)Message;
|
|
|
|
|
|
|
|
/* Check if we have messages on the reply chain */
|
|
|
|
if (!(WakeupThread->LpcExitThreadCalled) &&
|
|
|
|
!(IsListEmpty(&WakeupThread->LpcReplyChain)))
|
|
|
|
{
|
|
|
|
/* Remove us from it and reinitialize it */
|
|
|
|
RemoveEntryList(&WakeupThread->LpcReplyChain);
|
|
|
|
InitializeListHead(&WakeupThread->LpcReplyChain);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if this is the message the thread had received */
|
|
|
|
if ((Thread->LpcReceivedMsgIdValid) &&
|
|
|
|
(Thread->LpcReceivedMessageId == ReplyMessage->MessageId))
|
|
|
|
{
|
|
|
|
/* Clear this data */
|
|
|
|
Thread->LpcReceivedMessageId = 0;
|
|
|
|
Thread->LpcReceivedMsgIdValid = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Free any data information */
|
|
|
|
LpcpFreeDataInfoMessage(Port,
|
|
|
|
ReplyMessage->MessageId,
|
|
|
|
ReplyMessage->CallbackId,
|
|
|
|
ReplyMessage->ClientId);
|
|
|
|
|
|
|
|
/* Release the lock and release the LPC semaphore to wake up waiters */
|
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
LpcpCompleteWait(&WakeupThread->LpcReplySemaphore);
|
|
|
|
|
|
|
|
/* Now we can let go of the thread */
|
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
|
|
|
|
/* Dereference port object */
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return Status;
|
2000-06-04 17:27:39 +00:00
|
|
|
}
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
2000-06-04 17:27:39 +00:00
|
|
|
*/
|
2007-01-10 04:27:40 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2005-09-04 18:00:59 +00:00
|
|
|
NtReplyWaitReceivePortEx(IN HANDLE PortHandle,
|
|
|
|
OUT PVOID *PortContext OPTIONAL,
|
|
|
|
IN PPORT_MESSAGE ReplyMessage OPTIONAL,
|
|
|
|
OUT PPORT_MESSAGE ReceiveMessage,
|
2007-01-10 04:27:40 +00:00
|
|
|
IN PLARGE_INTEGER Timeout OPTIONAL)
|
2000-06-04 17:27:39 +00:00
|
|
|
{
|
2007-01-21 17:21:42 +00:00
|
|
|
PLPCP_PORT_OBJECT Port, ReceivePort, ConnectionPort = NULL;
|
2007-01-10 04:27:40 +00:00
|
|
|
KPROCESSOR_MODE PreviousMode = KeGetPreviousMode(), WaitMode = PreviousMode;
|
2009-08-24 19:58:15 +00:00
|
|
|
NTSTATUS Status;
|
2007-01-10 04:27:40 +00:00
|
|
|
PLPCP_MESSAGE Message;
|
|
|
|
PETHREAD Thread = PsGetCurrentThread(), WakeupThread;
|
|
|
|
PLPCP_CONNECTION_MESSAGE ConnectMessage;
|
|
|
|
ULONG ConnectionInfoLength;
|
2008-05-11 12:22:51 +00:00
|
|
|
//PORT_MESSAGE CapturedReplyMessage;
|
2008-05-11 09:39:26 +00:00
|
|
|
LARGE_INTEGER CapturedTimeout;
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
PAGED_CODE();
|
|
|
|
LPCTRACE(LPC_REPLY_DEBUG,
|
|
|
|
"Handle: %lx. Messages: %p/%p. Context: %p\n",
|
|
|
|
PortHandle,
|
|
|
|
ReplyMessage,
|
|
|
|
ReceiveMessage,
|
|
|
|
PortContext);
|
|
|
|
|
2008-05-11 09:39:26 +00:00
|
|
|
if (KeGetPreviousMode() == UserMode)
|
|
|
|
{
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2008-05-11 09:39:26 +00:00
|
|
|
{
|
|
|
|
if (ReplyMessage != NULL)
|
|
|
|
{
|
|
|
|
ProbeForRead(ReplyMessage, sizeof(PORT_MESSAGE), sizeof(ULONG));
|
2008-05-11 12:22:51 +00:00
|
|
|
/*RtlCopyMemory(&CapturedReplyMessage, ReplyMessage, sizeof(PORT_MESSAGE));
|
|
|
|
ReplyMessage = &CapturedReplyMessage;*/
|
2008-05-11 09:39:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Timeout != NULL)
|
|
|
|
{
|
|
|
|
ProbeForReadLargeInteger(Timeout);
|
|
|
|
RtlCopyMemory(&CapturedTimeout, Timeout, sizeof(LARGE_INTEGER));
|
|
|
|
Timeout = &CapturedTimeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PortContext != NULL)
|
|
|
|
ProbeForWritePointer(PortContext);
|
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2008-05-11 09:39:26 +00:00
|
|
|
{
|
2008-05-11 12:22:51 +00:00
|
|
|
DPRINT1("SEH crash [1]\n");
|
2009-08-24 19:58:15 +00:00
|
|
|
DbgBreakPoint();
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
2008-05-11 09:39:26 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2008-05-11 09:39:26 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* If this is a system thread, then let it page out its stack */
|
|
|
|
if (Thread->SystemThread) WaitMode = UserMode;
|
|
|
|
}
|
2007-01-10 04:27:40 +00:00
|
|
|
|
|
|
|
/* Check if caller has a reply message */
|
|
|
|
if (ReplyMessage)
|
|
|
|
{
|
|
|
|
/* Validate its length */
|
2007-08-04 08:49:47 +00:00
|
|
|
if (((ULONG)ReplyMessage->u1.s1.DataLength + sizeof(PORT_MESSAGE)) >
|
|
|
|
(ULONG)ReplyMessage->u1.s1.TotalLength)
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
|
|
|
/* Fail */
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure it has a valid ID */
|
|
|
|
if (!ReplyMessage->MessageId) return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the Port object */
|
|
|
|
Status = ObReferenceObjectByHandle(PortHandle,
|
|
|
|
0,
|
|
|
|
LpcPortObjectType,
|
|
|
|
PreviousMode,
|
|
|
|
(PVOID*)&Port,
|
|
|
|
NULL);
|
|
|
|
if (!NT_SUCCESS(Status)) return Status;
|
|
|
|
|
|
|
|
/* Check if the caller has a reply message */
|
|
|
|
if (ReplyMessage)
|
|
|
|
{
|
|
|
|
/* Validate its length in respect to the port object */
|
2007-08-04 08:49:47 +00:00
|
|
|
if (((ULONG)ReplyMessage->u1.s1.TotalLength > Port->MaxMessageLength) ||
|
|
|
|
((ULONG)ReplyMessage->u1.s1.TotalLength <=
|
|
|
|
(ULONG)ReplyMessage->u1.s1.DataLength))
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
|
|
|
/* Too large, fail */
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_PORT_MESSAGE_TOO_LONG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if this is anything but a client port */
|
|
|
|
if ((Port->Flags & LPCP_PORT_TYPE_MASK) != LPCP_CLIENT_PORT)
|
|
|
|
{
|
2007-01-21 17:21:42 +00:00
|
|
|
/* Check if this is the connection port */
|
|
|
|
if (Port->ConnectionPort == Port)
|
|
|
|
{
|
|
|
|
/* Use this port */
|
|
|
|
ConnectionPort = ReceivePort = Port;
|
|
|
|
ObReferenceObject(ConnectionPort);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Acquire the lock */
|
|
|
|
KeAcquireGuardedMutex(&LpcpLock);
|
|
|
|
|
|
|
|
/* Get the port */
|
|
|
|
ConnectionPort = ReceivePort = Port->ConnectionPort;
|
|
|
|
if (!ConnectionPort)
|
|
|
|
{
|
|
|
|
/* Fail */
|
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_PORT_DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release lock and reference */
|
2007-06-11 01:24:41 +00:00
|
|
|
ObReferenceObject(ConnectionPort);
|
2007-01-21 17:21:42 +00:00
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
}
|
2007-01-10 04:27:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Otherwise, use the port itself */
|
|
|
|
ReceivePort = Port;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if the caller gave a reply message */
|
|
|
|
if (ReplyMessage)
|
|
|
|
{
|
|
|
|
/* Get the ETHREAD corresponding to it */
|
|
|
|
Status = PsLookupProcessThreadByCid(&ReplyMessage->ClientId,
|
|
|
|
NULL,
|
|
|
|
&WakeupThread);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* No thread found, fail */
|
|
|
|
ObDereferenceObject(Port);
|
2007-01-21 17:21:42 +00:00
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
2007-01-10 04:27:40 +00:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate a message from the port zone */
|
|
|
|
Message = LpcpAllocateFromPortZone();
|
|
|
|
if (!Message)
|
|
|
|
{
|
|
|
|
/* Fail if we couldn't allocate a message */
|
2007-01-21 17:21:42 +00:00
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
2007-01-10 04:27:40 +00:00
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Keep the lock acquired */
|
|
|
|
KeAcquireGuardedMutex(&LpcpLock);
|
|
|
|
|
|
|
|
/* Make sure this is the reply the thread is waiting for */
|
2007-01-22 08:07:24 +00:00
|
|
|
if ((WakeupThread->LpcReplyMessageId != ReplyMessage->MessageId) ||
|
|
|
|
((LpcpGetMessageFromThread(WakeupThread)) &&
|
|
|
|
(LpcpGetMessageType(&LpcpGetMessageFromThread(WakeupThread)->
|
|
|
|
Request) != LPC_REQUEST)))
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
|
|
|
/* It isn't, fail */
|
2007-01-21 17:21:42 +00:00
|
|
|
LpcpFreeToPortZone(Message, 3);
|
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
2007-01-10 04:27:40 +00:00
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_REPLY_MESSAGE_MISMATCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the message */
|
|
|
|
LpcpMoveMessage(&Message->Request,
|
|
|
|
ReplyMessage,
|
|
|
|
ReplyMessage + 1,
|
|
|
|
LPC_REPLY,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* Reference the thread while we use it */
|
|
|
|
ObReferenceObject(WakeupThread);
|
|
|
|
Message->RepliedToThread = WakeupThread;
|
|
|
|
|
|
|
|
/* Set this as the reply message */
|
|
|
|
WakeupThread->LpcReplyMessageId = 0;
|
|
|
|
WakeupThread->LpcReplyMessage = (PVOID)Message;
|
|
|
|
|
|
|
|
/* Check if we have messages on the reply chain */
|
|
|
|
if (!(WakeupThread->LpcExitThreadCalled) &&
|
|
|
|
!(IsListEmpty(&WakeupThread->LpcReplyChain)))
|
|
|
|
{
|
|
|
|
/* Remove us from it and reinitialize it */
|
|
|
|
RemoveEntryList(&WakeupThread->LpcReplyChain);
|
|
|
|
InitializeListHead(&WakeupThread->LpcReplyChain);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if this is the message the thread had received */
|
|
|
|
if ((Thread->LpcReceivedMsgIdValid) &&
|
|
|
|
(Thread->LpcReceivedMessageId == ReplyMessage->MessageId))
|
|
|
|
{
|
|
|
|
/* Clear this data */
|
|
|
|
Thread->LpcReceivedMessageId = 0;
|
|
|
|
Thread->LpcReceivedMsgIdValid = FALSE;
|
|
|
|
}
|
|
|
|
|
2007-01-21 17:21:42 +00:00
|
|
|
/* Free any data information */
|
|
|
|
LpcpFreeDataInfoMessage(Port,
|
|
|
|
ReplyMessage->MessageId,
|
|
|
|
ReplyMessage->CallbackId,
|
|
|
|
ReplyMessage->ClientId);
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/* Release the lock and release the LPC semaphore to wake up waiters */
|
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
LpcpCompleteWait(&WakeupThread->LpcReplySemaphore);
|
|
|
|
|
|
|
|
/* Now we can let go of the thread */
|
|
|
|
ObDereferenceObject(WakeupThread);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now wait for someone to reply to us */
|
|
|
|
LpcpReceiveWait(ReceivePort->MsgQueue.Semaphore, WaitMode);
|
|
|
|
if (Status != STATUS_SUCCESS) goto Cleanup;
|
|
|
|
|
|
|
|
/* Wait done, get the LPC lock */
|
|
|
|
KeAcquireGuardedMutex(&LpcpLock);
|
|
|
|
|
|
|
|
/* Check if we've received nothing */
|
|
|
|
if (IsListEmpty(&ReceivePort->MsgQueue.ReceiveHead))
|
|
|
|
{
|
|
|
|
/* Check if this was a waitable port and wake it */
|
|
|
|
if (ReceivePort->Flags & LPCP_WAITABLE_PORT)
|
|
|
|
{
|
|
|
|
/* Reset its event */
|
|
|
|
KeResetEvent(&ReceivePort->WaitEvent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release the lock and fail */
|
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
2007-01-21 17:21:42 +00:00
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
2007-01-10 04:27:40 +00:00
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the message on the queue */
|
|
|
|
Message = CONTAINING_RECORD(RemoveHeadList(&ReceivePort->
|
|
|
|
MsgQueue.ReceiveHead),
|
|
|
|
LPCP_MESSAGE,
|
|
|
|
Entry);
|
|
|
|
|
|
|
|
/* Check if the queue is empty now */
|
|
|
|
if (IsListEmpty(&ReceivePort->MsgQueue.ReceiveHead))
|
|
|
|
{
|
|
|
|
/* Check if this was a waitable port */
|
|
|
|
if (ReceivePort->Flags & LPCP_WAITABLE_PORT)
|
|
|
|
{
|
|
|
|
/* Reset its event */
|
|
|
|
KeResetEvent(&ReceivePort->WaitEvent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Re-initialize the message's list entry */
|
|
|
|
InitializeListHead(&Message->Entry);
|
|
|
|
|
|
|
|
/* Set this as the received message */
|
|
|
|
Thread->LpcReceivedMessageId = Message->Request.MessageId;
|
|
|
|
Thread->LpcReceivedMsgIdValid = TRUE;
|
|
|
|
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
2008-05-11 09:39:26 +00:00
|
|
|
/* Check if this was a connection request */
|
|
|
|
if (LpcpGetMessageType(&Message->Request) == LPC_CONNECTION_REQUEST)
|
|
|
|
{
|
|
|
|
/* Get the connection message */
|
|
|
|
ConnectMessage = (PLPCP_CONNECTION_MESSAGE)(Message + 1);
|
|
|
|
LPCTRACE(LPC_REPLY_DEBUG,
|
|
|
|
"Request Messages: %p/%p\n",
|
|
|
|
Message,
|
|
|
|
ConnectMessage);
|
2007-01-10 04:27:40 +00:00
|
|
|
|
2008-05-11 09:39:26 +00:00
|
|
|
/* Get its length */
|
|
|
|
ConnectionInfoLength = Message->Request.u1.s1.DataLength -
|
|
|
|
sizeof(LPCP_CONNECTION_MESSAGE);
|
2007-01-10 04:27:40 +00:00
|
|
|
|
2008-05-11 09:39:26 +00:00
|
|
|
/* Return it as the receive message */
|
|
|
|
*ReceiveMessage = Message->Request;
|
|
|
|
|
|
|
|
/* Clear our stack variable so the message doesn't get freed */
|
2007-01-10 04:27:40 +00:00
|
|
|
Message = NULL;
|
2008-05-11 09:39:26 +00:00
|
|
|
|
|
|
|
/* Setup the receive message */
|
|
|
|
ReceiveMessage->u1.s1.TotalLength = (CSHORT)(sizeof(LPCP_MESSAGE) +
|
|
|
|
ConnectionInfoLength);
|
|
|
|
ReceiveMessage->u1.s1.DataLength = (CSHORT)ConnectionInfoLength;
|
|
|
|
RtlCopyMemory(ReceiveMessage + 1,
|
|
|
|
ConnectMessage + 1,
|
|
|
|
ConnectionInfoLength);
|
|
|
|
|
|
|
|
/* Clear the port context if the caller requested one */
|
|
|
|
if (PortContext) *PortContext = NULL;
|
|
|
|
}
|
|
|
|
else if (LpcpGetMessageType(&Message->Request) != LPC_REPLY)
|
|
|
|
{
|
|
|
|
/* Otherwise, this is a new message or event */
|
|
|
|
LPCTRACE(LPC_REPLY_DEBUG,
|
|
|
|
"Non-Reply Messages: %p/%p\n",
|
|
|
|
&Message->Request,
|
|
|
|
(&Message->Request) + 1);
|
|
|
|
|
|
|
|
/* Copy it */
|
|
|
|
LpcpMoveMessage(ReceiveMessage,
|
|
|
|
&Message->Request,
|
|
|
|
(&Message->Request) + 1,
|
|
|
|
0,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* Return its context */
|
|
|
|
if (PortContext) *PortContext = Message->PortContext;
|
|
|
|
|
|
|
|
/* And check if it has data information */
|
|
|
|
if (Message->Request.u2.s2.DataInfoOffset)
|
|
|
|
{
|
|
|
|
/* It does, save it, and don't free the message below */
|
|
|
|
LpcpSaveDataInfoMessage(Port, Message, 1);
|
|
|
|
Message = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* This is a reply message, should never happen! */
|
|
|
|
ASSERT(FALSE);
|
2007-01-10 04:27:40 +00:00
|
|
|
}
|
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2007-01-10 04:27:40 +00:00
|
|
|
{
|
2008-05-11 12:22:51 +00:00
|
|
|
DPRINT1("SEH crash [2]\n");
|
|
|
|
DbgBreakPoint();
|
2008-11-24 13:40:26 +00:00
|
|
|
Status = _SEH2_GetExceptionCode();
|
2007-01-10 04:27:40 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2007-01-10 04:27:40 +00:00
|
|
|
|
2007-01-21 17:21:42 +00:00
|
|
|
/* Check if we have a message pointer here */
|
|
|
|
if (Message)
|
|
|
|
{
|
|
|
|
/* Free it and release the lock */
|
|
|
|
LpcpFreeToPortZone(Message, 3);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Just release the lock */
|
|
|
|
KeReleaseGuardedMutex(&LpcpLock);
|
|
|
|
}
|
2007-01-10 04:27:40 +00:00
|
|
|
|
|
|
|
Cleanup:
|
|
|
|
/* All done, dereference the port and return the status */
|
|
|
|
LPCTRACE(LPC_REPLY_DEBUG,
|
|
|
|
"Port: %p. Status: %p\n",
|
|
|
|
Port,
|
|
|
|
Status);
|
2007-01-21 17:21:42 +00:00
|
|
|
if (ConnectionPort) ObDereferenceObject(ConnectionPort);
|
2007-01-10 04:27:40 +00:00
|
|
|
ObDereferenceObject(Port);
|
|
|
|
return Status;
|
2000-06-04 17:27:39 +00:00
|
|
|
}
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
2001-01-29 00:13:22 +00:00
|
|
|
*/
|
2007-01-10 04:27:40 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2005-09-04 18:00:59 +00:00
|
|
|
NtReplyWaitReceivePort(IN HANDLE PortHandle,
|
|
|
|
OUT PVOID *PortContext OPTIONAL,
|
|
|
|
IN PPORT_MESSAGE ReplyMessage OPTIONAL,
|
|
|
|
OUT PPORT_MESSAGE ReceiveMessage)
|
2001-01-29 00:13:22 +00:00
|
|
|
{
|
2007-01-10 04:27:40 +00:00
|
|
|
/* Call the newer API */
|
2005-09-04 18:00:59 +00:00
|
|
|
return NtReplyWaitReceivePortEx(PortHandle,
|
2007-01-10 04:27:40 +00:00
|
|
|
PortContext,
|
|
|
|
ReplyMessage,
|
|
|
|
ReceiveMessage,
|
|
|
|
NULL);
|
2001-01-29 00:13:22 +00:00
|
|
|
}
|
|
|
|
|
2007-01-10 04:27:40 +00:00
|
|
|
/*
|
|
|
|
* @unimplemented
|
2000-06-04 17:27:39 +00:00
|
|
|
*/
|
2007-01-10 04:27:40 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
NtReplyWaitReplyPort(IN HANDLE PortHandle,
|
|
|
|
IN PPORT_MESSAGE ReplyMessage)
|
2000-06-04 17:27:39 +00:00
|
|
|
{
|
2007-01-10 04:27:40 +00:00
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
2000-06-04 17:27:39 +00:00
|
|
|
}
|
|
|
|
|
2004-06-23 19:49:21 +00:00
|
|
|
/*
|
2007-01-10 04:27:40 +00:00
|
|
|
* @unimplemented
|
2004-06-23 19:49:21 +00:00
|
|
|
*/
|
|
|
|
NTSTATUS
|
2006-10-26 04:55:34 +00:00
|
|
|
NTAPI
|
2007-01-10 04:27:40 +00:00
|
|
|
NtReadRequestData(IN HANDLE PortHandle,
|
|
|
|
IN PPORT_MESSAGE Message,
|
|
|
|
IN ULONG Index,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG BufferLength,
|
|
|
|
OUT PULONG Returnlength)
|
2004-06-23 19:49:21 +00:00
|
|
|
{
|
2007-01-10 04:27:40 +00:00
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
NtWriteRequestData(IN HANDLE PortHandle,
|
|
|
|
IN PPORT_MESSAGE Message,
|
|
|
|
IN ULONG Index,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG BufferLength,
|
|
|
|
OUT PULONG ReturnLength)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
2004-06-23 19:49:21 +00:00
|
|
|
}
|
2000-06-04 17:27:39 +00:00
|
|
|
|
|
|
|
/* EOF */
|