windows-nt/Source/XPSP1/NT/com/rpc/locator/client/autohand.cxx
2020-09-26 16:20:57 +08:00

319 lines
6 KiB
C++
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (C) Microsoft Corporation, 1992 - 1999
Module Name:
autohand.cxx
Abstract:
This module implements the autohandle functions call by the compiler
generated stubs.
Author:
Steven Zeck (stevez) 03/10/92
Kamen Moutafov (KamenM) Feb 2000 - add support for multiple transfer syntaxes
--*/
#include <rpc.h>
#include <rpcdcep.h>
#ifdef __cplusplus
extern "C" {
#endif
#include <rpcnsi.h>
#include <rpcnsip.h>
#ifdef __cplusplus
}
#endif
typedef RPC_STATUS
(RPC_ENTRY *RPC_BIND)
(
IN PRPC_MESSAGE Message
);
RPC_STATUS
RpcpNsGetBuffer(
IN PRPC_MESSAGE Message,
RPC_BIND BindFn
);
// Interface can be only RPC_CLIENT_INTERFACE
inline BOOL DoesInterfaceSupportMultipleTransferSyntaxes(void *Interface)
{
// the client and server interface have the same layout - we can just
// use one of them
RPC_CLIENT_INTERFACE *ClientInterface = (RPC_CLIENT_INTERFACE *)Interface;
if (ClientInterface->Length == NT351_INTERFACE_SIZE)
return FALSE;
return (ClientInterface->Flags & RPCFLG_HAS_MULTI_SYNTAXES);
}
RPC_STATUS RPC_ENTRY
I_RpcNsGetBuffer(
IN PRPC_MESSAGE Message
)
/*++
Routine Description:
Auto bind to a given interface.
Arguments:
Message - describes the interface that we want to auto bind to.
Returns:
RPC_S_OK, RPC_S_CALL_FAILED_DNE, I_RpcGetBuffer()
--*/
{
if (!DoesInterfaceSupportMultipleTransferSyntaxes(Message->RpcInterfaceInformation))
Message->ImportContext = 0;
return RpcpNsGetBuffer(Message, I_RpcGetBuffer);
}
extern "C"
{
RPC_STATUS RPC_ENTRY
I_RpcNsNegotiateTransferSyntax(
IN PRPC_MESSAGE Message
);
}; // extern "C"
RPC_STATUS RPC_ENTRY
I_RpcNsNegotiateTransferSyntax(
IN PRPC_MESSAGE Message
)
/*++
Routine Description:
Auto bind to a given interface.
Arguments:
Message - describes the interface that we want to auto bind to.
Returns:
RPC_S_OK, RPC_S_CALL_FAILED_DNE, I_RpcGetBuffer()
--*/
{
Message->ImportContext = 0;
return RpcpNsGetBuffer(Message, I_RpcNegotiateTransferSyntax);
}
RPC_STATUS
RpcpNsGetBuffer(
IN PRPC_MESSAGE Message,
RPC_BIND BindFn
)
/*++
Routine Description:
Auto bind to a given interface.
Arguments:
Message - describes the interface that we want to auto bind to.
Returns:
RPC_S_OK or error
--*/
{
RPC_STATUS status;
PRPC_IMPORT_CONTEXT_P Import;
int fSetAge = 0;
// If there already is a handle, use it directly.
if (Message->Handle)
return(BindFn(Message));
Message->ImportContext = 0;
for (int cTry = 0; cTry < 2; cTry++)
{
status = RpcNsBindingImportBegin(RPC_C_NS_SYNTAX_DEFAULT, 0,
Message->RpcInterfaceInformation, 0, &Message->ImportContext);
Import = (PRPC_IMPORT_CONTEXT_P) Message->ImportContext;
if (status)
break;
// This will cause some Zecky Magic
// Also note that naively one might move this ExpAge to
// after the first lookup failed time- that is a bobo
if (fSetAge)
status = RpcNsMgmtHandleSetExpAge (Message->ImportContext, 0);
// If we found no bindings last time, tell the locator
// to look on the net.
// For each handle returned by import, try using it with I_RpcGetBuffer
// function until success or no more import handles.
while(RpcNsBindingImportNext(Import, &Import->ProposedHandle)
== RPC_S_OK)
{
Message->Handle = Import->ProposedHandle;
if (BindFn(Message) == RPC_S_OK)
return(RPC_S_OK);
RpcBindingFree(&Import->ProposedHandle);
}
fSetAge = 1;
RpcNsBindingImportDone(&Message->ImportContext);
}
if (status == RPC_S_NAME_SERVICE_UNAVAILABLE ||
status == RPC_S_OUT_OF_MEMORY ||
status == RPC_S_OUT_OF_RESOURCES)
return(status);
return(RPC_S_CALL_FAILED_DNE);
}
RPC_STATUS RPC_ENTRY
I_RpcNsSendReceive(
IN PRPC_MESSAGE Message,
OUT RPC_BINDING_HANDLE * Handle
)
/*++
Routine Description:
Make a call on a RPC server. If the call fails continue looking for
servers to bind to. This is only needed for protocols
which don't allocate the connection at I_RpcGetBuffer time.
Arguments:
Message - describes the interface that we want to Call. The
ImportContext field contains the active lookup handle, if any.
Handle - returns the binding handle used on the successfull call.
Returns:
RPC_S_CALL_FAILED_DNE, I_RpcSendReceive()
--*/
{
RPC_STATUS status;
PRPC_IMPORT_CONTEXT_P Import;
Import = (PRPC_IMPORT_CONTEXT_P) Message->ImportContext;
while ((status = I_RpcSendReceive(Message)) != RPC_S_OK && Import)
{
RpcBindingFree(&Import->ProposedHandle);
// The call failed, try the next binding handle
while(RpcNsBindingImportNext(Import, &Import->ProposedHandle)
== RPC_S_OK)
{
Message->Handle = Import->ProposedHandle;
I_RpcReBindBuffer(Message);
continue;
}
I_RpcFreeBuffer(Message);
status = RPC_S_CALL_FAILED_DNE;
}
if (Import)
{
*Handle = Import->ProposedHandle;
RpcNsBindingImportDone(&Message->ImportContext);
}
if (status)
*Handle = 0;
return(status);
}
void RPC_ENTRY
I_RpcNsRaiseException(
IN PRPC_MESSAGE Message,
IN RPC_STATUS Status
)
/*++
Routine Description:
Raise an Exception for an autobind interface. We simply cleanup any
open import context and raise the exception.
Arguments:
Message - continas the import context, if any.
--*/
{
RpcNsBindingImportDone(&Message->ImportContext);
RpcRaiseException(Status);
}
// BUG BUG implement this is the runtime.
RPC_STATUS RPC_ENTRY
I_RpcReBindBuffer(
IN PRPC_MESSAGE Message
)
{
(void __RPC_FAR *) Message;
RpcRaiseException(RPC_S_INTERNAL_ERROR);
return(0);
}