mirror of
https://github.com/reactos/reactos.git
synced 2025-04-05 21:21:33 +00:00
[CSRSRV]
- Comment on the size of some members of the CSR_WAIT_BLOCK structure. - Initialize the WaitBlock member of CSR_THREAD to a valid value when creating a wait block, and NULLify it when we release a wait block. - ALWAYS USE offsets in CSR_CAPTURE_BUFFER structure, instead of real pointers !! It is needed when their base address change (eg. during a CSR wait, their base address, corresponding to the address of an CSR API message, change) (found when testing CSR waits with the console). svn path=/branches/ros-csrss/; revision=57806
This commit is contained in:
parent
4ccda5290d
commit
ae20fbcb81
3 changed files with 34 additions and 18 deletions
|
@ -156,13 +156,13 @@ BOOLEAN
|
||||||
|
|
||||||
typedef struct _CSR_WAIT_BLOCK
|
typedef struct _CSR_WAIT_BLOCK
|
||||||
{
|
{
|
||||||
ULONG Size;
|
ULONG Size; // Size of the wait block (variable-sized)
|
||||||
LIST_ENTRY WaitList;
|
LIST_ENTRY WaitList;
|
||||||
LIST_ENTRY UserWaitList;
|
LIST_ENTRY UserWaitList;
|
||||||
PVOID WaitContext;
|
PVOID WaitContext;
|
||||||
PCSR_THREAD WaitThread;
|
PCSR_THREAD WaitThread;
|
||||||
CSR_WAIT_FUNCTION WaitFunction;
|
CSR_WAIT_FUNCTION WaitFunction;
|
||||||
CSR_API_MESSAGE WaitApiMessage;
|
CSR_API_MESSAGE WaitApiMessage; // Variable-sized CSR API message
|
||||||
} CSR_WAIT_BLOCK, *PCSR_WAIT_BLOCK;
|
} CSR_WAIT_BLOCK, *PCSR_WAIT_BLOCK;
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -504,7 +504,7 @@ CsrApiRequestThread(IN PVOID Parameter)
|
||||||
if ((ServerDll) && (ServerDll->HardErrorCallback))
|
if ((ServerDll) && (ServerDll->HardErrorCallback))
|
||||||
{
|
{
|
||||||
/* Call it */
|
/* Call it */
|
||||||
ServerDll->HardErrorCallback(NULL /* CsrThread == NULL */, HardErrorMsg);
|
ServerDll->HardErrorCallback(NULL /* == CsrThread */, HardErrorMsg);
|
||||||
|
|
||||||
/* If it's handled, get out of here */
|
/* If it's handled, get out of here */
|
||||||
if (HardErrorMsg->Response != ResponseNotHandled) break;
|
if (HardErrorMsg->Response != ResponseNotHandled) break;
|
||||||
|
@ -524,6 +524,7 @@ CsrApiRequestThread(IN PVOID Parameter)
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ReplyMsg = &ReceiveMsg;
|
ReplyMsg = &ReceiveMsg;
|
||||||
|
ReplyPort = CsrApiPort;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (MessageType == LPC_REQUEST)
|
else if (MessageType == LPC_REQUEST)
|
||||||
|
@ -1178,20 +1179,21 @@ CsrCaptureArguments(IN PCSR_THREAD CsrThread,
|
||||||
BufferDistance = (ULONG_PTR)RemoteCaptureBuffer - (ULONG_PTR)LocalCaptureBuffer;
|
BufferDistance = (ULONG_PTR)RemoteCaptureBuffer - (ULONG_PTR)LocalCaptureBuffer;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Convert all the pointer offsets into real pointers, and make
|
* All the pointer offsets correspond to pointers which point
|
||||||
* them point to the remote data buffer instead of the local one.
|
* to the remote data buffer instead of the local one.
|
||||||
*/
|
*/
|
||||||
for (i = 0 ; i < RemoteCaptureBuffer->PointerCount ; ++i)
|
for (i = 0 ; i < RemoteCaptureBuffer->PointerCount ; ++i)
|
||||||
{
|
{
|
||||||
if (RemoteCaptureBuffer->PointerOffsetsArray[i] != 0)
|
if (RemoteCaptureBuffer->PointerOffsetsArray[i] != 0)
|
||||||
{
|
{
|
||||||
|
/* Temporarily transform the offset into a pointer */
|
||||||
RemoteCaptureBuffer->PointerOffsetsArray[i] += (ULONG_PTR)ApiMessage;
|
RemoteCaptureBuffer->PointerOffsetsArray[i] += (ULONG_PTR)ApiMessage;
|
||||||
|
|
||||||
/* Validate the bounds of the current pointer */
|
/* Validate the bounds of the current pointed pointer */
|
||||||
if ((*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] >= CsrThread->Process->ClientViewBase) &&
|
if ((*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] >= CsrThread->Process->ClientViewBase) &&
|
||||||
(*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] < CsrThread->Process->ClientViewBounds))
|
(*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] < CsrThread->Process->ClientViewBounds))
|
||||||
{
|
{
|
||||||
/* Modify the pointer to take into account its new position */
|
/* Modify the pointed pointer to take into account its new position */
|
||||||
*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] += BufferDistance;
|
*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] += BufferDistance;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -1201,19 +1203,22 @@ CsrCaptureArguments(IN PCSR_THREAD CsrThread,
|
||||||
DbgBreakPoint();
|
DbgBreakPoint();
|
||||||
ApiMessage->Status = STATUS_INVALID_PARAMETER;
|
ApiMessage->Status = STATUS_INVALID_PARAMETER;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Transform back into an offset */
|
||||||
|
RemoteCaptureBuffer->PointerOffsetsArray[i] -= (ULONG_PTR)ApiMessage;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if we got success */
|
/* Check if we got success */
|
||||||
if (ApiMessage->Status != STATUS_SUCCESS)
|
if (ApiMessage->Status != STATUS_SUCCESS)
|
||||||
{
|
{
|
||||||
/* Failure. Free the buffer and return*/
|
/* Failure. Free the buffer and return */
|
||||||
RtlFreeHeap(CsrHeap, 0, RemoteCaptureBuffer);
|
RtlFreeHeap(CsrHeap, 0, RemoteCaptureBuffer);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* Success, save the previous buffer */
|
/* Success, save the previous buffer and use the remote capture buffer */
|
||||||
RemoteCaptureBuffer->PreviousCaptureBuffer = LocalCaptureBuffer;
|
RemoteCaptureBuffer->PreviousCaptureBuffer = LocalCaptureBuffer;
|
||||||
ApiMessage->CsrCaptureData = RemoteCaptureBuffer;
|
ApiMessage->CsrCaptureData = RemoteCaptureBuffer;
|
||||||
}
|
}
|
||||||
|
@ -1246,29 +1251,38 @@ CsrReleaseCapturedArguments(IN PCSR_API_MESSAGE ApiMessage)
|
||||||
SIZE_T BufferDistance;
|
SIZE_T BufferDistance;
|
||||||
ULONG i;
|
ULONG i;
|
||||||
|
|
||||||
/* Get the capture buffers */
|
/* Get the remote capture buffer */
|
||||||
RemoteCaptureBuffer = ApiMessage->CsrCaptureData;
|
RemoteCaptureBuffer = ApiMessage->CsrCaptureData;
|
||||||
LocalCaptureBuffer = RemoteCaptureBuffer->PreviousCaptureBuffer;
|
|
||||||
|
|
||||||
/* Do not continue if there is no captured buffer */
|
/* Do not continue if there is no captured buffer */
|
||||||
if (!RemoteCaptureBuffer) return;
|
if (!RemoteCaptureBuffer) return;
|
||||||
|
|
||||||
/* Free the previous one */
|
/* If there is one, get the corresponding local capture buffer */
|
||||||
|
LocalCaptureBuffer = RemoteCaptureBuffer->PreviousCaptureBuffer;
|
||||||
|
|
||||||
|
/* Free the previous one and use again the local capture buffer */
|
||||||
RemoteCaptureBuffer->PreviousCaptureBuffer = NULL;
|
RemoteCaptureBuffer->PreviousCaptureBuffer = NULL;
|
||||||
|
ApiMessage->CsrCaptureData = LocalCaptureBuffer;
|
||||||
|
|
||||||
/* Calculate the difference between our buffer and the client's */
|
/* Calculate the difference between our buffer and the client's */
|
||||||
BufferDistance = (ULONG_PTR)RemoteCaptureBuffer - (ULONG_PTR)LocalCaptureBuffer;
|
BufferDistance = (ULONG_PTR)RemoteCaptureBuffer - (ULONG_PTR)LocalCaptureBuffer;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Convert back all the pointers into pointer offsets, and make them
|
* All the pointer offsets correspond to pointers which point
|
||||||
* point to the local data buffer instead of the remote one (revert
|
* to the local data buffer instead of the remote one (revert
|
||||||
* the logic of CsrCaptureArguments).
|
* the logic of CsrCaptureArguments).
|
||||||
*/
|
*/
|
||||||
for (i = 0 ; i < RemoteCaptureBuffer->PointerCount ; ++i)
|
for (i = 0 ; i < RemoteCaptureBuffer->PointerCount ; ++i)
|
||||||
{
|
{
|
||||||
if (RemoteCaptureBuffer->PointerOffsetsArray[i] != 0)
|
if (RemoteCaptureBuffer->PointerOffsetsArray[i] != 0)
|
||||||
{
|
{
|
||||||
|
/* Temporarily transform the offset into a pointer */
|
||||||
|
RemoteCaptureBuffer->PointerOffsetsArray[i] += (ULONG_PTR)ApiMessage;
|
||||||
|
|
||||||
|
/* Modify the pointed pointer to take into account its new position */
|
||||||
*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] -= BufferDistance;
|
*(PULONG_PTR)RemoteCaptureBuffer->PointerOffsetsArray[i] -= BufferDistance;
|
||||||
|
|
||||||
|
/* Transform back into an offset */
|
||||||
RemoteCaptureBuffer->PointerOffsetsArray[i] -= (ULONG_PTR)ApiMessage;
|
RemoteCaptureBuffer->PointerOffsetsArray[i] -= (ULONG_PTR)ApiMessage;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1313,7 +1327,7 @@ CsrValidateMessageBuffer(IN PCSR_API_MESSAGE ApiMessage,
|
||||||
IN ULONG ElementSize)
|
IN ULONG ElementSize)
|
||||||
{
|
{
|
||||||
PCSR_CAPTURE_BUFFER CaptureBuffer = ApiMessage->CsrCaptureData;
|
PCSR_CAPTURE_BUFFER CaptureBuffer = ApiMessage->CsrCaptureData;
|
||||||
// SIZE_T BufferDistance = (ULONG_PTR)Buffer - (ULONG_PTR)ApiMessage;
|
SIZE_T BufferDistance = (ULONG_PTR)Buffer - (ULONG_PTR)ApiMessage;
|
||||||
ULONG i;
|
ULONG i;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1353,10 +1367,10 @@ CsrValidateMessageBuffer(IN PCSR_API_MESSAGE ApiMessage,
|
||||||
for (i = 0 ; i < CaptureBuffer->PointerCount ; ++i)
|
for (i = 0 ; i < CaptureBuffer->PointerCount ; ++i)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* If the pointer offset is in fact equal to the
|
* The pointer offset must be equal to the delta between
|
||||||
* real address of the buffer then it's OK.
|
* the addresses of the buffer and of the API message.
|
||||||
*/
|
*/
|
||||||
if (CaptureBuffer->PointerOffsetsArray[i] == (ULONG_PTR)Buffer /* BufferDistance + (ULONG_PTR)ApiMessage */)
|
if (CaptureBuffer->PointerOffsetsArray[i] == BufferDistance)
|
||||||
{
|
{
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,6 +73,7 @@ CsrInitializeWait(IN CSR_WAIT_FUNCTION WaitFunction,
|
||||||
/* Initialize it */
|
/* Initialize it */
|
||||||
WaitBlock->Size = Size;
|
WaitBlock->Size = Size;
|
||||||
WaitBlock->WaitThread = CsrWaitThread;
|
WaitBlock->WaitThread = CsrWaitThread;
|
||||||
|
CsrWaitThread->WaitBlock = WaitBlock;
|
||||||
WaitBlock->WaitContext = WaitContext;
|
WaitBlock->WaitContext = WaitContext;
|
||||||
WaitBlock->WaitFunction = WaitFunction;
|
WaitBlock->WaitFunction = WaitFunction;
|
||||||
WaitBlock->UserWaitList.Flink = NULL;
|
WaitBlock->UserWaitList.Flink = NULL;
|
||||||
|
@ -242,6 +243,7 @@ CsrCreateWait(IN PLIST_ENTRY WaitList,
|
||||||
if (CsrWaitThread->Flags & CsrThreadTerminated)
|
if (CsrWaitThread->Flags & CsrThreadTerminated)
|
||||||
{
|
{
|
||||||
/* Fail the wait */
|
/* Fail the wait */
|
||||||
|
CsrWaitThread->WaitBlock = NULL;
|
||||||
RtlFreeHeap(CsrHeap, 0, WaitBlock);
|
RtlFreeHeap(CsrHeap, 0, WaitBlock);
|
||||||
CsrReleaseWaitLock();
|
CsrReleaseWaitLock();
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
Loading…
Reference in a new issue