1865 lines
46 KiB
C
1865 lines
46 KiB
C
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
NdsLib32.c
|
|
|
|
Abstract:
|
|
|
|
This module implements the exposed user-mode link to
|
|
Netware NDS support in the Netware redirector. For
|
|
more comments, see ndslib32.h.
|
|
|
|
Author:
|
|
|
|
Cory West [CoryWest] 23-Feb-1995
|
|
|
|
--*/
|
|
|
|
#include <procs.h>
|
|
//#include <nwapilyr.h>
|
|
|
|
NTSTATUS
|
|
NwNdsOpenGenericHandle(
|
|
IN PUNICODE_STRING puNdsTree,
|
|
OUT LPDWORD lpdwHandleType,
|
|
OUT PHANDLE phNwRdrHandle
|
|
) {
|
|
|
|
NTSTATUS ntstatus, OpenStatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
ACCESS_MASK DesiredAccess = SYNCHRONIZE | FILE_LIST_DIRECTORY;
|
|
|
|
WCHAR DevicePreamble[] = L"\\Device\\Nwrdr\\";
|
|
UINT PreambleLength = 14;
|
|
|
|
WCHAR NameStr[128];
|
|
UNICODE_STRING uOpenName;
|
|
UINT i;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
BYTE RrpData[1024];
|
|
|
|
//
|
|
// Prepare the open name.
|
|
//
|
|
|
|
uOpenName.MaximumLength = sizeof( NameStr );
|
|
|
|
if (puNdsTree->Length > (MAX_NDS_TREE_NAME_LEN * sizeof(WCHAR)))
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
for ( i = 0; i < PreambleLength ; i++ )
|
|
NameStr[i] = DevicePreamble[i];
|
|
|
|
try {
|
|
|
|
for ( i = 0 ; i < ( puNdsTree->Length / sizeof( WCHAR ) ) ; i++ ) {
|
|
NameStr[i + PreambleLength] = puNdsTree->Buffer[i];
|
|
}
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
uOpenName.Length = (USHORT)(( i * sizeof( WCHAR ) ) +
|
|
( PreambleLength * sizeof( WCHAR ) ));
|
|
uOpenName.Buffer = NameStr;
|
|
|
|
//
|
|
// Set up the object attributes.
|
|
//
|
|
|
|
InitializeObjectAttributes(
|
|
&ObjectAttributes,
|
|
&uOpenName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
NULL,
|
|
NULL );
|
|
|
|
//
|
|
// Make the compiler happy about variable initialization.
|
|
//
|
|
|
|
RtlZeroMemory( &IoStatusBlock, sizeof( IO_STATUS_BLOCK ) );
|
|
|
|
ntstatus = NtOpenFile(
|
|
phNwRdrHandle,
|
|
DesiredAccess,
|
|
&ObjectAttributes,
|
|
&IoStatusBlock,
|
|
FILE_SHARE_VALID_FLAGS,
|
|
FILE_SYNCHRONOUS_IO_NONALERT
|
|
);
|
|
|
|
if ( !NT_SUCCESS(ntstatus) )
|
|
return ntstatus;
|
|
|
|
OpenStatus = IoStatusBlock.Status;
|
|
|
|
//
|
|
// Verify that this is a tree handle, not a server handle.
|
|
//
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET)RrpData;
|
|
|
|
Rrp->Version = 0;
|
|
|
|
RtlCopyMemory( &(Rrp->Parameters).VerifyTree,
|
|
puNdsTree,
|
|
sizeof( UNICODE_STRING ) );
|
|
|
|
RtlCopyMemory( (BYTE *)(&(Rrp->Parameters).VerifyTree) + sizeof( UNICODE_STRING ),
|
|
puNdsTree->Buffer,
|
|
puNdsTree->Length );
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( *phNwRdrHandle,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_VERIFY_TREE,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ) + puNdsTree->Length,
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
ntstatus = GetExceptionCode();
|
|
goto CloseAndExit2;
|
|
}
|
|
|
|
if ( !NT_SUCCESS( ntstatus ))
|
|
{
|
|
*lpdwHandleType = HANDLE_TYPE_NCP_SERVER;
|
|
}
|
|
else
|
|
{
|
|
*lpdwHandleType = HANDLE_TYPE_NDS_TREE;
|
|
}
|
|
|
|
return OpenStatus;
|
|
|
|
CloseAndExit2:
|
|
|
|
NtClose( *phNwRdrHandle );
|
|
*phNwRdrHandle = NULL;
|
|
|
|
return ntstatus;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
NwNdsSetTreeContext (
|
|
IN HANDLE hNdsRdr,
|
|
IN PUNICODE_STRING puTree,
|
|
IN PUNICODE_STRING puContext
|
|
)
|
|
/*+++
|
|
|
|
This sets the current context in the requested tree.
|
|
|
|
---*/
|
|
{
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
DWORD RrpSize;
|
|
BYTE *CurrentString;
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
RrpSize = sizeof( NWR_NDS_REQUEST_PACKET ) +
|
|
puTree->Length +
|
|
puContext->Length;
|
|
|
|
Rrp = LocalAlloc( LMEM_ZEROINIT, RrpSize );
|
|
|
|
if ( !Rrp ) {
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
try {
|
|
|
|
(Rrp->Parameters).SetContext.TreeNameLen = puTree->Length;
|
|
(Rrp->Parameters).SetContext.ContextLen = puContext->Length;
|
|
|
|
CurrentString = (BYTE *)(Rrp->Parameters).SetContext.TreeAndContextString;
|
|
|
|
RtlCopyMemory( CurrentString, puTree->Buffer, puTree->Length );
|
|
CurrentString += puTree->Length;
|
|
RtlCopyMemory( CurrentString, puContext->Buffer, puContext->Length );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
ntstatus = STATUS_INVALID_PARAMETER;
|
|
goto ExitWithCleanup;
|
|
}
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsRdr,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_SETCONTEXT,
|
|
(PVOID) Rrp,
|
|
RrpSize,
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
ntstatus = GetExceptionCode();
|
|
goto ExitWithCleanup;
|
|
}
|
|
|
|
ExitWithCleanup:
|
|
|
|
LocalFree( Rrp );
|
|
return ntstatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsGetTreeContext (
|
|
IN HANDLE hNdsRdr,
|
|
IN PUNICODE_STRING puTree,
|
|
OUT PUNICODE_STRING puContext
|
|
)
|
|
/*+++
|
|
|
|
This gets the current context of the requested tree.
|
|
|
|
---*/
|
|
{
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
DWORD RrpSize;
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
RrpSize = sizeof( NWR_NDS_REQUEST_PACKET ) + puTree->Length;
|
|
|
|
Rrp = LocalAlloc( LMEM_ZEROINIT, RrpSize );
|
|
|
|
if ( !Rrp ) {
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
try {
|
|
|
|
(Rrp->Parameters).GetContext.TreeNameLen = puTree->Length;
|
|
|
|
RtlCopyMemory( (BYTE *)(Rrp->Parameters).GetContext.TreeNameString,
|
|
puTree->Buffer,
|
|
puTree->Length );
|
|
|
|
(Rrp->Parameters).GetContext.Context.MaximumLength = puContext->MaximumLength;
|
|
(Rrp->Parameters).GetContext.Context.Length = 0;
|
|
(Rrp->Parameters).GetContext.Context.Buffer = puContext->Buffer;
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
ntstatus = STATUS_INVALID_PARAMETER;
|
|
goto ExitWithCleanup;
|
|
}
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsRdr,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_GETCONTEXT,
|
|
(PVOID) Rrp,
|
|
RrpSize,
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
ntstatus = GetExceptionCode();
|
|
goto ExitWithCleanup;
|
|
}
|
|
|
|
//
|
|
// Copy out the length; the buffer has already been written.
|
|
//
|
|
|
|
puContext->Length = (Rrp->Parameters).GetContext.Context.Length;
|
|
|
|
ExitWithCleanup:
|
|
|
|
LocalFree( Rrp );
|
|
return ntstatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsIsNdsConnection (
|
|
IN HANDLE hNdsRdr,
|
|
OUT BOOL * pfIsNds,
|
|
OUT PUNICODE_STRING puTree
|
|
)
|
|
/*+++
|
|
|
|
This tests the current connection handle to see if it is one that is
|
|
connected to a server in an NDS tree. If so, the name of the tree is
|
|
put into puTree.
|
|
|
|
---*/
|
|
{
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
PCONN_DETAILS2 Rrp;
|
|
DWORD RrpSize;
|
|
|
|
*pfIsNds = FALSE;
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
RrpSize = sizeof( CONN_DETAILS2 );
|
|
|
|
Rrp = LocalAlloc( LMEM_ZEROINIT, RrpSize );
|
|
|
|
if ( !Rrp )
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
try
|
|
{
|
|
ntstatus = NtFsControlFile( hNdsRdr,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_GET_CONN_DETAILS2,
|
|
NULL,
|
|
0,
|
|
(PVOID) Rrp,
|
|
RrpSize );
|
|
|
|
}
|
|
except ( EXCEPTION_EXECUTE_HANDLER )
|
|
{
|
|
ntstatus = GetExceptionCode();
|
|
goto ExitWithCleanup;
|
|
}
|
|
|
|
if ( ntstatus == STATUS_SUCCESS )
|
|
{
|
|
if ( Rrp->fNds )
|
|
{
|
|
puTree->Length = (USHORT) wcslen( Rrp->NdsTreeName );
|
|
|
|
if ( puTree->MaximumLength >= puTree->Length )
|
|
wcscpy( puTree->Buffer, Rrp->NdsTreeName );
|
|
else
|
|
puTree->Length = 0;
|
|
|
|
*pfIsNds = TRUE;
|
|
}
|
|
}
|
|
|
|
ExitWithCleanup:
|
|
|
|
LocalFree( Rrp );
|
|
return ntstatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsList (
|
|
IN HANDLE hNdsTree,
|
|
IN DWORD dwObjectId,
|
|
OUT DWORD *dwIterHandle,
|
|
OUT BYTE *pbReplyBuf,
|
|
IN DWORD dwReplyBufLen
|
|
) {
|
|
|
|
NTSTATUS Status;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
|
|
PNDS_RESPONSE_SUBORDINATE_LIST Rsp;
|
|
DWORD dwRspBufferLen;
|
|
|
|
BYTE RrpData[256];
|
|
BYTE RspData[1024];
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
|
|
Rrp->Parameters.ListSubordinates.ObjectId = dwObjectId;
|
|
Rrp->Parameters.ListSubordinates.IterHandle = *dwIterHandle;
|
|
|
|
if ( dwReplyBufLen != 0 &&
|
|
pbReplyBuf != NULL ) {
|
|
|
|
Rsp = ( PNDS_RESPONSE_SUBORDINATE_LIST ) pbReplyBuf;
|
|
dwRspBufferLen = dwReplyBufLen;
|
|
|
|
} else {
|
|
|
|
Rsp = ( PNDS_RESPONSE_SUBORDINATE_LIST ) RspData;
|
|
dwRspBufferLen = 1024;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
Status = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_LIST_SUBS,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ),
|
|
(PVOID) Rsp,
|
|
dwRspBufferLen );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
if ( Status == STATUS_SUCCESS )
|
|
{
|
|
*dwIterHandle = Rsp->IterationHandle;
|
|
}
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsReadObjectInfo(
|
|
IN HANDLE hNdsTree,
|
|
IN DWORD dwObjectId,
|
|
OUT BYTE *pbRawReply,
|
|
IN DWORD dwReplyBufLen
|
|
)
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
|
|
PNDS_RESPONSE_GET_OBJECT_INFO Rsp;
|
|
DWORD dwRspBufferLen;
|
|
|
|
BYTE RrpData[256];
|
|
BYTE RspData[1024];
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
|
|
Rrp->Parameters.GetObjectInfo.ObjectId = dwObjectId;
|
|
|
|
if ( dwReplyBufLen != 0 &&
|
|
pbRawReply != NULL ) {
|
|
|
|
Rsp = ( PNDS_RESPONSE_GET_OBJECT_INFO ) pbRawReply;
|
|
dwRspBufferLen = dwReplyBufLen;
|
|
|
|
} else {
|
|
|
|
Rsp = ( PNDS_RESPONSE_GET_OBJECT_INFO ) RspData;
|
|
dwRspBufferLen = 1024;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
Status = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_READ_INFO,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ),
|
|
(PVOID) Rsp,
|
|
dwRspBufferLen );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
NwNdsReadAttribute (
|
|
IN HANDLE hNdsTree,
|
|
IN DWORD dwObjectId,
|
|
IN DWORD *dwIterHandle,
|
|
IN PUNICODE_STRING puAttrName,
|
|
OUT BYTE *pbReplyBuf,
|
|
IN DWORD dwReplyBufLen
|
|
) {
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
PNDS_RESPONSE_READ_ATTRIBUTE Rsp = ( PNDS_RESPONSE_READ_ATTRIBUTE )
|
|
pbReplyBuf;
|
|
|
|
DWORD dwAttributeNameLen;
|
|
|
|
BYTE RrpData[1024];
|
|
|
|
//
|
|
// Check the incoming buffer.
|
|
//
|
|
if ( !dwReplyBufLen || !Rsp )
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
RtlZeroMemory( Rrp, 1024 );
|
|
|
|
(Rrp->Parameters).ReadAttribute.ObjectId = dwObjectId;
|
|
(Rrp->Parameters).ReadAttribute.IterHandle = *dwIterHandle;
|
|
|
|
//
|
|
// Nds strings are NULL terminated; watch the size.
|
|
//
|
|
|
|
dwAttributeNameLen = puAttrName->Length + sizeof( WCHAR );
|
|
if (dwAttributeNameLen > (MAX_NDS_SCHEMA_NAME_CHARS * sizeof(WCHAR))) {
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
(Rrp->Parameters).ReadAttribute.AttributeNameLength = dwAttributeNameLen;
|
|
|
|
try {
|
|
|
|
//
|
|
// But don't try to copy more than the user gave us.
|
|
//
|
|
|
|
memcpy( (Rrp->Parameters).ReadAttribute.AttributeName,
|
|
puAttrName->Buffer,
|
|
puAttrName->Length );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Send the request to the Redirector FSD.
|
|
//
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_READ_ATTR,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ) + dwAttributeNameLen,
|
|
(PVOID) Rsp,
|
|
dwReplyBufLen );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
if ( ntstatus == STATUS_SUCCESS )
|
|
{
|
|
*dwIterHandle = Rsp->IterationHandle;
|
|
}
|
|
|
|
//
|
|
// There's no buffer post processing on this one.
|
|
//
|
|
|
|
return ntstatus;
|
|
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsOpenStream (
|
|
IN HANDLE hNdsTree,
|
|
IN DWORD dwObjectId,
|
|
IN PUNICODE_STRING puStreamName,
|
|
IN DWORD dwOpenFlags,
|
|
OUT DWORD *pdwFileLength
|
|
) {
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
BYTE RrpData[1024];
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
RtlZeroMemory( Rrp, 1024 );
|
|
|
|
(Rrp->Parameters).OpenStream.StreamAccess = dwOpenFlags;
|
|
(Rrp->Parameters).OpenStream.ObjectOid = dwObjectId;
|
|
|
|
(Rrp->Parameters).OpenStream.StreamName.Length = puStreamName->Length;
|
|
(Rrp->Parameters).OpenStream.StreamName.MaximumLength =
|
|
sizeof( RrpData ) - FIELD_OFFSET(NWR_NDS_REQUEST_PACKET,Parameters.OpenStream.StreamNameString);
|
|
(Rrp->Parameters).OpenStream.StreamName.Buffer =
|
|
(Rrp->Parameters).OpenStream.StreamNameString;
|
|
|
|
//
|
|
// Make sure we're not trashing memory.
|
|
//
|
|
|
|
if ( (Rrp->Parameters).OpenStream.StreamName.Length >
|
|
(Rrp->Parameters).OpenStream.StreamName.MaximumLength ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
try {
|
|
|
|
//
|
|
// But don't try to copy more than the user gave us.
|
|
//
|
|
|
|
memcpy( (Rrp->Parameters).OpenStream.StreamNameString,
|
|
puStreamName->Buffer,
|
|
puStreamName->Length );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Send the request to the Redirector FSD.
|
|
//
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_OPEN_STREAM,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ) + puStreamName->Length,
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
if ( pdwFileLength ) {
|
|
*pdwFileLength = (Rrp->Parameters).OpenStream.FileLength;
|
|
}
|
|
|
|
return ntstatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsGetQueueInformation(
|
|
IN HANDLE hNdsTree,
|
|
IN PUNICODE_STRING puQueueName,
|
|
OUT PUNICODE_STRING puHostServer,
|
|
OUT PDWORD pdwQueueId
|
|
) {
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
BYTE RrpData[1024];
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
RtlZeroMemory( Rrp, sizeof( RrpData ) );
|
|
|
|
if ( puQueueName ) {
|
|
(Rrp->Parameters).GetQueueInfo.QueueName.Length = puQueueName->Length;
|
|
(Rrp->Parameters).GetQueueInfo.QueueName.MaximumLength = puQueueName->MaximumLength;
|
|
(Rrp->Parameters).GetQueueInfo.QueueName.Buffer = puQueueName->Buffer;
|
|
}
|
|
|
|
if ( puHostServer ) {
|
|
(Rrp->Parameters).GetQueueInfo.HostServer.Length = 0;
|
|
(Rrp->Parameters).GetQueueInfo.HostServer.MaximumLength = puHostServer->MaximumLength;
|
|
(Rrp->Parameters).GetQueueInfo.HostServer.Buffer = puHostServer->Buffer;
|
|
}
|
|
|
|
//
|
|
// Send the request to the Redirector FSD.
|
|
//
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_GET_QUEUE_INFO,
|
|
(PVOID) Rrp,
|
|
sizeof( NWR_NDS_REQUEST_PACKET ),
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
if ( NT_SUCCESS( ntstatus ) ) {
|
|
|
|
if ( pdwQueueId ) {
|
|
*pdwQueueId = (Rrp->Parameters).GetQueueInfo.QueueId;
|
|
}
|
|
|
|
if (puHostServer) {
|
|
puHostServer->Length = (Rrp->Parameters).GetQueueInfo.HostServer.Length;
|
|
}
|
|
|
|
}
|
|
|
|
return ntstatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsGetVolumeInformation(
|
|
IN HANDLE hNdsTree,
|
|
IN PUNICODE_STRING puVolumeName,
|
|
OUT PUNICODE_STRING puHostServer,
|
|
OUT PUNICODE_STRING puHostVolume
|
|
) {
|
|
|
|
NTSTATUS ntstatus;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
PNWR_NDS_REQUEST_PACKET Rrp;
|
|
DWORD RequestSize;
|
|
BYTE RrpData[1024];
|
|
BYTE ReplyData[1024];
|
|
PBYTE NameStr;
|
|
|
|
//
|
|
// Set up the request.
|
|
//
|
|
|
|
Rrp = (PNWR_NDS_REQUEST_PACKET) RrpData;
|
|
RtlZeroMemory( Rrp, sizeof( RrpData ) );
|
|
|
|
if ( !puVolumeName ||
|
|
puVolumeName->Length > MAX_NDS_NAME_SIZE ||
|
|
!puHostServer ||
|
|
!puHostVolume ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
try {
|
|
|
|
(Rrp->Parameters).GetVolumeInfo.VolumeNameLen = puVolumeName->Length;
|
|
RtlCopyMemory( &((Rrp->Parameters).GetVolumeInfo.VolumeName[0]),
|
|
puVolumeName->Buffer,
|
|
puVolumeName->Length );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Send the request to the Redirector FSD.
|
|
//
|
|
|
|
RequestSize = sizeof( NWR_NDS_REQUEST_PACKET ) +
|
|
(Rrp->Parameters).GetVolumeInfo.VolumeNameLen;
|
|
|
|
try {
|
|
|
|
ntstatus = NtFsControlFile( hNdsTree,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_GET_VOLUME_INFO,
|
|
(PVOID) Rrp,
|
|
RequestSize,
|
|
ReplyData,
|
|
sizeof( ReplyData ) );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
if ( NT_SUCCESS( ntstatus ) ) {
|
|
|
|
try {
|
|
|
|
if ( ( puHostServer->MaximumLength < (Rrp->Parameters).GetVolumeInfo.ServerNameLen ) ||
|
|
( puHostVolume->MaximumLength < (Rrp->Parameters).GetVolumeInfo.TargetVolNameLen ) ) {
|
|
|
|
return STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
|
|
puHostServer->Length = (USHORT)(Rrp->Parameters).GetVolumeInfo.ServerNameLen;
|
|
puHostVolume->Length = (USHORT)(Rrp->Parameters).GetVolumeInfo.TargetVolNameLen;
|
|
|
|
NameStr = &ReplyData[0];
|
|
|
|
RtlCopyMemory( puHostServer->Buffer, NameStr, puHostServer->Length );
|
|
NameStr += puHostServer->Length;
|
|
RtlCopyMemory( puHostVolume->Buffer, NameStr, puHostVolume->Length );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
}
|
|
|
|
return ntstatus;
|
|
|
|
}
|
|
|
|
//
|
|
// User mode fragment exchange.
|
|
//
|
|
|
|
int
|
|
_cdecl
|
|
FormatBuf(
|
|
char *buf,
|
|
int bufLen,
|
|
const char *format,
|
|
va_list args
|
|
);
|
|
|
|
int
|
|
_cdecl
|
|
CalculateBuf(
|
|
const char *format,
|
|
va_list args
|
|
);
|
|
|
|
|
|
int
|
|
_cdecl
|
|
FormatBuf(
|
|
char *buf,
|
|
int bufLen,
|
|
const char *format,
|
|
va_list args
|
|
)
|
|
/*
|
|
|
|
Routine Description:
|
|
|
|
Formats a buffer according to supplied the format string.
|
|
|
|
FormatString - Supplies an ANSI string which describes how to
|
|
convert from the input arguments into NCP request fields, and
|
|
from the NCP response fields into the output arguments.
|
|
|
|
Field types, request/response:
|
|
|
|
'b' byte ( byte / byte* )
|
|
'w' hi-lo word ( word / word* )
|
|
'd' hi-lo dword ( dword / dword* )
|
|
'W' lo-hi word ( word / word*)
|
|
'D' lo-hi dword ( dword / dword*)
|
|
'-' zero/skip byte ( void )
|
|
'=' zero/skip word ( void )
|
|
._. zero/skip string ( word )
|
|
'p' pstring ( char* )
|
|
'c' cstring ( char* )
|
|
'C' cstring followed skip word ( char*, word )
|
|
'V' sized NDS value ( byte *, dword / byte **, dword *)
|
|
'S' p unicode string copy as NDS_STRING (UNICODE_STRING *)
|
|
's' cstring copy as NDS_STRING (char* / char *, word)
|
|
'r' raw bytes ( byte*, word )
|
|
'u' p unicode string ( UNICODE_STRING * )
|
|
'U' p uppercase string( UNICODE_STRING * )
|
|
|
|
Routine Arguments:
|
|
|
|
char *buf - destination buffer.
|
|
int buflen - length of the destination buffer.
|
|
char *format - format string.
|
|
args - args to the format string.
|
|
|
|
Implementation Notes:
|
|
|
|
This comes verbatim from kernel mode.
|
|
|
|
*/
|
|
{
|
|
ULONG ix;
|
|
|
|
NTSTATUS status;
|
|
const char *z = format;
|
|
|
|
//
|
|
// Convert the input arguments into request packet.
|
|
//
|
|
|
|
ix = 0;
|
|
|
|
while ( *z )
|
|
{
|
|
switch ( *z )
|
|
{
|
|
case '=':
|
|
if ((ix + 1) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = 0;
|
|
// intentional fallthrough - '='= 2 bytes, '-'= 1 byte
|
|
case '-':
|
|
if ((ix + 1) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = 0;
|
|
break;
|
|
|
|
case '_':
|
|
{
|
|
WORD l = va_arg ( args, WORD );
|
|
if (ix + (ULONG)l > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
while ( l-- )
|
|
buf[ix++] = 0;
|
|
break;
|
|
}
|
|
|
|
case 'b':
|
|
if ((ix + 1) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = va_arg ( args, BYTE );
|
|
break;
|
|
|
|
case 'w':
|
|
{
|
|
WORD w = va_arg ( args, WORD );
|
|
if ((ix + 2) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = (BYTE) (w >> 8);
|
|
buf[ix++] = (BYTE) (w >> 0);
|
|
break;
|
|
}
|
|
|
|
case 'd':
|
|
{
|
|
DWORD d = va_arg ( args, DWORD );
|
|
if ((ix + 4) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = (BYTE) (d >> 24);
|
|
buf[ix++] = (BYTE) (d >> 16);
|
|
buf[ix++] = (BYTE) (d >> 8);
|
|
buf[ix++] = (BYTE) (d >> 0);
|
|
break;
|
|
}
|
|
|
|
case 'W':
|
|
{
|
|
WORD w = va_arg(args, WORD);
|
|
if ((ix + 2) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
(* (WORD *)&buf[ix]) = w;
|
|
ix += 2;
|
|
break;
|
|
}
|
|
|
|
case 'D':
|
|
{
|
|
DWORD d = va_arg (args, DWORD);
|
|
if ((ix + 4) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
(* (DWORD *)&buf[ix]) = d;
|
|
ix += 4;
|
|
break;
|
|
}
|
|
|
|
case 'c':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
WORD l = (WORD)strlen( c );
|
|
if ((ix + (ULONG)l + 1) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
RtlCopyMemory( &buf[ix], c, l+1 );
|
|
ix += l + 1;
|
|
break;
|
|
}
|
|
|
|
case 'C':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
WORD l = va_arg ( args, WORD );
|
|
WORD len = strlen( c ) + 1;
|
|
if (ix + (ULONG)l > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
RtlCopyMemory( &buf[ix], c, len > l? l : len);
|
|
ix += l;
|
|
buf[ix-1] = 0;
|
|
break;
|
|
}
|
|
|
|
case 'p':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
BYTE l = (BYTE)strlen( c );
|
|
if ((ix + (ULONG)l + 1) > (ULONG)bufLen)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
buf[ix++] = l;
|
|
RtlCopyMemory( &buf[ix], c, l );
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'u':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg ( args, PUNICODE_STRING );
|
|
OEM_STRING OemString;
|
|
ULONG Length;
|
|
|
|
//
|
|
// Calculate required string length, excluding trailing NUL.
|
|
//
|
|
|
|
Length = RtlUnicodeStringToOemSize( pUString ) - 1;
|
|
ASSERT( Length < 0x100 );
|
|
|
|
//
|
|
// We need to check for more then just "Length" because
|
|
// the MaximumLength we pass in has "+1" on it so even
|
|
// though we don't care about the ending NULL it is
|
|
// going to get put in there so we have to account for it
|
|
//
|
|
if ( (ix + Length + 1) > (ULONG)bufLen ) {
|
|
goto ErrorExit;
|
|
}
|
|
|
|
buf[ix++] = (UCHAR)Length;
|
|
OemString.Buffer = &buf[ix];
|
|
OemString.MaximumLength = (USHORT)Length + 1;
|
|
|
|
status = RtlUnicodeStringToOemString( &OemString, pUString, FALSE );
|
|
ASSERT( NT_SUCCESS( status ));
|
|
ix += (USHORT)Length;
|
|
break;
|
|
}
|
|
|
|
case 'S':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg (args, PUNICODE_STRING);
|
|
ULONG Length, rLength;
|
|
|
|
Length = pUString->Length;
|
|
rLength = ROUNDUP4(Length + sizeof( WCHAR ));
|
|
|
|
if (ix + sizeof(rLength) + rLength > (ULONG)bufLen) {
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// The VLM client uses the rounded up length and it seems to
|
|
// make a difference! Also, don't forget that NDS strings have
|
|
// to be NULL terminated.
|
|
//
|
|
|
|
*((DWORD *)&buf[ix]) = rLength;
|
|
ix += 4;
|
|
RtlCopyMemory(&buf[ix], pUString->Buffer, Length);
|
|
ix += Length;
|
|
rLength -= Length;
|
|
RtlFillMemory( &buf[ix], rLength, '\0' );
|
|
ix += rLength;
|
|
break;
|
|
|
|
}
|
|
|
|
case 's':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg (args, PUNICODE_STRING);
|
|
ULONG Length, rLength;
|
|
|
|
Length = pUString->Length;
|
|
rLength = Length + sizeof( WCHAR );
|
|
|
|
if (ix + sizeof(rLength) + rLength > (ULONG)bufLen) {
|
|
// DebugTrace( 0, Dbg, "FormatBuf: case 's' request buffer too small.\n", 0 );
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Don't use the padded size here, only the NDS null terminator.
|
|
//
|
|
|
|
*((DWORD *)&buf[ix]) = rLength;
|
|
ix += 4;
|
|
RtlCopyMemory(&buf[ix], pUString->Buffer, Length);
|
|
ix += Length;
|
|
rLength -= Length;
|
|
RtlFillMemory( &buf[ix], rLength, '\0' );
|
|
ix += rLength;
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
case 'V':
|
|
{
|
|
// too similar to 'S' - should be combined
|
|
BYTE* b = va_arg ( args, BYTE* );
|
|
DWORD l = va_arg ( args, DWORD );
|
|
if ( ix + l + sizeof(DWORD) > (ULONG)
|
|
bufLen )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
*((DWORD *)&buf[ix]) = l;
|
|
ix += sizeof(DWORD);
|
|
RtlCopyMemory( &buf[ix], b, l );
|
|
l = ROUNDUP4(l);
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'r':
|
|
{
|
|
BYTE* b = va_arg ( args, BYTE* );
|
|
WORD l = va_arg ( args, WORD );
|
|
if ( b == NULL || l == 0 )
|
|
{
|
|
break;
|
|
}
|
|
if ( ix + l > (ULONG)bufLen )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
RtlCopyMemory( &buf[ix], b, l );
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
if ( ix > (ULONG)bufLen )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
|
|
z++;
|
|
}
|
|
|
|
return(ix);
|
|
|
|
ErrorExit:
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
_cdecl
|
|
CalculateBuf(
|
|
const char *format,
|
|
va_list args
|
|
)
|
|
/*
|
|
|
|
Routine Description:
|
|
|
|
This routine calculates the buffer size needed to hold a request.
|
|
FormatString - Supplies an ANSI string which describes how to
|
|
convert from the input arguments into NCP request fields, and
|
|
from the NCP response fields into the output arguments.
|
|
|
|
Field types, request/response:
|
|
|
|
'b' byte ( byte / byte* )
|
|
'w' hi-lo word ( word / word* )
|
|
'd' hi-lo dword ( dword / dword* )
|
|
'W' lo-hi word ( word / word*)
|
|
'D' lo-hi dword ( dword / dword*)
|
|
'-' zero/skip byte ( void )
|
|
'=' zero/skip word ( void )
|
|
._. zero/skip string ( word )
|
|
'p' pstring ( char* )
|
|
'c' cstring ( char* )
|
|
'C' cstring followed skip word ( char*, word )
|
|
'V' sized NDS value ( byte *, dword / byte **, dword *)
|
|
'S' p unicode string copy as NDS_STRING (UNICODE_STRING *)
|
|
's' cstring copy as NDS_STRING (char* / char *, word)
|
|
'r' raw bytes ( byte*, word )
|
|
'u' p unicode string ( UNICODE_STRING * )
|
|
'U' p uppercase string( UNICODE_STRING * )
|
|
|
|
Routine Arguments:
|
|
|
|
char *format - format string.
|
|
args - args to the format string.
|
|
|
|
Implementation Notes:
|
|
|
|
This comes verbatim from kernel mode.
|
|
|
|
*/
|
|
{
|
|
ULONG ix;
|
|
|
|
const char *z = format;
|
|
|
|
//
|
|
// Convert the input arguments into request packet.
|
|
//
|
|
|
|
ix = 0;
|
|
|
|
while ( *z )
|
|
{
|
|
switch ( *z )
|
|
{
|
|
case '=':
|
|
ix++;
|
|
case '-':
|
|
ix++;
|
|
break;
|
|
|
|
case '_':
|
|
{
|
|
WORD l = va_arg ( args, WORD );
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'b':
|
|
{
|
|
char b = va_arg ( args, BYTE );
|
|
ix++;
|
|
break;
|
|
}
|
|
|
|
case 'w':
|
|
{
|
|
WORD w = va_arg ( args, WORD );
|
|
ix += 2;
|
|
break;
|
|
}
|
|
|
|
case 'd':
|
|
{
|
|
DWORD d = va_arg ( args, DWORD );
|
|
ix += 4;
|
|
break;
|
|
}
|
|
|
|
case 'W':
|
|
{
|
|
WORD w = va_arg(args, WORD);
|
|
ix += 2;
|
|
break;
|
|
}
|
|
|
|
case 'D':
|
|
{
|
|
DWORD d = va_arg (args, DWORD);
|
|
ix += 4;
|
|
break;
|
|
}
|
|
|
|
case 'c':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
WORD l = (WORD)strlen( c );
|
|
ix += l + 1;
|
|
break;
|
|
}
|
|
|
|
case 'C':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
WORD l = va_arg ( args, WORD );
|
|
WORD len = strlen( c ) + 1;
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'p':
|
|
{
|
|
char* c = va_arg ( args, char* );
|
|
BYTE l = (BYTE)strlen( c );
|
|
ix++;
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'u':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg ( args, PUNICODE_STRING );
|
|
OEM_STRING OemString;
|
|
ULONG Length;
|
|
|
|
//
|
|
// Calculate required string length, excluding trailing NUL.
|
|
//
|
|
|
|
Length = RtlUnicodeStringToOemSize( pUString ) - 1;
|
|
ASSERT( Length < 0x100 );
|
|
|
|
ix++;
|
|
ix += (USHORT)Length;
|
|
break;
|
|
}
|
|
|
|
case 'S':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg (args, PUNICODE_STRING);
|
|
ULONG Length, rLength;
|
|
|
|
Length = pUString->Length;
|
|
|
|
//
|
|
// The VLM client uses the rounded up length and it seems to
|
|
// make a difference! Also, don't forget that NDS strings have
|
|
// to be NULL terminated.
|
|
//
|
|
|
|
rLength = ROUNDUP4(Length + sizeof( WCHAR ));
|
|
ix += 4;
|
|
ix += Length;
|
|
rLength -= Length;
|
|
ix += rLength;
|
|
break;
|
|
|
|
}
|
|
|
|
case 's':
|
|
{
|
|
PUNICODE_STRING pUString = va_arg (args, PUNICODE_STRING);
|
|
ULONG Length, rLength;
|
|
|
|
Length = pUString->Length;
|
|
|
|
//
|
|
// Don't use the padded size here, only the NDS null terminator.
|
|
//
|
|
|
|
rLength = Length + sizeof( WCHAR );
|
|
ix += 4;
|
|
ix += Length;
|
|
rLength -= Length;
|
|
ix += rLength;
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
case 'V':
|
|
{
|
|
// too similar to 'S' - should be combined
|
|
BYTE* b = va_arg ( args, BYTE* );
|
|
DWORD l = va_arg ( args, DWORD );
|
|
ix += sizeof(DWORD);
|
|
l = ROUNDUP4(l);
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
case 'r':
|
|
{
|
|
BYTE* b = va_arg ( args, BYTE* );
|
|
WORD l = va_arg ( args, WORD );
|
|
if ( b == NULL || l == 0 )
|
|
{
|
|
break;
|
|
}
|
|
ix += l;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
z++;
|
|
}
|
|
|
|
return(ix);
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
_cdecl
|
|
ParseResponse(
|
|
PUCHAR Response,
|
|
ULONG ResponseLength,
|
|
char* FormatString,
|
|
... // format specific parameters
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine parse an NCP response.
|
|
|
|
Packet types:
|
|
|
|
'G' Generic packet ( )
|
|
|
|
Field types, request/response:
|
|
|
|
'b' byte ( byte* )
|
|
'w' hi-lo word ( word* )
|
|
'x' ordered word ( word* )
|
|
'd' hi-lo dword ( dword* )
|
|
'e' ordered dword ( dword* )
|
|
'-' zero/skip byte ( void )
|
|
'=' zero/skip word ( void )
|
|
._. zero/skip string ( word )
|
|
'p' pstring ( char* )
|
|
'c' cstring ( char* )
|
|
'r' raw bytes ( byte*, word )
|
|
|
|
Added 3/29/95 by CoryWest:
|
|
|
|
'W' lo-hi word ( word / word*)
|
|
'D' lo-hi dword ( dword / dword*)
|
|
'S' unicode string copy as NDS_STRING (UNICODE_STRING *)
|
|
'T' terminal unicode string copy as NDS_STRING (UNICODE_STRING *)
|
|
|
|
't' terminal unicode string with the nds null copied
|
|
as NDS_STRING (UNICODE_STRING *) (for GetUseName)
|
|
|
|
Return Value:
|
|
|
|
STATUS - Success or failure, depending on the response.
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
PCHAR FormatByte;
|
|
va_list Arguments;
|
|
ULONG Length = 0;
|
|
|
|
va_start( Arguments, FormatString );
|
|
|
|
//
|
|
// User mode parse response handles only generic packets.
|
|
//
|
|
|
|
if ( *FormatString != 'G' ) {
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
FormatByte = FormatString + 1;
|
|
while ( *FormatByte ) {
|
|
|
|
switch ( *FormatByte ) {
|
|
|
|
case '-':
|
|
Length += 1;
|
|
break;
|
|
|
|
case '=':
|
|
Length += 2;
|
|
break;
|
|
|
|
case '_':
|
|
{
|
|
WORD l = va_arg ( Arguments, WORD );
|
|
Length += l;
|
|
break;
|
|
}
|
|
|
|
case 'b':
|
|
{
|
|
BYTE* b = va_arg ( Arguments, BYTE* );
|
|
*b = Response[Length++];
|
|
break;
|
|
}
|
|
|
|
case 'w':
|
|
{
|
|
BYTE* b = va_arg ( Arguments, BYTE* );
|
|
b[1] = Response[Length++];
|
|
b[0] = Response[Length++];
|
|
break;
|
|
}
|
|
|
|
case 'x':
|
|
{
|
|
WORD* w = va_arg ( Arguments, WORD* );
|
|
*w = *(WORD UNALIGNED *)&Response[Length];
|
|
Length += 2;
|
|
break;
|
|
}
|
|
|
|
case 'd':
|
|
{
|
|
BYTE* b = va_arg ( Arguments, BYTE* );
|
|
b[3] = Response[Length++];
|
|
b[2] = Response[Length++];
|
|
b[1] = Response[Length++];
|
|
b[0] = Response[Length++];
|
|
break;
|
|
}
|
|
|
|
case 'e':
|
|
{
|
|
DWORD UNALIGNED * d = va_arg ( Arguments, DWORD* );
|
|
*d = *(DWORD UNALIGNED *)&Response[Length];
|
|
Length += 4;
|
|
break;
|
|
}
|
|
|
|
case 'c':
|
|
{
|
|
char* c = va_arg ( Arguments, char* );
|
|
WORD l = (WORD)strlen( &Response[Length] );
|
|
memcpy ( c, &Response[Length], l+1 );
|
|
Length += l+1;
|
|
break;
|
|
}
|
|
|
|
case 'p':
|
|
{
|
|
char* c = va_arg ( Arguments, char* );
|
|
BYTE l = Response[Length++];
|
|
memcpy ( c, &Response[Length], l );
|
|
c[l+1] = 0;
|
|
break;
|
|
}
|
|
|
|
case 'r':
|
|
{
|
|
BYTE* b = va_arg ( Arguments, BYTE* );
|
|
WORD l = va_arg ( Arguments, WORD );
|
|
RtlCopyMemory( b, &Response[Length], l );
|
|
Length += l;
|
|
break;
|
|
}
|
|
|
|
case 'W':
|
|
{
|
|
|
|
WORD *w = va_arg ( Arguments, WORD* );
|
|
*w = (* (WORD *)&Response[Length]);
|
|
Length += 2;
|
|
break;
|
|
|
|
}
|
|
|
|
case 'D':
|
|
{
|
|
|
|
DWORD *d = va_arg ( Arguments, DWORD* );
|
|
*d = (* (DWORD *)&Response[Length]);
|
|
Length += 4;
|
|
break;
|
|
|
|
}
|
|
|
|
case 'S':
|
|
{
|
|
|
|
PUNICODE_STRING pU = va_arg( Arguments, PUNICODE_STRING );
|
|
USHORT strl;
|
|
|
|
if (pU) {
|
|
|
|
strl = (USHORT)(* (DWORD *)&Response[Length]);
|
|
|
|
//
|
|
// Don't count the null terminator that is part of
|
|
// Novell's counted unicode string.
|
|
//
|
|
|
|
pU->Length = strl - sizeof( WCHAR );
|
|
Length += 4;
|
|
if (pU->Length >= pU->MaximumLength) {
|
|
pU->Length = pU->MaximumLength;
|
|
}
|
|
RtlCopyMemory( pU->Buffer, &Response[Length], pU->Length );
|
|
Length += ROUNDUP4(strl);
|
|
|
|
} else {
|
|
|
|
//
|
|
// Skip over the string since we don't want it.
|
|
//
|
|
|
|
Length += ROUNDUP4((* (DWORD *)&Response[Length] ));
|
|
Length += 4;
|
|
}
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 's':
|
|
{
|
|
|
|
PUNICODE_STRING pU = va_arg( Arguments, PUNICODE_STRING );
|
|
USHORT strl;
|
|
|
|
if (pU) {
|
|
|
|
strl = (USHORT)(* (DWORD *)&Response[Length]);
|
|
pU->Length = strl;
|
|
Length += 4;
|
|
if (pU->Length >= pU->MaximumLength) {
|
|
pU->Length = pU->MaximumLength;
|
|
}
|
|
RtlCopyMemory( pU->Buffer, &Response[Length], pU->Length );
|
|
Length += ROUNDUP4(strl);
|
|
|
|
} else {
|
|
|
|
//
|
|
// Skip over the string since we don't want it.
|
|
//
|
|
|
|
Length += ROUNDUP4((* (DWORD *)&Response[Length] ));
|
|
Length += 4;
|
|
}
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'T':
|
|
{
|
|
|
|
PUNICODE_STRING pU = va_arg( Arguments, PUNICODE_STRING );
|
|
USHORT strl;
|
|
|
|
if (pU) {
|
|
|
|
strl = (USHORT)(* (DWORD *)&Response[Length] );
|
|
strl -= sizeof( WCHAR ); // Don't count the NULL from NDS.
|
|
|
|
if ( strl <= pU->MaximumLength ) {
|
|
|
|
pU->Length = strl;
|
|
Length += 4;
|
|
RtlCopyMemory( pU->Buffer, &Response[Length], pU->Length );
|
|
|
|
//
|
|
// No need to advance the pointers since this is
|
|
// specifically a termination case!
|
|
//
|
|
|
|
} else {
|
|
|
|
pU->Length = 0;
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 't':
|
|
{
|
|
|
|
PUNICODE_STRING pU = va_arg( Arguments, PUNICODE_STRING );
|
|
USHORT strl;
|
|
|
|
if (pU) {
|
|
|
|
strl = (USHORT)(* (DWORD *)&Response[Length] );
|
|
|
|
if ( strl <= pU->MaximumLength ) {
|
|
|
|
pU->Length = strl;
|
|
Length += 4;
|
|
RtlCopyMemory( pU->Buffer, &Response[Length], pU->Length );
|
|
|
|
//
|
|
// No need to advance the pointers since this is
|
|
// specifically a termination case!
|
|
//
|
|
|
|
} else {
|
|
|
|
pU->Length = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( Length > ResponseLength ) {
|
|
return( STATUS_INVALID_PARAMETER );
|
|
}
|
|
|
|
FormatByte++;
|
|
|
|
}
|
|
|
|
va_end( Arguments );
|
|
return( Status );
|
|
|
|
}
|
|
|
|
NTSTATUS
|
|
NwNdsChangePassword(
|
|
IN HANDLE hNwRdr,
|
|
IN PUNICODE_STRING puTreeName,
|
|
IN PUNICODE_STRING puUserName,
|
|
IN PUNICODE_STRING puCurrentPassword,
|
|
IN PUNICODE_STRING puNewPassword
|
|
) {
|
|
|
|
NTSTATUS Status;
|
|
PNWR_NDS_REQUEST_PACKET pNdsRequest;
|
|
DWORD dwRequestLength;
|
|
PBYTE CurrentString;
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
//
|
|
// Allocate the request.
|
|
//
|
|
|
|
dwRequestLength = sizeof( NWR_NDS_REQUEST_PACKET ) +
|
|
puTreeName->Length +
|
|
puUserName->Length +
|
|
puCurrentPassword->Length +
|
|
puNewPassword->Length;
|
|
|
|
pNdsRequest = LocalAlloc( LMEM_ZEROINIT, dwRequestLength );
|
|
|
|
if ( !pNdsRequest) {
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
//
|
|
// Copy the parameters into the request buffer.
|
|
//
|
|
|
|
try {
|
|
|
|
(pNdsRequest->Parameters).ChangePass.NdsTreeNameLength =
|
|
puTreeName->Length;
|
|
(pNdsRequest->Parameters).ChangePass.UserNameLength =
|
|
puUserName->Length;
|
|
(pNdsRequest->Parameters).ChangePass.CurrentPasswordLength =
|
|
puCurrentPassword->Length;
|
|
(pNdsRequest->Parameters).ChangePass.NewPasswordLength =
|
|
puNewPassword->Length;
|
|
|
|
CurrentString = ( PBYTE ) &((pNdsRequest->Parameters).ChangePass.StringBuffer[0]);
|
|
RtlCopyMemory( CurrentString, puTreeName->Buffer, puTreeName->Length );
|
|
|
|
CurrentString += puTreeName->Length;
|
|
RtlCopyMemory( CurrentString, puUserName->Buffer, puUserName->Length );
|
|
|
|
CurrentString += puUserName->Length;
|
|
RtlCopyMemory( CurrentString, puCurrentPassword->Buffer, puCurrentPassword->Length );
|
|
|
|
CurrentString += puCurrentPassword->Length;
|
|
RtlCopyMemory( CurrentString, puNewPassword->Buffer, puNewPassword->Length );
|
|
|
|
Status = NtFsControlFile( hNwRdr,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&IoStatusBlock,
|
|
FSCTL_NWR_NDS_CHANGE_PASS,
|
|
(PVOID) pNdsRequest,
|
|
dwRequestLength,
|
|
NULL,
|
|
0 );
|
|
|
|
} except ( EXCEPTION_EXECUTE_HANDLER ) {
|
|
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
LocalFree( pNdsRequest );
|
|
return Status;
|
|
|
|
}
|
|
|