535 lines
20 KiB
C
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
/*++
Copyright (c) 1987-1993 Microsoft Corporation
Module Name:
Remote.c
Abstract:
Provides a support routine, RxRemoteAPI, for transporting
an API request to a remote API worker and translating its response.
Author:
John Rogers (JohnRo) and a cast of thousands.
Environment:
Portable to any flat, 32-bit environment. (Uses Win32 typedefs.)
Requires ANSI C extensions: slash-slash comments, long external names.
Revision History:
01-Apr-1991 JohnRo
Created portable LanMan (NT) version from LanMan 2.x.
03-May-1991 JohnRo
Really pass COPY of parm desc to convert args, like it expects.
Also pass it UNC server name (\\stuff) for ease of use.
Add check for valid computer name.
Use Unicode transitional types.
Changed to use three data descs: 16-bit, 32-bit and SMB versions.
Fixed receive buffer length problem.
RcvDataPresent flag is not needed.
Use RxpFatalErrorCode() rather than checking for specific errors.
Quiet debug output by default.
Reduced recompile hits from header files.
Don't use NET_API_FUNCTION for non-APIs.
07-May-1991 JohnRo
Made changes to reflect CliffV's code review.
Add validation of parm desc.
09-May-1991 JohnRo
Made LINT-suggested changes.
14-May-1991 JohnRo
Pass 3 aux descriptors to RxRemoteApi.
29-May-1991 JohnRo
Handle SendDataPtr16 and SendDataSize16 correctly for setinfo.
Print status if ConvertArgs failed.
13-Jun-1991 JohnRo
RxpConvertArgs needs DataDesc16 and AuxDesc16 to fix server set info
for level 102.
16-Jul-1991 JohnRo
Allow receive data buffer to be zero bytes long with nonnull pointer.
17-Jul-1991 JohnRo
Extracted RxpDebug.h from Rxp.h.
16-Aug-1991 rfirth
Changed interface (NoPermissionRequired => Flags)
25-Sep-1991 JohnRo
Quiet normal debug messages.
21-Nov-1991 JohnRo
Removed NT dependencies to reduce recompiles.
27-Nov-1991 JohnRo
Do some checking of ApiNumber.
31-Mar-1992 JohnRo
Prevent too large size requests.
06-May-1993 JohnRo
RAID 8849: Export RxRemoteApi for DEC and others.
Use NetpKdPrint() where possible.
Use PREFIX_ equates.
--*/
// These must be included first:
#include <windef.h> // IN, DWORD, LPTSTR, etc.
#include <rxp.h> // Private header file.
// These may be included in any order:
#include <apiworke.h> // REM_MAX_PARMS.
#include <limits.h> // CHAR_BIT.
#include <lmerr.h> // NERR_ and ERROR_ equates.
#include <names.h> // NetpIsComputerNameValid().
#include <netdebug.h> // NetpKdPrint(), FORMAT_ equates, etc.
#include <netlib.h> // NetpMemoryFree(), etc.
#include <prefix.h> // PREFIX_ equates.
#include <rap.h> // RapIsValidDescriptorSmb().
#include <rx.h> // My prototype, etc.
#include <rxpdebug.h> // IF_DEBUG().
NET_API_STATUS
RxRemoteApi(
IN DWORD ApiNumber,
IN LPCWSTR UncServerName, // this is not OPTIONAL!
IN LPDESC ParmDescString,
IN LPDESC DataDesc16 OPTIONAL,
IN LPDESC DataDesc32 OPTIONAL,
IN LPDESC DataDescSmb OPTIONAL,
IN LPDESC AuxDesc16 OPTIONAL,
IN LPDESC AuxDesc32 OPTIONAL,
IN LPDESC AuxDescSmb OPTIONAL,
IN DWORD Flags,
... // rest of API's arguments
)
// Define equate for last named (non-variable) argument, for use with
// va_start macro.
#define LAST_NAMED_ARGUMENT Flags
/*++
Routine Description:
RxRemoteApi (which is analogous to LanMan's NetIRemoteAPI) formats
parameter and data buffers for transporting a local API request to a
remote API worker and translates the remote response into the local
equivalent.
Arguments:
ApiNumber - Function number of the API required.
ServerName - Points to the name of server this API is to be executed on.
This MUST begin with "\\".
ParmDescString - A pointer to a ASCIIZ string describing the API call
parameters (other than server name).
DataDesc16 - A pointer to a ASCIIZ string describing the
structure of the data in the call, i.e. the return data structure
for a Enum or GetInfo call. This string is used for adjusting pointers
to data in the local buffers after transfer across the net. If there
is no structure involved in the call then DataDesc16 must be
a NULL pointer. DataDesc16 is a "modified" 16-bit descriptor,
which may contain "internal use only" characters.
DataDesc32 - A pointer to the 32-bit version of DataDesc16. Must be NULL
iff DataDesc16 is NULL.
DataDescSmb - An optional pointer to the SMB version of DataDesc16.
This must not contain any "internal use only" characters. Must be NULL
iff DataDesc16 is NULL.
AuxDesc16, AuxDesc32, AuxDescSmb - Will be NULL in most cases unless a
REM_AUX_COUNT descriptor char is present in the data descriptors in
which case these descriptors define a secondary data format as
DataDesc16/DataDescSmb define the primary.
Flags - bitmap of various control flags. Currently defined are:
NO_PERMISSION_REQUIRED (0x1) - This flag is TRUE if this API does not
require any permission on the remote machine, and that a null session
is to be used for this request.
ALLOCATE_RESPONSE (0x2) - Set if this routine and its subordinates
(viz RxpConvertArgs, RxpConvertBlock) allocate a response buffer.
We do this because at this level we know how much data is returned
from the down-level server in an Enum or GetInfo call. We can therefore
better estimate the size of buffer to allocate and return to the caller
with 32-bit data than can the individual RxNet routines. The upshot of
this is that we waste less space
... - The remainder of the parameters for the API call as given by the
application. (The "..." notation is from ANSI C, and refers to a
variable argument list. These arguments will be processing using the
ANSI <stdarg.h> macros.)
Return Value:
NET_API_STATUS.
--*/
{
BYTE parm_buf[REM_MAX_PARMS]; // Parameter buffer
LPDESC ParmDescCopy; // Copy of parm desc in buffer.
LPBYTE parm_pos; // Pointer into parm_buf
va_list ParmPtr; // Pointer to stack parms.
DWORD parm_len; // Length of send parameters
DWORD ret_parm_len; // Length of expected parms
DWORD rcv_data_length; // Length of caller's rcv buf
LPVOID rcv_data_ptr; // Pointer to callers rcv buf
LPVOID SendDataPtr16; // Ptr to send buffer to use
DWORD SendDataSize16; // Size of caller's send buf
LPBYTE SmbRcvBuffer = NULL; // Rcv buffer, 16-bit version.
DWORD SmbRcvBufferLength; // Length of the above.
NET_API_STATUS Status; // Return status from remote.
LPTSTR TransportName = NULL; // Optional transport name.
IF_DEBUG(REMOTE) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: entered, api num " FORMAT_DWORD "...\n",
ApiNumber ));
}
//
// Make sure API number doesn't get truncated.
// Note that we can't check against the MAX_API equate anymore, as
// that only includes APIs which we know about. Now that RxRemoteApi is
// being exported for use by anyone, we don't know the maximum API number
// which the app might be using.
//
if ( ((DWORD)(WORD)ApiNumber) != ApiNumber ) {
NetpKdPrint(( PREFIX_NETAPI
"RxpStartBuildingTransaction: API NUMBER "
"(" FORMAT_HEX_DWORD ") TOO LARGE, "
"returning ERROR_INVALID_PARAMETER.\n",
ApiNumber ));
return (ERROR_INVALID_PARAMETER);
}
#if DBG
// Code in this file depends on 16-bit words; the Remote Admin Protocol
// demands it.
NetpAssert( ( (sizeof(WORD)) * CHAR_BIT) == 16);
// We are removing these for the time being because the delay load handler for this
// function returns FALSE by default, so the ASSERT will fire incorrectly. For a future
// release, we should change our delayload handler.
//NetpAssert(RapIsValidDescriptorSmb(ParmDescString));
if (DataDescSmb != NULL) {
//NetpAssert(RapIsValidDescriptorSmb(DataDescSmb));
NetpAssert(DataDesc16 != NULL);
NetpAssert(DataDesc32 != NULL);
} else {
NetpAssert(DataDesc16 == NULL);
NetpAssert(DataDesc32 == NULL);
}
if (AuxDescSmb != NULL) {
//NetpAssert(RapIsValidDescriptorSmb(AuxDescSmb));
NetpAssert(AuxDesc16 != NULL);
NetpAssert(AuxDesc32 != NULL);
} else {
NetpAssert(AuxDesc16 == NULL);
NetpAssert(AuxDesc32 == NULL);
}
#endif
if (! NetpIsUncComputerNameValid((LPWSTR)UncServerName)) {
return (NERR_InvalidComputer);
}
//
// Set found parameter flags to FALSE and pointers to NULL.
//
rcv_data_length = 0;
rcv_data_ptr = NULL;
//
// First build the parameter block which will be sent to the API
// worker. This consists of the two descriptor strings, ParmDescString
// and DataDescSmb, followed by the parameters (or the data pointed
// to by the parameters) that were passed to RxRemoteApi.
//
parm_pos = parm_buf; // Start of parameter buffer.
parm_len = 0;
ret_parm_len = 2* sizeof(WORD); // Allow for return status & converter.
Status = RxpStartBuildingTransaction(
parm_buf, // buffer start
REM_MAX_PARMS, // buffer len
ApiNumber, // API number
ParmDescString, // parm desc (original)
DataDescSmb, // data desc (SMB version)
(LPVOID *)&parm_pos, // roving output ptr
& parm_len, // curr output len (updated)
NULL, // last string ptr (don't care)
& ParmDescCopy); // ptr to parm desc copy
if (Status != NERR_Success) {
//
// Consider increasing REM_MAX_PARMS...
//
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: Buffer overflow!\n" ));
NetpBreakPoint();
return (Status);
}
//
// Set up ParmPtr to point to first of the caller's parameters.
//
va_start(ParmPtr, LAST_NAMED_ARGUMENT);
//
// If this API has specified a transport name, load the transport
// name from the first parameter.
//
if (Flags & USE_SPECIFIC_TRANSPORT) {
//
// Set up ParmPtr to point to first of the caller's parameters.
//
TransportName = va_arg(ParmPtr, LPTSTR);
}
//
// Build the rest of the transaction by converting the rest of the
// 32-bit arguments.
//
Status = RxpConvertArgs(
ParmDescCopy, // copy of desc, in SMB buf, will be updated.
DataDesc16,
DataDesc32,
DataDescSmb,
AuxDesc16,
AuxDesc32,
AuxDescSmb,
REM_MAX_PARMS, // MaximumInputBlockLength
REM_MAX_PARMS, // MaximumOutputBlockLength
& ret_parm_len, // curr inp blk len (updated)
& parm_len, // curr output len (updated)
& parm_pos, // curr output ptr (updated)
& ParmPtr, // rest of API's arguments (after server name)
& SendDataSize16, // caller's send buff siz (set)
(LPBYTE *) & SendDataPtr16, // caller's send buff (set)
& rcv_data_length, // caller's rcv buff len (set)
//
// WARNING: Added complexity. Unfortunate, but there you go.
// If we are to allocate the 32-bit received data buffer for
// the caller then the value passed back in rcv_data_ptr will
// be the address of the caller's pointer to the buffer.
// Although this makes things more difficult to understand
// down here, it simplifies life for the caller. Such is the
// lot of the (put upon) systems programmer...
//
(LPBYTE *)&rcv_data_ptr, // caller's rcv buff (set)
Flags
);
va_end(ParmPtr);
//NetpAssert(RapIsValidDescriptorSmb(ParmDescCopy));
if (Status != NERR_Success) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: RxpConvertArgs failed, status="
FORMAT_API_STATUS "\n", Status ));
return (Status);
}
IF_DEBUG(REMOTE) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: RxpConvertArgs says r.data.len=" FORMAT_DWORD
"\n", rcv_data_length ));
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: RxpConvertArgs says r.parm.len=" FORMAT_DWORD
"\n", ret_parm_len ));
}
//
// Set SmbRcvBuffer, and SmbRcvBufferLength to some appropriate values.
// Because we now allow the final 32-bit receive buffer to be allocated
// somewhere along the codepath of this routine, it is legal to have a
// non-zero receive data length with a meaningless receive data pointer
// (which could be NULL)
//
if (rcv_data_length != 0) {
//
// Allocate the SMB receive buffer.
//
SmbRcvBufferLength = rcv_data_length;
SmbRcvBuffer = NetpMemoryAllocate( SmbRcvBufferLength );
if (SmbRcvBuffer == NULL) {
return ERROR_NOT_ENOUGH_MEMORY;
}
#if DBG
IF_DEBUG(REMOTE) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: allocated " FORMAT_DWORD
" bytes as SmbRcvBuffer @ " FORMAT_LPVOID "\n",
SmbRcvBufferLength, (LPVOID) SmbRcvBuffer ));
}
#endif
} else {
//
// Allow zero-length receive data buffer with nonnull addr.
//
SmbRcvBufferLength = 0;
SmbRcvBuffer = NULL;
//
// Don't understand the commentary here: its not a nonnull addr., and
// as far as I can tell never has been. JR?
// RLF 8-19
//
IF_DEBUG(REMOTE) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: using 0 len buffer w/ nonnull addr.\n" ));
}
}
//
// The parameter buffers and data buffers are now set up for
// sending to the API worker so call RxpTransactSmb to send them.
//
NetpAssert( parm_len <= MAX_TRANSACT_SEND_PARM_SIZE );
NetpAssert( SendDataSize16 <= MAX_TRANSACT_SEND_DATA_SIZE );
NetpAssert( ret_parm_len <= MAX_TRANSACT_RET_PARM_SIZE );
NetpAssert( SmbRcvBufferLength <= MAX_TRANSACT_RET_DATA_SIZE );
Status = RxpTransactSmb(
(LPWSTR)UncServerName, // computer name
TransportName,
parm_buf, // Send parm buffer
parm_len, // Send parm length
SendDataPtr16, // Send data buffer
SendDataSize16, // Send data size
parm_buf, // Rcv prm buffer
ret_parm_len, // Rcv parm length
SmbRcvBuffer, // Rcv data buffer
&SmbRcvBufferLength, // Rcv data length (returned actual bytes read)
//
// NoPermissionRequired is now a bit in the Flags word
//
(BOOL)(Flags & NO_PERMISSION_REQUIRED)
);
#if DBG
IF_DEBUG(REMOTE) {
NetpKdPrint(( PREFIX_NETAPI
"RxRemoteApi: back from RxpTransactSmb, status="
FORMAT_API_STATUS "\n", Status ));
}
NetpAssert( SmbRcvBufferLength <= rcv_data_length );
if (SmbRcvBuffer == NULL) {
NetpAssert( SmbRcvBufferLength == 0 );
}
#endif
if (Status != NERR_Success) {
switch (Status) {
case NERR_BufTooSmall: // No data returned from API worker
rcv_data_length = 0;
break;
case ERROR_MORE_DATA: // Just a warning for the caller
break;
case NERR_TooMuchData: // Just a warning for the caller
break;
default:
rcv_data_length = 0;
break;
}
}
NetpAssert( SmbRcvBufferLength <= MAX_TRANSACT_RET_DATA_SIZE );
// The API call was successful. Now translate the return buffers
// into the local API format.
//
// If we didn't have some fatal error (e.g. unable to talk to the remote
// machine at all), then convert back... Note that fatal errors don't
// include statuses returned by the remote machine.
//
if (! RxpFatalErrorCode(Status)) {
//
// Set up ParmPtr to point to first of the caller's parameters.
//
va_start(ParmPtr, LAST_NAMED_ARGUMENT);
if (Flags & USE_SPECIFIC_TRANSPORT) {
//
// Skip over the first argument - it's the transport.
//
(VOID) va_arg(ParmPtr, LPTSTR);
}
Status = RxpConvertBlock(
ApiNumber,
parm_buf, // response blk
ParmDescString, // parm desc (original)
DataDesc16,
DataDesc32,
AuxDesc16,
AuxDesc32,
& ParmPtr, // rest of API's args
SmbRcvBuffer, // 16-bit version of rcv buff
SmbRcvBufferLength, // amount of BYTES in SmbRcvBuffer
//
// WARNING: Increased complexity. If the ALLOCATE_RESPONSE
// flag is set, then the caller supplied us with the address
// of the pointer to the buffer (which is about to be
// allocated in RxpConvertBlock). Else, rcv_data_ptr is
// the address of the buffer that the caller has already
// allocated before calling this routine
//
rcv_data_ptr, // native version of rcv buff
rcv_data_length, // length of the above
Flags // allocate a 32-bit response buffer?
);
va_end(ParmPtr);
//
// Don't check Status; we'll just return it to caller.
//
}
if (SendDataPtr16 != NULL) {
NetpMemoryFree(SendDataPtr16); // If add'l mem alloc'ed, free it.
}
if (SmbRcvBuffer != NULL) {
NetpMemoryFree(SmbRcvBuffer);
}
return(Status); // Return status from RxpConvertBlock or
// RxpTransactSmb.
} // RxRemoteApi