mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1534 lines
34 KiB
1534 lines
34 KiB
/*++
|
|
|
|
Copyright (C) Microsoft Corporation, 1992 - 1999
|
|
|
|
Module Name:
|
|
|
|
lpcsvr.hxx
|
|
|
|
Abstract:
|
|
|
|
Class definition for the server side of RPC on LPC protocol
|
|
engine.
|
|
|
|
Author:
|
|
|
|
Steven Zeck (stevez) 12/17/91 (spcsvr.hxx)
|
|
|
|
Revision History:
|
|
|
|
16-Dec-1992 mikemon
|
|
|
|
Rewrote the majority of the code and added comments.
|
|
|
|
-- mazharm code fork from spcsvr.hxx
|
|
|
|
05/13/96 mazharm merged WMSG/LRPC into a common protocol
|
|
9/22/97 no more WMSG
|
|
|
|
Kamen Moutafov (kamenm) Jan-2000 Support for multiple transfer syntaxes
|
|
Kamen Moutafov (KamenM) Dec 99 - Feb 2000 - Support for cell debugging stuff
|
|
--*/
|
|
|
|
#ifndef __LPCSVR_HXX__
|
|
#define __LPCSVR_HXX__
|
|
|
|
class LRPC_SASSOCIATION;
|
|
class LRPC_CASSOCIATION;
|
|
class LRPC_ADDRESS;
|
|
class LRPC_SERVER ;
|
|
class LRPC_SCALL ;
|
|
class LRPC_SERVER ;
|
|
class LRPC_CASSOCIATION_DICT;
|
|
|
|
|
|
// the next four are protected by the LrpcMutex
|
|
extern LRPC_CASSOCIATION_DICT * LrpcAssociationDict;
|
|
|
|
// the number of associations in the LrpcAssociationDict
|
|
// dictionary that have their fAssociationLingered member set
|
|
extern long LrpcLingeredAssociations;
|
|
|
|
// the total number of LRPC associations destroyed in this process
|
|
// this is used to dynamically turn on garbage collection if
|
|
// necessary
|
|
extern ULONG LrpcDestroyedAssociations;
|
|
|
|
// each time the number of destroyed associations is divisible by
|
|
// NumberOfLrpcDestroyedAssociationsToSample, we check this timestamp
|
|
// and compare it to the current. If the difference is more than
|
|
// DestroyedLrpcAssociationBatchThreshold, we will turn on
|
|
// garbage collection automatically
|
|
extern ULARGE_INTEGER LastDestroyedAssociationsBatchTimestamp;
|
|
|
|
const ULONG NumberOfLrpcDestroyedAssociationsToSample = 128;
|
|
|
|
// in 100 nano-second intervals, this constant is 1 second
|
|
const DWORD DestroyedLrpcAssociationBatchThreshold = 1000 * 10 * 1000;
|
|
|
|
// 4 is an arbitrary number designed to ensure reasonably sized cache
|
|
// Feel free to change it if there is a perf reason for that
|
|
const long MaxLrpcLingeredAssociations = 4;
|
|
extern LRPC_SERVER *GlobalLrpcServer;
|
|
|
|
RPC_STATUS
|
|
InitializeLrpcServer(
|
|
) ;
|
|
|
|
extern RPC_STATUS
|
|
InitializeLrpcIfNecessary(
|
|
) ;
|
|
|
|
#define LrpcSetEndpoint(Endpoint) (\
|
|
GlobalLrpcServer->SetEndpoint(Endpoint))
|
|
#define LrpcGetEndpoint(Endpoint) (\
|
|
GlobalLrpcServer->GetEndpoint(Endpoint))
|
|
|
|
NEW_SDICT(LRPC_SASSOCIATION);
|
|
NEW_SDICT(LRPC_SCALL) ;
|
|
|
|
#define SERVER_KEY_MASK 0x8000
|
|
#define SERVERKEY(_x_) ((_x_) & 0x80000000)
|
|
|
|
#define CLEANUP goto cleanup
|
|
|
|
#define COPYMSG(LrpcCopy, LrpcMessage) \
|
|
LrpcCopy->LpcHeader.ClientId = LrpcMessage->LpcHeader.ClientId ; \
|
|
LrpcCopy->LpcHeader.CallbackId = LrpcMessage->LpcHeader.CallbackId ; \
|
|
LrpcCopy->LpcHeader.MessageId = LrpcMessage->LpcHeader.MessageId ; \
|
|
LrpcCopy->LpcHeader.u2.s2.DataInfoOffset = \
|
|
LrpcMessage->LpcHeader.u2.s2.DataInfoOffset
|
|
|
|
#define INITMSG(LrpcMessage, CId, Cbd, MId) \
|
|
(LrpcMessage)->LpcHeader.ClientId = ClientIdToMsgClientId(CId) ;\
|
|
(LrpcMessage)->LpcHeader.CallbackId = Cbd ;\
|
|
(LrpcMessage)->LpcHeader.MessageId = MId
|
|
|
|
#define AllocateMessage() ((LRPC_MESSAGE *) RpcAllocateBuffer(sizeof(LRPC_MESSAGE)))
|
|
#define FreeMessage(x) (RpcFreeBuffer((x)))
|
|
|
|
//
|
|
// The high bit of the sequence number
|
|
// is used to determine if it is a client or server
|
|
// key.
|
|
// CAUTION: Big endian implementations need to swap this
|
|
// around
|
|
//
|
|
typedef struct {
|
|
USHORT AssocKey;
|
|
USHORT SeqNumber;
|
|
} LPC_KEY;
|
|
|
|
|
|
class LRPC_SERVER
|
|
{
|
|
private:
|
|
LRPC_ADDRESS *Address ;
|
|
RPC_CHAR * Endpoint;
|
|
BOOL EndpointInitialized ;
|
|
MUTEX ServerMutex ;
|
|
|
|
public:
|
|
LRPC_SERVER(
|
|
IN OUT RPC_STATUS *Status
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
InitializeAsync (
|
|
) ;
|
|
|
|
void
|
|
GetEndpoint(
|
|
IN RPC_CHAR *Endpoint
|
|
)
|
|
{
|
|
RpcpStringCopy(Endpoint, this->Endpoint) ;
|
|
}
|
|
|
|
RPC_STATUS
|
|
SetEndpoint(
|
|
IN RPC_CHAR *Endpoint
|
|
)
|
|
{
|
|
if (EndpointInitialized == 0)
|
|
{
|
|
this->Endpoint = DuplicateString(Endpoint) ;
|
|
if (this->Endpoint == 0)
|
|
{
|
|
return RPC_S_OUT_OF_MEMORY ;
|
|
}
|
|
|
|
EndpointInitialized = 1 ;
|
|
}
|
|
|
|
return RPC_S_OK ;
|
|
}
|
|
|
|
void
|
|
AcquireMutex(
|
|
)
|
|
{
|
|
ServerMutex.Request() ;
|
|
}
|
|
|
|
void
|
|
ReleaseMutex(
|
|
)
|
|
{
|
|
ServerMutex.Clear() ;
|
|
}
|
|
|
|
void VerifyOwned(void)
|
|
{
|
|
ServerMutex.VerifyOwned();
|
|
}
|
|
|
|
BOOL
|
|
TryRequestMutex (
|
|
void
|
|
)
|
|
{
|
|
return ServerMutex.TryRequest();
|
|
}
|
|
} ;
|
|
|
|
inline void LrpcMutexRequest(void)
|
|
{
|
|
GlobalLrpcServer->AcquireMutex();
|
|
}
|
|
|
|
inline void LrpcMutexClear(void)
|
|
{
|
|
GlobalLrpcServer->ReleaseMutex();
|
|
}
|
|
|
|
inline void LrpcMutexVerifyOwned(void)
|
|
{
|
|
GlobalLrpcServer->VerifyOwned();
|
|
}
|
|
|
|
inline BOOL LrpcMutexTryRequest(void)
|
|
{
|
|
return GlobalLrpcServer->TryRequestMutex();
|
|
}
|
|
|
|
|
|
class LRPC_ADDRESS : public RPC_ADDRESS
|
|
/*++
|
|
|
|
Class Description:
|
|
|
|
Fields:
|
|
|
|
LpcAddressPort - Contains the connection port which this address will
|
|
use to wait for clients to connect.
|
|
|
|
CallThreadCount - Contains the number of call threads we have executing.
|
|
|
|
MinimumCallThreads - Contains the minimum number of call threads.
|
|
|
|
ServerListeningFlag - Contains a flag indicating whether or not the server
|
|
is listening for remote procedure calls. A non-zero value indicates
|
|
that it is listening.
|
|
|
|
ActiveCallCount - Contains the number of remote procedure calls active
|
|
on this address.
|
|
|
|
AssociationDictionary - Contains the dictionary of associations on this
|
|
address. We need this to map from an association key into the
|
|
correct association. This is necessary to prevent a race condition
|
|
between deleting an association and using it.
|
|
|
|
--*/
|
|
{
|
|
private:
|
|
|
|
HANDLE LpcAddressPort;
|
|
long CallThreadCount;
|
|
long MinimumCallThreads;
|
|
LRPC_SASSOCIATION_DICT AssociationDictionary;
|
|
int AssociationCount ;
|
|
USHORT SequenceNumber;
|
|
CellTag DebugCellTag;
|
|
DebugEndpointInfo *DebugCell;
|
|
|
|
// not protected - operated by interlocks. We push on the list,
|
|
// but we never pop, knowing addresses don't get deleted.
|
|
LRPC_ADDRESS *AddressChain;
|
|
|
|
// the next three are protected by the LrpcMutex
|
|
// If TickleMessage is non-zero, this means that the
|
|
// address has been prepared for tickling. Once it
|
|
// is initialized, fTickleMessageAvailable is used
|
|
// to tell whether the TickleMessage is currently posted
|
|
// or not. If it is already posted, no need to repost.
|
|
BOOL fTickleMessageAvailable;
|
|
LRPC_BIND_EXCHANGE *TickleMessage;
|
|
UNICODE_STRING ThisAddressLoopbackString;
|
|
|
|
// the number of threads on this address doing long wait
|
|
INTERLOCKED_INTEGER ThreadsDoingLongWait;
|
|
|
|
public:
|
|
unsigned int ServerListeningFlag ;
|
|
BOOL fServerThreadsStarted;
|
|
|
|
LRPC_ADDRESS (
|
|
OUT RPC_STATUS * Status
|
|
);
|
|
|
|
~LRPC_ADDRESS (
|
|
void
|
|
)
|
|
{
|
|
if (DebugCell)
|
|
{
|
|
FreeCell(DebugCell, &DebugCellTag);
|
|
DebugCell = NULL;
|
|
}
|
|
}
|
|
|
|
virtual RPC_STATUS
|
|
ServerStartingToListen (
|
|
IN unsigned int MinimumCallThreads,
|
|
IN unsigned int MaximumConcurrentCalls
|
|
);
|
|
|
|
virtual void
|
|
ServerStoppedListening (
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
ServerSetupAddress (
|
|
IN RPC_CHAR * NetworkAddress,
|
|
IN RPC_CHAR * *Endpoint,
|
|
IN unsigned int PendingQueueSize,
|
|
IN void * SecurityDescriptor, OPTIONAL
|
|
IN unsigned long EndpointFlags,
|
|
IN unsigned long NICFlags,
|
|
OUT NETWORK_ADDRESS_VECTOR **ppNetworkAddressVector
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
LRPC_ADDRESS::CompleteListen (
|
|
);
|
|
|
|
inline void
|
|
DereferenceAssociation (
|
|
IN LRPC_SASSOCIATION * Association
|
|
);
|
|
|
|
friend BOOL
|
|
RecvLotsaCallsWrapper(
|
|
IN LRPC_ADDRESS * Address
|
|
);
|
|
|
|
RPC_STATUS
|
|
BeginLongCall(
|
|
void
|
|
);
|
|
|
|
BOOL
|
|
EndLongCall(
|
|
void
|
|
);
|
|
|
|
inline BOOL
|
|
PrepareForLoopbackTicklingIfNecessary (
|
|
void
|
|
)
|
|
{
|
|
LrpcMutexVerifyOwned();
|
|
|
|
if (TickleMessage)
|
|
return TRUE;
|
|
|
|
return PrepareForLoopbackTickling();
|
|
}
|
|
|
|
BOOL
|
|
LoopbackTickle (
|
|
void
|
|
);
|
|
|
|
inline BOOL
|
|
IsPreparedForLoopbackTickling (
|
|
void
|
|
)
|
|
{
|
|
return (TickleMessage != NULL);
|
|
}
|
|
|
|
inline LRPC_ADDRESS *
|
|
GetNextAddress (
|
|
void
|
|
)
|
|
{
|
|
return AddressChain;
|
|
}
|
|
|
|
inline int
|
|
GetNumberOfThreadsDoingShortWait (
|
|
void
|
|
)
|
|
{
|
|
return CallThreadCount - ThreadsDoingLongWait.GetInteger();
|
|
}
|
|
|
|
private:
|
|
|
|
RPC_STATUS
|
|
ActuallySetupAddress (
|
|
IN RPC_CHAR *Endpoint,
|
|
IN void * SecurityDescriptor
|
|
);
|
|
|
|
inline LRPC_SASSOCIATION *
|
|
ReferenceAssociation (
|
|
IN unsigned long AssociationKey
|
|
);
|
|
|
|
inline LRPC_CASSOCIATION *
|
|
ReferenceClientAssoc (
|
|
IN unsigned long AssociationKey
|
|
);
|
|
|
|
void
|
|
ReceiveLotsaCalls (
|
|
);
|
|
|
|
void
|
|
DealWithNewClient (
|
|
IN LRPC_MESSAGE * ConnectionRequest
|
|
);
|
|
|
|
void
|
|
DealWithConnectResponse (
|
|
IN LRPC_MESSAGE * ConnectResponse
|
|
) ;
|
|
|
|
void
|
|
RejectNewClient (
|
|
IN LRPC_MESSAGE * ConnectionRequest,
|
|
IN RPC_STATUS Status
|
|
);
|
|
|
|
BOOL
|
|
DealWithLRPCRequest (
|
|
IN LRPC_MESSAGE * LrpcMessage,
|
|
IN LRPC_MESSAGE * LrpcReplyMessage,
|
|
IN LRPC_SASSOCIATION *Association,
|
|
OUT LRPC_MESSAGE **LrpcResponse
|
|
) ;
|
|
|
|
BOOL fKeepThread(
|
|
)
|
|
{
|
|
// we keep at least two threads per address, to avoid creating the
|
|
// second thread each time a call arrives. Note that the second
|
|
// thread is free to timeout on the port and exit, the question
|
|
// is that it doesn't exit immediately after the call, as this will
|
|
// cause unnecessary creation/deletions of threads
|
|
return (((CallThreadCount - ActiveCallCount) <= 2)
|
|
|| (CallThreadCount <= MinimumCallThreads));
|
|
}
|
|
|
|
typedef enum
|
|
{
|
|
asctDestroyContextHandle = 0,
|
|
asctCleanupIdleSContext
|
|
} AssociationCallbackType;
|
|
|
|
void
|
|
EnumerateAndCallEachAssociation (
|
|
IN AssociationCallbackType asctType,
|
|
IN OUT void *Context OPTIONAL
|
|
);
|
|
|
|
virtual void
|
|
DestroyContextHandlesForInterface (
|
|
IN RPC_SERVER_INTERFACE PAPI * RpcInterfaceInformation,
|
|
IN BOOL RundownContextHandles
|
|
);
|
|
|
|
virtual void
|
|
CleanupIdleSContexts (
|
|
void
|
|
);
|
|
|
|
BOOL
|
|
PrepareForLoopbackTickling (
|
|
void
|
|
);
|
|
|
|
void HandleInvalidAssociationReference (
|
|
IN LRPC_MESSAGE *RequestMessage,
|
|
IN OUT LRPC_MESSAGE **ReplyMessage,
|
|
IN ULONG AssociationKey
|
|
);
|
|
};
|
|
|
|
|
|
class LRPC_SBINDING
|
|
/*++
|
|
|
|
Class Description:
|
|
|
|
Each object of this class represents a binding to an interface by a
|
|
client.
|
|
|
|
Fields:
|
|
|
|
RpcInterface - Contains a pointer to the bound interface.
|
|
|
|
PresentationContext - Contains the key which the client will send when
|
|
it wants to use this binding.
|
|
|
|
--*/
|
|
{
|
|
friend class LRPC_SASSOCIATION;
|
|
friend class LRPC_SCALL;
|
|
friend class LRPC_ADDRESS;
|
|
|
|
private:
|
|
|
|
RPC_INTERFACE * RpcInterface;
|
|
int PresentationContext;
|
|
int SelectedTransferSyntaxIndex;
|
|
unsigned long SequenceNumber ;
|
|
|
|
public:
|
|
|
|
LRPC_SBINDING (
|
|
IN RPC_INTERFACE * RpcInterface,
|
|
IN int SelectedTransferSyntax
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
We will construct a LRPC_SBINDING.
|
|
|
|
Arguments:
|
|
|
|
RpcInterface - Supplies the bound interface.
|
|
|
|
TransferSyntax - Supplies the transfer syntax which the client will use
|
|
over this binding.
|
|
|
|
--*/
|
|
{
|
|
this->RpcInterface = RpcInterface;
|
|
SequenceNumber = 0;
|
|
SelectedTransferSyntaxIndex = SelectedTransferSyntax;
|
|
}
|
|
|
|
RPC_STATUS
|
|
CheckSecurity (
|
|
SCALL * Context
|
|
);
|
|
|
|
inline unsigned short GetOnTheWirePresentationContext(void)
|
|
{
|
|
return (unsigned short)PresentationContext;
|
|
}
|
|
|
|
inline void SetPresentationContextFromPacket(unsigned short PresentContext)
|
|
{
|
|
PresentationContext = (int) PresentContext;
|
|
}
|
|
|
|
inline int GetPresentationContext (void)
|
|
{
|
|
return PresentationContext;
|
|
}
|
|
|
|
inline void SetPresentationContext (int PresentContext)
|
|
{
|
|
PresentationContext = PresentContext;
|
|
}
|
|
|
|
inline void GetSelectedTransferSyntaxAndDispatchTable(
|
|
OUT RPC_SYNTAX_IDENTIFIER **SelectedTransferSyntax,
|
|
OUT PRPC_DISPATCH_TABLE *SelectedDispatchTable)
|
|
{
|
|
RpcInterface->GetSelectedTransferSyntaxAndDispatchTable(SelectedTransferSyntaxIndex,
|
|
SelectedTransferSyntax, SelectedDispatchTable);
|
|
}
|
|
|
|
DWORD GetInterfaceFirstDWORD(void)
|
|
{
|
|
return RpcInterface->GetInterfaceFirstDWORD();;
|
|
}
|
|
};
|
|
|
|
|
|
|
|
typedef void * LRPC_BUFFER;
|
|
|
|
NEW_SDICT(LRPC_SBINDING);
|
|
NEW_SDICT(LRPC_CLIENT_BUFFER);
|
|
NEW_SDICT2(LRPC_SCALL, PVOID);
|
|
NEW_NAMED_SDICT2(LRPC_SCALL_THREAD, LRPC_SCALL, HANDLE);
|
|
|
|
#define USER_NAME_LEN 256
|
|
#define DOMAIN_NAME_LEN 128
|
|
|
|
extern const SID AnonymousSid;
|
|
|
|
|
|
class LRPC_SCONTEXT
|
|
{
|
|
public:
|
|
HANDLE hToken;
|
|
LUID ClientLuid;
|
|
RPC_CHAR *ClientName;
|
|
int RefCount;
|
|
LRPC_SASSOCIATION *Association;
|
|
AUTHZ_CLIENT_CONTEXT_HANDLE AuthzClientContext;
|
|
|
|
LRPC_SCONTEXT (
|
|
IN HANDLE MyToken,
|
|
IN LUID *UserLuid,
|
|
IN LRPC_SASSOCIATION *MyAssociation,
|
|
IN BOOL fDefaultLogonId,
|
|
IN BOOL fAnonymousToken
|
|
);
|
|
|
|
~LRPC_SCONTEXT (
|
|
void
|
|
);
|
|
|
|
void AddReference();
|
|
void RemoveReference();
|
|
void Destroy();
|
|
|
|
RPC_STATUS
|
|
GetUserName (
|
|
IN OUT ULONG *ClientPrincipalNameBufferLength OPTIONAL,
|
|
OUT RPC_CHAR **UserName,
|
|
IN HANDLE hUserToken OPTIONAL
|
|
);
|
|
|
|
static const unsigned int Deleted = 1;
|
|
static const unsigned int DefaultLogonId = 2;
|
|
static const unsigned int Anonymous = 4;
|
|
static const unsigned int ServerSideOnly = 8;
|
|
|
|
inline void SetDeletedFlag(void)
|
|
{
|
|
Flags.SetFlagUnsafe(Deleted);
|
|
}
|
|
inline void ClearDeletedFlag(void)
|
|
{
|
|
Flags.ClearFlagUnsafe(Deleted);
|
|
}
|
|
inline BOOL GetDeletedFlag(void)
|
|
{
|
|
return Flags.GetFlag(Deleted);
|
|
}
|
|
|
|
inline void SetDefaultLogonIdFlag(void)
|
|
{
|
|
Flags.SetFlagUnsafe(DefaultLogonId);
|
|
}
|
|
inline void ClearDefaultLogonIdFlag(void)
|
|
{
|
|
Flags.ClearFlagUnsafe(DefaultLogonId);
|
|
}
|
|
inline BOOL GetDefaultLogonIdFlag(void)
|
|
{
|
|
return Flags.GetFlag(DefaultLogonId);
|
|
}
|
|
|
|
inline void SetAnonymousFlag(void)
|
|
{
|
|
Flags.SetFlagUnsafe(Anonymous);
|
|
}
|
|
inline void ClearAnonymousFlag(void)
|
|
{
|
|
Flags.ClearFlagUnsafe(Anonymous);
|
|
}
|
|
inline BOOL GetAnonymousFlag(void)
|
|
{
|
|
return Flags.GetFlag(Anonymous);
|
|
}
|
|
|
|
inline void SetServerSideOnlyFlag(void)
|
|
{
|
|
Flags.SetFlagUnsafe(ServerSideOnly);
|
|
}
|
|
inline void ClearServerSideOnlyFlag(void)
|
|
{
|
|
Flags.ClearFlagUnsafe(ServerSideOnly);
|
|
}
|
|
inline BOOL GetServerSideOnlyFlag(void)
|
|
{
|
|
return Flags.GetFlag(ServerSideOnly);
|
|
}
|
|
|
|
static const ULONG CONTEXT_IDLE_TIMEOUT = 2 * 60 * 1000; // 2 minutes
|
|
|
|
inline void UpdateLastAccessTime (void)
|
|
{
|
|
LastAccessTime = NtGetTickCount();
|
|
}
|
|
|
|
inline BOOL IsIdle (void)
|
|
{
|
|
// make sure if the tick count wraps the calculation is still valid
|
|
return (NtGetTickCount() - LastAccessTime > CONTEXT_IDLE_TIMEOUT);
|
|
}
|
|
|
|
private:
|
|
static TOKEN_USER *
|
|
GetSID (
|
|
IN HANDLE hToken
|
|
);
|
|
|
|
static RPC_STATUS
|
|
LookupUser (
|
|
IN SID *pSid,
|
|
OUT RPC_CHAR **UserName
|
|
);
|
|
|
|
// changed either on creation, or within
|
|
// the Association mutex
|
|
CompositeFlags Flags;
|
|
|
|
// the last access time for server side only contexts.
|
|
// Undefined if the context is not server side only
|
|
// It is used to determine whether the context is idle
|
|
ULONG LastAccessTime;
|
|
};
|
|
|
|
NEW_SDICT(LRPC_SCONTEXT);
|
|
|
|
|
|
class LRPC_SASSOCIATION : public ASSOCIATION_HANDLE
|
|
/*++
|
|
|
|
Class Description:
|
|
|
|
|
|
Fields:
|
|
|
|
LpcServerPort - Contains the LPC server communication port.
|
|
|
|
Bindings - Contains the dictionary of bindings with the client. This
|
|
information is necessary to dispatch remote procedure calls to the
|
|
correct stub.
|
|
|
|
Address - Contains the address which this association is over.
|
|
|
|
AssociationReferenceCount - Contains a count of the number of objects
|
|
referencing this association. This will be the number of outstanding
|
|
remote procedure calls, and one for LPC (because of the context
|
|
pointer). We will protect this fielding using the global mutex.
|
|
|
|
Buffers - Contains the dictionary of buffers to be written into the
|
|
client's address space on demand.
|
|
|
|
AssociationKey - Contains the key for this association in the dictionary
|
|
of associations maintained by the address.
|
|
|
|
ClientThreadDict - This dictionary contains one entry per client thread. Each entry
|
|
contains a list of SCALLS representing calls made by that client thread. The
|
|
calls need to be causally ordered.
|
|
|
|
--*/
|
|
{
|
|
friend class LRPC_ADDRESS;
|
|
friend class LRPC_SCALL;
|
|
friend class LRPC_SCONTEXT;
|
|
friend void
|
|
SetFaultPacket (
|
|
IN LRPC_MESSAGE *LrpcMessage,
|
|
IN RPC_STATUS Status,
|
|
IN int Flags,
|
|
IN LRPC_SCALL *CurrentCall OPTIONAL
|
|
);
|
|
|
|
private:
|
|
|
|
long AssociationReferenceCount;
|
|
USHORT DictionaryKey;
|
|
USHORT SequenceNumber;
|
|
HANDLE LpcServerPort;
|
|
HANDLE LpcReplyPort ;
|
|
LRPC_ADDRESS * Address;
|
|
LRPC_SBINDING_DICT Bindings;
|
|
int Aborted ;
|
|
long Deleted ;
|
|
LRPC_SCALL *CachedSCall;
|
|
LONG CachedSCallAvailable;
|
|
BOOL fFirstCall;
|
|
LRPC_CLIENT_BUFFER_DICT Buffers ;
|
|
MUTEX AssociationMutex ;
|
|
QUEUE FreeSCallQueue ;
|
|
LRPC_SCALL_THREAD_DICT2 ClientThreadDict ;
|
|
LRPC_SCONTEXT_DICT SContextDict;
|
|
|
|
public:
|
|
|
|
LRPC_SCALL_DICT2 SCallDict ;
|
|
|
|
|
|
LRPC_SASSOCIATION (
|
|
IN LRPC_ADDRESS * Address,
|
|
IN RPC_STATUS *Status
|
|
);
|
|
|
|
~LRPC_SASSOCIATION (
|
|
);
|
|
|
|
RPC_STATUS
|
|
AddBinding (
|
|
IN OUT LRPC_BIND_EXCHANGE * BindExchange
|
|
);
|
|
|
|
void
|
|
DealWithBindMessage (
|
|
IN LRPC_MESSAGE * LrpcMessage
|
|
);
|
|
|
|
LRPC_MESSAGE *
|
|
DealWithCopyMessage (
|
|
IN LRPC_COPY_MESSAGE * LrpcMessage
|
|
) ;
|
|
|
|
NTSTATUS
|
|
ReplyMessage (
|
|
IN LRPC_MESSAGE * LrpcMessage
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
SaveToken (
|
|
IN LRPC_MESSAGE *LrpcMessage,
|
|
OUT HANDLE *pTokenHandle,
|
|
IN BOOL fRestoreToken = 0
|
|
) ;
|
|
|
|
LRPC_MESSAGE *
|
|
DealWithBindBackMessage (
|
|
IN LRPC_MESSAGE *BindBack
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
BindBack (
|
|
IN RPC_CHAR *Endpoint,
|
|
IN DWORD pAssocKey
|
|
) ;
|
|
|
|
LRPC_MESSAGE *
|
|
DealWithPartialRequest (
|
|
IN LRPC_MESSAGE **LrpcMessage
|
|
) ;
|
|
|
|
void
|
|
Delete(
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
AllocateSCall (
|
|
IN LRPC_MESSAGE *LrpcMessage,
|
|
IN LRPC_MESSAGE *LrpcReply,
|
|
IN unsigned int Flags,
|
|
IN LRPC_SCALL **SCall
|
|
) ;
|
|
|
|
void
|
|
FreeSCall (
|
|
LRPC_SCALL *SCall
|
|
) ;
|
|
|
|
int
|
|
MaybeQueueSCall (
|
|
IN LRPC_SCALL *SCall
|
|
) ;
|
|
|
|
LRPC_SCALL *
|
|
GetNextSCall (
|
|
IN LRPC_SCALL *SCall
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
GetClientName (
|
|
IN LRPC_SCALL *SCall,
|
|
IN OUT ULONG *ClientPrincipalNameBufferLength OPTIONAL, // in bytes
|
|
OUT RPC_CHAR **ClientPrincipalName
|
|
);
|
|
|
|
void
|
|
AbortAssociation (
|
|
)
|
|
{
|
|
Aborted = 1 ;
|
|
}
|
|
|
|
LRPC_SCONTEXT *
|
|
FindSecurityContext (
|
|
ULONG SecurityContextId
|
|
)
|
|
{
|
|
LRPC_SCONTEXT *SContext;
|
|
|
|
AssociationMutex.Request();
|
|
SContext = SContextDict.Find(SecurityContextId);
|
|
if (SContext)
|
|
{
|
|
SContext->AddReference();
|
|
}
|
|
AssociationMutex.Clear();
|
|
|
|
return SContext;
|
|
}
|
|
|
|
int
|
|
IsAborted(
|
|
)
|
|
{
|
|
return Aborted ;
|
|
}
|
|
|
|
void
|
|
MaybeDereference (
|
|
)
|
|
{
|
|
if (Aborted)
|
|
{
|
|
Delete() ;
|
|
}
|
|
}
|
|
|
|
LRPC_ADDRESS *
|
|
InqAddress(
|
|
)
|
|
{
|
|
return Address ;
|
|
}
|
|
|
|
virtual RPC_STATUS CreateThread(void);
|
|
virtual void RundownNotificationCompleted(void);
|
|
|
|
void
|
|
CleanupIdleSContexts (
|
|
void
|
|
);
|
|
};
|
|
|
|
|
|
|
|
class LRPC_SCALL : public SCALL
|
|
/*++
|
|
|
|
Class Description:
|
|
|
|
Fields:
|
|
|
|
Association - Contains the association over which the remote procedure
|
|
call was received. We need this information to make callbacks and
|
|
to send the reply.
|
|
|
|
LrpcMessage - Contains the request message. We need this to send callbacks
|
|
as well as the reply.
|
|
|
|
SBinding - Contains the binding being used for this remote procedure call.
|
|
|
|
ObjectUuidFlag - Contains a flag indicting whether or not an object
|
|
uuid was specified for this remote procedure call. A non-zero
|
|
value indicates that an object uuid was specified.
|
|
|
|
ObjectUuid - Optionally contains the object uuid for this call, if one
|
|
was specified.
|
|
|
|
ClientId - Contains the thread identifier of the thread which made the
|
|
remote procedure call.
|
|
|
|
MessageId - Contains an identifier used by LPC to identify the current
|
|
remote procedure call.
|
|
|
|
PushedResponse - When the client needs to send a large response to the
|
|
server it must be transfered via a request. This holds the pushed
|
|
response until the request gets here.
|
|
|
|
--*/
|
|
{
|
|
friend class LRPC_SASSOCIATION;
|
|
friend class LRPC_ADDRESS;
|
|
friend void
|
|
SetFaultPacket (
|
|
IN LRPC_MESSAGE *LrpcMessage,
|
|
IN RPC_STATUS Status,
|
|
IN int Flags,
|
|
IN LRPC_SCALL *CurrentCall OPTIONAL
|
|
);
|
|
|
|
private:
|
|
CLIENT_AUTH_INFO AuthInfo;
|
|
LRPC_SASSOCIATION * Association;
|
|
LRPC_MESSAGE * LrpcRequestMessage;
|
|
LRPC_MESSAGE * LrpcReplyMessage;
|
|
LRPC_SBINDING * SBinding;
|
|
unsigned int ObjectUuidFlag;
|
|
ULONG CallId;
|
|
CLIENT_ID ClientId;
|
|
ULONG MessageId;
|
|
ULONG CallbackId;
|
|
void * PushedResponse;
|
|
ULONG CurrentBufferLength ;
|
|
int BufferComplete ;
|
|
LRPC_MESSAGE *LrpcAsyncReplyMessage;
|
|
unsigned int Flags ;
|
|
BOOL FirstSend ;
|
|
BOOL PipeSendCalled ;
|
|
int Deleted ;
|
|
RPC_UUID ObjectUuid;
|
|
|
|
DebugCallInfo *DebugCell;
|
|
|
|
// Async RPC
|
|
EVENT *ReceiveEvent ;
|
|
MUTEX *CallMutex ;
|
|
|
|
// the cell tag to the DebugCell above
|
|
CellTag DebugCellTag;
|
|
|
|
ULONG RcvBufferLength ;
|
|
BOOL ReceiveComplete ;
|
|
BOOL AsyncReply ;
|
|
LRPC_SCALL *NextSCall ;
|
|
LRPC_SCALL *LastSCall ;
|
|
ULONG NeededLength;
|
|
|
|
BOOL fSyncDispatch ;
|
|
BOOL Choked ;
|
|
long CancelPending;
|
|
|
|
RPC_MESSAGE RpcMessage;
|
|
RPC_RUNTIME_INFO RuntimeInfo;
|
|
QUEUE BufferQueue ;
|
|
|
|
RPC_CHAR *ClientPrincipalName;
|
|
public:
|
|
LRPC_SCONTEXT *SContext;
|
|
|
|
LRPC_SCALL (OUT RPC_STATUS *RpcStatus)
|
|
{
|
|
ObjectType = LRPC_SCALL_TYPE;
|
|
ReceiveEvent = 0;
|
|
CallMutex = 0;
|
|
LrpcAsyncReplyMessage = 0;
|
|
ClientPrincipalName = NULL;
|
|
*RpcStatus = RPC_S_OK;
|
|
|
|
if (IsServerSideDebugInfoEnabled())
|
|
{
|
|
DebugCell = (DebugCallInfo *)AllocateCell(&DebugCellTag);
|
|
if (DebugCell == NULL)
|
|
*RpcStatus = RPC_S_OUT_OF_MEMORY;
|
|
else
|
|
{
|
|
memset(DebugCell, 0, sizeof(DebugCallInfo));
|
|
DebugCell->FastInit = 0;
|
|
DebugCell->Type = dctCallInfo;
|
|
DebugCell->Status = (BYTE)csAllocated;
|
|
DebugCell->CallFlags = DBGCELL_CACHED_CALL | DBGCELL_LRPC_CALL;
|
|
DebugCell->LastUpdateTime = NtGetTickCount();
|
|
}
|
|
}
|
|
else
|
|
DebugCell = NULL;
|
|
}
|
|
|
|
~LRPC_SCALL (
|
|
)
|
|
{
|
|
if (ReceiveEvent)
|
|
{
|
|
if (CallId != (ULONG) -1)
|
|
{
|
|
Association->AssociationMutex.Request() ;
|
|
Association->SCallDict.Delete(ULongToPtr(CallId));
|
|
Association->AssociationMutex.Clear() ;
|
|
}
|
|
|
|
delete ReceiveEvent;
|
|
delete CallMutex ;
|
|
}
|
|
|
|
if (DebugCell)
|
|
{
|
|
FreeCell(DebugCell, &DebugCellTag);
|
|
}
|
|
|
|
if (LrpcAsyncReplyMessage)
|
|
{
|
|
FreeMessage(LrpcAsyncReplyMessage);
|
|
}
|
|
}
|
|
|
|
virtual RPC_STATUS
|
|
NegotiateTransferSyntax (
|
|
IN OUT PRPC_MESSAGE Message
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
GetBuffer (
|
|
IN OUT PRPC_MESSAGE Message,
|
|
IN UUID *ObjectUuid
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
SendReceive (
|
|
IN OUT PRPC_MESSAGE Message
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
Send (
|
|
IN OUT PRPC_MESSAGE Messsage
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
Receive (
|
|
IN PRPC_MESSAGE Message,
|
|
IN unsigned int Size
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
AsyncSend (
|
|
IN OUT PRPC_MESSAGE Message
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
AsyncReceive (
|
|
IN OUT PRPC_MESSAGE Message,
|
|
IN unsigned int Size
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
SetAsyncHandle (
|
|
IN PRPC_ASYNC_STATE pAsync
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
AbortAsyncCall (
|
|
IN PRPC_ASYNC_STATE pAsync,
|
|
IN unsigned long ExceptionCode
|
|
) ;
|
|
|
|
virtual void
|
|
FreeBuffer (
|
|
IN PRPC_MESSAGE Message
|
|
);
|
|
|
|
virtual void
|
|
FreePipeBuffer (
|
|
IN PRPC_MESSAGE Message
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
ReallocPipeBuffer (
|
|
IN PRPC_MESSAGE Message,
|
|
IN unsigned int NewSize
|
|
) ;
|
|
|
|
virtual RPC_STATUS
|
|
ImpersonateClient (
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
RevertToSelf (
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
GetAuthorizationContext (
|
|
IN BOOL ImpersonateOnReturn,
|
|
IN AUTHZ_RESOURCE_MANAGER_HANDLE AuthzResourceManager,
|
|
IN PLARGE_INTEGER pExpirationTime OPTIONAL,
|
|
IN LUID Identifier,
|
|
IN DWORD Flags,
|
|
IN PVOID DynamicGroupArgs OPTIONAL,
|
|
OUT PAUTHZ_CLIENT_CONTEXT_HANDLE pAuthzClientContext
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
IsClientLocal (
|
|
OUT unsigned int * ClientLocalFlag
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
ConvertToServerBinding (
|
|
OUT RPC_BINDING_HANDLE __RPC_FAR * ServerBinding
|
|
);
|
|
|
|
virtual void
|
|
InquireObjectUuid (
|
|
OUT RPC_UUID * ObjectUuid
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
ToStringBinding (
|
|
OUT RPC_CHAR ** StringBinding
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
GetAssociationContextCollection (
|
|
OUT ContextCollection **CtxCollection
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
InquireAuthClient (
|
|
OUT RPC_AUTHZ_HANDLE * Privileges,
|
|
OUT RPC_CHAR * * ServerPrincipalName, OPTIONAL
|
|
OUT unsigned long * AuthenticationLevel,
|
|
OUT unsigned long * AuthenticationService,
|
|
OUT unsigned long * AuthorizationService,
|
|
IN unsigned long Flags
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
InquireCallAttributes (
|
|
IN OUT void *RpcCallAttributes
|
|
);
|
|
|
|
virtual RPC_STATUS
|
|
InqTransportType(
|
|
OUT unsigned int __RPC_FAR * Type
|
|
)
|
|
{
|
|
*Type = TRANSPORT_TYPE_LPC ;
|
|
|
|
return (RPC_S_OK) ;
|
|
}
|
|
|
|
virtual BOOL
|
|
IsSyncCall (
|
|
)
|
|
{
|
|
return !pAsync ;
|
|
}
|
|
|
|
virtual unsigned
|
|
TestCancel(
|
|
)
|
|
{
|
|
return InterlockedExchange(&CancelPending, 0);
|
|
}
|
|
|
|
virtual void
|
|
FreeObject (
|
|
) ;
|
|
|
|
BOOL
|
|
IsClientAsync (
|
|
)
|
|
{
|
|
return ((Flags & LRPC_SYNC_CLIENT) == 0);
|
|
}
|
|
|
|
RPC_STATUS
|
|
ActivateCall (
|
|
IN LRPC_SASSOCIATION * Association,
|
|
IN LRPC_MESSAGE * LrpcMessage,
|
|
IN LRPC_MESSAGE * LrpcReplyMessage,
|
|
IN unsigned int Flags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
description
|
|
|
|
Arguments:
|
|
|
|
arg1 - description
|
|
|
|
--*/
|
|
|
|
{
|
|
RPC_STATUS Status = RPC_S_OK;
|
|
|
|
this->Association = Association;
|
|
this->LrpcRequestMessage = LrpcMessage;
|
|
this->LrpcReplyMessage = LrpcReplyMessage;
|
|
ObjectUuidFlag = 0;
|
|
PushedResponse = 0;
|
|
CurrentBufferLength = 0;
|
|
BufferComplete = 0;
|
|
this->Flags = Flags ;
|
|
FirstSend = 1;
|
|
PipeSendCalled = 0;
|
|
Deleted = 0;
|
|
SBinding = 0;
|
|
SetReferenceCount(2);
|
|
pAsync = 0;
|
|
NextSCall = 0;
|
|
LastSCall = 0;
|
|
CancelPending = 0;
|
|
|
|
if (DebugCell)
|
|
{
|
|
DebugCell->Status = (BYTE)csActive;
|
|
if (Association->CachedSCall == this)
|
|
DebugCell->CallFlags = DBGCELL_CACHED_CALL | DBGCELL_LRPC_CALL;
|
|
else
|
|
DebugCell->CallFlags = DBGCELL_CACHED_CALL;
|
|
DebugCell->LastUpdateTime = NtGetTickCount();
|
|
}
|
|
|
|
// Async RPC stuff
|
|
|
|
RuntimeInfo.Length = sizeof(RPC_RUNTIME_INFO) ;
|
|
RpcMessage.ReservedForRuntime = &RuntimeInfo;
|
|
RpcMessage.RpcFlags = 0;
|
|
|
|
if (LrpcMessage->Rpc.RpcHeader.SecurityContextId != -1)
|
|
{
|
|
SContext = Association->FindSecurityContext(
|
|
LrpcMessage->Rpc.RpcHeader.SecurityContextId);
|
|
if (SContext == NULL)
|
|
{
|
|
Status = RPC_S_ACCESS_DENIED;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SContext = NULL;
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
inline void DeactivateCall (void)
|
|
{
|
|
if (DebugCell)
|
|
{
|
|
DebugCell->Status = (BYTE)csAllocated;
|
|
DebugCell->LastUpdateTime = NtGetTickCount();
|
|
}
|
|
}
|
|
|
|
RPC_STATUS
|
|
ProcessResponse (
|
|
IN LRPC_MESSAGE **LrpcMessage
|
|
) ;
|
|
|
|
inline RPC_STATUS
|
|
LrpcMessageToRpcMessage (
|
|
IN LRPC_MESSAGE * LrpcMessage,
|
|
IN OUT PRPC_MESSAGE Message
|
|
) ;
|
|
|
|
void
|
|
DealWithRequestMessage (
|
|
);
|
|
|
|
void
|
|
HandleRequest (
|
|
) ;
|
|
|
|
LRPC_SASSOCIATION *
|
|
InqAssociation (
|
|
)
|
|
{
|
|
return Association;
|
|
}
|
|
|
|
LRPC_MESSAGE *
|
|
InitMsg (
|
|
)
|
|
{
|
|
INITMSG(LrpcReplyMessage,
|
|
ClientId,
|
|
CallbackId,
|
|
MessageId) ;
|
|
|
|
LrpcReplyMessage->LpcHeader.u1.s1.TotalLength =
|
|
LrpcReplyMessage->LpcHeader.u1.s1.DataLength+sizeof(PORT_MESSAGE);
|
|
|
|
return LrpcReplyMessage;
|
|
}
|
|
|
|
virtual RPC_STATUS
|
|
InqConnection (
|
|
OUT void **ConnId,
|
|
OUT BOOL *pfFirstCall
|
|
)
|
|
{
|
|
ASSERT(Association);
|
|
*ConnId = Association;
|
|
|
|
if (InterlockedIncrement((LONG *) &(Association->fFirstCall)) == 1)
|
|
{
|
|
*pfFirstCall = 1;
|
|
}
|
|
else
|
|
{
|
|
*pfFirstCall = 0;
|
|
}
|
|
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
inline HANDLE InqLocalClientPID (
|
|
void
|
|
)
|
|
{
|
|
return ClientId.UniqueProcess;
|
|
}
|
|
|
|
private:
|
|
RPC_STATUS
|
|
GetBufferDo(
|
|
IN OUT PRPC_MESSAGE Message,
|
|
IN unsigned long NewSize,
|
|
IN BOOL fDataValid
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
SetupCall (
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
GetCoalescedBuffer (
|
|
IN PRPC_MESSAGE Message,
|
|
IN BOOL BufferValid
|
|
) ;
|
|
|
|
RPC_STATUS
|
|
SendRequest (
|
|
IN OUT PRPC_MESSAGE Messsage,
|
|
OUT BOOL *Shutup
|
|
) ;
|
|
|
|
void
|
|
SendReply (
|
|
);
|
|
|
|
NTSTATUS
|
|
SendDGReply (
|
|
IN LRPC_MESSAGE *LrpcMessage
|
|
)
|
|
{
|
|
NTSTATUS NtStatus ;
|
|
|
|
LrpcMessage->LpcHeader.u2.ZeroInit = 0;
|
|
LrpcMessage->LpcHeader.CallbackId = 0;
|
|
LrpcMessage->LpcHeader.MessageId = 0;
|
|
LrpcMessage->LpcHeader.u1.s1.TotalLength =
|
|
LrpcMessage->LpcHeader.u1.s1.DataLength+sizeof(PORT_MESSAGE);
|
|
|
|
NtStatus = NtRequestPort(Association->LpcReplyPort,
|
|
(PORT_MESSAGE *) LrpcMessage);
|
|
|
|
if (!NT_SUCCESS(NtStatus))
|
|
{
|
|
#if DBG
|
|
if ((NtStatus != STATUS_INVALID_CID)
|
|
&& (NtStatus != STATUS_REPLY_MESSAGE_MISMATCH)
|
|
&& (NtStatus != STATUS_PORT_DISCONNECTED))
|
|
{
|
|
PrintToDebugger("RPC : NtRequestPort : %lx, 0x%x, 0x%x\n",
|
|
NtStatus, this, Association->LpcReplyPort);
|
|
ASSERT(NtStatus != STATUS_INVALID_HANDLE);
|
|
}
|
|
#endif // DBG
|
|
Association->Delete();
|
|
}
|
|
|
|
|
|
return NtStatus ;
|
|
}
|
|
|
|
LRPC_SBINDING *
|
|
LookupBinding (
|
|
IN unsigned short PresentContextId
|
|
);
|
|
};
|
|
|
|
|
|
inline NTSTATUS
|
|
LRPC_SASSOCIATION::ReplyMessage(
|
|
IN LRPC_MESSAGE * LrpcMessage
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Send a reply to a request
|
|
|
|
Arguments:
|
|
|
|
LrpcMessage - Reply message
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - Function succeeded
|
|
RPC_S_OUT_OF_MEMORY - we ran out of memory
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS NtStatus ;
|
|
|
|
LrpcMessage->LpcHeader.u1.s1.TotalLength =
|
|
LrpcMessage->LpcHeader.u1.s1.DataLength + sizeof(PORT_MESSAGE);
|
|
|
|
NtStatus = NtReplyPort(LpcServerPort,
|
|
(PORT_MESSAGE *) LrpcMessage) ;
|
|
|
|
if (!NT_SUCCESS(NtStatus))
|
|
{
|
|
#if DBG
|
|
PrintToDebugger("RPC : NtReplyPort : %lx, 0x%x, 0x%x\n",
|
|
NtStatus,
|
|
this,
|
|
LpcReplyPort);
|
|
ASSERT(NtStatus != STATUS_INVALID_HANDLE);
|
|
#endif
|
|
Delete();
|
|
}
|
|
|
|
return NtStatus ;
|
|
}
|
|
|
|
inline void
|
|
LRPC_SCONTEXT::AddReference()
|
|
{
|
|
Association->AssociationMutex.VerifyOwned();
|
|
ASSERT(RefCount);
|
|
|
|
RefCount++;
|
|
}
|
|
|
|
inline void
|
|
LRPC_SCONTEXT::RemoveReference()
|
|
{
|
|
LRPC_SASSOCIATION *MyAssoc = Association;
|
|
|
|
Association->AssociationMutex.Request();
|
|
RefCount--;
|
|
if (RefCount == 0)
|
|
{
|
|
delete this;
|
|
}
|
|
MyAssoc->AssociationMutex.Clear();
|
|
}
|
|
|
|
inline void
|
|
LRPC_SCONTEXT::Destroy()
|
|
{
|
|
Association->AssociationMutex.VerifyOwned();
|
|
if (GetDeletedFlag() == 0)
|
|
{
|
|
SetDeletedFlag();
|
|
RefCount--;
|
|
if (RefCount == 0)
|
|
{
|
|
delete this;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif //__LPCSVR_HXX__
|