Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1056 lines
28 KiB

/*++
Copyright (c) 1987-1996 Microsoft Corporation
Module Name:
announce.c
Abstract:
Routines to handle ssi announcements.
Author:
Ported from Lan Man 2.0
Environment:
User mode only.
Contains NT-specific code.
Requires ANSI C extensions: slash-slash comments, long external names.
Revision History:
21-May-1991 (cliffv)
Ported to NT. Converted to NT style.
02-Jan-1992 (madana)
added support for builtin/multidomain replication.
--*/
//
// Common include files.
//
#include "logonsrv.h" // Include files common to entire service
#pragma hdrstop
//
// Include files specific to this .c file
//
//
// Maximum number of pulses that we allow a BDC to ignore before ignoring it.
//
#define MAX_PULSE_TIMEOUT 3
VOID
NlRemovePendingBdc(
IN PSERVER_SESSION ServerSession
)
/*++
Routine Description:
Remove the specified Server Session from the list of pending BDCs.
Enter with the ServerSessionTable Sem locked
Arguments:
ServerSession -- Pointer to the server session structure to remove from the
list.
Return Value:
None.
--*/
{
//
// Ensure the server session is really on the list.
//
if ( (ServerSession->SsFlags & SS_PENDING_BDC) == 0 ) {
return;
}
//
// Decrement the count of pending BDCs
//
NlAssert( NlGlobalPendingBdcCount > 0 );
NlGlobalPendingBdcCount --;
//
// If this is the last BDC in the pending list,
// turn off the timer.
//
if ( NlGlobalPendingBdcCount == 0 ) {
NlGlobalPendingBdcTimer.Period = (DWORD) MAILSLOT_WAIT_FOREVER;
}
//
// Remove the pending BDC from the list of pending BDCs.
//
RemoveEntryList( &ServerSession->SsPendingBdcList );
//
// Turn off the flag indicating we're in the list.
//
ServerSession->SsFlags &= ~SS_PENDING_BDC;
NlPrint((NL_PULSE_MORE,
"NlRemovePendingBdc: %s: Removed from pending list. Count: %ld\n",
ServerSession->SsComputerName,
NlGlobalPendingBdcCount ));
}
VOID
NlAddPendingBdc(
IN PSERVER_SESSION ServerSession
)
/*++
Routine Description:
Add the specified Server Session to the list of pending BDCs.
Enter with the ServerSessionTable Sem locked
Arguments:
ServerSession -- Pointer to the server session structure to add to the
list.
Return Value:
None.
--*/
{
//
// Ensure the server session is really off the list.
//
if ( ServerSession->SsFlags & SS_PENDING_BDC ) {
return;
}
//
// If this is the first pending BDC,
// start the timer.
//
if ( NlGlobalPendingBdcCount == 0 ) {
// Run the timer at twice the frequency of the timeout to ensure that
// entries don't have to wait nearly twice the timeout period before
// they expire.
NlGlobalPendingBdcTimer.Period = NlGlobalParameters.PulseTimeout1 * 500;
NlQuerySystemTime( &NlGlobalPendingBdcTimer.StartTime );
//
// Tell the main thread that I've changed a timer.
//
if ( !SetEvent( NlGlobalTimerEvent ) ) {
NlPrint(( NL_CRITICAL,
"NlAddPendingBdc: %s: SetEvent2 failed %ld\n",
ServerSession->SsComputerName,
GetLastError() ));
}
}
//
// Increment the count of pending BDCs
//
NlGlobalPendingBdcCount ++;
//
// Add the pending BDC to the list of pending BDCs.
//
InsertTailList( &NlGlobalPendingBdcList, &ServerSession->SsPendingBdcList );
//
// Turn on the flag indicating we're in the list.
//
ServerSession->SsFlags |= SS_PENDING_BDC;
NlPrint((NL_PULSE_MORE,
"NlAddPendingBdc: %s: Added to pending list. Count: %ld\n",
ServerSession->SsComputerName,
NlGlobalPendingBdcCount ));
}
VOID
NetpLogonPutDBInfo(
IN PDB_CHANGE_INFO DBInfo,
IN OUT PCHAR * Where
)
/*++
Routine Description:
Put Database info structure in mailslot buffer.
Arguments:
DBInfo : database info structure pointer.
Where : indirect pointer to mailslot buffer. Database info
is copied over here. When returned this pointer is
updated to point the end of mailslot buffer.
Return Value:
None.
--*/
{
NetpLogonPutBytes( &DBInfo->DBIndex, sizeof(DBInfo->DBIndex), Where);
NetpLogonPutBytes( &(DBInfo->LargeSerialNumber),
sizeof(DBInfo->LargeSerialNumber),
Where);
NetpLogonPutBytes( &(DBInfo->NtDateAndTime),
sizeof(DBInfo->NtDateAndTime),
Where);
}
VOID
NetpLogonUpdateDBInfo(
IN PLARGE_INTEGER SerialNumber,
IN OUT PCHAR * Where
)
/*++
Routine Description:
Update the Serial Number within the already packed mailslot buffer.
Arguments:
SerialNumber: New SerialNumber.
Where : indirect pointer to mailslot buffer. Database info
is copied over here. When returned this pointer is
updated to point the end of mailslot buffer.
Return Value:
None.
--*/
{
*Where += sizeof(DWORD);
NetpLogonPutBytes( SerialNumber, sizeof(LARGE_INTEGER), Where);
*Where += sizeof(LARGE_INTEGER);
}
BOOLEAN
NlAllocatePrimaryAnnouncement(
OUT PNETLOGON_DB_CHANGE *UasChangeBuffer,
OUT LPDWORD UasChangeSize,
OUT PCHAR *DbChangeInfoPointer
)
/*++
Routine Description:
Build and allocate an UAS_CHANGE message which describes the latest
account database changes.
Arguments:
UasChangeBuffer - Returns a pointer to the buffer containing the message.
The caller is responsible for freeing the buffer using NetpMemoryFree.
UasChangeSize - Returns the size (in bytes) of the allocated buffer.
DbChangeInfoPointer - Returns the address of the DB Change info
within the allocated buffer. The field is not properly aligned.
Return Value:
TRUE - iff the buffer could be successfully allocated.
--*/
{
PNETLOGON_DB_CHANGE UasChange;
DB_CHANGE_INFO DBChangeInfo;
ULONG DateAndTime1970;
DWORD NumDBs;
PCHAR Where;
DWORD i;
DWORD DomainSidSize;
//
// allocate space for this message.
//
DomainSidSize = RtlLengthSid( NlGlobalDomainInfo->DomAccountDomainId );
UasChange = NetpMemoryAllocate(
sizeof(NETLOGON_DB_CHANGE)+
(NUM_DBS - 1) * sizeof(DB_CHANGE_INFO) +
(DomainSidSize - 1) +
sizeof(DWORD) // for DWORD alignment of SID
);
if( UasChange == NULL ) {
NlPrint((NL_CRITICAL, "NlAllocatePrimaryAnnouncement can't allocate memory\n" ));
return FALSE;
}
//
// Build the UasChange message using the latest domain modified
// information from SAM.
//
UasChange->Opcode = LOGON_UAS_CHANGE;
LOCK_CHANGELOG();
SmbPutUlong( &UasChange->LowSerialNumber,
NlGlobalChangeLogDesc.SerialNumber[SAM_DB].LowPart);
if (!RtlTimeToSecondsSince1970( &NlGlobalDBInfoArray[SAM_DB].CreationTime,
&DateAndTime1970 )) {
NlPrint((NL_CRITICAL, "DomainCreationTime can't be converted\n" ));
DateAndTime1970 = 0;
}
SmbPutUlong( &UasChange->DateAndTime, DateAndTime1970 );
// Tell the BDC we only intend to send pulses infrequently
SmbPutUlong( &UasChange->Pulse, NlGlobalParameters.PulseMaximum);
// Caller will change this field as appropriate
SmbPutUlong( &UasChange->Random, 0 );
Where = UasChange->PrimaryDCName;
NetpLogonPutOemString( NlGlobalDomainInfo->DomOemComputerName,
sizeof(UasChange->PrimaryDCName),
&Where );
NetpLogonPutOemString( NlGlobalDomainInfo->DomOemDomainName,
sizeof(UasChange->DomainName),
&Where );
//
// builtin domain support
//
NetpLogonPutUnicodeString( NlGlobalDomainInfo->DomUnicodeComputerNameString.Buffer,
sizeof(UasChange->UnicodePrimaryDCName),
&Where );
NetpLogonPutUnicodeString( NlGlobalDomainInfo->DomUnicodeDomainName,
sizeof(UasChange->UnicodeDomainName),
&Where );
//
// number of database info that follow
//
NumDBs = NUM_DBS;
NetpLogonPutBytes( &NumDBs, sizeof(NumDBs), &Where );
*DbChangeInfoPointer = Where;
for( i = 0; i < NUM_DBS; i++) {
DBChangeInfo.DBIndex =
NlGlobalDBInfoArray[i].DBIndex;
DBChangeInfo.LargeSerialNumber =
NlGlobalChangeLogDesc.SerialNumber[i];
DBChangeInfo.NtDateAndTime =
NlGlobalDBInfoArray[i].CreationTime;
NetpLogonPutDBInfo( &DBChangeInfo, &Where );
}
//
// place domain SID in the message.
//
NetpLogonPutBytes( &DomainSidSize, sizeof(DomainSidSize), &Where );
NetpLogonPutDomainSID( NlGlobalDomainInfo->DomAccountDomainId, DomainSidSize, &Where );
NetpLogonPutNtToken( &Where, 0 );
UNLOCK_CHANGELOG();
*UasChangeSize = (DWORD)(Where - (PCHAR)UasChange);
*UasChangeBuffer = UasChange;
return TRUE;
}
VOID
NlPrimaryAnnouncementFinish(
IN PSERVER_SESSION ServerSession,
IN DWORD DatabaseId,
IN PLARGE_INTEGER SerialNumber
)
/*++
Routine Description:
Indicate that the specified BDC has completed replication of the specified
database.
Note: this BDC might not be on the pending list at at if it was doing the
replication on its own accord. This routine is designed to handle that
eventuality.
Arguments:
ServerSession -- Pointer to the server session structure to remove from the
list.
DatabaseID -- Database ID of the database
SerialNumber -- SerialNumber of the latest delta returned to the BDC.
NULL indicates a full sync just completed
Return Value:
None.
--*/
{
BOOLEAN SendPulse = FALSE;
//
// Mark the session that the replication of this particular database
// has finished.
//
LOCK_SERVER_SESSION_TABLE( ServerSession->SsDomainInfo );
ServerSession->SsFlags &= ~NlGlobalDBInfoArray[DatabaseId].DBSessionFlag;
//
// If all of the databases are now replicated, OR
// the BDC just finished a full sync on one or more of its database,
// remove this BDC from the pending list.
//
if ( (ServerSession->SsFlags & SS_REPL_MASK) == 0 || SerialNumber == NULL ) {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncementFinish: %s: all databases are now in sync on BDC\n",
ServerSession->SsComputerName ));
NlRemovePendingBdc( ServerSession );
SendPulse = TRUE;
}
//
// If a full sync just completed,
// force a partial sync so we can update our serial numbers.
//
if ( SerialNumber == NULL ) {
ServerSession->SsBdcDbSerialNumber[DatabaseId].QuadPart = 0;
ServerSession->SsFlags |= NlGlobalDBInfoArray[DatabaseId].DBSessionFlag;
//
// Save the current serial number for this BDC.
//
} else {
ServerSession->SsBdcDbSerialNumber[DatabaseId] = *SerialNumber;
}
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncementFinish: %s: " FORMAT_LPWSTR " SerialNumber: %lx %lx\n",
ServerSession->SsComputerName,
NlGlobalDBInfoArray[DatabaseId].DBName,
ServerSession->SsBdcDbSerialNumber[DatabaseId].HighPart,
ServerSession->SsBdcDbSerialNumber[DatabaseId].LowPart ));
UNLOCK_SERVER_SESSION_TABLE( ServerSession->SsDomainInfo );
//
// If this BDC is finished,
// try to send a pulse to more BDCs.
//
if ( SendPulse ) {
NlPrimaryAnnouncement( ANNOUNCE_CONTINUE );
}
}
VOID
NlPrimaryAnnouncementTimeout(
VOID
)
/*++
Routine Description:
The primary announcement timer has expired.
Handle timing out any BDC's that haven't responded yet.
Arguments:
None.
Return Value:
None.
--*/
{
LARGE_INTEGER TimeNow;
BOOLEAN SendPulse = FALSE;
PLIST_ENTRY ListEntry;
PSERVER_SESSION ServerSession;
//
// Get the current time of day
//
NlQuerySystemTime( &TimeNow );
//
// Handle each BDC that has a pulse pending.
//
LOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
for ( ListEntry = NlGlobalPendingBdcList.Flink ;
ListEntry != &NlGlobalPendingBdcList ;
ListEntry = ListEntry->Flink) {
ServerSession = CONTAINING_RECORD( ListEntry, SERVER_SESSION, SsPendingBdcList );
//
// Ignore entries that haven't timed out yet.
//
if ( ServerSession->SsLastPulseTime.QuadPart +
NlGlobalParameters.PulseTimeout1_100ns.QuadPart >
TimeNow.QuadPart ) {
continue;
}
//
// If the pulse has been sent and there has been no response at all,
// OR there hasn't been another response in a VERY long time
// time this entry out.
//
if ( (ServerSession->SsFlags & SS_PULSE_SENT) ||
(ServerSession->SsLastPulseTime.QuadPart +
NlGlobalParameters.PulseTimeout2_100ns.QuadPart <=
TimeNow.QuadPart) ) {
//
// Increment the count of times this BDC has timed out.
//
if ( ServerSession->SsPulseTimeoutCount < MAX_PULSE_TIMEOUT ) {
ServerSession->SsPulseTimeoutCount++;
}
//
// Remove this entry from the queue.
//
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncementTimeout: %s: BDC didn't respond to pulse.\n",
ServerSession->SsComputerName ));
NlRemovePendingBdc( ServerSession );
//
// Indicate we should send more pulses
//
SendPulse = TRUE;
}
}
UNLOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
//
// If any entry has timed out,
// try to send a pulse to more BDCs.
//
if ( SendPulse ) {
NlPrimaryAnnouncement( ANNOUNCE_CONTINUE );
}
}
VOID
NlPrimaryAnnouncement(
IN DWORD AnnounceFlags
)
/*++
Routine Description:
Periodic broadcast of messages to domain containing latest
account database changes.
Arguments:
AnnounceFlags - Flags requesting special handling of the announcement.
ANNOUNCE_FORCE -- set to indicate that the pulse should be forced to
all BDCs in the domain.
ANNOUNCE_CONTINUE -- set to indicate that this call is a
continuation of a previous call to process all entries.
ANNOUNCE_IMMEDIATE -- set to indicate that this call is a result
of a request for immediate replication
Return Value:
None.
--*/
{
NTSTATUS Status;
PNETLOGON_DB_CHANGE UasChange;
DWORD UasChangeSize;
PCHAR DbChangeInfoPointer;
LARGE_INTEGER TimeNow;
DWORD SessionFlags;
PSERVER_SESSION ServerSession;
PLIST_ENTRY ListEntry;
static ULONG EntriesHandled = 0;
static BOOLEAN ImmediateAnnouncement;
NlPrint((NL_PULSE_MORE, "NlPrimaryAnnouncement: Entered %ld\n", AnnounceFlags ));
//
// If the DS is recovering from a backup,
// avoid announcing that we're available.
//
if ( NlGlobalDsPaused ) {
NlPrint((NL_PULSE_MORE, "NlPrimaryAnnouncement: Ds is paused\n" ));
return;
}
//
// Allocate the UAS_CHANGE message to send.
//
if ( !NlAllocatePrimaryAnnouncement( &UasChange,
&UasChangeSize,
&DbChangeInfoPointer ) ) {
return;
}
//
// If we need to force the pulse to all the BDCs,
// mark that we've not done any entries yet, and
// mark each entry that a pulse is needed.
//
LOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
if ( AnnounceFlags & ANNOUNCE_FORCE ) {
EntriesHandled = 0;
for ( ListEntry = NlGlobalBdcServerSessionList.Flink ;
ListEntry != &NlGlobalBdcServerSessionList ;
ListEntry = ListEntry->Flink) {
ServerSession = CONTAINING_RECORD( ListEntry, SERVER_SESSION, SsBdcList );
ServerSession->SsFlags |= SS_FORCE_PULSE;
}
}
//
// If this isn't a continuation of a previous request to send out pulses,
// Reset the count of BDCs that have been handled.
//
if ( (AnnounceFlags & ANNOUNCE_CONTINUE) == 0 ) {
EntriesHandled = 0;
//
// Remember whether this was an immediate announcement for the
// initial call and all of the continuations.
//
ImmediateAnnouncement = (AnnounceFlags & ANNOUNCE_IMMEDIATE) != 0;
}
//
// Loop sending announcements until
// we have the maximum number of announcements outstanding, OR
// we've processed all the entries in the list.
//
while ( NlGlobalPendingBdcCount < NlGlobalParameters.PulseConcurrency &&
EntriesHandled < NlGlobalBdcServerSessionCount ) {
BOOLEAN SendPulse;
LPWSTR TransportName;
DWORD MaxSessionFlags;
//
// If netlogon is exitting,
// stop sending announcements
//
if ( NlGlobalTerminate ) {
break;
}
//
// Get the server session entry for the next BDC in the list.
//
// The BDC Server Session list is maintained in the order pulses should
// be sent. As a pulse is sent (or is skipped), the entry is placed
// at the tail of the list. This gives each BDC a chance at a pulse
// before any BDC is repeated.
ListEntry = NlGlobalBdcServerSessionList.Flink ;
ServerSession = CONTAINING_RECORD( ListEntry, SERVER_SESSION, SsBdcList );
SendPulse = FALSE;
SessionFlags = 0;
// Only replicate those databases that negotiation says to replicate
MaxSessionFlags = NlMaxReplMask(ServerSession->SsNegotiatedFlags);
if ( ServerSession->SsTransport == NULL ) {
TransportName = NULL;
} else {
TransportName = ServerSession->SsTransport->TransportName;
}
//
// Determine if we should send an announcement to this BDC.
//
// Send a pluse unconditionally if a pulse is being forced.
//
if ( ServerSession->SsFlags & SS_FORCE_PULSE ) {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncement: %s: pulse forced to be sent\n",
ServerSession->SsComputerName ));
SendPulse = TRUE;
ServerSession->SsFlags &= ~SS_FORCE_PULSE;
SessionFlags = MaxSessionFlags;
TransportName = NULL; // Send on all transports
//
// Only send to any other BDC if there isn't a pulse outstanding and
// previous announcements haven't been ignored.
//
} else if ( (ServerSession->SsFlags & SS_PENDING_BDC) == 0 &&
ServerSession->SsPulseTimeoutCount < MAX_PULSE_TIMEOUT ) {
ULONG i;
SessionFlags = 0;
//
// Only send an announcement if at least one database is out
// of sync.
//
for( i = 0; i < NUM_DBS; i++) {
//
// If this BDC isn't interested in this database,
// just skip it.
//
if ( (NlGlobalDBInfoArray[i].DBSessionFlag & MaxSessionFlags) == 0 ) {
continue;
}
//
// If we need to know the serial number of the BDC,
// force the replication.
//
if ( ServerSession->SsFlags &
NlGlobalDBInfoArray[i].DBSessionFlag ) {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncement: %s: " FORMAT_LPWSTR " database serial number needed. Pulse sent.\n",
ServerSession->SsComputerName,
NlGlobalDBInfoArray[i].DBName ));
SendPulse = TRUE;
SessionFlags |= NlGlobalDBInfoArray[i].DBSessionFlag;
//
// If the BDC is out of sync with us,
// tell it.
//
} else if ( NlGlobalChangeLogDesc.SerialNumber[i].QuadPart >
ServerSession->SsBdcDbSerialNumber[i].QuadPart ) {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncement: %s: " FORMAT_LPWSTR " database is out of sync. Pulse sent.\n",
ServerSession->SsComputerName,
NlGlobalDBInfoArray[i].DBName ));
SendPulse = TRUE;
SessionFlags |= NlGlobalDBInfoArray[i].DBSessionFlag;
}
}
//
// Fix a timing window on NT 3.1 BDCs.
//
// During promotion of a BDC to PDC, the following events occur:
// Two server accounts are changed on the old PDC and
// are marked for immediate replication.
// The Server manager asks the new PDC to partial sync.
//
// If the first immediate replication starts immediately, and the
// second immediate replication pulse is ignored because replication
// is in progress, and the first replication has finished the SAM
// database and is working on the LSA database when the server
// manager partial sync request comes in, then that request will be
// ignored (rightfully) since replication is still in progress.
// However, an NT 3.1 BDC replicator thread will not go back to
// do the SAM database once it finishes with the LSA database. So
// the replicator thread terminates with the SAM database still
// needing replication. The server manager (rightfully) interprets
// this as an error.
//
// Our solution is to set the backoff period on such "immediate"
// replication attempts to the same value an NT 3.1 PDC would use.
// This typically prevents the initial replication from starting in
// the first place.
//
// Only do it for NT 3.1 BDCs since we're risking being overloaded.
//
if ( ImmediateAnnouncement &&
SendPulse &&
(ServerSession->SsFlags & SS_AUTHENTICATED) &&
(ServerSession->SsNegotiatedFlags & NETLOGON_SUPPORTS_PERSISTENT_BDC) == 0 ) {
SessionFlags = 0;
}
}
//
// Send a pulse unconditionally if it has been PulseMaximum since the
// latest pulse.
//
// Avoid this if the BDC is an NT 5 BDC that doesn't use Netlogon to replicate
// from us.
//
if ( !SendPulse &&
MaxSessionFlags != 0 &&
NlTimeHasElapsedEx( &ServerSession->SsLastPulseTime,
&NlGlobalParameters.PulseMaximum_100ns,
NULL ) ) {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncement: %s: Maximum pulse since previous pulse. Pulse sent.\n",
ServerSession->SsComputerName ));
SendPulse = TRUE;
SessionFlags = 0;
TransportName = NULL; // Send on all transports
}
//
// Put this entry at the tail of the list regardless of whether
// we'll actually send an announcement to it.
//
RemoveEntryList( ListEntry );
InsertTailList( &NlGlobalBdcServerSessionList, ListEntry );
EntriesHandled ++;
//
// Send the announcement.
//
if ( SendPulse ) {
WCHAR LocalComputerName[CNLEN+1];
PCHAR Where;
ULONG i;
//
// Add this BDC to the list of BDCs that have a pulse pending.
//
// Don't add this BDC to the list if we don't expect a response.
// We don't expect a response from an LM BDC. We don't expect
// a response from a BDC that is merely getting its PulseMaximum
// pulse.
//
// If we don't expect a response, set the backoff period to a
// large value to prevent a large load on the PDC in the case
// that the BDC does actually respond.
//
// If we expect a response, set the backoff period to almost
// immediately since we're waiting for them.
//
if ( SessionFlags == 0 ) {
SmbPutUlong( &UasChange->Random,
max(NlGlobalParameters.Randomize,
NlGlobalParameters.Pulse/10) );
} else {
NlAddPendingBdc( ServerSession );
SmbPutUlong( &UasChange->Random, NlGlobalParameters.Randomize );
}
//
// Indicate that the pulse has been sent.
// This flag is set from the time we send the pulse until the
// first time the BDC responds. We use this to detect failed
// BDCs that have a secure channel up.
//
ServerSession->SsFlags &= ~SS_REPL_MASK;
ServerSession->SsFlags |= SS_PULSE_SENT | SessionFlags;
NlQuerySystemTime( &TimeNow );
ServerSession->SsLastPulseTime = TimeNow;
//
// Don't keep the server session locked since sending the mailslot
// message takes a long time.
//
NetpCopyStrToWStr( LocalComputerName,
ServerSession->SsComputerName );
UNLOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
//
// Update the message to be specific to this BDC.
//
// If we need the BDC to respond,
// set the serial number to make the BDC think it has a lot
// of deltas to pick up.
//
LOCK_CHANGELOG();
Where = DbChangeInfoPointer;
for( i = 0; i < NUM_DBS; i++) {
LARGE_INTEGER SerialNumber;
SerialNumber = NlGlobalChangeLogDesc.SerialNumber[i];
if ( NlGlobalDBInfoArray[i].DBSessionFlag & SessionFlags ) {
//
// Don't change the high part since
// a) NT 3.1 BDCs will do a full sync if there are too
// many changes.
// b) The high part contains the PDC promotion count.
//
SerialNumber.LowPart = 0xFFFFFFFF;
}
NetpLogonUpdateDBInfo( &SerialNumber, &Where );
}
UNLOCK_CHANGELOG();
//
// Send the datagram to this BDC.
// Failure isn't fatal
//
#if NETLOGONDBG
NlPrintDom((NL_MAILSLOT, NlGlobalDomainInfo,
"Sent '%s' message to %ws[%s] on %ws.\n",
NlMailslotOpcode(UasChange->Opcode),
LocalComputerName,
NlDgrNameType(ComputerName),
TransportName ));
#endif // NETLOGONDBG
Status = NlBrowserSendDatagram(
NlGlobalDomainInfo,
0,
LocalComputerName,
ComputerName,
TransportName,
NETLOGON_LM_MAILSLOT_A,
UasChange,
UasChangeSize,
NULL ); // Don't flush Netbios cache
if ( !NT_SUCCESS(Status) ) {
NlPrint((NL_CRITICAL,
"Cannot send datagram to '%ws' 0x%lx\n",
LocalComputerName,
Status ));
}
LOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
} else {
NlPrint((NL_PULSE_MORE,
"NlPrimaryAnnouncement: %s: pulse not needed at this time.\n",
ServerSession->SsComputerName ));
}
}
UNLOCK_SERVER_SESSION_TABLE( NlGlobalDomainInfo );
//
// Free up message memory.
//
NetpMemoryFree( UasChange );
NlPrint((NL_PULSE_MORE, "NlPrimaryAnnouncement: Return\n" ));
return;
}