Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

3238 lines
93 KiB

/*++
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;
}