3239 lines
93 KiB
C
3239 lines
93 KiB
C
/*++
|
||
|
||
Copyright (c) 1997-1999 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
info.c
|
||
|
||
Abstract:
|
||
|
||
Support the RPC interface that provides internal info to the caller.
|
||
|
||
Author:
|
||
|
||
Billy J Fuller 27-Mar-1998
|
||
|
||
Environment
|
||
|
||
User mode, winnt32
|
||
|
||
--*/
|
||
|
||
#include <ntreppch.h>
|
||
#pragma hdrstop
|
||
|
||
|
||
#include <ntdsapi.h>
|
||
#include <frs.h>
|
||
#include <ntdsapip.h> // ms internal flags for DsCrackNames()
|
||
#include <ntfrsapi.h>
|
||
#include <info.h>
|
||
#include <tablefcn.h>
|
||
#include <lmaccess.h>
|
||
#include <lmapibuf.h>
|
||
|
||
#ifdef SECURITY_WIN32
|
||
#include <security.h>
|
||
#else
|
||
#define SECURITY_WIN32
|
||
#include <security.h>
|
||
#undef SECURITY_WIN32
|
||
#endif
|
||
|
||
|
||
|
||
extern PCHAR LatestChanges[];
|
||
extern PCHAR CoLocationNames[];
|
||
|
||
//
|
||
// Useful macros
|
||
//
|
||
#define IPRINTGNAME(_I_, _G_, _F_, _GUID_, _P_) \
|
||
{ \
|
||
if (_G_) { \
|
||
GuidToStr(_G_->Guid, _GUID_); \
|
||
IPRINT3(_I_, _F_, _P_, _G_->Name, _GUID_); \
|
||
} \
|
||
}
|
||
|
||
//
|
||
// Try to avoid splitting a record struct across calls.
|
||
//
|
||
#define INFO_HAS_SPACE(_info_) (((_info_)->SizeInChars - (_info_)->OffsetToFree) >= 2000)
|
||
|
||
|
||
extern OSVERSIONINFOEX OsInfo;
|
||
extern SYSTEM_INFO SystemInfo;
|
||
extern PCHAR ProcessorArchName[12];
|
||
|
||
extern FLAG_NAME_TABLE CxtionOptionsFlagNameTable[];
|
||
|
||
//
|
||
// DC name for LDAP binding
|
||
//
|
||
WCHAR InfoDcName[MAX_PATH + 1];
|
||
|
||
//
|
||
// Member Subscriber Links
|
||
//
|
||
typedef struct _INFO_DN INFO_DN, *PINFO_DN;
|
||
struct _INFO_DN {
|
||
PINFO_DN Next;
|
||
PWCHAR Dn;
|
||
PWCHAR SetType;
|
||
};
|
||
|
||
|
||
//
|
||
// This table is used to keep contexts across multiple calls from ntfrsutl.exe
|
||
//
|
||
PGEN_TABLE FrsInfoContextTable = NULL;
|
||
//
|
||
// This counter is used to create context handles for each caller.
|
||
//
|
||
ULONG FrsInfoContextNum = 0;
|
||
|
||
//
|
||
// To avoid a DOS attack, limit number of active contexts.
|
||
//
|
||
#define FRS_INFO_MAX_CONTEXT_ACTIVE (1000)
|
||
|
||
|
||
|
||
VOID
|
||
DbsDisplayRecordIPrint(
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PINFO_TABLE InfoTable,
|
||
IN BOOL Read,
|
||
IN PULONG RecordFieldx,
|
||
IN ULONG FieldCount
|
||
);
|
||
|
||
//
|
||
// From frs\ds.c
|
||
//
|
||
PVOID *
|
||
FrsDsFindValues(
|
||
IN PLDAP ldap,
|
||
IN PLDAPMessage Entry,
|
||
IN PWCHAR DesiredAttr,
|
||
IN BOOL DoBerVals
|
||
);
|
||
|
||
PWCHAR
|
||
FrsDsExtendDn(
|
||
IN PWCHAR Dn,
|
||
IN PWCHAR Cn
|
||
);
|
||
|
||
PWCHAR
|
||
FrsDsExtendDnOu(
|
||
IN PWCHAR Dn,
|
||
IN PWCHAR Ou
|
||
);
|
||
|
||
PWCHAR
|
||
FrsDsFindValue(
|
||
IN PLDAP ldap,
|
||
IN PLDAPMessage Entry,
|
||
IN PWCHAR DesiredAttr
|
||
);
|
||
|
||
GUID *
|
||
FrsDsFindGuid(
|
||
IN PLDAP Ldap,
|
||
IN PLDAPMessage LdapEntry
|
||
);
|
||
|
||
PWCHAR
|
||
FrsDsMakeRdn(
|
||
IN PWCHAR DN
|
||
);
|
||
|
||
PWCHAR
|
||
FrsDsConvertToSettingsDn(
|
||
IN PWCHAR Dn
|
||
);
|
||
|
||
PSCHEDULE
|
||
FrsDsFindSchedule(
|
||
IN PLDAP Ldap,
|
||
IN PLDAPMessage LdapEntry,
|
||
OUT PULONG Len
|
||
);
|
||
|
||
VOID
|
||
FrsPrintRpcStats(
|
||
IN ULONG Severity,
|
||
IN PNTFRSAPI_INFO Info, OPTIONAL
|
||
IN DWORD Tabs
|
||
);
|
||
|
||
VOID
|
||
FrsPrintThreadStats(
|
||
IN ULONG Severity,
|
||
IN PNTFRSAPI_INFO Info, OPTIONAL
|
||
IN DWORD Tabs
|
||
);
|
||
|
||
|
||
VOID
|
||
InfoPrint(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN PCHAR Format,
|
||
IN ... )
|
||
/*++
|
||
Routine Description:
|
||
Format and print a line of information output into the info buffer.
|
||
|
||
Arguments:
|
||
Info - Info buffer
|
||
Format - printf format
|
||
|
||
Return Value:
|
||
None.
|
||
--*/
|
||
{
|
||
PCHAR Line;
|
||
ULONG LineLen;
|
||
LONG LineSize;
|
||
|
||
//
|
||
// varargs stuff
|
||
//
|
||
va_list argptr;
|
||
va_start(argptr, Format);
|
||
|
||
//
|
||
// Print the line into the info buffer
|
||
//
|
||
try {
|
||
if (!FlagOn(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
//
|
||
// Calc offset to start of free buffer space.
|
||
// And calc max space left in buffer.
|
||
//
|
||
Line = ((PCHAR)Info) + Info->OffsetToFree;
|
||
LineSize = (Info->SizeInChars - (ULONG)(Line - (PCHAR)Info)) - 1;
|
||
|
||
if ((LineSize <= 0) || (_vsnprintf(Line, LineSize, Format, argptr) < 0)) {
|
||
//
|
||
// Buffer is filled. Set the terminating NULL and the buffer full flag.
|
||
//
|
||
Line[LineSize - 1] = '\0';
|
||
SetFlag(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL);
|
||
} else {
|
||
|
||
LineLen = strlen(Line) + 1;
|
||
if (Info->CharsToSkip > 0) {
|
||
//
|
||
// Still skipping chars that we previously returned. Barf.
|
||
//
|
||
Info->CharsToSkip = (LineLen > Info->CharsToSkip) ?
|
||
0 : Info->CharsToSkip - LineLen;
|
||
} else {
|
||
//
|
||
// The line fits. Bump freespace offset and TotalChars returned.
|
||
//
|
||
Info->OffsetToFree += LineLen;
|
||
Info->TotalChars += LineLen;
|
||
}
|
||
}
|
||
}
|
||
} except(EXCEPTION_EXECUTE_HANDLER) {
|
||
}
|
||
va_end(argptr);
|
||
}
|
||
|
||
|
||
|
||
#define Tab L" "
|
||
VOID
|
||
InfoTabs(
|
||
IN DWORD Tabs,
|
||
IN PWCHAR TabW
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Create a string of tabs for prettyprint
|
||
|
||
Arguments:
|
||
Tabs - number of tabs
|
||
TabW - preallocated string to receive tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
DWORD i;
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
Tabs = (Tabs >= MAX_TABS) ? MAX_TABS : Tabs;
|
||
for (TabW[0] = L'\0', i = 0; i < Tabs; ++i) {
|
||
wcscat(TabW, Tab);
|
||
}
|
||
}
|
||
|
||
|
||
|
||
DWORD
|
||
InfoPrintDbSets(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on replica sets (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintDbSets:"
|
||
PVOID Key;
|
||
PREPLICA Replica;
|
||
CHAR Guid[GUID_CHAR_LEN + 1];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
extern PGEN_TABLE ReplicasByGuid;
|
||
extern PGEN_TABLE DeletedReplicas;
|
||
|
||
InfoTabs(Tabs, TabW);
|
||
IPRINT1(Info, "%wsACTIVE REPLICA SETS\n", TabW);
|
||
Key = NULL;
|
||
while (Replica = GTabNextDatum(ReplicasByGuid, &Key)) {
|
||
if (REPLICA_IS_ACTIVE(Replica)) {
|
||
FrsPrintTypeReplica(0, Info, Tabs, Replica, NULL, 0);
|
||
} else {
|
||
//
|
||
// If the replica set is not active, one or more of the GNames
|
||
// could have freed guid pointers (feeefeee, bug 319600) so
|
||
// don't print the replica set in this case, just the name and state.
|
||
//
|
||
if (Replica->SetName) {
|
||
IPRINT3(Info, "%ws %ws in state %s\n",
|
||
TabW, Replica->SetName->Name, RSS_NAME(Replica->ServiceState));
|
||
}
|
||
}
|
||
}
|
||
|
||
IPRINT0(Info, "\nDELETED REPLICA SETS\n");
|
||
Key = NULL;
|
||
if (DeletedReplicas) {
|
||
while (Replica = GTabNextDatum(DeletedReplicas, &Key)) {
|
||
if (Replica->SetName) {
|
||
IPRINT2(Info, "%ws %ws\n", TabW, Replica->SetName->Name);
|
||
}
|
||
}
|
||
}
|
||
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
|
||
|
||
BOOL
|
||
InfoSearch(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR Base,
|
||
IN ULONG Scope,
|
||
IN PWCHAR Filter,
|
||
IN PWCHAR Attrs[],
|
||
IN ULONG AttrsOnly,
|
||
IN LDAPMessage **Res
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Perform ldap_search_s
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
Ldap - bound ldap handle
|
||
.
|
||
.
|
||
.
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoSearch:"
|
||
DWORD LStatus;
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
InfoTabs(Tabs, TabW);
|
||
LStatus = ldap_search_s(Ldap, Base, Scope, Filter, Attrs, AttrsOnly, Res);
|
||
|
||
if (LStatus != LDAP_SUCCESS) {
|
||
IPRINT5(Info, "%wsWARN - ldap_search_s(%ws, %d, %ws); %ws\n",
|
||
TabW, Base, Scope, ComputerName, ldap_err2string(LStatus));
|
||
return FALSE;
|
||
}
|
||
return TRUE;
|
||
}
|
||
|
||
|
||
PCONFIG_NODE
|
||
InfoAllocBasicNode(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN PWCHAR TabW,
|
||
IN PWCHAR NodeType,
|
||
IN PWCHAR ParentDn,
|
||
IN PWCHAR Filter,
|
||
IN BOOL *FirstError,
|
||
IN PLDAP Ldap,
|
||
IN PLDAPMessage LdapEntry
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Allocate a node and fill in the basic info (dn and name)
|
||
|
||
Arguments:
|
||
Info - text buffer
|
||
TabW - Prettyprint
|
||
NodeType - Prettyprint
|
||
Ldap - openned, bound ldap
|
||
LdapEntry - returned from ldap_first/next_entry()
|
||
|
||
Return Value:
|
||
NULL if basic info is not available.
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoAllocBasicNode:"
|
||
PCONFIG_NODE Node = NULL;
|
||
CHAR Guid[GUID_CHAR_LEN + 1];
|
||
|
||
|
||
//
|
||
// Fetch values from the DS
|
||
//
|
||
Node = FrsAllocType(CONFIG_NODE_TYPE);
|
||
Node->Dn = FrsDsFindValue(Ldap, LdapEntry, ATTR_DN);
|
||
FRS_WCSLWR(Node->Dn);
|
||
|
||
//
|
||
// Name
|
||
//
|
||
Node->Name = FrsBuildGName(FrsDsFindGuid(Ldap, LdapEntry),
|
||
FrsDsMakeRdn(Node->Dn));
|
||
if (!Node->Dn || !Node->Name->Name || !Node->Name->Guid) {
|
||
IPRINT5(Info, "\n%ws%ws: ERROR - The object returned by the DS"
|
||
" lacks a dn (%08x), Rdn (%08x), or Guid(%08x)\n",
|
||
TabW, NodeType, Node->Dn, Node->Name->Name, Node->Name->Guid);
|
||
if (*FirstError) {
|
||
*FirstError = FALSE;
|
||
IPRINT5(Info, "%ws%ws: ERROR - Using ldp, bind to %ws and search the "
|
||
"container %ws using the filter "
|
||
"%ws for more information.\n",
|
||
TabW, NodeType, &InfoDcName[2], ParentDn, Filter);
|
||
}
|
||
return FrsFreeType(Node);
|
||
}
|
||
IPRINT3(Info, "\n%ws%ws: %ws\n", TabW, NodeType, Node->Name->Name);
|
||
|
||
IPRINT2(Info, "%ws DN : %ws\n", TabW, Node->Dn);
|
||
GuidToStr(Node->Name->Guid, Guid);
|
||
IPRINT2(Info, "%ws Guid : %s\n", TabW, Guid);
|
||
|
||
return Node;
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintDsCxtions(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR Base,
|
||
IN BOOL IsSysVol
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Print the cxtions from the DS.
|
||
|
||
Arguments:
|
||
ldap - opened and bound ldap connection
|
||
Base - Name of object or container in DS
|
||
|
||
Return Value:
|
||
None.
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintDsCxtions:"
|
||
PWCHAR Attrs[10];
|
||
PLDAPMessage LdapEntry;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR CxtionOptionsWStr = NULL;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
CHAR FlagBuffer[120];
|
||
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Search the DS for cxtions
|
||
//
|
||
Attrs[0] = ATTR_DN;
|
||
Attrs[1] = ATTR_SCHEDULE;
|
||
Attrs[2] = ATTR_FROM_SERVER;
|
||
Attrs[3] = ATTR_OBJECT_GUID;
|
||
Attrs[4] = ATTR_USN_CHANGED;
|
||
Attrs[5] = ATTR_ENABLED_CXTION;
|
||
Attrs[6] = ATTR_OPTIONS;
|
||
Attrs[7] = ATTR_WHEN_CHANGED;
|
||
Attrs[8] = ATTR_WHEN_CREATED;
|
||
Attrs[9] = NULL;
|
||
if (!InfoSearch(Info, Tabs, Ldap, Base, LDAP_SCOPE_ONELEVEL,
|
||
CATEGORY_CXTION, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"CXTION", Base,
|
||
CATEGORY_CXTION, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// Node's partner's name
|
||
//
|
||
Node->PartnerDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_FROM_SERVER);
|
||
FRS_WCSLWR(Node->PartnerDn);
|
||
IPRINT2(Info, "%ws Partner Dn : %ws\n", TabW, Node->PartnerDn);
|
||
Node->PartnerName = FrsBuildGName(NULL, FrsDsMakeRdn(Node->PartnerDn));
|
||
IPRINT2(Info, "%ws Partner Rdn : %ws\n", TabW, Node->PartnerName->Name);
|
||
|
||
//
|
||
// Enabled
|
||
//
|
||
Node->EnabledCxtion = FrsDsFindValue(Ldap, LdapEntry, ATTR_ENABLED_CXTION);
|
||
IPRINT2(Info, "%ws Enabled : %ws\n", TabW, Node->EnabledCxtion);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Options
|
||
//
|
||
CxtionOptionsWStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_OPTIONS);
|
||
if (CxtionOptionsWStr != NULL) {
|
||
Node->CxtionOptions = _wtoi(CxtionOptionsWStr);
|
||
CxtionOptionsWStr = FrsFree(CxtionOptionsWStr);
|
||
} else {
|
||
Node->CxtionOptions = 0;
|
||
}
|
||
|
||
FrsFlagsToStr(Node->CxtionOptions, CxtionOptionsFlagNameTable, sizeof(FlagBuffer), FlagBuffer);
|
||
IPRINT3(Info, "%ws Options : 0x%08x [%s]\n",
|
||
TabW, Node->CxtionOptions, FlagBuffer);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoCrack(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PWCHAR Dn,
|
||
IN HANDLE Handle,
|
||
IN PWCHAR DomainDnsName,
|
||
IN DWORD DesiredFormat
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Find the NT4 account name for Dn. Dn should be the Dn
|
||
of a computer object.
|
||
|
||
Arguments:
|
||
Dn - Of computer object
|
||
Handle - From DsBind
|
||
DomainDnsName - If !NULL, produce new local handle
|
||
DesiredFormat - DS_NT4_ACCOUNT_NAME or DS_STRING_SID_NAME
|
||
|
||
Return Value:
|
||
NT4 Account Name or NULL
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoCrack:"
|
||
DWORD WStatus;
|
||
DS_NAME_RESULT *Cracked;
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
HANDLE LocalHandle = NULL;
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Computer's Dn not available
|
||
//
|
||
if (!Dn) {
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Need something to go on!
|
||
//
|
||
if (!HANDLE_IS_VALID(Handle) && !DomainDnsName) {
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Bind to Ds
|
||
//
|
||
if (DomainDnsName) {
|
||
WStatus = DsBind(NULL, DomainDnsName, &LocalHandle);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
IPRINT4(Info, "%ws ERROR - DsBind(%ws, %08x); WStatus %s\n",
|
||
TabW, DomainDnsName, DesiredFormat, ErrLabelW32(WStatus));
|
||
return;
|
||
}
|
||
Handle = LocalHandle;
|
||
}
|
||
|
||
//
|
||
// Crack the computer's distinguished name into its NT4 Account Name
|
||
//
|
||
WStatus = DsCrackNames(Handle, // in hDS,
|
||
DS_NAME_NO_FLAGS, // in flags,
|
||
DS_FQDN_1779_NAME, // in formatOffered,
|
||
DesiredFormat, // in formatDesired,
|
||
1, // in cNames,
|
||
&Dn, // in *rpNames,
|
||
&Cracked); // out *ppResult
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
IPRINT4(Info, "%ws ERROR - DsCrackNames(%ws, %08x); WStatus %s\n",
|
||
TabW, Dn, DesiredFormat, ErrLabelW32(WStatus));
|
||
//
|
||
// What else can we do?
|
||
//
|
||
if (HANDLE_IS_VALID(LocalHandle)) {
|
||
DsUnBind(&LocalHandle);
|
||
}
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Might have it
|
||
//
|
||
if (Cracked && Cracked->cItems && Cracked->rItems) {
|
||
//
|
||
// Got it!
|
||
//
|
||
if (Cracked->rItems->status == DS_NAME_NO_ERROR) {
|
||
IPRINT2(Info, "%ws Cracked Domain : %ws\n",
|
||
TabW, Cracked->rItems->pDomain);
|
||
IPRINT3(Info, "%ws Cracked Name : %08x %ws\n",
|
||
TabW, DesiredFormat, Cracked->rItems->pName);
|
||
//
|
||
// Only got the domain; rebind and try again
|
||
//
|
||
} else if (Cracked->rItems->status == DS_NAME_ERROR_DOMAIN_ONLY) {
|
||
InfoCrack(Info, Tabs, Dn, NULL, Cracked->rItems->pDomain, DesiredFormat);
|
||
} else {
|
||
IPRINT4(Info, "%ws ERROR - DsCrackNames(%ws, %08x); internal status %d\n",
|
||
TabW, Dn, DesiredFormat, Cracked->rItems->status);
|
||
}
|
||
DsFreeNameResult(Cracked);
|
||
} else {
|
||
IPRINT3(Info, "%ws ERROR - DsCrackNames(%ws, %08x); no status\n",
|
||
TabW, Dn, DesiredFormat);
|
||
}
|
||
if (HANDLE_IS_VALID(LocalHandle)) {
|
||
DsUnBind(&LocalHandle);
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoCrackDns(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR Base
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Find the DNS name for Base. Base should be the Dn
|
||
of a computer object.
|
||
|
||
Arguments:
|
||
Info
|
||
Tabs
|
||
Ldap
|
||
Base
|
||
|
||
Return Value:
|
||
Prints a message into Info.
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoCrackDns:"
|
||
PWCHAR Attrs[2];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
PLDAPMessage LdapEntry;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PWCHAR DnsName = NULL;
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Computer's Dn not available
|
||
//
|
||
if (!Base) {
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Search the DS for the DNS attribute of Base
|
||
//
|
||
Attrs[0] = ATTR_DNS_HOST_NAME;
|
||
Attrs[1] = NULL;
|
||
if (!InfoSearch(Info, Tabs, Ldap, Base, LDAP_SCOPE_BASE,
|
||
CATEGORY_ANY, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
if (!LdapEntry) {
|
||
IPRINT2(Info, "%ws ERROR - No LdapEntry for Dns name on %ws\n", TabW, Base);
|
||
goto cleanup;
|
||
}
|
||
|
||
DnsName = FrsDsFindValue(Ldap, LdapEntry, ATTR_DNS_HOST_NAME);
|
||
if (!DnsName) {
|
||
IPRINT2(Info, "%ws ERROR - No DNS name on %ws\n", TabW, Base);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Got it!
|
||
//
|
||
IPRINT2(Info, "%ws Computer's DNS : %ws\n", TabW, DnsName);
|
||
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFree(DnsName);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintMembers(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN BOOL IsSysVol,
|
||
IN PWCHAR Base,
|
||
IN HANDLE DsHandle
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Print the members
|
||
|
||
Arguments:
|
||
ldap - opened and bound ldap connection
|
||
Base - Name of object or container in DS
|
||
|
||
Return Value:
|
||
None.
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintMembers:"
|
||
PWCHAR Attrs[9];
|
||
PLDAPMessage LdapEntry;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Search the DS for members
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_USN_CHANGED;
|
||
Attrs[4] = ATTR_SERVER_REF;
|
||
Attrs[5] = ATTR_COMPUTER_REF;
|
||
Attrs[6] = ATTR_WHEN_CHANGED;
|
||
Attrs[7] = ATTR_WHEN_CREATED;
|
||
Attrs[8] = NULL;
|
||
if (!InfoSearch(Info, Tabs, Ldap, Base, LDAP_SCOPE_ONELEVEL,
|
||
CATEGORY_MEMBER, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"MEMBER", Base,
|
||
CATEGORY_MEMBER, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// NTDS Settings (DSA) Reference
|
||
//
|
||
Node->SettingsDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_SERVER_REF);
|
||
IPRINT2(Info, "%ws Server Ref : %ws\n", TabW, Node->SettingsDn);
|
||
|
||
//
|
||
// Computer Reference
|
||
//
|
||
Node->ComputerDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_COMPUTER_REF);
|
||
FRS_WCSLWR(Node->ComputerDn);
|
||
IPRINT2(Info, "%ws Computer Ref : %ws\n", TabW, Node->ComputerDn);
|
||
|
||
InfoCrack(Info, Tabs, Node->ComputerDn, DsHandle, NULL, DS_NT4_ACCOUNT_NAME);
|
||
InfoCrack(Info, Tabs, Node->ComputerDn, DsHandle, NULL, DS_STRING_SID_NAME);
|
||
InfoCrackDns(Info, Tabs, Ldap, Node->ComputerDn);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
//
|
||
// Get the inbound cxtions
|
||
//
|
||
InfoPrintDsCxtions(Info, Tabs + 1, Ldap, Node->Dn, FALSE);
|
||
if (IsSysVol) {
|
||
if (Node->SettingsDn) {
|
||
InfoPrintDsCxtions(Info, Tabs + 1, Ldap, Node->SettingsDn, TRUE);
|
||
} else {
|
||
IPRINT2(Info, "%ws WARN - %ws lacks a settings reference\n",
|
||
TabW, Node->Name->Name);
|
||
}
|
||
}
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintDsSets(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR SetDnAddr,
|
||
IN HANDLE DsHandle,
|
||
IN OUT PINFO_DN *InfoSets
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Print replica sets from the ds
|
||
|
||
Arguments:
|
||
ldap - opened and bound ldap connection
|
||
Base - Name of object or container in DS
|
||
|
||
Return Value:
|
||
None
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintDsSets:"
|
||
PWCHAR Attrs[12];
|
||
DWORD i;
|
||
PINFO_DN InfoSet;
|
||
PLDAPMessage LdapEntry;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Have we processed this settings before?
|
||
//
|
||
for (InfoSet = *InfoSets; InfoSet; InfoSet = InfoSet->Next) {
|
||
if (WSTR_EQ(InfoSet->Dn, SetDnAddr)) {
|
||
IPRINT2(Info, "%ws %ws processed previously\n", TabW, SetDnAddr);
|
||
break;
|
||
}
|
||
}
|
||
//
|
||
// Yep; get the sets
|
||
//
|
||
if (InfoSet) {
|
||
//
|
||
// Recurse to the next level in the DS hierarchy
|
||
//
|
||
InfoPrintMembers(Info,
|
||
Tabs + 1,
|
||
Ldap,
|
||
FRS_RSTYPE_IS_SYSVOLW(InfoSet->SetType),
|
||
InfoSet->Dn,
|
||
DsHandle);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Search the DS beginning at Base for sets
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_USN_CHANGED;
|
||
Attrs[4] = ATTR_SET_TYPE;
|
||
Attrs[5] = ATTR_PRIMARY_MEMBER;
|
||
Attrs[6] = ATTR_FILE_FILTER;
|
||
Attrs[7] = ATTR_DIRECTORY_FILTER;
|
||
Attrs[8] = ATTR_WHEN_CHANGED;
|
||
Attrs[9] = ATTR_WHEN_CREATED;
|
||
Attrs[10] = ATTR_FRS_FLAGS;
|
||
Attrs[11] = NULL;
|
||
|
||
if (!InfoSearch(Info, Tabs, Ldap, SetDnAddr, LDAP_SCOPE_BASE,
|
||
CATEGORY_REPLICA_SET, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"SET", SetDnAddr,
|
||
CATEGORY_REPLICA_SET, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// Replica set type
|
||
//
|
||
Node->SetType = FrsDsFindValue(Ldap, LdapEntry, ATTR_SET_TYPE);
|
||
IPRINT2(Info, "%ws Type : %ws\n", TabW, Node->SetType);
|
||
|
||
//
|
||
// Primary member
|
||
//
|
||
Node->MemberDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_PRIMARY_MEMBER);
|
||
IPRINT2(Info, "%ws Primary Member: %ws\n", TabW, Node->MemberDn);
|
||
|
||
//
|
||
// File filter
|
||
//
|
||
Node->FileFilterList = FrsDsFindValue(Ldap, LdapEntry, ATTR_FILE_FILTER);
|
||
IPRINT2(Info, "%ws File Filter : %ws\n", TabW, Node->FileFilterList);
|
||
|
||
//
|
||
// Directory filter
|
||
//
|
||
Node->DirFilterList = FrsDsFindValue(Ldap, LdapEntry, ATTR_DIRECTORY_FILTER);
|
||
IPRINT2(Info, "%ws Dir Filter : %ws\n", TabW, Node->DirFilterList);
|
||
|
||
//
|
||
// FRS Flags value.
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_FRS_FLAGS);
|
||
IPRINT2(Info, "%ws FRS Flags : %ws\n", TabW, WStr);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
|
||
InfoSet = FrsAlloc(sizeof(INFO_DN));
|
||
InfoSet->Dn = FrsWcsDup(Node->Dn);
|
||
InfoSet->SetType = FrsWcsDup(Node->SetType);
|
||
InfoSet->Next = *InfoSets;
|
||
*InfoSets = InfoSet;
|
||
|
||
//
|
||
// Recurse to the next level in the DS hierarchy
|
||
//
|
||
InfoPrintMembers(Info,
|
||
Tabs + 1,
|
||
Ldap,
|
||
FRS_RSTYPE_IS_SYSVOLW(Node->SetType),
|
||
Node->Dn,
|
||
DsHandle);
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintSettings(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR MemberDn,
|
||
IN HANDLE DsHandle,
|
||
IN OUT PINFO_DN *InfoSettings,
|
||
IN OUT PINFO_DN *InfoSets
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Scan the DS tree for NTFRS-Settings
|
||
|
||
Arguments:
|
||
ldap - opened and bound ldap connection
|
||
Base - Name of object or container in DS
|
||
|
||
Return Value:
|
||
WIN32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintSettings:"
|
||
PWCHAR Attrs[7];
|
||
PLDAPMessage LdapEntry;
|
||
PWCHAR MemberDnAddr;
|
||
PWCHAR SetDnAddr;
|
||
PWCHAR SettingsDnAddr;
|
||
PINFO_DN InfoSetting;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Find the member component
|
||
//
|
||
MemberDnAddr = wcsstr(MemberDn, L"cn=");
|
||
if (!MemberDnAddr) {
|
||
IPRINT2(Info, "%ws ERROR - No MemberDnAddr in %ws\n", TabW, MemberDn);
|
||
goto cleanup;
|
||
}
|
||
//
|
||
// Find the set component
|
||
//
|
||
SetDnAddr = wcsstr(MemberDnAddr + 3, L"cn=");
|
||
if (!SetDnAddr) {
|
||
IPRINT2(Info, "%ws ERROR - No SetDnAddr in %ws\n", TabW, MemberDn);
|
||
goto cleanup;
|
||
}
|
||
//
|
||
// Find the settings component
|
||
//
|
||
SettingsDnAddr = wcsstr(SetDnAddr + 3, L"cn=");
|
||
if (!SettingsDnAddr) {
|
||
IPRINT2(Info, "%ws ERROR - No SettingsDnAddr in %ws\n", TabW, MemberDn);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Have we processed this settings before?
|
||
//
|
||
for (InfoSetting = *InfoSettings; InfoSetting; InfoSetting = InfoSetting->Next) {
|
||
if (WSTR_EQ(InfoSetting->Dn, SettingsDnAddr)) {
|
||
IPRINT2(Info, "%ws %ws processed previously\n", TabW, SettingsDnAddr);
|
||
break;
|
||
}
|
||
}
|
||
//
|
||
// Yep; get the sets
|
||
//
|
||
if (InfoSetting) {
|
||
InfoPrintDsSets(Info, Tabs + 1, Ldap, SetDnAddr, DsHandle, InfoSets);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Search the DS beginning at Base for settings
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_USN_CHANGED;
|
||
Attrs[4] = ATTR_WHEN_CHANGED;
|
||
Attrs[5] = ATTR_WHEN_CREATED;
|
||
Attrs[6] = NULL;
|
||
if (!InfoSearch(Info, Tabs, Ldap, SettingsDnAddr, LDAP_SCOPE_BASE,
|
||
CATEGORY_NTFRS_SETTINGS, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"SETTINGS", SettingsDnAddr,
|
||
CATEGORY_NTFRS_SETTINGS, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
|
||
InfoSetting = FrsAlloc(sizeof(INFO_DN));
|
||
InfoSetting->Dn = FrsWcsDup(Node->Dn);
|
||
InfoSetting->Next = *InfoSettings;
|
||
*InfoSettings = InfoSetting;
|
||
|
||
//
|
||
// Recurse to the next level in the DS hierarchy
|
||
//
|
||
InfoPrintDsSets(Info, Tabs + 1, Ldap, SetDnAddr, DsHandle, InfoSets);
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintSubscribers(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR SubscriptionDn,
|
||
IN PINFO_DN *InfoSubs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Print subscribers
|
||
|
||
Arguments:
|
||
Ldap - opened and bound ldap connection
|
||
SubscriptionDn - distininguished name of subscriptions object
|
||
|
||
Return Value:
|
||
WIN32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintSubscribers:"
|
||
PWCHAR Attrs[10];
|
||
PLDAPMessage LdapEntry;
|
||
PINFO_DN InfoSub;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Search the DS beginning at Base for the entries of class "Filter"
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_USN_CHANGED;
|
||
Attrs[4] = ATTR_REPLICA_ROOT;
|
||
Attrs[5] = ATTR_REPLICA_STAGE;
|
||
Attrs[6] = ATTR_MEMBER_REF;
|
||
Attrs[7] = ATTR_WHEN_CHANGED;
|
||
Attrs[8] = ATTR_WHEN_CREATED;
|
||
Attrs[9] = NULL;
|
||
if (!InfoSearch(Info, Tabs, Ldap, SubscriptionDn, LDAP_SCOPE_ONELEVEL,
|
||
CATEGORY_SUBSCRIBER, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"SUBSCRIBER", SubscriptionDn,
|
||
CATEGORY_SUBSCRIBER, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// Member reference
|
||
//
|
||
Node->MemberDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_MEMBER_REF);
|
||
IPRINT2(Info, "%ws Member Ref: %ws\n", TabW, Node->MemberDn);
|
||
|
||
if (Node->MemberDn) {
|
||
InfoSub = FrsAlloc(sizeof(INFO_DN ));
|
||
InfoSub->Dn = FrsWcsDup(Node->MemberDn);
|
||
InfoSub->Next = *InfoSubs;
|
||
*InfoSubs = InfoSub;
|
||
}
|
||
|
||
//
|
||
// Root pathname
|
||
//
|
||
Node->Root = FrsDsFindValue(Ldap, LdapEntry, ATTR_REPLICA_ROOT);
|
||
FRS_WCSLWR(Node->Root);
|
||
IPRINT2(Info, "%ws Root : %ws\n", TabW, Node->Root);
|
||
|
||
//
|
||
// Staging pathname
|
||
//
|
||
Node->Stage = FrsDsFindValue(Ldap, LdapEntry, ATTR_REPLICA_STAGE);
|
||
FRS_WCSLWR(Node->Stage);
|
||
IPRINT2(Info, "%ws Stage : %ws\n", TabW, Node->Stage);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
VOID
|
||
InfoPrintSubscriptions(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR ComputerDn,
|
||
IN PINFO_DN *InfoSubs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Recursively scan the DS tree beginning at computer
|
||
|
||
Arguments:
|
||
Info
|
||
Tabs
|
||
Ldap
|
||
ComputerDn
|
||
|
||
Return Value:
|
||
WIN32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintSubscriptions:"
|
||
PWCHAR Attrs[8];
|
||
PLDAPMessage LdapEntry;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PCONFIG_NODE Node = NULL;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Search the DS beginning at Base for the entries of class "Filter"
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_USN_CHANGED;
|
||
Attrs[4] = ATTR_WORKING;
|
||
Attrs[5] = ATTR_WHEN_CHANGED;
|
||
Attrs[6] = ATTR_WHEN_CREATED;
|
||
Attrs[7] = NULL;
|
||
if (!InfoSearch(Info, Tabs + 1, Ldap, ComputerDn, LDAP_SCOPE_SUBTREE,
|
||
CATEGORY_SUBSCRIPTIONS, Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL;
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"SUBSCRIPTION", ComputerDn,
|
||
CATEGORY_SUBSCRIPTIONS, &FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// Working Directory
|
||
//
|
||
Node->Working = FrsDsFindValue(Ldap, LdapEntry, ATTR_WORKING);
|
||
IPRINT2(Info, "%ws Working : %ws\n", TabW, Node->Working);
|
||
IPRINT2(Info, "%ws Actual Working: %ws\n", TabW, WorkingPath);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
|
||
//
|
||
// Recurse to the next level in the DS hierarchy
|
||
//
|
||
InfoPrintSubscribers(Info, Tabs + 1, Ldap, Node->Dn, InfoSubs);
|
||
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
cleanup:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
}
|
||
|
||
|
||
BOOL
|
||
InfoPrintComputer(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs,
|
||
IN PLDAP Ldap,
|
||
IN PWCHAR FindDn,
|
||
IN PWCHAR ObjectCategory,
|
||
IN ULONG Scope,
|
||
OUT PINFO_DN *InfoSubs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on DS computer objects.
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
Ldap - bound ldap handle
|
||
DefaultNcDn - DN of the DCs default naming context
|
||
FindDn - Base Dn for search
|
||
ObjectCategory - Object class (computer or user)
|
||
A user object serves the same purpose as the computer
|
||
object *sometimes* following a NT4 to NT5 upgrade.
|
||
Scope - Scope of search (currently BASE or SUBTREE)
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintComputer:"
|
||
DWORD i;
|
||
DWORD LStatus;
|
||
PLDAPMessage LdapEntry;
|
||
PWCHAR UserAccountControl;
|
||
DWORD NumVals;
|
||
PINFO_DN InfoSub;
|
||
BOOL FoundAComputer = FALSE;
|
||
PCONFIG_NODE Node = NULL;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PWCHAR *Values = NULL;
|
||
DWORD WStatus = ERROR_SUCCESS;
|
||
BOOL FirstError = TRUE;
|
||
PWCHAR WStr = NULL;
|
||
DWORD ComputerFqdnLen;
|
||
PWCHAR Attrs[12];
|
||
CHAR TBuff[100];
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
WCHAR Filter[MAX_PATH + 1];
|
||
WCHAR ComputerFqdn[MAX_PATH + 1];
|
||
|
||
//
|
||
// Initialize return value
|
||
//
|
||
*InfoSubs = NULL;
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
|
||
//
|
||
// Filter that locates our computer object
|
||
//
|
||
if (_snwprintf(Filter, sizeof(Filter)/sizeof(WCHAR) - 1 ,L"(&%s(sAMAccountName=%s$))", ObjectCategory, ComputerName) <0) {
|
||
IPRINT1(Info, "%wsWARN - Buffer too small to hold filter.\n",TabW);
|
||
goto CLEANUP;
|
||
}
|
||
|
||
Filter[sizeof(Filter)/sizeof(WCHAR) - 1] = UNICODE_NULL;
|
||
|
||
//
|
||
// Search the DS beginning at Base for the entries of class "Filter"
|
||
//
|
||
Attrs[0] = ATTR_OBJECT_GUID;
|
||
Attrs[1] = ATTR_DN;
|
||
Attrs[2] = ATTR_SCHEDULE;
|
||
Attrs[3] = ATTR_COMPUTER_REF_BL;
|
||
Attrs[4] = ATTR_SERVER_REF;
|
||
Attrs[5] = ATTR_SERVER_REF_BL;
|
||
Attrs[6] = ATTR_USER_ACCOUNT_CONTROL;
|
||
Attrs[7] = ATTR_DNS_HOST_NAME;
|
||
Attrs[8] = ATTR_WHEN_CHANGED;
|
||
Attrs[9] = ATTR_WHEN_CREATED;
|
||
Attrs[10] = NULL;
|
||
InfoSearch(Info, Tabs + 1, Ldap, FindDn, Scope, Filter, Attrs, 0, &LdapMsg);
|
||
|
||
if (!LdapMsg) {
|
||
goto CLEANUP;
|
||
}
|
||
//
|
||
// Scan the entries returned from ldap_search
|
||
//
|
||
for (LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
LdapEntry != NULL && WIN_SUCCESS(WStatus);
|
||
LdapEntry = ldap_next_entry(Ldap, LdapEntry)) {
|
||
FoundAComputer = TRUE;
|
||
|
||
//
|
||
// Basic info (dn, rdn, and guid)
|
||
//
|
||
Node = InfoAllocBasicNode(Info, TabW, L"COMPUTER", FindDn, Filter,
|
||
&FirstError, Ldap, LdapEntry);
|
||
if (!Node) {
|
||
continue;
|
||
}
|
||
UserAccountControl = FrsDsFindValue(Ldap, LdapEntry, ATTR_USER_ACCOUNT_CONTROL);
|
||
if (UserAccountControl) {
|
||
IPRINT2(Info, "%ws UAC : 0x%08x\n",
|
||
TabW, wcstoul(UserAccountControl, NULL, 10));
|
||
UserAccountControl = FrsFree(UserAccountControl);
|
||
}
|
||
|
||
//
|
||
// Server reference
|
||
//
|
||
Node->SettingsDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_SERVER_REF_BL);
|
||
IPRINT2(Info, "%ws Server BL : %ws\n", TabW, Node->SettingsDn);
|
||
if (!Node->SettingsDn) {
|
||
Node->SettingsDn = FrsDsFindValue(Ldap, LdapEntry, ATTR_SERVER_REF);
|
||
IPRINT2(Info, "%ws Server Ref: %ws\n", TabW, Node->SettingsDn);
|
||
}
|
||
//
|
||
// Make sure it references the settings; not the server
|
||
//
|
||
Node->SettingsDn = FrsDsConvertToSettingsDn(Node->SettingsDn);
|
||
IPRINT2(Info, "%ws Settings : %ws\n", TabW, Node->SettingsDn);
|
||
|
||
//
|
||
// DNS Host Name
|
||
//
|
||
Node->DnsName = FrsDsFindValue(Ldap, LdapEntry, ATTR_DNS_HOST_NAME);
|
||
IPRINT2(Info, "%ws DNS Name : %ws\n", TabW, Node->DnsName);
|
||
|
||
//
|
||
// Created and Modified
|
||
//
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CREATED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenCreated : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
WStr = FrsDsFindValue(Ldap, LdapEntry, ATTR_WHEN_CHANGED);
|
||
FormatGeneralizedTime(WStr, sizeof(TBuff), TBuff);
|
||
IPRINT2(Info, "%ws WhenChanged : %s\n", TabW, TBuff);
|
||
FrsFree(WStr);
|
||
|
||
//
|
||
// Schedule, if any
|
||
//
|
||
Node->Schedule = FrsDsFindSchedule(Ldap, LdapEntry, &Node->ScheduleLength);
|
||
if (Node->Schedule) {
|
||
IPRINT1(Info, "%ws Schedule\n", TabW);
|
||
FrsPrintTypeSchedule(0, Info, Tabs + 1, Node->Schedule, NULL, 0);
|
||
}
|
||
|
||
InfoPrintSubscriptions(Info, Tabs + 1, Ldap, Node->Dn, InfoSubs);
|
||
|
||
//
|
||
// Subscriber Member Bls
|
||
//
|
||
if (!*InfoSubs) {
|
||
IPRINT2(Info, "%ws %ws IS NOT A MEMBER OF ANY SET!\n",
|
||
TabW, ComputerName);
|
||
} else {
|
||
IPRINT1(Info, "%ws Subscriber Member Back Links:\n", TabW);
|
||
for (InfoSub = *InfoSubs; InfoSub; InfoSub = InfoSub->Next) {
|
||
FRS_WCSLWR(InfoSub->Dn);
|
||
IPRINT2(Info, "%ws %ws\n", TabW, InfoSub->Dn);
|
||
}
|
||
}
|
||
|
||
//
|
||
// Next computer
|
||
//
|
||
Node = FrsFreeType(Node);
|
||
}
|
||
|
||
CLEANUP:
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
FrsFreeType(Node);
|
||
return FoundAComputer;
|
||
}
|
||
|
||
|
||
DWORD
|
||
InfoPrintDs(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on DS (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintDs:"
|
||
DWORD WStatus;
|
||
DWORD LStatus;
|
||
DWORD i;
|
||
PWCHAR DcAddr;
|
||
PWCHAR DcName;
|
||
PWCHAR DcDnsName;
|
||
DWORD NumVals;
|
||
PWCHAR Config;
|
||
PLDAPMessage LdapEntry;
|
||
BOOL PrintedComputers;
|
||
PINFO_DN InfoSub;
|
||
PINFO_DN InfoSetting;
|
||
PINFO_DN InfoSet;
|
||
PINFO_DN InfoSubs = NULL;
|
||
PINFO_DN InfoSettings = NULL;
|
||
PINFO_DN InfoSets = NULL;
|
||
PWCHAR SitesDn = NULL;
|
||
PWCHAR ServicesDn = NULL;
|
||
PWCHAR DefaultNcDn = NULL;
|
||
PWCHAR ComputersDn = NULL;
|
||
PWCHAR DomainControllersDn = NULL;
|
||
PLDAPMessage LdapMsg = NULL;
|
||
PWCHAR *Values = NULL;
|
||
PLDAP Ldap = NULL;
|
||
HANDLE LocalDsHandle = INVALID_HANDLE_VALUE;
|
||
WCHAR ComputerFqdn[MAX_PATH + 1];
|
||
DWORD ComputerFqdnLen;
|
||
WCHAR TabW[MAX_TAB_WCHARS + 1];
|
||
CHAR Guid[GUID_CHAR_LEN + 1];
|
||
PWCHAR Attrs[3];
|
||
PDOMAIN_CONTROLLER_INFO DcInfo = NULL;
|
||
struct l_timeval Timeout;
|
||
|
||
DWORD InfoFlags;
|
||
CHAR FlagBuffer[220];
|
||
ULONG ulOptions;
|
||
|
||
extern PWCHAR DsDomainControllerName;
|
||
extern FLAG_NAME_TABLE DsGetDcInfoFlagNameTable[];
|
||
//
|
||
// Client side ldap_connect timeout in seconds. Reg value "Ldap Bind Timeout In Seconds". Default is 30 seconds.
|
||
//
|
||
extern DWORD LdapBindTimeoutInSeconds;
|
||
|
||
//
|
||
// Adjust indentation
|
||
//
|
||
InfoTabs(Tabs, TabW);
|
||
IPRINT1(Info, "%wsNTFRS CONFIGURATION IN THE DS\n", TabW);
|
||
|
||
Ldap = NULL;
|
||
if (IsADc) {
|
||
DcAddr = NULL;
|
||
DcName = ComputerName;
|
||
DcDnsName = ComputerDnsName;
|
||
IPRINT1(Info, "%wsSUBSTITUTE DCINFO FOR DC\n", TabW);
|
||
IPRINT2(Info, "%ws FRS DomainControllerName: %ws\n", TabW, DsDomainControllerName);
|
||
IPRINT2(Info, "%ws Computer Name : %ws\n", TabW, DcName);
|
||
IPRINT2(Info, "%ws Computer DNS Name : %ws\n", TabW, DcDnsName);
|
||
} else {
|
||
//
|
||
// Domain Controller
|
||
//
|
||
WStatus = DsGetDcName(NULL,
|
||
NULL,
|
||
NULL,
|
||
NULL,
|
||
DS_DIRECTORY_SERVICE_REQUIRED |
|
||
DS_WRITABLE_REQUIRED |
|
||
DS_BACKGROUND_ONLY,
|
||
&DcInfo);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
DcInfo = NULL;
|
||
IPRINT2(Info, "%wsWARN - DsGetDcName WStatus %s; Flushing cache...\n",
|
||
TabW, ErrLabelW32(WStatus));
|
||
WStatus = DsGetDcName(NULL,
|
||
NULL,
|
||
NULL,
|
||
NULL,
|
||
DS_DIRECTORY_SERVICE_REQUIRED |
|
||
DS_WRITABLE_REQUIRED |
|
||
DS_FORCE_REDISCOVERY,
|
||
&DcInfo);
|
||
}
|
||
//
|
||
// Report the error and retry for any DC
|
||
//
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
DcInfo = NULL;
|
||
IPRINT3(Info, "%wsERROR - DsGetDcName(%ws); WStatus %s\n",
|
||
TabW, ComputerName, ErrLabelW32(WStatus));
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Dump dcinfo
|
||
//
|
||
IPRINT1(Info, "%wsDCINFO\n", TabW);
|
||
IPRINT2(Info, "%ws LAST DomainControllerName: %ws\n", TabW, DsDomainControllerName);
|
||
IPRINT2(Info, "%ws DomainControllerName : %ws\n", TabW, DcInfo->DomainControllerName);
|
||
IPRINT2(Info, "%ws DomainControllerAddress : %ws\n", TabW, DcInfo->DomainControllerAddress);
|
||
IPRINT2(Info, "%ws DomainControllerType : %08x\n",TabW, DcInfo->DomainControllerAddressType);
|
||
IPRINT2(Info, "%ws DomainName : %ws\n", TabW, DcInfo->DomainName);
|
||
IPRINT2(Info, "%ws DnsForestName : %ws\n", TabW, DcInfo->DnsForestName);
|
||
IPRINT2(Info, "%ws DcSiteName : %ws\n", TabW, DcInfo->DcSiteName);
|
||
IPRINT2(Info, "%ws ClientSiteName : %ws\n", TabW, DcInfo->ClientSiteName);
|
||
|
||
InfoFlags = DcInfo->Flags;
|
||
FrsFlagsToStr(InfoFlags, DsGetDcInfoFlagNameTable, sizeof(FlagBuffer), FlagBuffer);
|
||
IPRINT3(Info, "%ws Flags : %08x [%s]\n",TabW, InfoFlags, FlagBuffer);
|
||
|
||
|
||
if (!DsDomainControllerName ||
|
||
!DcInfo->DomainControllerName ||
|
||
WSTR_NE(DsDomainControllerName, DcInfo->DomainControllerName)) {
|
||
IPRINT3(Info, "%wsWARN - Using DC %ws; not %ws\n",
|
||
TabW, DcInfo->DomainControllerName, DsDomainControllerName);
|
||
}
|
||
|
||
//
|
||
// Binding address
|
||
//
|
||
DcAddr = DcInfo->DomainControllerAddress;
|
||
DcDnsName = DcInfo->DomainControllerName;
|
||
}
|
||
wcsncpy(InfoDcName, DcDnsName, ARRAY_SZ(InfoDcName)-1);
|
||
InfoDcName[ARRAY_SZ(InfoDcName)-1] = L'\0';
|
||
|
||
//
|
||
// BIND to the DS
|
||
//
|
||
IPRINT1(Info, "\n%wsBINDING TO THE DS:\n", TabW);
|
||
|
||
//
|
||
// if ldap_open is called with a server name the api will call DsGetDcName
|
||
// passing the server name as the domainname parm...bad, because
|
||
// DsGetDcName will make a load of DNS queries based on the server name,
|
||
// it is designed to construct these queries from a domain name...so all
|
||
// these queries will be bogus, meaning they will waste network bandwidth,
|
||
// time to fail, and worst case cause expensive on demand links to come up
|
||
// as referrals/forwarders are contacted to attempt to resolve the bogus
|
||
// names. By setting LDAP_OPT_AREC_EXCLUSIVE to on using ldap_set_option
|
||
// after the ldap_init but before any other operation using the ldap
|
||
// handle from ldap_init, the delayed connection setup will not call
|
||
// DsGetDcName, just gethostbyname, or if an IP is passed, the ldap client
|
||
// will detect that and use the address directly.
|
||
//
|
||
|
||
//
|
||
// Remove the leading \\ if they exist.
|
||
//
|
||
FRS_TRIM_LEADING_2SLASH(DcDnsName);
|
||
FRS_TRIM_LEADING_2SLASH(DcAddr);
|
||
|
||
ulOptions = PtrToUlong(LDAP_OPT_ON);
|
||
Timeout.tv_sec = LdapBindTimeoutInSeconds;
|
||
Timeout.tv_usec = 0;
|
||
|
||
//
|
||
// Try using DcDnsName first.
|
||
//
|
||
if ((Ldap == NULL) && (DcDnsName != NULL)) {
|
||
|
||
Ldap = ldap_init(DcDnsName, LDAP_PORT);
|
||
if (Ldap != NULL) {
|
||
ldap_set_option(Ldap, LDAP_OPT_AREC_EXCLUSIVE, &ulOptions);
|
||
LStatus = ldap_connect(Ldap, &Timeout);
|
||
if (LStatus != LDAP_SUCCESS) {
|
||
IPRINT4(Info, "%ws WARN - ldap_connect(%ws); (ldap error %08x = %ws)\n",
|
||
TabW, DcDnsName, LStatus, ldap_err2string(LStatus));
|
||
ldap_unbind_s(Ldap);
|
||
Ldap = NULL;
|
||
} else {
|
||
IPRINT2(Info, "%ws ldap_connect : %ws\n", TabW, DcDnsName);
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// Try using DcAddr next.
|
||
//
|
||
if ((Ldap == NULL) && (DcAddr != NULL)) {
|
||
|
||
Ldap = ldap_init(DcAddr, LDAP_PORT);
|
||
if (Ldap != NULL) {
|
||
ldap_set_option(Ldap, LDAP_OPT_AREC_EXCLUSIVE, &ulOptions);
|
||
LStatus = ldap_connect(Ldap, &Timeout);
|
||
if (LStatus != LDAP_SUCCESS) {
|
||
IPRINT4(Info, "%ws WARN - ldap_connect(%ws); (ldap error %08x = %ws)\n",
|
||
TabW, DcAddr, LStatus, ldap_err2string(LStatus));
|
||
ldap_unbind_s(Ldap);
|
||
Ldap = NULL;
|
||
} else {
|
||
IPRINT2(Info, "%ws ldap_connect : %ws\n", TabW, DcAddr);
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// Try using DcName finally.
|
||
//
|
||
if ((Ldap == NULL) && (DcName != NULL)) {
|
||
|
||
Ldap = ldap_init(DcName, LDAP_PORT);
|
||
if (Ldap != NULL) {
|
||
ldap_set_option(Ldap, LDAP_OPT_AREC_EXCLUSIVE, &ulOptions);
|
||
LStatus = ldap_connect(Ldap, &Timeout);
|
||
if (LStatus != LDAP_SUCCESS) {
|
||
IPRINT4(Info, "%ws WARN - ldap_connect(%ws); (ldap error %08x = %ws)\n",
|
||
TabW, DcName, LStatus, ldap_err2string(LStatus));
|
||
ldap_unbind_s(Ldap);
|
||
Ldap = NULL;
|
||
} else {
|
||
IPRINT2(Info, "%ws ldap_connect : %ws\n", TabW, DcName);
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// Whatever it is, we can't find it.
|
||
//
|
||
if (!Ldap) {
|
||
IPRINT6(Info, "%ws ERROR - ldap_connect(DNS %ws, BIOS %ws, IP %ws); (ldap error %08x = %ws)\n",
|
||
TabW, DcDnsName, DcName, DcAddr, LStatus, ldap_err2string(LStatus));
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Bind to the ldap server
|
||
//
|
||
LStatus = ldap_bind_s(Ldap, NULL, NULL, LDAP_AUTH_NEGOTIATE);
|
||
|
||
//
|
||
// No luck; report error and carry on
|
||
//
|
||
if (LStatus != LDAP_SUCCESS) {
|
||
IPRINT4(Info, "%ws ERROR - ldap_bind_s(%ws); (ldap error %08x = %ws)\n",
|
||
TabW, ComputerName, LStatus, ldap_err2string(LStatus));
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Bind to the Ds (for various Ds calls such as DsCrackName())
|
||
//
|
||
//
|
||
// DC's Dns Name
|
||
//
|
||
WStatus = ERROR_RETRY;
|
||
if (!WIN_SUCCESS(WStatus) && DcDnsName) {
|
||
WStatus = DsBind(DcDnsName, NULL, &LocalDsHandle);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
LocalDsHandle = NULL;
|
||
IPRINT3(Info, "%ws WARN - DsBind(DcDnsName %ws); WStatus %s\n",
|
||
TabW, DcDnsName, ErrLabelW32(WStatus));
|
||
} else {
|
||
IPRINT2(Info, "%ws DsBind : %ws\n", TabW, DcDnsName);
|
||
}
|
||
}
|
||
|
||
//
|
||
// DC's Computer Name
|
||
//
|
||
if (!WIN_SUCCESS(WStatus) && DcName) {
|
||
WStatus = DsBind(DcName, NULL, &LocalDsHandle);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
LocalDsHandle = NULL;
|
||
IPRINT3(Info, "%ws WARN - DsBind(DcName %ws); WStatus %s\n",
|
||
TabW, DcName, ErrLabelW32(WStatus));
|
||
} else {
|
||
IPRINT2(Info, "%ws DsBind : %ws\n", TabW, DcName);
|
||
}
|
||
}
|
||
|
||
//
|
||
// DC's IP Address
|
||
//
|
||
if (!WIN_SUCCESS(WStatus) && DcAddr) {
|
||
WStatus = DsBind(DcAddr, NULL, &LocalDsHandle);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
LocalDsHandle = NULL;
|
||
IPRINT3(Info, "%ws WARN - DsBind(DcAddr %ws); WStatus %s\n",
|
||
TabW, DcAddr, ErrLabelW32(WStatus));
|
||
} else {
|
||
IPRINT2(Info, "%ws DsBind : %ws\n", TabW, DcAddr);
|
||
}
|
||
}
|
||
|
||
//
|
||
// Whatever it is, we can't find it
|
||
//
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
IPRINT5(Info, "%ws ERROR - DsBind(DNS %ws, BIOS %ws, IP %ws); WStatus %s\n",
|
||
TabW, DcDnsName, DcName, DcAddr, ErrLabelW32(WStatus));
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// NAMING CONTEXTS
|
||
//
|
||
IPRINT1(Info, "\n%wsNAMING CONTEXTS:\n", TabW);
|
||
|
||
//
|
||
// Find the naming contexts and the default naming context
|
||
//
|
||
Attrs[0] = ATTR_NAMING_CONTEXTS;
|
||
Attrs[1] = ATTR_DEFAULT_NAMING_CONTEXT;
|
||
Attrs[2] = NULL;
|
||
if (!InfoSearch(Info, Tabs + 1, Ldap, CN_ROOT, LDAP_SCOPE_BASE, CATEGORY_ANY,
|
||
Attrs, 0, &LdapMsg)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
LdapEntry = ldap_first_entry(Ldap, LdapMsg);
|
||
if (!LdapEntry) {
|
||
IPRINT2(Info, "%ws ERROR - ldap_first_entry(contexts, %ws) no entry\n",
|
||
TabW, ComputerName);
|
||
goto cleanup;
|
||
}
|
||
Values = (PWCHAR *)FrsDsFindValues(Ldap, LdapEntry, ATTR_NAMING_CONTEXTS, FALSE);
|
||
if (!Values) {
|
||
IPRINT2(Info, "%ws ERROR - FrsDsFindValues(contexts, %ws) no entry\n",
|
||
TabW, ComputerName);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Now, find the naming context that begins with "CN=configuration"
|
||
//
|
||
NumVals = ldap_count_values(Values);
|
||
while (NumVals--) {
|
||
FRS_WCSLWR(Values[NumVals]);
|
||
Config = wcsstr(Values[NumVals], CONFIG_NAMING_CONTEXT);
|
||
if (Config && Config == Values[NumVals]) {
|
||
//
|
||
// Build the pathname for "configuration\sites & services"
|
||
//
|
||
SitesDn = FrsDsExtendDn(Config, CN_SITES);
|
||
ServicesDn = FrsDsExtendDn(Config, CN_SERVICES);
|
||
break;
|
||
}
|
||
}
|
||
LDAP_FREE_VALUES(Values);
|
||
|
||
//
|
||
// Finally, find the default naming context
|
||
//
|
||
Values = (PWCHAR *)FrsDsFindValues(Ldap,
|
||
LdapEntry,
|
||
ATTR_DEFAULT_NAMING_CONTEXT,
|
||
FALSE);
|
||
if (!Values) {
|
||
IPRINT2(Info, "%ws ERROR - FrsDsFindValues(default naming context, %ws) no entry\n",
|
||
TabW, ComputerName);
|
||
goto cleanup;
|
||
}
|
||
|
||
DefaultNcDn = FrsWcsDup(Values[0]);
|
||
ComputersDn = FrsDsExtendDn(DefaultNcDn, CN_COMPUTERS);
|
||
DomainControllersDn = FrsDsExtendDnOu(DefaultNcDn, CN_DOMAIN_CONTROLLERS);
|
||
LDAP_FREE_VALUES(Values);
|
||
|
||
IPRINT2(Info, "%ws SitesDn : %ws\n", TabW, SitesDn);
|
||
IPRINT2(Info, "%ws ServicesDn : %ws\n", TabW, ServicesDn);
|
||
IPRINT2(Info, "%ws DefaultNcDn: %ws\n", TabW, DefaultNcDn);
|
||
IPRINT2(Info, "%ws ComputersDn: %ws\n", TabW, ComputersDn);
|
||
IPRINT2(Info, "%ws DomainCtlDn: %ws\n", TabW, DomainControllersDn);
|
||
|
||
//
|
||
// Retrieve the computer's fully qualified Dn
|
||
//
|
||
ComputerFqdnLen = MAX_PATH;
|
||
if (!GetComputerObjectName(NameFullyQualifiedDN, ComputerFqdn, &ComputerFqdnLen)) {
|
||
IPRINT4(Info, "%ws ERROR - GetComputerObjectName(%ws); Len %d, WStatus %s\n",
|
||
TabW, ComputerName, ComputerFqdnLen, ErrLabelW32(GetLastError()));
|
||
ComputerFqdn[0] = L'\0';
|
||
} else {
|
||
IPRINT2(Info, "%ws Fqdn : %ws\n", TabW, ComputerFqdn);
|
||
}
|
||
|
||
//
|
||
// Find and print the computer info
|
||
//
|
||
PrintedComputers = FALSE;
|
||
if (!PrintedComputers && ComputerFqdn[0]) {
|
||
IPRINT1(Info, "%ws Searching : Fqdn\n", TabW);
|
||
PrintedComputers = InfoPrintComputer(Info, Tabs, Ldap, ComputerFqdn,
|
||
CATEGORY_COMPUTER, LDAP_SCOPE_BASE, &InfoSubs);
|
||
}
|
||
if (!PrintedComputers && ComputersDn) {
|
||
IPRINT1(Info, "%ws Searching : Computers\n", TabW);
|
||
PrintedComputers = InfoPrintComputer(Info, Tabs, Ldap, ComputersDn,
|
||
CATEGORY_COMPUTER, LDAP_SCOPE_SUBTREE, &InfoSubs);
|
||
}
|
||
if (!PrintedComputers && DomainControllersDn) {
|
||
IPRINT1(Info, "%ws Searching : Domain Controllers\n", TabW);
|
||
PrintedComputers = InfoPrintComputer(Info, Tabs, Ldap, DomainControllersDn,
|
||
CATEGORY_COMPUTER, LDAP_SCOPE_SUBTREE, &InfoSubs);
|
||
}
|
||
if (!PrintedComputers && DefaultNcDn) {
|
||
IPRINT1(Info, "%ws Searching : Default Naming Context\n", TabW);
|
||
PrintedComputers = InfoPrintComputer(Info, Tabs, Ldap, DefaultNcDn,
|
||
CATEGORY_COMPUTER, LDAP_SCOPE_SUBTREE, &InfoSubs);
|
||
}
|
||
if (!PrintedComputers && DefaultNcDn) {
|
||
IPRINT1(Info, "%ws Searching : Default Naming Context for USER\n", TabW);
|
||
PrintedComputers = InfoPrintComputer(Info, Tabs, Ldap, DefaultNcDn,
|
||
CATEGORY_USER, LDAP_SCOPE_SUBTREE, &InfoSubs);
|
||
}
|
||
|
||
for (InfoSub = InfoSubs; InfoSub; InfoSub = InfoSub->Next) {
|
||
InfoPrintSettings(Info, Tabs, Ldap, InfoSub->Dn, LocalDsHandle, &InfoSettings,
|
||
&InfoSets);
|
||
}
|
||
|
||
cleanup:
|
||
//
|
||
// Cleanup
|
||
//
|
||
LDAP_FREE_VALUES(Values);
|
||
LDAP_FREE_MSG(LdapMsg);
|
||
if (DcInfo) {
|
||
NetApiBufferFree(DcInfo);
|
||
DcInfo = NULL;
|
||
}
|
||
if (Ldap) {
|
||
ldap_unbind_s(Ldap);
|
||
}
|
||
if (HANDLE_IS_VALID(LocalDsHandle)) {
|
||
DsUnBind(&LocalDsHandle);
|
||
}
|
||
FrsFree(SitesDn);
|
||
FrsFree(ServicesDn);
|
||
FrsFree(DefaultNcDn);
|
||
FrsFree(ComputersDn);
|
||
FrsFree(DomainControllersDn);
|
||
|
||
while (InfoSub = InfoSubs) {
|
||
InfoSubs = InfoSub->Next;
|
||
FrsFree(InfoSub->Dn);
|
||
FrsFree(InfoSub->SetType);
|
||
FrsFree(InfoSub);
|
||
}
|
||
while (InfoSetting = InfoSettings) {
|
||
InfoSettings = InfoSetting->Next;
|
||
FrsFree(InfoSetting->Dn);
|
||
FrsFree(InfoSetting->SetType);
|
||
FrsFree(InfoSetting);
|
||
}
|
||
while (InfoSet = InfoSets) {
|
||
InfoSets = InfoSet->Next;
|
||
FrsFree(InfoSet->Dn);
|
||
FrsFree(InfoSet->SetType);
|
||
FrsFree(InfoSet);
|
||
}
|
||
|
||
//
|
||
// Real error messages are in the info buffer
|
||
//
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
PVOID
|
||
InfoFreeInfoTable(
|
||
IN PINFO_TABLE InfoTable,
|
||
IN PNTFRSAPI_INFO Info
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Free the info IDTable
|
||
|
||
Arguments:
|
||
InfoTable
|
||
Info
|
||
|
||
Return Value:
|
||
NULL
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoFreeInfoTable:"
|
||
JET_ERR jerr;
|
||
|
||
if (InfoTable == NULL) {
|
||
return NULL;
|
||
}
|
||
|
||
if (InfoTable->TableCtx != NULL) {
|
||
DbsFreeTableContext(InfoTable->TableCtx, InfoTable->ThreadCtx->JSesid);
|
||
}
|
||
|
||
if (InfoTable->ThreadCtx != NULL) {
|
||
jerr = DbsCloseJetSession(InfoTable->ThreadCtx);
|
||
if (!JET_SUCCESS(jerr)) {
|
||
IPRINT1(Info, "DbsCloseJetSession jet error = %s\n", ErrLabelJet(jerr));
|
||
}
|
||
|
||
InfoTable->ThreadCtx = FrsFreeType(InfoTable->ThreadCtx);
|
||
}
|
||
|
||
return FrsFree(InfoTable);
|
||
|
||
}
|
||
|
||
|
||
JET_ERR
|
||
InfoConfigTableWorker(
|
||
IN PTHREAD_CTX ThreadCtx,
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PCONFIG_TABLE_RECORD ConfigRecord,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is a worker function passed to FrsEnumerateTable(). Each time
|
||
it is called it prints an entry into the info buffer.
|
||
|
||
Arguments:
|
||
|
||
ThreadCtx - Needed to access Jet.
|
||
TableCtx - A ptr to an ConfigTable context struct.
|
||
ConfigRecord - A ptr to a config table record.
|
||
InfoTable
|
||
|
||
Thread Return Value:
|
||
|
||
A Jet error status. Success means call us with the next record.
|
||
Failure means don't call again and pass our status back to the
|
||
caller of FrsEnumerateTable().
|
||
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoConfigTableWorker:"
|
||
|
||
PINFO_TABLE InfoTable = FrsInfoContext->InfoTable;
|
||
|
||
//
|
||
// Check if there is enough room for another record.
|
||
//
|
||
if (!INFO_HAS_SPACE(InfoTable->Info)) {
|
||
SetFlag(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL);
|
||
}
|
||
|
||
if (FrsInfoContext->KeyValue == NULL) {
|
||
FrsInfoContext->KeyValue = FrsAlloc(sizeof(ULONG));
|
||
}
|
||
|
||
CopyMemory(FrsInfoContext->KeyValue, &ConfigRecord->ReplicaNumber, sizeof(ULONG));
|
||
|
||
if (FlagOn(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
return JET_errNoCurrentRecord;
|
||
}
|
||
|
||
IPRINT0(InfoTable->Info, "\n\n");
|
||
|
||
DbsDisplayRecordIPrint(TableCtx, InfoTable, TRUE, NULL, 0);
|
||
|
||
return JET_errSuccess;
|
||
}
|
||
|
||
|
||
JET_ERR
|
||
InfoIDTableWorker(
|
||
IN PTHREAD_CTX ThreadCtx,
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PIDTABLE_RECORD IDTableRec,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is a worker function passed to FrsEnumerateTable(). Each time
|
||
it is called it prints an entry into the info buffer.
|
||
|
||
Arguments:
|
||
|
||
ThreadCtx - Needed to access Jet.
|
||
TableCtx - A ptr to an IDTable context struct.
|
||
IDTableRec - A ptr to a IDTable record.
|
||
InfoTable
|
||
|
||
Thread Return Value:
|
||
|
||
A Jet error status. Success means call us with the next record.
|
||
Failure means don't call again and pass our status back to the
|
||
caller of FrsEnumerateTable().
|
||
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoIDTableWorker:"
|
||
|
||
PINFO_TABLE InfoTable = FrsInfoContext->InfoTable;
|
||
|
||
//
|
||
// Check if there is enough room for another record.
|
||
//
|
||
if (!INFO_HAS_SPACE(InfoTable->Info)) {
|
||
SetFlag(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL);
|
||
}
|
||
|
||
if (FrsInfoContext->KeyValue == NULL) {
|
||
FrsInfoContext->KeyValue = FrsAlloc(sizeof(GUID));
|
||
}
|
||
|
||
COPY_GUID(FrsInfoContext->KeyValue, &IDTableRec->FileGuid);
|
||
|
||
if (FlagOn(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
return JET_errNoCurrentRecord;
|
||
}
|
||
|
||
//
|
||
// Table Descriptor
|
||
//
|
||
IPRINT2(InfoTable->Info, "\nTable Type: ID Table for %ws (%d)\n",
|
||
InfoTable->Replica->ReplicaName->Name, InfoTable->Replica->ReplicaNumber);
|
||
|
||
DbsDisplayRecordIPrint(TableCtx, InfoTable, TRUE, NULL, 0);
|
||
|
||
|
||
return JET_errSuccess;
|
||
|
||
|
||
}
|
||
|
||
|
||
JET_ERR
|
||
InfoInOutLogTableWorker(
|
||
IN PTHREAD_CTX ThreadCtx,
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PCHANGE_ORDER_COMMAND Coc,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext,
|
||
IN PWCHAR TableDescriptor
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is a worker function passed to FrsEnumerateTable(). Each time
|
||
it is called it prints an entry into the info buffer.
|
||
|
||
Arguments:
|
||
|
||
ThreadCtx - Needed to access Jet.
|
||
TableCtx - A ptr to an IDTable context struct.
|
||
Coc - A ptr to a inbound log record (change order)
|
||
InfoTable
|
||
TableDescriptor
|
||
|
||
Thread Return Value:
|
||
|
||
A Jet error status. Success means call us with the next record.
|
||
Failure means don't call again and pass our status back to the
|
||
caller of FrsEnumerateTable().
|
||
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoInOutLogTableWorker:"
|
||
|
||
PREPLICA Replica;
|
||
|
||
PCXTION Cxtion = NULL;
|
||
PWSTR CxtName = L"<null>";
|
||
PWSTR PartnerName = L"<null>";
|
||
PWSTR PartSrvName = L"<null>";
|
||
PCHAR CxtionState = "<null>";
|
||
BOOL PrintCxtion;
|
||
PINFO_TABLE InfoTable = FrsInfoContext->InfoTable;
|
||
|
||
//
|
||
// Check if there is enough room for another record.
|
||
//
|
||
if (!INFO_HAS_SPACE(InfoTable->Info)) {
|
||
SetFlag(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL);
|
||
}
|
||
|
||
if (FrsInfoContext->KeyValue == NULL) {
|
||
FrsInfoContext->KeyValue = FrsAlloc(sizeof(ULONG));
|
||
}
|
||
|
||
CopyMemory(FrsInfoContext->KeyValue, &Coc->SequenceNumber, sizeof(ULONG));
|
||
|
||
if (FlagOn(InfoTable->Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
return JET_errNoCurrentRecord;
|
||
}
|
||
|
||
//
|
||
// Table Descriptor
|
||
//
|
||
IPRINT3(InfoTable->Info, "\nTable Type: %ws for %ws (%d)\n",
|
||
TableDescriptor, InfoTable->Replica->ReplicaName->Name, InfoTable->Replica->ReplicaNumber);
|
||
|
||
//
|
||
// Dump the change order record.
|
||
//
|
||
DbsDisplayRecordIPrint(TableCtx, InfoTable, TRUE, NULL, 0);
|
||
|
||
|
||
Replica = InfoTable->Replica;
|
||
//
|
||
// Find the cxtion for this CO
|
||
//
|
||
LOCK_CXTION_TABLE(Replica);
|
||
|
||
Cxtion = GTabLookupNoLock(Replica->Cxtions, &Coc->CxtionGuid, NULL);
|
||
|
||
PrintCxtion = (Cxtion != NULL) && (Cxtion->Inbound);
|
||
|
||
if (PrintCxtion) {
|
||
CxtionState = GetCxtionStateName(Cxtion);
|
||
|
||
if (Cxtion->Name != NULL) {
|
||
|
||
if (Cxtion->Name->Name != NULL) {
|
||
CxtName = Cxtion->Name->Name;
|
||
}
|
||
}
|
||
|
||
if ((Cxtion->Partner != NULL) && (Cxtion->Partner->Name != NULL)) {
|
||
PartnerName = Cxtion->Partner->Name;
|
||
}
|
||
|
||
if (Cxtion->PartSrvName != NULL) {
|
||
PartSrvName = Cxtion->PartSrvName;
|
||
}
|
||
}
|
||
UNLOCK_CXTION_TABLE(Replica);
|
||
|
||
if (PrintCxtion) {
|
||
IPRINT3(InfoTable->Info, "Cxtion Name : %ws <- %ws\\%ws\n",
|
||
CxtName, PartnerName, PartSrvName);
|
||
|
||
IPRINT1(InfoTable->Info, "Cxtion State : %s\n", CxtionState);
|
||
}
|
||
|
||
return JET_errSuccess;
|
||
|
||
}
|
||
|
||
|
||
JET_ERR
|
||
InfoInLogTableWorker(
|
||
IN PTHREAD_CTX ThreadCtx,
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PCHANGE_ORDER_COMMAND Coc,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is a worker function passed to FrsEnumerateTable(). Each time
|
||
it is called it prints an entry into the info buffer.
|
||
|
||
Arguments:
|
||
|
||
ThreadCtx - Needed to access Jet.
|
||
TableCtx - A ptr to an IDTable context struct.
|
||
Coc - A ptr to a inbound log record (change order)
|
||
InfoTable
|
||
|
||
Thread Return Value:
|
||
|
||
A Jet error status. Success means call us with the next record.
|
||
Failure means don't call again and pass our status back to the
|
||
caller of FrsEnumerateTable().
|
||
|
||
--*/
|
||
{
|
||
return InfoInOutLogTableWorker(ThreadCtx, TableCtx, Coc, FrsInfoContext,
|
||
L"Inbound Log Table");
|
||
}
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
JET_ERR
|
||
InfoOutLogTableWorker(
|
||
IN PTHREAD_CTX ThreadCtx,
|
||
IN PTABLE_CTX TableCtx,
|
||
IN PCHANGE_ORDER_COMMAND Coc,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is a worker function passed to FrsEnumerateTable(). Each time
|
||
it is called it prints an entry into the info buffer.
|
||
|
||
Arguments:
|
||
|
||
ThreadCtx - Needed to access Jet.
|
||
TableCtx - A ptr to an IDTable context struct.
|
||
Coc - A ptr to a inbound log record (change order)
|
||
InfoTable
|
||
|
||
Thread Return Value:
|
||
|
||
A Jet error status. Success means call us with the next record.
|
||
Failure means don't call again and pass our status back to the
|
||
caller of FrsEnumerateTable().
|
||
|
||
--*/
|
||
{
|
||
return InfoInOutLogTableWorker(ThreadCtx, TableCtx, Coc, FrsInfoContext,
|
||
L"Outbound Log Table");
|
||
}
|
||
|
||
|
||
DWORD
|
||
InfoPrintSingleTable(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext,
|
||
IN PREPLICA Replica,
|
||
IN PENUMERATE_TABLE_ROUTINE InfoTableWorker
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
|
||
Display data for the specified table using the InfoPrint interface.
|
||
|
||
Arguments:
|
||
Info - ptr to the API Info ctx.
|
||
FrsInfoContext - Context saved by the service.
|
||
Replica, -- ptr to the replica struct for the replica set.
|
||
InfoTableWorker -- The function to call to display each record.
|
||
|
||
Return Value:
|
||
jet error Status
|
||
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintSingleTable:"
|
||
|
||
JET_ERR jerr = JET_errSuccess;
|
||
PINFO_TABLE InfoTable = NULL;
|
||
|
||
|
||
try {
|
||
|
||
InfoTable = FrsAlloc(sizeof(*InfoTable));
|
||
FrsInfoContext->InfoTable = InfoTable;
|
||
InfoTable->ThreadCtx = FrsAllocType(THREAD_CONTEXT_TYPE);
|
||
InfoTable->TableCtx = DbsCreateTableContext(FrsInfoContext->TableType);
|
||
InfoTable->Info = Info;
|
||
InfoTable->Tabs = 0; /* Tabs + 1*/ // Pitch this tabs stuff.
|
||
|
||
if (FrsInfoContext->KeyValue == NULL) {
|
||
//
|
||
// Print the replica name only if this is the first call.
|
||
//
|
||
if ((FrsInfoContext->TableType == ConfigTablex) ||
|
||
(FrsInfoContext->TableType == ServiceTablex)) {
|
||
|
||
IPRINT1(Info, "\n***** %ws\n", FrsInfoContext->TableName);
|
||
} else {
|
||
IPRINT1(Info, "\n***** %ws\n", Replica->ReplicaName->Name);
|
||
}
|
||
}
|
||
|
||
//
|
||
// Setup a Jet Session (returning the session ID in ThreadCtx).
|
||
//
|
||
jerr = DbsCreateJetSession(InfoTable->ThreadCtx);
|
||
if (!JET_SUCCESS(jerr)) {
|
||
IPRINT2(Info,"ERROR - %ws: DbsCreateJetSession jet error %s.\n",
|
||
FrsInfoContext->TableName, ErrLabelJet(jerr));
|
||
goto RETURN;
|
||
}
|
||
//
|
||
// Init the table context and open the table.
|
||
//
|
||
jerr = DbsOpenTable(InfoTable->ThreadCtx,
|
||
InfoTable->TableCtx,
|
||
ReplicaAddrToId(Replica),
|
||
FrsInfoContext->TableType,
|
||
NULL);
|
||
if (!JET_SUCCESS(jerr)) {
|
||
IPRINT2(Info,"ERROR - %ws: DbsOpenTable jet error %s.\n",
|
||
FrsInfoContext->TableName, ErrLabelJet(jerr));
|
||
goto RETURN;
|
||
}
|
||
|
||
InfoTable->Replica = Replica;
|
||
|
||
//
|
||
// Scan thru the Table
|
||
//
|
||
jerr = FrsEnumerateTableFrom(InfoTable->ThreadCtx,
|
||
InfoTable->TableCtx,
|
||
FrsInfoContext->Indexx,
|
||
FrsInfoContext->KeyValue,
|
||
FrsInfoContext->ScanDirection,
|
||
InfoTableWorker,
|
||
FrsInfoContext);
|
||
//
|
||
// We're done. Return success if we made it to the end
|
||
//
|
||
if (jerr != JET_errNoCurrentRecord &&
|
||
jerr != JET_wrnTableEmpty) {
|
||
IPRINT2(Info,"ERROR - %ws: FrsEnumerateTableFrom jet error %s.\n",
|
||
FrsInfoContext->TableName, ErrLabelJet(jerr));
|
||
}
|
||
|
||
RETURN:;
|
||
|
||
} finally {
|
||
//
|
||
// Make sure we close jet and free the memory.
|
||
//
|
||
InfoTable = InfoFreeInfoTable(InfoTable, Info);
|
||
}
|
||
|
||
return jerr;
|
||
}
|
||
|
||
|
||
DWORD
|
||
InfoPrintTables(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN PFRS_INFO_CONTEXT FrsInfoContext,
|
||
IN PWCHAR TableDescriptor,
|
||
IN TABLE_TYPE TableType,
|
||
IN ULONG InfoIndexx,
|
||
IN PENUMERATE_TABLE_ROUTINE InfoTableWorker
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on a DB Table (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
FrsInfoContext - Context saved by the service.
|
||
TableDescriptor - Text string for output
|
||
TableType - Table type code (from schema.h)
|
||
InfoIndexx - Table index to use for enumeration (from schema.h)
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintTables:"
|
||
PVOID Key;
|
||
PREPLICA Replica = NULL;
|
||
PREPLICA NextReplica = NULL;
|
||
extern PGEN_TABLE ReplicasByGuid;
|
||
|
||
FrsFree(FrsInfoContext->TableName);
|
||
|
||
FrsInfoContext->TableName = FrsWcsDup(TableDescriptor);
|
||
FrsInfoContext->TableType = TableType;
|
||
FrsInfoContext->Indexx = InfoIndexx;
|
||
|
||
//
|
||
// Check for single instance tables.
|
||
//
|
||
if ((TableType == ConfigTablex) ||
|
||
(TableType == ServiceTablex)) {
|
||
|
||
InfoPrintSingleTable(Info,
|
||
FrsInfoContext,
|
||
Replica,
|
||
InfoTableWorker);
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
//
|
||
// For the given table type, dump info for all replica sets.
|
||
//
|
||
if (FrsInfoContext->KeyValue == NULL) {
|
||
//
|
||
// Print the header only for the first call.
|
||
//
|
||
IPRINT1(Info, "NTFRS %ws\n", TableDescriptor);
|
||
}
|
||
|
||
|
||
do {
|
||
NextReplica = NULL;
|
||
|
||
Key = NULL;
|
||
while (Replica = GTabNextDatum(ReplicasByGuid, &Key)) {
|
||
if (Replica->ReplicaNumber == FrsInfoContext->ReplicaNumber) {
|
||
//
|
||
// Found the replica we were looking for. Break and process it.
|
||
//
|
||
NextReplica = Replica;
|
||
break;
|
||
} else if ((Replica->ReplicaNumber > FrsInfoContext->ReplicaNumber) &&
|
||
((NextReplica == NULL) ||
|
||
(Replica->ReplicaNumber < NextReplica->ReplicaNumber))) {
|
||
//
|
||
// We are getting closer. Pick this one instead.
|
||
//
|
||
NextReplica = Replica;
|
||
}
|
||
}
|
||
|
||
if (NextReplica != NULL) {
|
||
|
||
FrsInfoContext->ReplicaNumber = NextReplica->ReplicaNumber;
|
||
|
||
InfoPrintSingleTable(Info,
|
||
FrsInfoContext,
|
||
NextReplica,
|
||
InfoTableWorker);
|
||
|
||
if (!FlagOn(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
//
|
||
// Buffer is not full yet. We must have completed processing the replica.
|
||
// Move to the next one.
|
||
//
|
||
FrsInfoContext->ReplicaNumber+=1;
|
||
FrsInfoContext->KeyValue = FrsFree(FrsInfoContext->KeyValue);
|
||
|
||
} else {
|
||
//
|
||
// Buffer is full. We will come back here looking for the same replica.
|
||
//
|
||
break;
|
||
}
|
||
}
|
||
|
||
} while ( NextReplica != NULL );
|
||
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
|
||
DWORD
|
||
InfoPrintMemory(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on memory usage (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintMemory:"
|
||
FrsPrintAllocStats(0, Info, Tabs);
|
||
FrsPrintRpcStats(0, Info, Tabs);
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
|
||
|
||
|
||
DWORD
|
||
InfoPrintThreads(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on thread usage (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintThreads:"
|
||
FrsPrintThreadStats(0, Info, Tabs);
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
VOID
|
||
FrsPrintStageStats(
|
||
IN ULONG Severity,
|
||
IN PNTFRSAPI_INFO Info, OPTIONAL
|
||
IN DWORD Tabs
|
||
);
|
||
DWORD
|
||
InfoPrintStage(
|
||
IN PNTFRSAPI_INFO Info,
|
||
IN DWORD Tabs
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info on thread usage (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
Info - RPC output buffer
|
||
Tabs - number of tabs
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoPrintStage:"
|
||
FrsPrintStageStats(0, Info, Tabs);
|
||
return ERROR_SUCCESS;
|
||
}
|
||
|
||
|
||
DWORD
|
||
InfoVerify(
|
||
IN ULONG BlobSize,
|
||
IN OUT PBYTE Blob
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Verify the consistency of the blob.
|
||
|
||
Arguments:
|
||
BlobSize - total bytes of Blob
|
||
Blob - details desired info and provides buffer for info
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoVerify:"
|
||
DWORD WStatus = ERROR_SUCCESS;
|
||
PBYTE EoB;
|
||
PBYTE EoI;
|
||
PBYTE BoL;
|
||
PBYTE BoF;
|
||
PNTFRSAPI_INFO Info = (PNTFRSAPI_INFO)Blob;
|
||
|
||
//
|
||
// Not a valid blob
|
||
//
|
||
if (BlobSize < NTFRSAPI_INFO_HEADER_SIZE) {
|
||
WStatus = FRS_ERR_INVALID_SERVICE_PARAMETER;
|
||
goto CLEANUP;
|
||
}
|
||
|
||
//
|
||
// BlobSize must include the entire Blob
|
||
//
|
||
if (BlobSize != Info->SizeInChars) {
|
||
WStatus = FRS_ERR_INVALID_SERVICE_PARAMETER;
|
||
goto CLEANUP;
|
||
}
|
||
|
||
//
|
||
// Return our info version
|
||
//
|
||
Info->NtFrsMajor = NTFRS_MAJOR;
|
||
Info->NtFrsMinor = NTFRS_MINOR;
|
||
SetFlag(Info->Flags, NTFRSAPI_INFO_FLAGS_VERSION);
|
||
|
||
//
|
||
// Bad major
|
||
//
|
||
if (Info->Major != Info->NtFrsMajor) {
|
||
DPRINT2(4,"NTFRSAPI major rev mismatch (dll=%d), (svc=%d)\n",
|
||
Info->Major, Info->NtFrsMajor);
|
||
WStatus = FRS_ERR_INVALID_SERVICE_PARAMETER;
|
||
goto CLEANUP;
|
||
}
|
||
//
|
||
// Bad minor -- put a message in the debug log.
|
||
//
|
||
if (Info->Minor != Info->NtFrsMinor) {
|
||
DPRINT2(4,"NTFRSAPI minor rev mismatch (dll=%d), (svc=%d)\n",
|
||
Info->Minor, Info->NtFrsMinor);
|
||
}
|
||
|
||
//
|
||
// Not large enough to verify internal consistency (or return any data).
|
||
//
|
||
if (Info->SizeInChars < sizeof(NTFRSAPI_INFO)) {
|
||
WStatus = FRS_ERR_INVALID_SERVICE_PARAMETER;
|
||
goto CLEANUP;
|
||
}
|
||
|
||
//
|
||
// Buffer full; done
|
||
//
|
||
if (FlagOn(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
goto CLEANUP;
|
||
}
|
||
|
||
//
|
||
// Verify internal offsets
|
||
//
|
||
// make this into a subroutine (table driven?)
|
||
//
|
||
EoB = Blob + BlobSize;
|
||
EoI = ((PBYTE)Info) + (Info->SizeInChars);
|
||
BoL = (PBYTE)(((PCHAR)Info) + Info->OffsetToLines);
|
||
BoF = (PBYTE)(((PCHAR)Info) + Info->OffsetToFree);
|
||
if (EoI > EoB ||
|
||
BoL > EoB ||
|
||
BoF > EoB ||
|
||
EoI < Blob ||
|
||
BoL < Blob ||
|
||
BoF < Blob) {
|
||
WStatus = FRS_ERR_INVALID_SERVICE_PARAMETER;
|
||
goto CLEANUP;
|
||
}
|
||
|
||
//
|
||
// No free space in buffer; done
|
||
//
|
||
if (BoF == EoB) {
|
||
SetFlag(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL);
|
||
goto CLEANUP;
|
||
}
|
||
|
||
CLEANUP:
|
||
return WStatus;
|
||
}
|
||
|
||
|
||
|
||
PVOID
|
||
InfoFrsInfoContextFree(
|
||
PFRS_INFO_CONTEXT FrsInfoContext
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Frees the FRS_INFO_CONTEXT structure.
|
||
|
||
Arguments:
|
||
FrsInfoContext - Context to free.
|
||
|
||
Return Value:
|
||
None
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "InfoFrsInfoContextFree:"
|
||
|
||
if (FrsInfoContext == NULL) {
|
||
return NULL;
|
||
}
|
||
|
||
FrsFree(FrsInfoContext->KeyValue);
|
||
FrsFree(FrsInfoContext->TableName);
|
||
FrsFree(FrsInfoContext);
|
||
|
||
return NULL;
|
||
}
|
||
|
||
|
||
DWORD
|
||
Info(
|
||
IN ULONG BlobSize,
|
||
IN OUT PBYTE Blob
|
||
)
|
||
/*++
|
||
Routine Description:
|
||
Return internal info (see private\net\inc\ntfrsapi.h).
|
||
|
||
Arguments:
|
||
BlobSize - total bytes of Blob
|
||
Blob - details desired info and provides buffer for info
|
||
|
||
Return Value:
|
||
Win32 Status
|
||
--*/
|
||
{
|
||
#undef DEBSUB
|
||
#define DEBSUB "Info:"
|
||
|
||
FILETIME Now;
|
||
ULARGE_INTEGER ULNow;
|
||
ULARGE_INTEGER ULLastAccessTime;
|
||
ULARGE_INTEGER TimeSinceLastAccess;
|
||
|
||
DWORD WStatus;
|
||
ULONG i;
|
||
ULONG ProductType;
|
||
ULONG Arch;
|
||
BOOL HaveLock = FALSE;
|
||
|
||
PNTFRSAPI_INFO Info = (PNTFRSAPI_INFO)Blob;
|
||
PFRS_INFO_CONTEXT FrsInfoContext = NULL;
|
||
PVOID Key;
|
||
CHAR TimeString[TIME_STRING_LENGTH];
|
||
|
||
|
||
try {
|
||
//
|
||
// Verify the blob
|
||
//
|
||
WStatus = InfoVerify(BlobSize, Blob);
|
||
if (!WIN_SUCCESS(WStatus)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// The table is initialized in the startup code.
|
||
// If it not yet initialized then return an error.
|
||
//
|
||
if (FrsInfoContextTable == NULL) {
|
||
WStatus = ERROR_RETRY;
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// Free table of all contexts that have not been accessed in
|
||
// the last 1 hour.
|
||
//
|
||
GTabLockTable(FrsInfoContextTable);
|
||
|
||
//
|
||
// Get the current time AFTER we grab the table lock.
|
||
// If we got the time before grabbing the lock, then this thread
|
||
// could be context switched out and another could come and grab the
|
||
// lock. The time stamp on that thread's InfoContext would be later
|
||
// than this thread's ULNow. That would cause the calculation below
|
||
// to screw up since it assumes ULLastAccessTime < ULNow.
|
||
//
|
||
GetSystemTimeAsFileTime((PFILETIME) &ULNow);
|
||
|
||
HaveLock = TRUE;
|
||
Key = NULL;
|
||
|
||
while ((FrsInfoContext = GTabNextDatumNoLock(FrsInfoContextTable, &Key)) != NULL) {
|
||
|
||
FileTimeToString(&FrsInfoContext->LastAccessTime, TimeString);
|
||
|
||
COPY_TIME(&ULLastAccessTime, &FrsInfoContext->LastAccessTime);
|
||
|
||
//
|
||
// This is an unsigned value and thus the calculation assumes
|
||
// that ULLastAccessTime is less than ULNow.
|
||
//
|
||
FRS_ASSERT(ULLastAccessTime.QuadPart <= ULNow.QuadPart);
|
||
TimeSinceLastAccess.QuadPart =
|
||
(ULNow.QuadPart - ULLastAccessTime.QuadPart) / (CONVERT_FILETIME_TO_MINUTES);
|
||
|
||
if (TimeSinceLastAccess.QuadPart > 60) {
|
||
GTabDeleteNoLock(FrsInfoContextTable,
|
||
&FrsInfoContext->ContextIndex,
|
||
NULL,
|
||
InfoFrsInfoContextFree);
|
||
//
|
||
// Reset table enum scan.
|
||
//
|
||
Key = NULL;
|
||
}
|
||
}
|
||
|
||
//
|
||
// The caller's context handle was returned in Info->TotalChars
|
||
// when the first call was made. Use it to find the context.
|
||
//
|
||
FrsInfoContext = GTabLookupNoLock(FrsInfoContextTable, &Info->TotalChars, NULL);
|
||
|
||
if (FrsInfoContext == NULL) {
|
||
//
|
||
// Check for DOS attack.
|
||
//
|
||
if (GTabNumberInTable(FrsInfoContextTable) >= FRS_INFO_MAX_CONTEXT_ACTIVE) {
|
||
HaveLock = FALSE;
|
||
GTabUnLockTable(FrsInfoContextTable);
|
||
goto cleanup;
|
||
}
|
||
|
||
//
|
||
// First call. Initialize a new context.
|
||
//
|
||
FrsInfoContext = FrsAlloc(sizeof(FRS_INFO_CONTEXT));
|
||
FrsInfoContext->ContextIndex = InterlockedIncrement(&FrsInfoContextNum);
|
||
DPRINT1(4,"Creating new ContextIndex = %d\n", FrsInfoContext->ContextIndex);
|
||
FrsInfoContext->ReplicaNumber = 0;
|
||
FrsInfoContext->ScanDirection = 1;
|
||
FrsInfoContext->SaveTotalChars = 0;
|
||
FrsInfoContext->KeyValue = NULL;
|
||
FrsInfoContext->TableName = NULL;
|
||
FrsInfoContext->TableType = TABLE_TYPE_INVALID;
|
||
|
||
GTabInsertEntryNoLock(FrsInfoContextTable,
|
||
FrsInfoContext,
|
||
&FrsInfoContext->ContextIndex,
|
||
NULL);
|
||
|
||
} else {
|
||
DPRINT1(4,"Using existing contextIndex = %d\n", FrsInfoContext->ContextIndex);
|
||
}
|
||
|
||
//
|
||
// Update LastAccessTime and pickup CharsToSkip. We now use the TotalChars field
|
||
// in the NTFRSAPI_INFO structure to store the ContextIndex so we save the
|
||
// TotalChars in the new FRS_INFO_CONTEXT structure and pick it up here.
|
||
//
|
||
if (FrsInfoContext->KeyValue != NULL) {
|
||
//
|
||
// If we are planning to scan to a specific record in the table then
|
||
// no need to skip characters.
|
||
//
|
||
Info->CharsToSkip = 0;
|
||
} else {
|
||
//
|
||
// Otherwise restore value from save area in caller's context
|
||
// since value passed in by caller is likely bogus.
|
||
//
|
||
Info->CharsToSkip = FrsInfoContext->SaveTotalChars;
|
||
}
|
||
|
||
//
|
||
// Restore TotalChars from save area in FrsInfoContext.
|
||
//
|
||
Info->TotalChars = FrsInfoContext->SaveTotalChars;
|
||
|
||
GetSystemTimeAsFileTime(&FrsInfoContext->LastAccessTime);
|
||
|
||
HaveLock = FALSE;
|
||
GTabUnLockTable(FrsInfoContextTable);
|
||
|
||
//
|
||
// Full buffer; done
|
||
//
|
||
if (FlagOn(Info->Flags, NTFRSAPI_INFO_FLAGS_FULL)) {
|
||
goto cleanup;
|
||
}
|
||
|
||
if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_VERSION) {
|
||
IPRINT0(Info, "NtFrs Version Information\n");
|
||
IPRINT1(Info, " NtFrs Major : %d\n", NtFrsMajor);
|
||
IPRINT1(Info, " NtFrs Minor : %d\n", NtFrsMinor);
|
||
// IPRINT1(Info, " NtFrs Module : %s\n", NtFrsModule);
|
||
IPRINT2(Info, " NtFrs Compiled on : %s %s\n", NtFrsDate, NtFrsTime);
|
||
#if NTFRS_TEST
|
||
IPRINT0(Info, " NTFRS_TEST Enabled\n");
|
||
#endif NTFRS_TEST
|
||
|
||
i = 0;
|
||
while (LatestChanges[i] != NULL) {
|
||
IPRINT1(Info, " %s\n", LatestChanges[i]);
|
||
i++;
|
||
}
|
||
|
||
|
||
IPRINT4(Info, "OS Version %d.%d (%d) - %w\n",
|
||
OsInfo.dwMajorVersion, OsInfo.dwMinorVersion,
|
||
OsInfo.dwBuildNumber, OsInfo.szCSDVersion);
|
||
|
||
ProductType = (ULONG) OsInfo.wProductType;
|
||
IPRINT4(Info, "SP (%hd.%hd) SM: 0x%04hx PT: 0x%02x\n",
|
||
OsInfo.wServicePackMajor, OsInfo.wServicePackMinor,
|
||
OsInfo.wSuiteMask, ProductType);
|
||
|
||
Arch = SystemInfo.wProcessorArchitecture;
|
||
if (Arch >= ARRAY_SZ(ProcessorArchName)) {
|
||
Arch = ARRAY_SZ(ProcessorArchName)-1;
|
||
}
|
||
|
||
IPRINT5(Info, "Processor: %s Level: 0x%04hx Revision: 0x%04hx Processor num/mask: %d/%08x\n",
|
||
ProcessorArchName[Arch], SystemInfo.wProcessorLevel,
|
||
SystemInfo.wProcessorRevision, SystemInfo.dwNumberOfProcessors,
|
||
SystemInfo.dwActiveProcessorMask);
|
||
|
||
goto cleanup;
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_SETS) {
|
||
WStatus = InfoPrintDbSets(Info, 0);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_DS) {
|
||
WStatus = InfoPrintDs(Info, 0);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_MEMORY) {
|
||
WStatus = InfoPrintMemory(Info, 0);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_IDTABLE) {
|
||
WStatus = InfoPrintTables(Info,
|
||
FrsInfoContext,
|
||
L"ID TABLES",
|
||
IDTablex,
|
||
GuidIndexx,
|
||
InfoIDTableWorker);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_INLOG) {
|
||
WStatus = InfoPrintTables(Info,
|
||
FrsInfoContext,
|
||
L"INLOG TABLES",
|
||
INLOGTablex,
|
||
ILSequenceNumberIndexx,
|
||
InfoInLogTableWorker);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_OUTLOG) {
|
||
WStatus = InfoPrintTables(Info,
|
||
FrsInfoContext,
|
||
L"OUTLOG TABLES",
|
||
OUTLOGTablex,
|
||
OLSequenceNumberIndexx,
|
||
InfoOutLogTableWorker);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_CONFIGTABLE) {
|
||
WStatus = InfoPrintTables(Info,
|
||
FrsInfoContext,
|
||
L"CONFIG TABLE",
|
||
ConfigTablex,
|
||
ReplicaNumberIndexx,
|
||
InfoConfigTableWorker);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_THREADS) {
|
||
WStatus = InfoPrintThreads(Info, 0);
|
||
|
||
} else if (Info->TypeOfInfo == NTFRSAPI_INFO_TYPE_STAGE) {
|
||
WStatus = InfoPrintStage(Info, 0);
|
||
|
||
} else {
|
||
IPRINT1(Info, "NtFrs Doesn't understand TypeOfInfo %d\n", Info->TypeOfInfo);
|
||
WStatus = ERROR_INVALID_PARAMETER;
|
||
}
|
||
|
||
cleanup:;
|
||
|
||
} except (EXCEPTION_EXECUTE_HANDLER) {
|
||
|
||
GET_EXCEPTION_CODE(WStatus);
|
||
//
|
||
// Make sure we drop the lock so we don't hang all other callers.
|
||
//
|
||
if (HaveLock) {
|
||
GTabUnLockTable(FrsInfoContextTable);
|
||
}
|
||
}
|
||
|
||
//
|
||
// Save the value of TotalChars in the context strut for this caller.
|
||
// Return the ContextIndex (handle) to the caller for use in subsequent calls.
|
||
//
|
||
if (FrsInfoContext != NULL) {
|
||
FrsInfoContext->SaveTotalChars = Info->TotalChars;
|
||
Info->TotalChars = FrsInfoContext->ContextIndex;
|
||
}
|
||
|
||
return WStatus;
|
||
}
|
||
|
||
|
||
|