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.
1023 lines
32 KiB
1023 lines
32 KiB
/*++
|
|
|
|
Copyright (c) 1991 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ApiWksta.c
|
|
|
|
Abstract:
|
|
|
|
This module contains individual API handlers for the NetWksta
|
|
APIs.
|
|
|
|
SUPPORTED : NetWkstaGetInfo, NetWkstaSetInfo.
|
|
|
|
UNSUPPORTED : NetWkstaSetUid.
|
|
|
|
SEE ALSO : NetWkstaUserLogon, NetWkstaUserLogoff - in ApiLogon.c.
|
|
|
|
Author:
|
|
|
|
Shanku Niyogi (w-shanku) 25-Feb-1991
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "XactSrvP.h"
|
|
|
|
//
|
|
// Declaration of descriptor strings.
|
|
//
|
|
|
|
STATIC const LPDESC Desc16_wksta_info_0 = REM16_wksta_info_0;
|
|
STATIC const LPDESC Desc16_wksta_info_1 = REM16_wksta_info_1;
|
|
STATIC const LPDESC Desc16_wksta_info_10 = REM16_wksta_info_10;
|
|
|
|
//
|
|
// The size of the heuristics is actually 55 chars but we add one
|
|
// for padding.
|
|
//
|
|
|
|
#define SIZE_HEURISTICS 56
|
|
|
|
NTSTATUS
|
|
XsNetWkstaGetInfo (
|
|
API_HANDLER_PARAMETERS
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine sets up a call to NetWkstaGetInfo. Because of the differences
|
|
between 16- and 32-bit structures, this routine does not use the normal
|
|
conversion process.
|
|
|
|
Arguments:
|
|
|
|
API_HANDLER_PARAMETERS - information about the API call. See
|
|
XsTypes.h for details.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS - STATUS_SUCCESS or reason for failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
NET_API_STATUS status;
|
|
|
|
PXS_NET_WKSTA_GET_INFO parameters = Parameters;
|
|
LPWKSTA_INFO_100 wksta_100 = NULL; // Native parameters
|
|
LPWKSTA_INFO_101 wksta_101 = NULL;
|
|
LPWKSTA_INFO_502 wksta_502 = NULL;
|
|
|
|
LPBYTE stringLocation = NULL; // Conversion variables
|
|
DWORD bytesRequired = 0;
|
|
BOOL varWrite;
|
|
LPWKSTA_16_INFO_1 entry1;
|
|
LPWKSTA_16_INFO_10 entry10;
|
|
TCHAR heuristics[SIZE_HEURISTICS];
|
|
DWORD i;
|
|
USHORT level;
|
|
|
|
WCHAR lanroot[PATHLEN+1];
|
|
|
|
API_HANDLER_PARAMETERS_REFERENCE; // Avoid warnings
|
|
|
|
IF_DEBUG(WKSTA) {
|
|
NetpKdPrint(( "XsNetWkstaGetInfo: header at %lx, "
|
|
"params at %lx, level %ld\n",
|
|
Header, parameters, SmbGetUshort( ¶meters->Level ) ));
|
|
}
|
|
|
|
try {
|
|
//
|
|
// Check for errors.
|
|
//
|
|
|
|
level = SmbGetUshort( ¶meters->Level );
|
|
if ( (level != 0) && (level != 1) && (level != 10) ) {
|
|
Header->Status = ERROR_INVALID_LEVEL;
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// we return the system directory as the lanroot
|
|
//
|
|
|
|
*lanroot = 0;
|
|
GetSystemDirectory(lanroot, sizeof(lanroot)/sizeof(*lanroot));
|
|
|
|
//
|
|
// Gather the requested data by making local GetInfo calls.
|
|
//
|
|
|
|
switch ( level ) {
|
|
|
|
case 10:
|
|
status = NetWkstaGetInfo(
|
|
NULL,
|
|
(DWORD)100,
|
|
(LPBYTE *)&wksta_100
|
|
);
|
|
|
|
if ( !XsApiSuccess ( status )) {
|
|
|
|
IF_DEBUG(API_ERRORS) {
|
|
NetpKdPrint(( "XsWkstaGetInfo: WkstaGetInfo (level 100) "
|
|
"failed: %X\n", status));
|
|
}
|
|
|
|
Header->Status = (WORD) status;
|
|
goto cleanup;
|
|
}
|
|
|
|
break;
|
|
|
|
case 0:
|
|
case 1:
|
|
status = NetWkstaGetInfo(
|
|
NULL,
|
|
(DWORD)101,
|
|
(LPBYTE *)&wksta_101
|
|
);
|
|
|
|
if ( !XsApiSuccess( status )) {
|
|
|
|
IF_DEBUG(API_ERRORS) {
|
|
NetpKdPrint(( "XsWkstaGetInfo: WkstaGetInfo (level 101) "
|
|
"failed: %X\n", status));
|
|
}
|
|
|
|
Header->Status = (WORD) status;
|
|
goto cleanup;
|
|
}
|
|
|
|
status = NetWkstaGetInfo(
|
|
NULL,
|
|
(DWORD)502,
|
|
(LPBYTE *)&wksta_502
|
|
);
|
|
|
|
if ( !XsApiSuccess( status )) {
|
|
|
|
IF_DEBUG(API_ERRORS) {
|
|
NetpKdPrint(( "XsWkstaGetInfo: WkstaGetInfo (level 502) "
|
|
"failed: %X\n", status));
|
|
}
|
|
|
|
Header->Status = (WORD) status;
|
|
goto cleanup;
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
//
|
|
// Calculate the amount of space required to hold the fixed and
|
|
// variable data. Since this is the only place where we get one
|
|
// case for each valid level, we will also get the source structure
|
|
// descriptor here.
|
|
//
|
|
|
|
switch ( level ) {
|
|
|
|
case 0:
|
|
|
|
StructureDesc = Desc16_wksta_info_0;
|
|
bytesRequired = sizeof( WKSTA_16_INFO_0 )
|
|
+ NetpUnicodeToDBCSLen( lanroot )
|
|
+ NetpUnicodeToDBCSLen( wksta_101->wki101_computername )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_username )
|
|
+ NetpUnicodeToDBCSLen( wksta_101->wki101_langroup )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_logon_server )
|
|
+ SIZE_HEURISTICS
|
|
+ 6; // for terminating nulls
|
|
break;
|
|
|
|
case 1:
|
|
|
|
StructureDesc = Desc16_wksta_info_1;
|
|
bytesRequired = sizeof( WKSTA_16_INFO_1 )
|
|
+ NetpUnicodeToDBCSLen( lanroot )
|
|
+ NetpUnicodeToDBCSLen( wksta_101->wki101_computername )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_username )
|
|
+ NetpUnicodeToDBCSLen( wksta_101->wki101_langroup )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_logon_server )
|
|
+ SIZE_HEURISTICS
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_logon_domain )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_oth_domains )
|
|
+ 8; // for terminating nulls
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
StructureDesc = Desc16_wksta_info_10;
|
|
bytesRequired = sizeof( WKSTA_16_INFO_10 )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_username )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_logon_domain )
|
|
+ NetpUnicodeToDBCSLen( wksta_100->wki100_computername )
|
|
+ NetpUnicodeToDBCSLen( wksta_100->wki100_langroup )
|
|
+ NetpUnicodeToDBCSLen( DEF16_wk_oth_domains )
|
|
+ 5; // for terminating nulls
|
|
break;
|
|
}
|
|
|
|
//
|
|
// If there isn't enough room in the buffer for this, don't write any
|
|
// variable data.
|
|
//
|
|
|
|
varWrite = ( (DWORD)SmbGetUshort( ¶meters->BufLen )
|
|
>= bytesRequired ) ? TRUE : FALSE;
|
|
|
|
stringLocation = (LPBYTE)( XsSmbGetPointer( ¶meters->Buffer )
|
|
+ RapStructureSize( StructureDesc, Response, FALSE ));
|
|
|
|
//
|
|
// Return NERR_BufTooSmall if fixed structure will not fit.
|
|
//
|
|
|
|
if ( !XsCheckBufferSize(
|
|
SmbGetUshort( ¶meters->BufLen ),
|
|
StructureDesc,
|
|
FALSE // not in native format
|
|
)) {
|
|
|
|
IF_DEBUG(ERRORS) {
|
|
NetpKdPrint(( "XsNetWkstaGetInfo: Buffer too small.\n" ));
|
|
}
|
|
Header->Status = NERR_BufTooSmall;
|
|
SmbPutUshort( ¶meters->TotalAvail, (WORD)bytesRequired );
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
//
|
|
// Based on the level, fill the appropriate information directly into
|
|
// 16-bit buffer.
|
|
//
|
|
|
|
entry1 = (LPWKSTA_16_INFO_1) XsSmbGetPointer( ¶meters->Buffer );
|
|
entry10 = (LPWKSTA_16_INFO_10) entry1;
|
|
|
|
switch ( level ) {
|
|
|
|
case 1:
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_logon_domain,
|
|
&entry1->wki1_logon_domain,
|
|
entry1
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_oth_domains,
|
|
&entry1->wki1_oth_domains,
|
|
entry1
|
|
);
|
|
}
|
|
SmbPutUshort( &entry1->wki1_numdgrambuf,
|
|
DEF16_wk_numdgrambuf );
|
|
|
|
//
|
|
// Fill the rest of the level 1 structure just like a
|
|
// level 0 structure.
|
|
//
|
|
|
|
case 0:
|
|
|
|
//
|
|
// Zero the reserved words.
|
|
//
|
|
|
|
SmbPutUshort( &entry1->wki1_reserved_1, (WORD) 0 );
|
|
SmbPutUlong( &entry1->wki1_reserved_2, (DWORD) 0 );
|
|
SmbPutUlong( &entry1->wki1_reserved_3, (DWORD) 0 );
|
|
SmbPutUshort( &entry1->wki1_reserved_4, (WORD) 0 );
|
|
SmbPutUshort( &entry1->wki1_reserved_5, (WORD) 0 );
|
|
SmbPutUshort( &entry1->wki1_reserved_6, (WORD) 0 );
|
|
|
|
//
|
|
// Fill in the fields which have analogues in NT.
|
|
//
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
lanroot,
|
|
&entry1->wki1_root,
|
|
entry1
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
wksta_101->wki101_computername,
|
|
&entry1->wki1_computername,
|
|
entry1
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
wksta_101->wki101_langroup,
|
|
&entry1->wki1_langroup,
|
|
entry1
|
|
);
|
|
}
|
|
|
|
entry1->wki1_ver_major = (BYTE) wksta_101->wki101_ver_major;
|
|
entry1->wki1_ver_minor = (BYTE) wksta_101->wki101_ver_minor;
|
|
|
|
SmbPutUshort( &entry1->wki1_charwait,
|
|
XsDwordToWord( wksta_502->wki502_char_wait ) );
|
|
SmbPutUlong( &entry1->wki1_chartime,
|
|
(DWORD) wksta_502->wki502_collection_time );
|
|
SmbPutUshort( &entry1->wki1_charcount,
|
|
XsDwordToWord( wksta_502->
|
|
wki502_maximum_collection_count ) );
|
|
SmbPutUshort( &entry1->wki1_keepconn,
|
|
XsDwordToWord( wksta_502->wki502_keep_conn ) );
|
|
SmbPutUshort( &entry1->wki1_maxthreads,
|
|
XsDwordToWord( wksta_502->wki502_max_threads ) );
|
|
SmbPutUshort( &entry1->wki1_maxcmds,
|
|
XsDwordToWord( wksta_502->wki502_max_cmds ) );
|
|
SmbPutUshort( &entry1->wki1_sesstimeout,
|
|
XsDwordToWord( wksta_502->wki502_sess_timeout ) );
|
|
|
|
//
|
|
// Construct the heuristics string.
|
|
//
|
|
|
|
// Request opportunistic locking of files.
|
|
heuristics[0] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_use_opportunistic_locking ));
|
|
// Optimize performance for command files.
|
|
heuristics[1] = MAKE_TCHAR('1');
|
|
// Unlock and WriteUnlock asynchronously.
|
|
heuristics[2] = MAKE_TCHAR('1'); // default
|
|
// Close and WriteClose asynchronously.
|
|
heuristics[3] = MAKE_TCHAR('1'); // default
|
|
// Buffer named pipes and communication devices.
|
|
heuristics[4] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_buf_named_pipes ));
|
|
// LockRead and WriteUnlock.
|
|
heuristics[5] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_use_lock_read_unlock ));
|
|
// Use Open and Read.
|
|
heuristics[6] = MAKE_TCHAR('0');
|
|
// Read-ahead to sector boundary.
|
|
heuristics[7] = MAKE_TCHAR('1');
|
|
// Use the "chain send" NetBIOS NCB.
|
|
heuristics[8] = MAKE_TCHAR('2');
|
|
// Buffer small read/write requests.
|
|
heuristics[9] = MAKE_TCHAR('1');
|
|
// Use buffer mode.
|
|
heuristics[10] = MAKE_TCHAR('3');
|
|
// Use raw data transfer read/write server message block protocols.
|
|
heuristics[11] = MAKE_TCHAR('1');
|
|
// Use large RAW read-ahead buffer.
|
|
heuristics[12] = MAKE_TCHAR('1');
|
|
// Use large RAW write-behind buffer.
|
|
heuristics[13] = MAKE_TCHAR('1');
|
|
// Use read multiplex SMB protocols.
|
|
heuristics[14] = MAKE_TCHAR('0');
|
|
// Use write multiplex SMB protocols.
|
|
heuristics[15] = MAKE_TCHAR('0');
|
|
// Use big buffer for large core reads.
|
|
heuristics[16] = MAKE_TCHAR('1');
|
|
// Set the read-ahead size.
|
|
heuristics[17] = MAKE_TCHAR('0');
|
|
// Set the write-behind size.
|
|
heuristics[18] = MAKE_TCHAR('0');
|
|
// Force 512-byte maximum transfers to and from core servers.
|
|
heuristics[19] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_use_512_byte_max_transfer ));
|
|
// Flush pipes and devices on DosBufReset or DosClose.
|
|
heuristics[20] = MAKE_TCHAR('0');
|
|
// Use encryption if the server supports it.
|
|
heuristics[21] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_use_encryption ));
|
|
// Control log entries for multiple occurences of an error.
|
|
heuristics[22] = MAKE_TCHAR('1');
|
|
// Buffer all files opened with "deny write" rights.
|
|
heuristics[23] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_buf_files_deny_write ));
|
|
// Buffer all files opened with R attribute.
|
|
heuristics[24] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_buf_read_only_files ));
|
|
// Read ahead when opening a file for execution.
|
|
heuristics[25] = MAKE_TCHAR('0');
|
|
// Handle Ctrl-C.
|
|
heuristics[26] = MAKE_TCHAR('2');
|
|
// Force correct open mode when creating files on a core server.
|
|
heuristics[27] = MAKE_TCHAR(XsBoolToDigit(
|
|
wksta_502->wki502_force_core_create_mode ));
|
|
// Use NetBIOS NoAck mode.
|
|
heuristics[28] = MAKE_TCHAR('0');
|
|
// Send data along with SMB write-block-RAW requests.
|
|
heuristics[29] = MAKE_TCHAR('1');
|
|
// Send a popup when the workstation logs an error.
|
|
heuristics[30] = MAKE_TCHAR('1');
|
|
// Close the print job, causing the remote spooler to print if no
|
|
// activity occurs on the printer for the time specified.
|
|
heuristics[31] = MAKE_TCHAR('0');
|
|
// Controls BufReset and SMBFlush behavior for the MS-DOS
|
|
// compatibility box.
|
|
heuristics[32] = MAKE_TCHAR('2');
|
|
// Controls the time-out value for performing logon validation from a
|
|
// domain controller.
|
|
heuristics[33] = MAKE_TCHAR('0');
|
|
|
|
for ( i = 34; i <= 54; i++ ) {
|
|
heuristics[i] = MAKE_TCHAR('0');
|
|
}
|
|
|
|
heuristics[SIZE_HEURISTICS-1] = MAKE_TCHAR('\0');
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
heuristics,
|
|
&entry1->wki1_wrkheuristics,
|
|
entry1
|
|
);
|
|
}
|
|
|
|
//
|
|
// Put default values in the fields that are meaningless in NT.
|
|
//
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_logon_server,
|
|
&entry1->wki1_logon_server,
|
|
entry1
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_username,
|
|
&entry1->wki1_username,
|
|
entry1
|
|
);
|
|
}
|
|
|
|
SmbPutUshort( &entry1->wki1_keepsearch,
|
|
(WORD) DEF16_wk_keepsearch );
|
|
SmbPutUshort( &entry1->wki1_numworkbuf,
|
|
(WORD) DEF16_wk_numworkbuf );
|
|
SmbPutUshort( &entry1->wki1_sizworkbuf,
|
|
(WORD) DEF16_wk_sizeworkbuf );
|
|
SmbPutUshort( &entry1->wki1_maxwrkcache,
|
|
(WORD) DEF16_wk_maxwrkcache );
|
|
SmbPutUshort( &entry1->wki1_sizerror,
|
|
(WORD) DEF16_wk_sizerror );
|
|
SmbPutUshort( &entry1->wki1_numalerts,
|
|
(WORD) DEF16_wk_numalerts );
|
|
SmbPutUshort( &entry1->wki1_numservices,
|
|
(WORD) DEF16_wk_numservices );
|
|
SmbPutUshort( &entry1->wki1_errlogsz,
|
|
(WORD) DEF16_wk_errlogsz );
|
|
SmbPutUshort( &entry1->wki1_printbuftime,
|
|
(WORD) DEF16_wk_printbuftime );
|
|
SmbPutUshort( &entry1->wki1_numcharbuf,
|
|
(WORD) DEF16_wk_numcharbuf );
|
|
SmbPutUshort( &entry1->wki1_sizcharbuf,
|
|
(WORD) DEF16_wk_sizcharbuf );
|
|
SmbPutUshort( &entry1->wki1_mailslots,
|
|
(WORD) DEF16_wk_mailslots );
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
//
|
|
// Fill in the fields which have analogues in NT.
|
|
//
|
|
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
wksta_100->wki100_computername,
|
|
&entry10->wki10_computername,
|
|
entry10
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
wksta_100->wki100_langroup,
|
|
&entry10->wki10_langroup,
|
|
entry10
|
|
);
|
|
}
|
|
|
|
entry10->wki10_ver_major = XsDwordToByte( wksta_100->wki100_ver_major );
|
|
entry10->wki10_ver_minor = XsDwordToByte( wksta_100->wki100_ver_minor );
|
|
|
|
//
|
|
// Put default values in the fields that are meaningless in NT.
|
|
//
|
|
|
|
|
|
if ( varWrite ) {
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_username,
|
|
&entry10->wki10_username,
|
|
entry10
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_logon_domain,
|
|
&entry10->wki10_logon_domain,
|
|
entry10
|
|
);
|
|
XsAddVarString(
|
|
stringLocation,
|
|
DEF16_wk_oth_domains,
|
|
&entry10->wki10_oth_domains,
|
|
entry10
|
|
);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
SmbPutUshort( ¶meters->TotalAvail, (WORD)bytesRequired );
|
|
|
|
if ( varWrite == 0 ) {
|
|
Header->Status = ERROR_MORE_DATA;
|
|
} else {
|
|
Header->Status = NERR_Success;
|
|
Header->Converter = 0;
|
|
}
|
|
|
|
cleanup:
|
|
;
|
|
} except( EXCEPTION_EXECUTE_HANDLER ) {
|
|
Header->Status = (WORD)RtlNtStatusToDosError( GetExceptionCode() );
|
|
}
|
|
|
|
if ( wksta_100 != NULL ) {
|
|
NetApiBufferFree( wksta_100 );
|
|
}
|
|
|
|
if ( wksta_101 != NULL ) {
|
|
NetApiBufferFree( wksta_101 );
|
|
}
|
|
|
|
if ( wksta_502 != NULL ) {
|
|
NetApiBufferFree( wksta_502 );
|
|
}
|
|
|
|
//
|
|
// Determine return buffer size.
|
|
//
|
|
|
|
XsSetDataCount(
|
|
¶meters->BufLen,
|
|
StructureDesc,
|
|
Header->Converter,
|
|
1,
|
|
Header->Status
|
|
);
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
} // XsNetWkstaGetInfo
|
|
|
|
|
|
|
|
NTSTATUS
|
|
XsNetWkstaSetInfo (
|
|
API_HANDLER_PARAMETERS
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine handles a call to NetWkstaSetInfo.
|
|
|
|
Arguments:
|
|
|
|
API_HANDLER_PARAMETERS - information about the API call. See
|
|
XsTypes.h for details.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS - STATUS_SUCCESS or reason for failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
NET_API_STATUS status;
|
|
|
|
PXS_NET_WKSTA_SET_INFO parameters = Parameters;
|
|
DWORD data;
|
|
BOOL flag;
|
|
DWORD nativeParmNum;
|
|
|
|
LPVOID buffer = NULL; // Conversion variables
|
|
DWORD bufferSize;
|
|
LPWKSTA_16_INFO_1 entry1;
|
|
LPWKSTA_16_INFO_10 entry10;
|
|
BOOL error;
|
|
DWORD parmNum;
|
|
|
|
API_HANDLER_PARAMETERS_REFERENCE; // Avoid warnings
|
|
|
|
try {
|
|
//
|
|
// Check for errors. We will filter out wrong levels now.
|
|
//
|
|
|
|
parmNum = SmbGetUshort( ¶meters->ParmNum );
|
|
|
|
switch ( SmbGetUshort( ¶meters->Level )) {
|
|
|
|
case 0:
|
|
StructureDesc = Desc16_wksta_info_0;
|
|
if ( parmNum == WKSTA_OTH_DOMAINS_PARMNUM ) {
|
|
Header->Status = ERROR_INVALID_LEVEL;
|
|
goto cleanup;
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
StructureDesc = Desc16_wksta_info_1;
|
|
break;
|
|
|
|
case 10:
|
|
StructureDesc = Desc16_wksta_info_10;
|
|
if ( parmNum == WKSTA_CHARWAIT_PARMNUM
|
|
|| parmNum == WKSTA_CHARTIME_PARMNUM
|
|
|| parmNum == WKSTA_CHARCOUNT_PARMNUM
|
|
|| parmNum == WKSTA_ERRLOGSZ_PARMNUM
|
|
|| parmNum == WKSTA_PRINTBUFTIME_PARMNUM
|
|
|| parmNum == WKSTA_WRKHEURISTICS_PARMNUM ) {
|
|
|
|
Header->Status = ERROR_INVALID_LEVEL;
|
|
goto cleanup;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Header->Status = ERROR_INVALID_LEVEL;
|
|
goto cleanup;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Check input buffer size if parmnum is PARMNUM_ALL.
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL ) {
|
|
|
|
if ( !XsCheckBufferSize(
|
|
SmbGetUshort( ¶meters->BufLen ),
|
|
StructureDesc,
|
|
FALSE // not in native format
|
|
)) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
buffer = (LPVOID)XsSmbGetPointer( ¶meters->Buffer );
|
|
bufferSize = (DWORD)SmbGetUshort( ¶meters->BufLen );
|
|
entry1 = (LPWKSTA_16_INFO_1)buffer;
|
|
entry10 = (LPWKSTA_16_INFO_10)buffer;
|
|
|
|
//
|
|
// Processing of this API depends on the value of the ParmNum
|
|
// parameter. Because of all the discrepancies between NT and downlevel
|
|
// info structures, we will handle each instance by hand.
|
|
//
|
|
|
|
error = TRUE;
|
|
|
|
//
|
|
// charwait - source data is in a WORD - convert to DWORD
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_CHARWAIT_PARMNUM ) {
|
|
|
|
if ( bufferSize < sizeof(WORD) ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
data = ( parmNum == PARMNUM_ALL )
|
|
? (DWORD)SmbGetUshort( &entry1->wki1_charwait )
|
|
: (DWORD)SmbGetUshort( (LPWORD)buffer );
|
|
|
|
status = NetWkstaSetInfo(
|
|
NULL,
|
|
PARMNUM_BASE_INFOLEVEL + WKSTA_CHARWAIT_PARMNUM,
|
|
(LPBYTE)&data,
|
|
NULL
|
|
);
|
|
|
|
if ( status != NERR_Success ) {
|
|
IF_DEBUG(ERRORS) {
|
|
NetpKdPrint(( "XsNetWkstaSetInfo : SetInfo of charwait failed"
|
|
"%X\n", status ));
|
|
}
|
|
Header->Status = (WORD)status;
|
|
goto cleanup;
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// chartime - source data is in a DWORD - convert to DWORD
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_CHARTIME_PARMNUM ) {
|
|
|
|
if ( bufferSize < sizeof(DWORD) ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
data = ( parmNum == PARMNUM_ALL )
|
|
? SmbGetUlong( &entry1->wki1_chartime )
|
|
: SmbGetUlong( (LPDWORD)buffer );
|
|
|
|
status = NetWkstaSetInfo(
|
|
NULL,
|
|
PARMNUM_BASE_INFOLEVEL + WKSTA_CHARTIME_PARMNUM,
|
|
(LPBYTE)&data,
|
|
NULL
|
|
);
|
|
|
|
if ( status != NERR_Success ) {
|
|
IF_DEBUG(ERRORS) {
|
|
NetpKdPrint(( "XsNetWkstaSetInfo : SetInfo of chartime failed"
|
|
"%X\n", status ));
|
|
}
|
|
Header->Status = (WORD)status;
|
|
goto cleanup;
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// charcount - source data is in a WORD - convert to DWORD
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_CHARCOUNT_PARMNUM ) {
|
|
|
|
if ( bufferSize < sizeof(WORD) ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
data = ( parmNum == PARMNUM_ALL )
|
|
? (DWORD)SmbGetUshort( &entry1->wki1_charcount )
|
|
: (DWORD)SmbGetUshort( (LPWORD)buffer );
|
|
|
|
status = NetWkstaSetInfo(
|
|
NULL,
|
|
PARMNUM_BASE_INFOLEVEL + WKSTA_CHARCOUNT_PARMNUM,
|
|
(LPBYTE)&data,
|
|
NULL
|
|
);
|
|
|
|
if ( status != NERR_Success ) {
|
|
IF_DEBUG(ERRORS) {
|
|
NetpKdPrint(( "XsNetWkstaSetInfo : SetInfo of charcount failed"
|
|
"%X\n", status ));
|
|
}
|
|
Header->Status = (WORD)status;
|
|
goto cleanup;
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// errlogsz, printbuftime - source data is in a WORD.
|
|
//
|
|
// We can't set this, but downlevel can, so indicate success,
|
|
// as long as something was sent.
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_ERRLOGSZ_PARMNUM
|
|
|| parmNum == WKSTA_PRINTBUFTIME_PARMNUM ) {
|
|
|
|
if ( bufferSize < sizeof(WORD) ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// othdomains - source data is a string.
|
|
//
|
|
// We can't set this, but downlevel can, so indicate success,
|
|
// as long as something was sent.
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_OTH_DOMAINS_PARMNUM ) {
|
|
|
|
if ( bufferSize == 0 ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// wrkheuristics - source data is in a string.
|
|
//
|
|
// There are some elements of this that we can set. We go through a loop,
|
|
// setting these.
|
|
//
|
|
|
|
if ( parmNum == PARMNUM_ALL || parmNum == WKSTA_WRKHEURISTICS_PARMNUM ) {
|
|
|
|
LPBYTE heuristics;
|
|
DWORD i;
|
|
|
|
if ( bufferSize < 54 ) {
|
|
Header->Status = NERR_BufTooSmall;
|
|
goto cleanup;
|
|
}
|
|
|
|
heuristics = ( parmNum == PARMNUM_ALL )
|
|
? (LPBYTE)XsSmbGetPointer( &entry1->wki1_wrkheuristics )
|
|
: (LPBYTE)buffer;
|
|
|
|
//
|
|
// Nothing to be changed
|
|
//
|
|
|
|
if ( heuristics == NULL ) {
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// Make sure we have the right size of string.
|
|
//
|
|
|
|
if ( strlen( heuristics ) != 54 ) {
|
|
|
|
Header->Status = ERROR_INVALID_PARAMETER;
|
|
goto cleanup;
|
|
}
|
|
|
|
for ( i = 0; i < 54; i++ ) {
|
|
|
|
//
|
|
// Make sure heuristics string is valid.
|
|
//
|
|
|
|
if ( !isdigit( heuristics[i] )) {
|
|
|
|
Header->Status = ERROR_INVALID_PARAMETER;
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// Check if we can set this field.
|
|
//
|
|
|
|
switch ( i ) {
|
|
|
|
case 0:
|
|
nativeParmNum = WKSTA_USEOPPORTUNISTICLOCKING_PARMNUM;
|
|
break;
|
|
case 4:
|
|
nativeParmNum = WKSTA_BUFFERNAMEDPIPES_PARMNUM;
|
|
break;
|
|
case 5:
|
|
nativeParmNum = WKSTA_USELOCKANDREADANDUNLOCK_PARMNUM;
|
|
break;
|
|
case 19:
|
|
nativeParmNum = WKSTA_USE512BYTESMAXTRANSFER_PARMNUM;
|
|
break;
|
|
case 21:
|
|
nativeParmNum = WKSTA_USEENCRYPTION_PARMNUM;
|
|
break;
|
|
case 23:
|
|
nativeParmNum = WKSTA_BUFFILESWITHDENYWRITE_PARMNUM;
|
|
break;
|
|
case 24:
|
|
nativeParmNum = WKSTA_BUFFERREADONLYFILES_PARMNUM;
|
|
break;
|
|
case 27:
|
|
nativeParmNum = WKSTA_FORCECORECREATEMODE_PARMNUM;
|
|
break;
|
|
default:
|
|
nativeParmNum = 0;
|
|
break;
|
|
|
|
}
|
|
|
|
//
|
|
// If we can set the field, set it.
|
|
//
|
|
|
|
if ( nativeParmNum != 0 ) {
|
|
|
|
if ( heuristics[i] != '0' && heuristics[i] != '1' ) {
|
|
|
|
Header->Status = ERROR_INVALID_PARAMETER;
|
|
goto cleanup;
|
|
}
|
|
|
|
flag = XsDigitToBool( heuristics[i] );
|
|
|
|
status = NetWkstaSetInfo(
|
|
NULL,
|
|
PARMNUM_BASE_INFOLEVEL + nativeParmNum,
|
|
(LPBYTE)&flag,
|
|
NULL
|
|
);
|
|
|
|
if ( status != NERR_Success ) {
|
|
IF_DEBUG(ERRORS) {
|
|
NetpKdPrint(( "XsNetWkstaSetInfo : SetInfo of a "
|
|
"heuristic failed: %X\n", status ));
|
|
}
|
|
Header->Status = (WORD)status;
|
|
goto cleanup;
|
|
}
|
|
}
|
|
}
|
|
|
|
error = FALSE;
|
|
}
|
|
|
|
//
|
|
// Tried all possible parmnums. If error is still set, we have an
|
|
// invalid parmnum on our hands.
|
|
//
|
|
|
|
if ( error ) {
|
|
|
|
Header->Status = ERROR_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
//
|
|
// No return information for this API.
|
|
//
|
|
|
|
cleanup:
|
|
;
|
|
} except( EXCEPTION_EXECUTE_HANDLER ) {
|
|
Header->Status = (WORD)RtlNtStatusToDosError( GetExceptionCode() );
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
} // XsNetWkstaSetInfo
|
|
|
|
|
|
NTSTATUS
|
|
XsNetWkstaSetUID (
|
|
API_HANDLER_PARAMETERS
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This temporary routine just returns STATUS_NOT_IMPLEMENTED.
|
|
|
|
Arguments:
|
|
|
|
API_HANDLER_PARAMETERS - information about the API call. See
|
|
XsTypes.h for details.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS - STATUS_SUCCESS or reason for failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
API_HANDLER_PARAMETERS_REFERENCE; // Avoid warnings
|
|
|
|
Header->Status = (WORD)NERR_InvalidAPI;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
} // XsNetWkstaSetUID
|