windows-nt/Source/XPSP1/NT/ds/netapi/rpcxlate/rxapi/svcgtinf.c
2020-09-26 16:20:57 +08:00

194 lines
5.8 KiB
C

/*++
Copyright (c) 1991-92 Microsoft Corporation
Module Name:
SvcGtInf.c
Abstract:
This file contains the RpcXlate code to handle the NetServiceGetInfo API.
Author:
John Rogers (JohnRo) 11-Sep-1991
Environment:
Portable to any flat, 32-bit environment. (Uses Win32 typedefs.)
Requires ANSI C extensions: slash-slash comments, long external names.
Revision History:
11-Sep-1991 JohnRo
Implement downlevel NetService APIs.
16-Sep-1991 JohnRo
Fixed level check.
22-Oct-1991 JohnRo
Free buffer on error.
07-Feb-1992 JohnRo
Use NetApiBufferAllocate() instead of private version.
--*/
// These must be included first:
#include <windef.h> // IN, DWORD, etc.
#include <lmcons.h> // LM20_ equates, NET_API_STATUS, etc.
// These may be included in any order:
#include <apinums.h> // API_ equates.
#include <lmapibuf.h> // NetApiBufferAllocate().
#include <lmerr.h> // ERROR_ and NERR_ equates.
#include <lmsvc.h>
#include <rxp.h> // RxpFatalErrorCode().
#include <netdebug.h> // DBGSTATIC, NetpKdPrint(()), FORMAT_ equates.
#include <rap.h> // LPDESC.
#include <remdef.h> // REM16_, REM32_, REMSmb_ equates.
#include <rx.h> // RxRemoteApi().
#include <rxpdebug.h> // IF_DEBUG().
#include <rxsvc.h> // My prototype.
#include <strucinf.h> // NetpServiceStructureInfo().
NET_API_STATUS
RxNetServiceGetInfo (
IN LPTSTR UncServerName,
IN LPTSTR Service,
IN DWORD Level,
OUT LPBYTE *BufPtr
)
/*++
Routine Description:
RxNetServiceGetInfo performs the same function as NetServiceGetInfo, except
that the server name is known to refer to a downlevel server.
Arguments:
(Same as NetServiceGetInfo, except UncServerName must not be null, and
must not refer to the local computer.)
Return Value:
(Same as NetServiceGetInfo.)
--*/
{
LPDESC DataDesc16, DataDesc32, DataDescSmb;
LPBYTE ApiBuffer32; // Buffer to be returned to caller.
DWORD ApiBufferSize32;
NET_API_STATUS Status;
DWORD TotalAvail;
LPSERVICE_INFO_2 serviceInfo2;
IF_DEBUG(SERVICE) {
NetpKdPrint(("RxNetServiceGetInfo: starting, server=" FORMAT_LPTSTR
", lvl=" FORMAT_DWORD ".\n", UncServerName, Level));
}
//
// Error check DLL stub and the app.
//
NetpAssert(UncServerName != NULL);
if (BufPtr == NULL) {
return (ERROR_INVALID_PARAMETER);
}
*BufPtr = NULL; // assume error; it makes error handlers easy to code.
// This also forces possible GP fault before we allocate memory.
//
// Learn about info level.
//
Status = NetpServiceStructureInfo (
Level, // level to learn about
PARMNUM_ALL, // No parmnum with this.
TRUE, // Need native sizes.
& DataDesc16,
& DataDesc32,
& DataDescSmb,
& ApiBufferSize32, // max buffer size (native)
NULL, // don't need fixed size.
NULL // don't need string size.
);
if (Status != NERR_Success) {
return (Status);
}
//
// Allocate memory for 32-bit version of info, which we'll use to get
// data from the remote computer.
//
Status = NetApiBufferAllocate(
ApiBufferSize32,
(LPVOID *) & ApiBuffer32);
if (Status != NERR_Success) {
return (Status);
}
IF_DEBUG(SERVICE) {
NetpKdPrint(( "RxNetServiceGetInfo: allocated buffer at "
FORMAT_LPVOID "\n", (LPVOID) ApiBuffer32 ));
}
//
// Actually remote the API, which will get back the
// data in native format.
//
Status = RxRemoteApi(
API_WServiceGetInfo, // API number
UncServerName, // Required, with \\name.
REMSmb_NetServiceGetInfo_P, // parm desc
DataDesc16,
DataDesc32,
DataDescSmb,
NULL, // no aux data desc 16
NULL, // no aux data desc 32
NULL, // no aux data desc SMB
FALSE, // not a null session API
// rest of API's arguments, in 32-bit LM 2.x format:
Service,
Level,
ApiBuffer32,
ApiBufferSize32,
& TotalAvail); // total size
NetpAssert( Status != ERROR_MORE_DATA );
NetpAssert( Status != NERR_BufTooSmall );
if (Status == NERR_Success) {
*BufPtr = ApiBuffer32;
if ((! RxpFatalErrorCode(Status)) && ((Level == 2) || (Level==1 ))) {
serviceInfo2 = (LPSERVICE_INFO_2)*BufPtr;
if (serviceInfo2 != NULL) {
DWORD installState;
if (Level == 2) {
serviceInfo2->svci2_display_name = serviceInfo2->svci2_name;
}
//
// if INSTALL or UNINSTALL is PENDING, then force the upper
// bits to 0. This is to prevent the upper bits of the wait
// hint from getting accidentally set. Downlevel should never
// use more than FF for waithint.
//
installState = serviceInfo2->svci2_status & SERVICE_INSTALL_STATE;
if ((installState == SERVICE_INSTALL_PENDING) ||
(installState == SERVICE_UNINSTALL_PENDING)) {
serviceInfo2->svci2_code &= SERVICE_RESRV_MASK;
}
}
}
} else {
(void) NetApiBufferFree( ApiBuffer32 );
}
return (Status);
} // RxNetServiceGetInfo