/*++ Copyright (c) 1987-2001 Microsoft Corporation Module Name: ftnfoctx.c Abstract: Utility routines to manipulate the forest trust context Author: 27-Jul-00 (cliffv) Environment: User mode only. Contains NT-specific code. Requires ANSI C extensions: slash-slash comments, long external names. Revision History: --*/ #include #include #include #include #include #include #include // ROUND_UP_POINTER #include // MIDL_user_free #include // qsort VOID NetpInitFtinfoContext( OUT PNL_FTINFO_CONTEXT FtinfoContext ) /*++ Routine Description: Routine to initialize the Ftinfo context structure. Arguments: FtinfoContext - Context to initialize Return Value: None --*/ { RtlZeroMemory( FtinfoContext, sizeof(*FtinfoContext) ); InitializeListHead( &FtinfoContext->FtinfoList ); } VOID NetpMarshalFtinfoEntry ( IN PLSA_FOREST_TRUST_RECORD InFtinfoRecord, OUT PLSA_FOREST_TRUST_RECORD OutFtinfoRecord, IN OUT LPBYTE *WherePtr ) /*++ Routine Description: Routine to marshalls a single Ftinfo entry Arguments: InFtinfoRecord - Template to copy into InFtinfoRecord OutFtinfoRecord - Entry to fill in On input, points to a zeroed buffer. WherePtr - On input, specifies where to marshal to. On output, points to the first byte past the marshalled data. Return Value: TRUE - Success FALSE - if no memory can be allocated --*/ { LPBYTE Where = *WherePtr; ULONG Size; ULONG SidSize; ULONG NameSize; NetpAssert( Where == ROUND_UP_POINTER( Where, ALIGN_WORST )); // // Copy the fixed size data // OutFtinfoRecord->ForestTrustType = InFtinfoRecord->ForestTrustType; OutFtinfoRecord->Flags = InFtinfoRecord->Flags; OutFtinfoRecord->Time = InFtinfoRecord->Time; // // Fill in a domain entry // switch( InFtinfoRecord->ForestTrustType ) { case ForestTrustDomainInfo: // // Copy the DWORD aligned data // if ( InFtinfoRecord->ForestTrustData.DomainInfo.Sid != NULL ) { SidSize = RtlLengthSid( InFtinfoRecord->ForestTrustData.DomainInfo.Sid ); OutFtinfoRecord->ForestTrustData.DomainInfo.Sid = (PISID) Where; RtlCopyMemory( Where, InFtinfoRecord->ForestTrustData.DomainInfo.Sid, SidSize ); Where += SidSize; } // // Copy the WCHAR aligned data // NameSize = InFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Length; if ( NameSize != 0 ) { OutFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Buffer = (LPWSTR) Where; OutFtinfoRecord->ForestTrustData.DomainInfo.DnsName.MaximumLength = (USHORT) (NameSize+sizeof(WCHAR)); OutFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Length = (USHORT)NameSize; RtlCopyMemory( Where, InFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Buffer, NameSize ); Where += NameSize; *((LPWSTR)Where) = L'\0'; Where += sizeof(WCHAR); } NameSize = InFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Length; if ( NameSize != 0 ) { OutFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Buffer = (LPWSTR) Where; OutFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.MaximumLength = (USHORT) (NameSize+sizeof(WCHAR)); OutFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Length = (USHORT)NameSize; RtlCopyMemory( Where, InFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Buffer, NameSize ); Where += NameSize; *((LPWSTR)Where) = L'\0'; Where += sizeof(WCHAR); } break; // // Fill in a TLN entry // case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: // // Copy the WCHAR aligned data // NameSize = InFtinfoRecord->ForestTrustData.TopLevelName.Length; if ( NameSize != 0 ) { OutFtinfoRecord->ForestTrustData.TopLevelName.Buffer = (LPWSTR) Where; OutFtinfoRecord->ForestTrustData.TopLevelName.MaximumLength = (USHORT) (NameSize+sizeof(WCHAR)); OutFtinfoRecord->ForestTrustData.TopLevelName.Length = (USHORT)NameSize; RtlCopyMemory( Where, InFtinfoRecord->ForestTrustData.TopLevelName.Buffer, NameSize ); Where += NameSize; *((LPWSTR)Where) = L'\0'; Where += sizeof(WCHAR); } break; default: NetpAssert( FALSE ); } Where = ROUND_UP_POINTER( Where, ALIGN_WORST ); *WherePtr = Where; } VOID NetpCompareHelper ( IN PUNICODE_STRING Name, IN OUT PULONG Index, OUT PUNICODE_STRING CurrentLabel ) /*++ Routine Description: This routine is a helper routine for finding the next rightmost label in a string. Arguments: Name - The input dns name. The dns name should not have a trailing . Index - On input, should contain the value returned by the previous call to this routine. On input for the first call, should be set to Name->Length/sizeof(WCHAR). On output, zero is returned to indicate that this is the last of the name. The caller should not call again. Any other value output is a context for the next call to this routine. CurrentLabel - Returns a descriptor describing the substring which is the next label. Return Value: None. --*/ { ULONG PreviousIndex = *Index; ULONG CurrentIndex = *Index; ULONG LabelIndex; NetpAssert( CurrentIndex != 0 ); // // Find the beginning of the next label // while ( CurrentIndex > 0 ) { CurrentIndex--; if ( Name->Buffer[CurrentIndex] == L'.' ) { break; } } if ( CurrentIndex == 0 ) { LabelIndex = CurrentIndex; } else { LabelIndex = CurrentIndex + 1; } // // Return it to the caller // CurrentLabel->Buffer = &Name->Buffer[LabelIndex]; CurrentLabel->Length = (USHORT)((PreviousIndex - LabelIndex) * sizeof(WCHAR)); CurrentLabel->MaximumLength = CurrentLabel->Length; *Index = CurrentIndex; } int NetpCompareDnsNameWithSortOrder( IN PUNICODE_STRING Name1, IN PUNICODE_STRING Name2 ) /*++ Routine Description: Routine to compare two DNS names. The DNS names must not have a trailing "." Labels are compare right to left to present a pleasent viewing order. Arguments: Name1 - First name to compare. Name2 - Second name to compare. Return Value: Signed value that gives the results of the comparison: Zero - String1 equals String2 < Zero - String1 less than String2 > Zero - String1 greater than String2 --*/ { ULONG Index1 = Name1->Length/sizeof(WCHAR); ULONG Index2 = Name2->Length/sizeof(WCHAR); UNICODE_STRING Label1; UNICODE_STRING Label2; LONG Result; // // Loop comparing labels // while ( Index1 != 0 && Index2 != 0 ) { // // Get the next label from each string // NetpCompareHelper ( Name1, &Index1, &Label1 ); NetpCompareHelper ( Name2, &Index2, &Label2 ); // // If the labels are different, // return that result to the caller. // Result = RtlCompareUnicodeString( &Label1, &Label2, TRUE ); if ( Result != 0 ) { return (int)Result; } } // // ASSERT: one label is a (proper) substring of the other // // If the first name is longer, indicate it is greater than the second // return Index1-Index2; } int __cdecl NetpCompareFtinfoEntryDns( const void *String1, const void *String2 ) /*++ Routine Description: qsort comparison routine for Dns string in Ftinfo entries Arguments: String1: First string to compare String2: Second string to compare Return Value: Signed value that gives the results of the comparison: Zero - String1 equals String2 < Zero - String1 less than String2 > Zero - String1 greater than String2 --*/ { PLSA_FOREST_TRUST_RECORD Entry1 = *((PLSA_FOREST_TRUST_RECORD *)String1); PLSA_FOREST_TRUST_RECORD Entry2 = *((PLSA_FOREST_TRUST_RECORD *)String2); PUNICODE_STRING Name1; PUNICODE_STRING Name2; int Result; // // Get the name from the entry // switch ( Entry1->ForestTrustType ) { case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: Name1 = &Entry1->ForestTrustData.TopLevelName; break; case ForestTrustDomainInfo: Name1 = &Entry1->ForestTrustData.DomainInfo.DnsName; break; default: // // If Entry2 can be recognized, // then entry 2 is less than this one. // switch ( Entry2->ForestTrustType ) { case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: case ForestTrustDomainInfo: return 1; // This name is greater than the other } // // Otherwise simply leave them in the same order // if ((Entry1 - Entry2) < 0 ) { return -1; } else if ((Entry1 - Entry2) > 0 ) { return 1; } else { return 0; } } switch ( Entry2->ForestTrustType ) { case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: Name2 = &Entry2->ForestTrustData.TopLevelName; break; case ForestTrustDomainInfo: Name2 = &Entry2->ForestTrustData.DomainInfo.DnsName; break; default: // // Since Entry1 is a recognized type, // this Entry2 is greater. // return -1; // This name is greater than the other } // // If the labels are different, // return the difference to the caller. // Result = NetpCompareDnsNameWithSortOrder( Name1, Name2 ); if ( Result != 0 ) { return Result; } // // If the labels are the same, // indicate TLNs are before domain info records. // return Entry1->ForestTrustType - Entry2->ForestTrustType; } int NetpCompareSid( PSID Sid1, PSID Sid2 ) /*++ Routine description: SID comparison routine that actually indicates if one sid is greater than another Arguments: Sid1 - First Sid Sid2 - Second Sid Returns: Signed value that gives the results of the comparison: Zero - String1 equals String2 < Zero - String1 less than String2 > Zero - String1 greater than String2 --*/ { DWORD Size1; DWORD Size2; LPBYTE Byte1; LPBYTE Byte2; ULONG i; NetpAssert( Sid1 && RtlValidSid( Sid1 )); NetpAssert( Sid2 && RtlValidSid( Sid2 )); // // The NULL SID is smaller // if ( Sid1 == NULL ) { if ( Sid2 != NULL ) { return -1; } else { return 0; } } if ( Sid2 == NULL ) { if ( Sid1 != NULL ) { return 1; } else { return 0; } } // // The longer sid is greater // Size1 = RtlLengthSid( Sid1 ); Size2 = RtlLengthSid( Sid2 ); if ( Size1 != Size2 ) { return Size1 - Size2; } // // Otherwise compare the bytes // Byte1 = (LPBYTE)Sid1; Byte2 = (LPBYTE)Sid2; for ( i=0; i Zero - String1 greater than String2 --*/ { PLSA_FOREST_TRUST_RECORD Entry1 = *((PLSA_FOREST_TRUST_RECORD *)String1); PLSA_FOREST_TRUST_RECORD Entry2 = *((PLSA_FOREST_TRUST_RECORD *)String2); PSID Sid1; PSID Sid2; int Result; // // Get the Sid from the entry // switch ( Entry1->ForestTrustType ) { case ForestTrustDomainInfo: Sid1 = Entry1->ForestTrustData.DomainInfo.Sid; break; default: // // If Entry2 can be recognized, // then entry 2 is less than this one. // switch ( Entry2->ForestTrustType ) { case ForestTrustDomainInfo: return 1; // This name is greater than the other } // // Otherwise simply leave them in the same order // if ((Entry1 - Entry2) < 0 ) { return -1; } else if ((Entry1 - Entry2) > 0 ) { return 1; } else { return 0; } } switch ( Entry2->ForestTrustType ) { case ForestTrustDomainInfo: Sid2 = Entry2->ForestTrustData.DomainInfo.Sid; break; default: // // Since Entry1 is a recognized type, // this Entry2 is greater. // return -1; // This name is greater than the other } // // Simply return the different of the sids. // return NetpCompareSid( Sid1, Sid2 ); } int __cdecl NetpCompareFtinfoEntryNetbios( const void *String1, const void *String2 ) /*++ Routine Description: qsort comparison routine for Netbios name in Ftinfo entries Arguments: String1: First string to compare String2: Second string to compare Return Value: Signed value that gives the results of the comparison: Zero - String1 equals String2 < Zero - String1 less than String2 > Zero - String1 greater than String2 --*/ { PLSA_FOREST_TRUST_RECORD Entry1 = *((PLSA_FOREST_TRUST_RECORD *)String1); PLSA_FOREST_TRUST_RECORD Entry2 = *((PLSA_FOREST_TRUST_RECORD *)String2); PUNICODE_STRING Name1; PUNICODE_STRING Name2; int Result; // // Get the Sid from the entry // switch ( Entry1->ForestTrustType ) { case ForestTrustDomainInfo: Name1 = &Entry1->ForestTrustData.DomainInfo.NetbiosName; if ( Name1->Length != 0 && Name1->Buffer != NULL ) { break; } default: // // If Entry2 can be recognized, // then entry 2 is less than this one. // switch ( Entry2->ForestTrustType ) { case ForestTrustDomainInfo: return 1; // This name is greater than the other } // // Otherwise simply leave them in the same order // if ((Entry1 - Entry2) < 0 ) { return -1; } else if ((Entry1 - Entry2) > 0 ) { return 1; } else { return 0; } } switch ( Entry2->ForestTrustType ) { case ForestTrustDomainInfo: Name2 = &Entry2->ForestTrustData.DomainInfo.NetbiosName; if ( Name2->Length != 0 && Name2->Buffer != NULL ) { break; } default: // // Since Entry1 is a recognized type, // this Entry2 is greater. // return -1; // This name is greater than the other } // // Simply return the difference of the names // return RtlCompareUnicodeString( Name1, Name2, TRUE ); } PLSA_FOREST_TRUST_INFORMATION NetpCopyFtinfoContext( IN PNL_FTINFO_CONTEXT FtinfoContext ) /*++ Routine Description: Routine to allocate an FTinfo array from an FTinfo context. Arguments: FtinfoContext - Context to use The caller must have previously called NetpInitFtinfoContext Return Value: FTinfo array. The caller should free this array using MIDL_user_free. If NULL, not enough memory was available. --*/ { PNL_FTINFO_ENTRY FtinfoEntry; PLIST_ENTRY ListEntry; PLSA_FOREST_TRUST_INFORMATION LocalForestTrustInfo; LPBYTE Where; ULONG Size; ULONG i; PLSA_FOREST_TRUST_RECORD Entries; // // Allocate a structure to return to the caller. // Size = ROUND_UP_COUNT( sizeof( *LocalForestTrustInfo ), ALIGN_WORST) + ROUND_UP_COUNT( FtinfoContext->FtinfoCount * sizeof(LSA_FOREST_TRUST_RECORD), ALIGN_WORST) + ROUND_UP_COUNT( FtinfoContext->FtinfoCount * sizeof(PLSA_FOREST_TRUST_RECORD), ALIGN_WORST) + FtinfoContext->FtinfoSize; LocalForestTrustInfo = MIDL_user_allocate( Size ); if ( LocalForestTrustInfo == NULL ) { return NULL; } RtlZeroMemory( LocalForestTrustInfo, Size ); Where = (LPBYTE)(LocalForestTrustInfo+1); Where = ROUND_UP_POINTER( Where, ALIGN_WORST ); // // Fill it in // LocalForestTrustInfo->RecordCount = FtinfoContext->FtinfoCount; // // Grab a huge chunk of ALIGN_WORST // (We fill it in during the loop below.) // Entries = (PLSA_FOREST_TRUST_RECORD) Where; Where = (LPBYTE)(&Entries[FtinfoContext->FtinfoCount]); Where = ROUND_UP_POINTER( Where, ALIGN_WORST ); // // Grab a huge chunk of dword aligned // (We fill it in during the loop below.) // LocalForestTrustInfo->Entries = (PLSA_FOREST_TRUST_RECORD *) Where; Where = (LPBYTE)(&LocalForestTrustInfo->Entries[FtinfoContext->FtinfoCount]); Where = ROUND_UP_POINTER( Where, ALIGN_WORST ); // // Fill in the individual entries // i = 0; for ( ListEntry = FtinfoContext->FtinfoList.Flink ; ListEntry != &FtinfoContext->FtinfoList ; ListEntry = ListEntry->Flink) { FtinfoEntry = CONTAINING_RECORD( ListEntry, NL_FTINFO_ENTRY, Next ); LocalForestTrustInfo->Entries[i] = &Entries[i]; NetpMarshalFtinfoEntry ( &FtinfoEntry->Record, &Entries[i], &Where ); i++; } NetpAssert( i == FtinfoContext->FtinfoCount ); NetpAssert( Where == ((LPBYTE)LocalForestTrustInfo) + Size ); // // Sort them into alphabetical order // qsort( LocalForestTrustInfo->Entries, LocalForestTrustInfo->RecordCount, sizeof(PLSA_FOREST_TRUST_RECORD), NetpCompareFtinfoEntryDns ); // // Return the allocated buffer to the caller. // return LocalForestTrustInfo; } VOID NetpCleanFtinfoContext( IN PNL_FTINFO_CONTEXT FtinfoContext ) /*++ Routine Description: Routine to cleanup the Ftinfo context structure. Arguments: FtinfoContext - Context to clean The caller must have previously called NetpInitFtinfoContext Return Value: None --*/ { PLIST_ENTRY ListEntry; PNL_FTINFO_ENTRY FtinfoEntry; // // Loop freeing the entries // while ( !IsListEmpty( &FtinfoContext->FtinfoList ) ) { // // Delink an entry // ListEntry = RemoveHeadList( &FtinfoContext->FtinfoList ); FtinfoEntry = CONTAINING_RECORD( ListEntry, NL_FTINFO_ENTRY, Next ); FtinfoContext->FtinfoCount -= 1; FtinfoContext->FtinfoSize -= FtinfoEntry->Size; RtlFreeHeap( RtlProcessHeap(), 0, FtinfoEntry ); } NetpAssert( FtinfoContext->FtinfoCount == 0 ); NetpAssert( FtinfoContext->FtinfoSize == 0 ); } PLSA_FOREST_TRUST_RECORD NetpAllocFtinfoEntry2 ( IN PNL_FTINFO_CONTEXT FtinfoContext, IN PLSA_FOREST_TRUST_RECORD InFtinfoRecord ) /*++ Routine Description: Same as NetpAllocFtinfoEntry except takes a template of an FTinfo entry on input. Arguments: FtinfoContext - Context to link the entry onto. InFtinfoRecord - Template to copy into InFtinfoRecord Return Value: Returns the address of the allocated forest trust record. The caller should not and cannot deallocate this buffer. It has a header and is linked into the FtinfoContext. Returns NULL if no memory can be allocated. --*/ { PNL_FTINFO_ENTRY FtinfoEntry; ULONG Size = ROUND_UP_COUNT(sizeof(NL_FTINFO_ENTRY), ALIGN_WORST); ULONG DataSize = 0; LPBYTE Where; // // Compute the size of the entry. // switch( InFtinfoRecord->ForestTrustType ) { case ForestTrustDomainInfo: if ( InFtinfoRecord->ForestTrustData.DomainInfo.Sid != NULL ) { DataSize += RtlLengthSid( InFtinfoRecord->ForestTrustData.DomainInfo.Sid ); } if ( InFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Length != 0 ) { DataSize += InFtinfoRecord->ForestTrustData.DomainInfo.DnsName.Length + sizeof(WCHAR); } if ( InFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Length != 0 ) { DataSize += InFtinfoRecord->ForestTrustData.DomainInfo.NetbiosName.Length + sizeof(WCHAR); } break; case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: if ( InFtinfoRecord->ForestTrustData.TopLevelName.Length != 0 ) { DataSize += InFtinfoRecord->ForestTrustData.TopLevelName.Length + sizeof(WCHAR); } break; default: NetpAssert( FALSE ); return NULL; } DataSize = ROUND_UP_COUNT(DataSize, ALIGN_WORST); // // Allocate an entry // Size += DataSize; FtinfoEntry = RtlAllocateHeap( RtlProcessHeap(), 0, Size ); if ( FtinfoEntry == NULL ) { return NULL; } RtlZeroMemory( FtinfoEntry, Size ); Where = (LPBYTE)(FtinfoEntry+1); // // Fill it in. // FtinfoEntry->Size = DataSize; NetpMarshalFtinfoEntry ( InFtinfoRecord, &FtinfoEntry->Record, &Where ); NetpAssert( Where == ((LPBYTE)FtinfoEntry) + Size ) // // Link it onto the list // InsertHeadList( &FtinfoContext->FtinfoList, &FtinfoEntry->Next ); FtinfoContext->FtinfoSize += FtinfoEntry->Size; FtinfoContext->FtinfoCount += 1; return &FtinfoEntry->Record; } BOOLEAN NetpAllocFtinfoEntry ( IN PNL_FTINFO_CONTEXT FtinfoContext, IN LSA_FOREST_TRUST_RECORD_TYPE ForestTrustType, IN PUNICODE_STRING Name, IN PSID Sid, IN PUNICODE_STRING NetbiosName ) /*++ Routine Description: Routine to allocate a single Ftinfo entry and link it onto the context. Arguments: FtinfoContext - Context to link the entry onto. ForestTypeType - Specifies the type of record to allocate. This must be ForestTrustTopLevelName or ForestTrustDomainInfo. Name - Specifies the name for the record. Sid - Specifies the SID for the record. (Ignored for ForestTrustTopLevelName.) NetbiosName - Specifies the netbios name for the record. (Ignored for ForestTrustTopLevelName.) Return Value: TRUE - Success FALSE - if no memory can be allocated --*/ { LSA_FOREST_TRUST_RECORD FtinfoRecord = {0}; // // Initialize the template Ftinfo entry // FtinfoRecord.ForestTrustType = ForestTrustType; switch( ForestTrustType ) { case ForestTrustDomainInfo: FtinfoRecord.ForestTrustData.DomainInfo.Sid = Sid; FtinfoRecord.ForestTrustData.DomainInfo.DnsName = *Name; FtinfoRecord.ForestTrustData.DomainInfo.NetbiosName = *NetbiosName; break; case ForestTrustTopLevelName: case ForestTrustTopLevelNameEx: FtinfoRecord.ForestTrustData.TopLevelName = *Name; break; default: NetpAssert( FALSE ); return FALSE; } // // Call the routine that takes a template and does the rest of the job // return (NetpAllocFtinfoEntry2( FtinfoContext, &FtinfoRecord ) != NULL); } BOOLEAN NetpIsSubordinate( IN const UNICODE_STRING * Subordinate, IN const UNICODE_STRING * Superior, IN BOOLEAN EqualReturnsTrue ) /*++ Routine Description: Determines if Subordinate string is indeed subordinate to Superior For example, "NY.acme.com" is subordinate to "acme.com", but "NY.acme.com" is NOT subordinate to "me.com" or "NY.acme.com" Arguments: Subordinate name to test for subordinate status Superior name to test for superior status EqualReturnsTrue - TRUE if equal names should return TRUE also Returns: TRUE is Subordinate is subordinate to Superior FALSE otherwise --*/ { USHORT SubIndex, SupIndex; UNICODE_STRING Temp; ASSERT( Subordinate && Subordinate->Buffer ); ASSERT( Superior && Superior->Buffer ); // // If equal names are to be considered subordinate, // compare the names for equality. // if ( EqualReturnsTrue && RtlEqualUnicodeString( Subordinate, Superior, TRUE )) { return TRUE; } // // A subordinate name must be longer than the superior name // if ( Subordinate->Length <= Superior->Length ) { return FALSE; } // // Subordinate name must be separated from the superior part by a period // if ( Subordinate->Buffer[( Subordinate->Length - Superior->Length ) / sizeof( WCHAR ) - 1] != L'.' ) { return FALSE; } // // Ensure the trailing part of the two names are the same. // Temp = *Subordinate; Temp.Buffer += ( Subordinate->Length - Superior->Length ) / sizeof( WCHAR ); Temp.Length = Superior->Length; Temp.MaximumLength = Temp.Length; if ( !RtlEqualUnicodeString( &Temp, Superior, TRUE )) { return FALSE; } return TRUE; } BOOLEAN NetpAddTlnFtinfoEntry ( IN PNL_FTINFO_CONTEXT FtinfoContext, IN PUNICODE_STRING Name ) /*++ Routine Description: Routine to add a TLN Ftinfo entry to the list. If there is already a TLN that is equal to or superior to this one, this TLN is ignored. (e.g., a TLN of a.acme.com is ignored of acme.com already exists in the list.) If there is already a TLN that is inferior to this one, the inferior TLN is removed and this one is added. (e.g., a TLN of acme.com causes an existing TLN of a.acme.com to be replaced by the new entry.) Arguments: FtinfoContext - Context to link the entry onto. Name - Specifies the name for the record. Return Value: TRUE - Success FALSE - if no memory can be allocated --*/ { PNL_FTINFO_ENTRY FtinfoEntry; PLIST_ENTRY ListEntry; // // Loop through the list of existing entries // for ( ListEntry = FtinfoContext->FtinfoList.Flink ; ListEntry != &FtinfoContext->FtinfoList ; ) { FtinfoEntry = CONTAINING_RECORD( ListEntry, NL_FTINFO_ENTRY, Next ); ListEntry = ListEntry->Flink; // // Ignore entries that aren't TLNs. // if ( FtinfoEntry->Record.ForestTrustType != ForestTrustTopLevelName ) { continue; } // // If the new name is subordinate (or equal to) to one already in the list, // ignore the new name. // if ( NetpIsSubordinate( Name, &FtinfoEntry->Record.ForestTrustData.TopLevelName, TRUE ) ) { return TRUE; } // // If the existing name is subordinate to the new name, // remove the existing name. // if ( NetpIsSubordinate( &FtinfoEntry->Record.ForestTrustData.TopLevelName, Name, FALSE ) ) { RemoveEntryList( &FtinfoEntry->Next ); FtinfoContext->FtinfoCount -= 1; FtinfoContext->FtinfoSize -= FtinfoEntry->Size; RtlFreeHeap( RtlProcessHeap(), 0, FtinfoEntry ); // continue looping since there may be more names to remove } } // // Add the new entry to the list // return NetpAllocFtinfoEntry( FtinfoContext, ForestTrustTopLevelName, Name, NULL, // No sid NULL ); // No Netbios name } VOID NetpMergeFtinfoHelper( IN PLSA_FOREST_TRUST_INFORMATION NewForestTrustInfo, IN PLSA_FOREST_TRUST_INFORMATION OldForestTrustInfo, IN OUT PULONG NewIndex, IN OUT PULONG OldIndex, OUT PLSA_FOREST_TRUST_RECORD *NewEntry, OUT PLSA_FOREST_TRUST_RECORD *OldEntry, OUT PULONG OldFlags, IN int (__cdecl *Routine) (const void *, const void *) ) /*++ Routine Description: This routine walks a pair of FTinfo arrays in sorted order and returns the next entry. If both entries are the same in the sort order, this routine returns an entry from both arrays Arguments: NewForestTrustInfo - Pointer to the first array OldForestTrustInfo - Pointer to the second array NewIndex - Current index into the first sorted array OldIndex - Current index into the second sorted array Before calling this routine the first time, the caller should set these parameters to zero. Both indices zero triggers this routine to qsort the arrays. The caller should *not* call this routine if both NewIndex and OldIndex are greater than the corresponding record count. NewEntry - Returns a pointer to an entry to be processed from the first sorted array. OldEntry - Returns a pointer to an entry to be processed from the second sorted array. Returns NULL if no entry is to be processed from the corresponding array. OldFlags - Returns the Flags field that corresponds to OldEntry. If there are duplicates of OldEntry, those duplicates are silently ignored by this routine. This field returns the logical OR of the Flags field of those entries. Routine - Comparison routine to passed to qsort to sort the FTinfo arrays. Return Value: None. --*/ { int RetVal; // // Sort the arrays // if ( *NewIndex == 0 && *OldIndex == 0 ) { qsort( NewForestTrustInfo->Entries, NewForestTrustInfo->RecordCount, sizeof(PLSA_FOREST_TRUST_RECORD), Routine ); qsort( OldForestTrustInfo->Entries, OldForestTrustInfo->RecordCount, sizeof(PLSA_FOREST_TRUST_RECORD), Routine ); } // // Compare the first entry at the front of each list to determine which list // to consume an entry from. // *NewEntry = NULL; *OldEntry = NULL; *OldFlags = 0; if ( *NewIndex < NewForestTrustInfo->RecordCount ) { // // If neither list is empty, // compare the entries to determine which is next. // if ( *OldIndex < OldForestTrustInfo->RecordCount ) { RetVal = (*Routine)( &NewForestTrustInfo->Entries[*NewIndex], &OldForestTrustInfo->Entries[*OldIndex] ); // // If the new entry is less than or equal to the old entry, // consume the new entry. // if ( RetVal <= 0 ) { *NewEntry = NewForestTrustInfo->Entries[*NewIndex]; (*NewIndex) ++; } // // If the old entry is less than or equal to the new entry, // consume the old entry. // if ( RetVal >= 0 ) { *OldEntry = OldForestTrustInfo->Entries[*OldIndex]; (*OldIndex) ++; } // // If the old list is empty and the new list isn't, // consume an entry from the new list. // } else { *NewEntry = NewForestTrustInfo->Entries[*NewIndex]; (*NewIndex) ++; } } else { // // If the new list is empty and the old list isn't, // consume an entry from the old list. // if ( *OldIndex < OldForestTrustInfo->RecordCount ) { *OldEntry = OldForestTrustInfo->Entries[*OldIndex]; (*OldIndex) ++; } } // // If we're returning an "OldEntry", // weed out all duplicates of that OldEntry. // if ( *OldEntry != NULL ) { *OldFlags |= (*OldEntry)->Flags; while ( *OldIndex < OldForestTrustInfo->RecordCount ) { // // Stop as soon as we hit an entry that isn't a duplicate. // RetVal = (*Routine)( OldEntry, &OldForestTrustInfo->Entries[*OldIndex] ); if ( RetVal != 0 ) { break; } *OldFlags |= (*OldEntry)->Flags; } } } NTSTATUS NetpMergeFtinfo( IN PUNICODE_STRING TrustedDomainName, IN PLSA_FOREST_TRUST_INFORMATION InNewForestTrustInfo, IN PLSA_FOREST_TRUST_INFORMATION InOldForestTrustInfo OPTIONAL, OUT PLSA_FOREST_TRUST_INFORMATION *MergedForestTrustInfo ) /*++ Routine Description: This function merges the changes from a new FTinfo into an old FTinfo and produces the resultant FTinfo. The merged FTinfo records are a combinition of the new and old records. Here's where the merged records come from: * The TLN exclusion records are copied from the TDO intact. * The TLN record from the trusted domain that maps to the dns domain name of the TDO is copied enabled. This reflects the LSA requirement that such a TLN not be disabled. For instance, if the TDO is for a.acme.com and there is a TLN for a.acme.com that TLN will be enabled. Also, if the TDO is for a.acme.com and there is a TLN for acme.com, that TLN will be enabled. * All other TLN records from the trusted domain are copied disabled with the following exceptions. If there is an enabled TLN on the TDO, all TLNs from the trusted domain that equal (or are subordinate to) the TDO TLN are marked as disabled. This follows the philosophy that new TLNs are imported as enabled. For instance, if the TDO had an enabled TLN for a.acme.com that TLN will still be enabled after the automatic update. If the TDO had an enabled TLN for acme.com and the trusted forest now has a TLN for a.acme.com, the resultant FTinfo will have an enabled TLN for a.acme.com. * The domain records from the trusted domain are copied enabled with the following exceptions. If there is a disabled domain record on the TDO whose dns domain name, or domain sid exactly matches the domain record, then the domain remains disabled. If there is a domain record on the TDO whose netbios name is disabled and whose netbios name exactly matches the netbios name on a domain record, then the netbios name is disabled. * Finally, orphaned exclusion records (those that are not subordinate to any TLN) are removed (Bug #707630). Arguments: TrustedDomainName - Trusted domain that is to be updated. NewForestTrustInfo - Specified the new array of FTinfo records as returned from the TrustedDomainName. The Flags field and Time field of the TLN entries are ignored. OldForestTrustInfo - Specified the array of FTinfo records as returned from the TDO. This field may be NULL if there is no existing records. MergedForestTrustInfo - Returns the resulant FTinfo records. The caller should free this buffer using MIDL_user_free. Return Value: STATUS_SUCCESS: Success. STATUS_INVALID_PARAMETER: One of the following happened: * There was no New TLN that TrustedDomainName is subordinate to. --*/ { NTSTATUS Status; LSA_FOREST_TRUST_INFORMATION OldForestTrustInfo; LSA_FOREST_TRUST_INFORMATION NewForestTrustInfo; LSA_FOREST_TRUST_INFORMATION NetbiosForestTrustInfo; NL_FTINFO_CONTEXT FtinfoContext; ULONG NewIndex; ULONG OldIndex; ULONG OldFlags; PLSA_FOREST_TRUST_RECORD NewEntry; PLSA_FOREST_TRUST_RECORD PreviousNewEntry; PLSA_FOREST_TRUST_RECORD OldEntry; BOOLEAN DomainTlnFound = FALSE; PLSA_FOREST_TRUST_RECORD OldTlnPrefix; ULONG Index; // // Initialization // *MergedForestTrustInfo = NULL; NetpInitFtinfoContext( &FtinfoContext ); RtlZeroMemory( &OldForestTrustInfo, sizeof(OldForestTrustInfo) ); RtlZeroMemory( &NewForestTrustInfo, sizeof(NewForestTrustInfo) ); RtlZeroMemory( &NetbiosForestTrustInfo, sizeof(NetbiosForestTrustInfo) ); // // Make a copy of the data that'll be qsorted so that we don't modify the caller's buffer. // if ( InOldForestTrustInfo != NULL ) { OldForestTrustInfo.RecordCount = InOldForestTrustInfo->RecordCount; OldForestTrustInfo.Entries = RtlAllocateHeap( RtlProcessHeap(), 0, OldForestTrustInfo.RecordCount * sizeof(PLSA_FOREST_TRUST_RECORD) ); if ( OldForestTrustInfo.Entries == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } RtlCopyMemory( OldForestTrustInfo.Entries, InOldForestTrustInfo->Entries, OldForestTrustInfo.RecordCount * sizeof(PLSA_FOREST_TRUST_RECORD) ); } NewForestTrustInfo.RecordCount = InNewForestTrustInfo->RecordCount; NewForestTrustInfo.Entries = RtlAllocateHeap( RtlProcessHeap(), 0, NewForestTrustInfo.RecordCount * sizeof(PLSA_FOREST_TRUST_RECORD) ); if ( NewForestTrustInfo.Entries == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } RtlCopyMemory( NewForestTrustInfo.Entries, InNewForestTrustInfo->Entries, NewForestTrustInfo.RecordCount * sizeof(PLSA_FOREST_TRUST_RECORD) ); // // Allocate a temporary Ftinfo array containing all of the domain entries. // Allocate it a worst case size. // NetbiosForestTrustInfo.Entries = RtlAllocateHeap( RtlProcessHeap(), 0, (OldForestTrustInfo.RecordCount+NewForestTrustInfo.RecordCount) * sizeof(PLSA_FOREST_TRUST_RECORD) ); if ( NetbiosForestTrustInfo.Entries == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } // // Loop through each list in DNS canonical order processing the least entry. // // This loop handles TLN and TLNEX entries only // NewIndex = 0; OldIndex = 0; OldTlnPrefix = NULL; PreviousNewEntry = NULL; while ( NewIndex < NewForestTrustInfo.RecordCount || OldIndex < OldForestTrustInfo.RecordCount ) { // // Grab the next entry from each of the sorted arrays // NetpMergeFtinfoHelper( &NewForestTrustInfo, &OldForestTrustInfo, &NewIndex, &OldIndex, &NewEntry, &OldEntry, &OldFlags, NetpCompareFtinfoEntryDns ); // // Process the old entry // if ( OldEntry != NULL ) { // // Remember to most recent TLN record from the old array. // if ( OldEntry->ForestTrustType == ForestTrustTopLevelName ) { OldTlnPrefix = OldEntry; // // TLN exclusion records are taken from the old entries // } else if ( OldEntry->ForestTrustType == ForestTrustTopLevelNameEx ) { if ( NetpAllocFtinfoEntry2( &FtinfoContext, OldEntry ) == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } } } // // Process the new entry // if ( NewEntry != NULL ) { // // Handle TLN entries // if ( NewEntry->ForestTrustType == ForestTrustTopLevelName ) { BOOLEAN SetTlnNewFlag; LSA_FOREST_TRUST_RECORD NewEntryCopy; // // Make a copy of the new entry. // // We modify the entry to get the time and flags right. We don't want // to modify the callers buffer. // NewEntryCopy = *NewEntry; // // Ignore duplicate new entries // // If the name of this new entry is subordinate to the previous new entry, // then this TLN can be quietly dropped. // // This is the case where the trusted domain sent us a TLN for both // acme.com and a.acme.com. The second entry is a duplicate. // if ( PreviousNewEntry != NULL && PreviousNewEntry->ForestTrustType == ForestTrustTopLevelName ) { if ( NetpIsSubordinate( &NewEntry->ForestTrustData.TopLevelName, &PreviousNewEntry->ForestTrustData.TopLevelName, TRUE ) ) { continue; } } // // By default any TLN from the new list should be marked as new. // SetTlnNewFlag = TRUE; // // Set the flags and timestamp on the new entry. // // If we're processing an entry from both lists, // grab the flags and timestamp from the old entry. // if ( OldEntry != NULL ) { NewEntryCopy.Flags = OldFlags; NewEntryCopy.Time = OldEntry->Time; // This entry isn't 'new'. SetTlnNewFlag = FALSE; // // Otherwise indicate that we have no information // } else { NewEntryCopy.Flags = 0; NewEntryCopy.Time.QuadPart = 0; } // // If this new entry is subordinate to the most recent old TLN record, // use the flag bits from that most recent old TLN record. // if ( OldTlnPrefix != NULL && NetpIsSubordinate( &NewEntryCopy.ForestTrustData.TopLevelName, &OldTlnPrefix->ForestTrustData.TopLevelName, FALSE ) ) { // // If the old TLN was disabled by the admin, // so should the new entry. // if ( OldTlnPrefix->Flags & LSA_TLN_DISABLED_ADMIN ) { NewEntryCopy.Flags |= LSA_TLN_DISABLED_ADMIN; SetTlnNewFlag = FALSE; // // If the old TLN was enabled, // so should the new entry. // } else if ( (OldTlnPrefix->Flags & LSA_FTRECORD_DISABLED_REASONS) == 0 ) { SetTlnNewFlag = FALSE; } } // // If the name of the forest is subordinate of or equal to the TLN name, // enable the entry. // if ( NetpIsSubordinate( TrustedDomainName, &NewEntryCopy.ForestTrustData.TopLevelName, TRUE )) { SetTlnNewFlag = FALSE; DomainTlnFound = TRUE; } // // If this is a new TLN, // mark it as such. // if ( SetTlnNewFlag ) { NewEntryCopy.Flags |= LSA_TLN_DISABLED_NEW; } // // Merge the new entry into the list // if ( NetpAllocFtinfoEntry2( &FtinfoContext, &NewEntryCopy ) == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } // // Remember this previous entry for the next iteration. // PreviousNewEntry = NewEntry; } } } // // Loop through each list in SID canonical order processing the least entry. // // This loop handles DOMAIN entries only // // This is in a separate loop since we want to process domain entries in SID order // to ensure the correct disabled bits are merged from the old list even though the // DNS domain name changes. // NewIndex = 0; OldIndex = 0; PreviousNewEntry = NULL; while ( NewIndex < NewForestTrustInfo.RecordCount || OldIndex < OldForestTrustInfo.RecordCount ) { // // Grab the next entry from each of the sorted arrays // NetpMergeFtinfoHelper( &NewForestTrustInfo, &OldForestTrustInfo, &NewIndex, &OldIndex, &NewEntry, &OldEntry, &OldFlags, NetpCompareFtinfoEntrySid ); // // Ignore the netbios bits for now (We'll get them on the next pass through the data.) // OldFlags &= ~(LSA_NB_DISABLED_ADMIN|LSA_NB_DISABLED_CONFLICT); // // Process the old entry // if ( OldEntry != NULL ) { // // Don't let the lack of a new entry allow an admin disabled entry to be deleted. // if ( OldEntry->ForestTrustType == ForestTrustDomainInfo && (OldFlags & LSA_SID_DISABLED_ADMIN) != 0 && NewEntry == NULL ) { // // Make a copy of the entry to ensure we don't modify the caller's buffer // LSA_FOREST_TRUST_RECORD OldEntryCopy; OldEntryCopy = *OldEntry; OldEntryCopy.Flags = OldFlags; // // Allocate entry. // // Remember the address of the entry for the netbios pass. // NetbiosForestTrustInfo.Entries[NetbiosForestTrustInfo.RecordCount] = NetpAllocFtinfoEntry2( &FtinfoContext, &OldEntryCopy ); if ( NetbiosForestTrustInfo.Entries[NetbiosForestTrustInfo.RecordCount] == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } NetbiosForestTrustInfo.RecordCount++; } } // // Process the new entry // if ( NewEntry != NULL ) { // // Handle domain entries // if ( NewEntry->ForestTrustType == ForestTrustDomainInfo ) { LSA_FOREST_TRUST_RECORD NewEntryCopy; // // Make a copy of the new entry. // // We modify the entry to get the time and flags right. We don't want // to modify the callers buffer. // NewEntryCopy = *NewEntry; // // Ignore duplicate new entries // // If the name of this new entry is subordinate to the previous new entry, // then this entry can be quietly dropped. // // We arbitrarily drop the second entry even though the other fields of the // triple might be different. // if ( PreviousNewEntry != NULL && PreviousNewEntry->ForestTrustType == ForestTrustDomainInfo ) { if ( RtlEqualSid( NewEntryCopy.ForestTrustData.DomainInfo.Sid, PreviousNewEntry->ForestTrustData.DomainInfo.Sid ) ) { continue; } } // // Set the flags and timestamp on the new entry. // // If we're processing an entry from both lists, // grab the flags and timestamp from the old entry. // if ( OldEntry != NULL ) { NewEntryCopy.Flags = OldFlags; NewEntryCopy.Time = OldEntry->Time; // // Otherwise indicate that we have no information // } else { NewEntryCopy.Flags = 0; NewEntryCopy.Time.QuadPart = 0; } // // Merge the new entry into the list // // Remember the address of the entry for the netbios pass. // NetbiosForestTrustInfo.Entries[NetbiosForestTrustInfo.RecordCount] = NetpAllocFtinfoEntry2( &FtinfoContext, &NewEntryCopy ); if ( NetbiosForestTrustInfo.Entries[NetbiosForestTrustInfo.RecordCount] == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } NetbiosForestTrustInfo.RecordCount++; // // Ensure there is a TLN for this domain entry // if ( !NetpAddTlnFtinfoEntry ( &FtinfoContext, &NewEntryCopy.ForestTrustData.DomainInfo.DnsName ) ) { Status = STATUS_NO_MEMORY; goto Cleanup; } // // Remember this previous entry for the next iteration. // PreviousNewEntry = NewEntry; } } } // // Loop through each list in Netbios canonical order processing the least entry. // // This loop handle the Netbios name in the domain entries. // // This is in a separate loop since we want to process domain entries in Netbios order // to ensure the correct disabled bits are merged from the old list even though the // DNS domain name or domain sid changes. // // This iteration is fundamentally different than the previous two. This iteration // uses NetbiosForestTrustInfo as the 'new' array. It is a psuedo ftinfo array that // is built as the list of all the domain entries that have been copied into FtinfoContext. // So, this iteration simply has to find that pre-existing entry and set the flags // appropriately. // NewIndex = 0; OldIndex = 0; PreviousNewEntry = NULL; while ( NewIndex < NetbiosForestTrustInfo.RecordCount || OldIndex < OldForestTrustInfo.RecordCount ) { // // Grab the next entry from each of the sorted arrays // NetpMergeFtinfoHelper( &NetbiosForestTrustInfo, &OldForestTrustInfo, &NewIndex, &OldIndex, &NewEntry, &OldEntry, &OldFlags, NetpCompareFtinfoEntryNetbios ); // // Ignore everything except the netbios bits. // // Everything else was processed on the previous iteration. // OldFlags &= (LSA_NB_DISABLED_ADMIN|LSA_NB_DISABLED_CONFLICT); // // This loop preserves the netbios disabled bits. // If there is no old entry, there's nothing to preserve. // if ( OldEntry == NULL ) { continue; } // // If there is no new entry, // ensure the *admin* disabled bit it preserved anyway. // if ( NewEntry == NULL ) { // // Don't let the lack of a new entry allow an admin disabled entry to be deleted. // // Note that the newly added entry might have a duplicate DNS name or SID. // if ( OldEntry->ForestTrustType == ForestTrustDomainInfo && (OldFlags & LSA_NB_DISABLED_ADMIN) != 0 ) { // // Make a copy of the entry to ensure we don't modify the caller's buffer // LSA_FOREST_TRUST_RECORD OldEntryCopy; OldEntryCopy = *OldEntry; OldEntryCopy.Flags = OldFlags; if ( !NetpAllocFtinfoEntry2( &FtinfoContext, &OldEntryCopy ) ) { Status = STATUS_NO_MEMORY; goto Cleanup; } } // // Copy any netbios disabled bits to the existing new entry. // } else { // // The NetbiosForestTrustInfo array only has domain entries. // And the entries are equal so both must be domain entries. // NetpAssert( NewEntry->ForestTrustType == ForestTrustDomainInfo ); NetpAssert( OldEntry->ForestTrustType == ForestTrustDomainInfo ); NewEntry->Flags |= OldFlags; } } // // Ensure there is a TLN that DomainName is subordinate to // if ( !DomainTlnFound ) { Status = STATUS_INVALID_PARAMETER; goto Cleanup; } // // Return the collected entries to the caller. // *MergedForestTrustInfo = NetpCopyFtinfoContext( &FtinfoContext ); if ( *MergedForestTrustInfo == NULL ) { Status = STATUS_NO_MEMORY; goto Cleanup; } // // Remove orphaned TLN exclusion records from the merged information // for ( Index = 0; Index < (*MergedForestTrustInfo)->RecordCount; Index++ ) { PLSA_FOREST_TRUST_RECORD This = (*MergedForestTrustInfo)->Entries[Index]; UNICODE_STRING * ExclusionName; ULONG Index2; BOOL Subordinate = FALSE; if ( This->ForestTrustType != ForestTrustTopLevelNameEx ) { // // Only interested in orphaned exclusions // continue; } ExclusionName = &This->ForestTrustData.TopLevelName; for ( Index2 = 0; Index2 < (*MergedForestTrustInfo)->RecordCount; Index2++ ) { PLSA_FOREST_TRUST_RECORD Other = (*MergedForestTrustInfo)->Entries[Index2]; UNICODE_STRING * TopLevelName; if ( Other->ForestTrustType != ForestTrustTopLevelName ) { // // Only interested in top level names (exclusion must be subordinate to it) // continue; } TopLevelName = &Other->ForestTrustData.TopLevelName; // // First perform a subordinate check where equality is not enough // if ( NetpIsSubordinate( ExclusionName, TopLevelName, FALSE )) { Subordinate = TRUE; // // Now check for equality // } else if ( RtlEqualUnicodeString( ExclusionName, TopLevelName, TRUE )) { // // A top level name is the same as an exclusion name. // Throw away the exclusion record, but ensure that // the top level name record is marked "disabled". // if (( Other->Flags & LSA_FTRECORD_DISABLED_REASONS ) == 0 ) { Other->Flags |= LSA_TLN_DISABLED_NEW; } break; } if ( Subordinate ) { break; } } if ( !Subordinate ) { // // This is an orphaned exclusion record. Remove it. // (*MergedForestTrustInfo)->RecordCount -= 1; (*MergedForestTrustInfo)->Entries[Index] = (*MergedForestTrustInfo)->Entries[(*MergedForestTrustInfo)->RecordCount]; Index -= 1; } } Status = STATUS_SUCCESS; Cleanup: // // Clean FtInfoContext // NetpCleanFtinfoContext( &FtinfoContext ); if ( OldForestTrustInfo.Entries != NULL ) { RtlFreeHeap( RtlProcessHeap(), 0, OldForestTrustInfo.Entries ); } if ( NewForestTrustInfo.Entries != NULL ) { RtlFreeHeap( RtlProcessHeap(), 0, NewForestTrustInfo.Entries ); } if ( NetbiosForestTrustInfo.Entries != NULL ) { RtlFreeHeap( RtlProcessHeap(), 0, NetbiosForestTrustInfo.Entries ); } return Status; }