xref: /reactos/dll/win32/samsrv/samrpc.c (revision 2b7246fd)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         Security Account Manager (SAM) Server
4  * FILE:            reactos/dll/win32/samsrv/samrpc.c
5  * PURPOSE:         RPC interface functions
6  *
7  * PROGRAMMERS:     Eric Kohl
8  */
9 
10 #include "samsrv.h"
11 
12 /* GLOBALS *******************************************************************/
13 
14 static SID_IDENTIFIER_AUTHORITY NtSidAuthority = {SECURITY_NT_AUTHORITY};
15 
16 static GENERIC_MAPPING ServerMapping =
17 {
18     SAM_SERVER_READ,
19     SAM_SERVER_WRITE,
20     SAM_SERVER_EXECUTE,
21     SAM_SERVER_ALL_ACCESS
22 };
23 
24 static GENERIC_MAPPING DomainMapping =
25 {
26     DOMAIN_READ,
27     DOMAIN_WRITE,
28     DOMAIN_EXECUTE,
29     DOMAIN_ALL_ACCESS
30 };
31 
32 static GENERIC_MAPPING AliasMapping =
33 {
34     ALIAS_READ,
35     ALIAS_WRITE,
36     ALIAS_EXECUTE,
37     ALIAS_ALL_ACCESS
38 };
39 
40 static GENERIC_MAPPING GroupMapping =
41 {
42     GROUP_READ,
43     GROUP_WRITE,
44     GROUP_EXECUTE,
45     GROUP_ALL_ACCESS
46 };
47 
48 static GENERIC_MAPPING UserMapping =
49 {
50     USER_READ,
51     USER_WRITE,
52     USER_EXECUTE,
53     USER_ALL_ACCESS
54 };
55 
56 PGENERIC_MAPPING pServerMapping = &ServerMapping;
57 
58 
59 /* FUNCTIONS *****************************************************************/
60 
61 static
62 LARGE_INTEGER
63 SampAddRelativeTimeToTime(IN LARGE_INTEGER AbsoluteTime,
64                           IN LARGE_INTEGER RelativeTime)
65 {
66     LARGE_INTEGER NewTime;
67 
68     NewTime.QuadPart = AbsoluteTime.QuadPart - RelativeTime.QuadPart;
69 
70     if (NewTime.QuadPart < 0)
71         NewTime.QuadPart = 0;
72 
73     return NewTime;
74 }
75 
76 
77 VOID
78 SampStartRpcServer(VOID)
79 {
80     RPC_STATUS Status;
81 
82     TRACE("SampStartRpcServer() called\n");
83 
84     Status = RpcServerUseProtseqEpW(L"ncacn_np",
85                                     RPC_C_PROTSEQ_MAX_REQS_DEFAULT,
86                                     L"\\pipe\\samr",
87                                     NULL);
88     if (Status != RPC_S_OK)
89     {
90         WARN("RpcServerUseProtseqEpW() failed (Status %lx)\n", Status);
91         return;
92     }
93 
94     Status = RpcServerRegisterIf(samr_v1_0_s_ifspec,
95                                  NULL,
96                                  NULL);
97     if (Status != RPC_S_OK)
98     {
99         WARN("RpcServerRegisterIf() failed (Status %lx)\n", Status);
100         return;
101     }
102 
103     Status = RpcServerListen(1, 20, TRUE);
104     if (Status != RPC_S_OK)
105     {
106         WARN("RpcServerListen() failed (Status %lx)\n", Status);
107         return;
108     }
109 
110     TRACE("SampStartRpcServer() done\n");
111 }
112 
113 
114 void __RPC_FAR * __RPC_USER midl_user_allocate(SIZE_T len)
115 {
116     return HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, len);
117 }
118 
119 
120 void __RPC_USER midl_user_free(void __RPC_FAR * ptr)
121 {
122     HeapFree(GetProcessHeap(), 0, ptr);
123 }
124 
125 
126 void __RPC_USER SAMPR_HANDLE_rundown(SAMPR_HANDLE hHandle)
127 {
128     FIXME("SAMPR_HANDLE_rundown(%p)\n", hHandle);
129 }
130 
131 
132 /* Function 0 */
133 NTSTATUS
134 NTAPI
135 SamrConnect(IN PSAMPR_SERVER_NAME ServerName,
136             OUT SAMPR_HANDLE *ServerHandle,
137             IN ACCESS_MASK DesiredAccess)
138 {
139     SAMPR_REVISION_INFO InRevisionInfo, OutRevisionInfo;
140     ULONG OutVersion;
141 
142     TRACE("SamrConnect(%p %p %lx)\n",
143           ServerName, ServerHandle, DesiredAccess);
144 
145     InRevisionInfo.V1.Revision = 0;
146     InRevisionInfo.V1.SupportedFeatures = 0;
147 
148     return SamrConnect5(ServerName,
149                         DesiredAccess,
150                         1,
151                         &InRevisionInfo,
152                         &OutVersion,
153                         &OutRevisionInfo,
154                         ServerHandle);
155 }
156 
157 
158 /* Function 1 */
159 NTSTATUS
160 NTAPI
161 SamrCloseHandle(IN OUT SAMPR_HANDLE *SamHandle)
162 {
163     PSAM_DB_OBJECT DbObject;
164     NTSTATUS Status = STATUS_SUCCESS;
165 
166     TRACE("SamrCloseHandle(%p)\n", SamHandle);
167 
168     RtlAcquireResourceShared(&SampResource,
169                              TRUE);
170 
171     Status = SampValidateDbObject(*SamHandle,
172                                   SamDbIgnoreObject,
173                                   0,
174                                   &DbObject);
175     if (Status == STATUS_SUCCESS)
176     {
177         Status = SampCloseDbObject(DbObject);
178         *SamHandle = NULL;
179     }
180 
181     RtlReleaseResource(&SampResource);
182 
183     TRACE("SamrCloseHandle done (Status 0x%08lx)\n", Status);
184 
185     return Status;
186 }
187 
188 
189 /* Function 2 */
190 NTSTATUS
191 NTAPI
192 SamrSetSecurityObject(IN SAMPR_HANDLE ObjectHandle,
193                       IN SECURITY_INFORMATION SecurityInformation,
194                       IN PSAMPR_SR_SECURITY_DESCRIPTOR SecurityDescriptor)
195 {
196     PSAM_DB_OBJECT DbObject = NULL;
197     ACCESS_MASK DesiredAccess = 0;
198     PSECURITY_DESCRIPTOR RelativeSd = NULL;
199     ULONG RelativeSdSize = 0;
200     HANDLE TokenHandle = NULL;
201     PGENERIC_MAPPING Mapping;
202     NTSTATUS Status;
203 
204     TRACE("SamrSetSecurityObject(%p %lx %p)\n",
205           ObjectHandle, SecurityInformation, SecurityDescriptor);
206 
207     if ((SecurityDescriptor == NULL) ||
208         (SecurityDescriptor->SecurityDescriptor == NULL) ||
209         !RtlValidSecurityDescriptor((PSECURITY_DESCRIPTOR)SecurityDescriptor->SecurityDescriptor))
210         return ERROR_INVALID_PARAMETER;
211 
212     if (SecurityInformation == 0 ||
213         SecurityInformation & ~(OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION
214         | DACL_SECURITY_INFORMATION | SACL_SECURITY_INFORMATION))
215         return ERROR_INVALID_PARAMETER;
216 
217     if (SecurityInformation & SACL_SECURITY_INFORMATION)
218         DesiredAccess |= ACCESS_SYSTEM_SECURITY;
219 
220     if (SecurityInformation & DACL_SECURITY_INFORMATION)
221         DesiredAccess |= WRITE_DAC;
222 
223     if (SecurityInformation & (OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION))
224         DesiredAccess |= WRITE_OWNER;
225 
226     if ((SecurityInformation & OWNER_SECURITY_INFORMATION) &&
227         (((PISECURITY_DESCRIPTOR)SecurityDescriptor)->Owner == NULL))
228         return ERROR_INVALID_PARAMETER;
229 
230     if ((SecurityInformation & GROUP_SECURITY_INFORMATION) &&
231         (((PISECURITY_DESCRIPTOR)SecurityDescriptor)->Group == NULL))
232         return ERROR_INVALID_PARAMETER;
233 
234     /* Validate the server handle */
235     Status = SampValidateDbObject(ObjectHandle,
236                                   SamDbIgnoreObject,
237                                   DesiredAccess,
238                                   &DbObject);
239     if (!NT_SUCCESS(Status))
240         goto done;
241 
242     /* Get the mapping for the object type */
243     switch (DbObject->ObjectType)
244     {
245         case SamDbServerObject:
246             Mapping = &ServerMapping;
247             break;
248 
249         case SamDbDomainObject:
250             Mapping = &DomainMapping;
251             break;
252 
253         case SamDbAliasObject:
254             Mapping = &AliasMapping;
255             break;
256 
257         case SamDbGroupObject:
258             Mapping = &GroupMapping;
259             break;
260 
261         case SamDbUserObject:
262             Mapping = &UserMapping;
263             break;
264 
265         default:
266             return STATUS_INVALID_HANDLE;
267     }
268 
269     /* Get the size of the SD */
270     Status = SampGetObjectAttribute(DbObject,
271                                     L"SecDesc",
272                                     NULL,
273                                     NULL,
274                                     &RelativeSdSize);
275     if (!NT_SUCCESS(Status))
276         return Status;
277 
278     /* Allocate a buffer for the SD */
279     RelativeSd = RtlAllocateHeap(RtlGetProcessHeap(), 0, RelativeSdSize);
280     if (RelativeSd == NULL)
281         return STATUS_INSUFFICIENT_RESOURCES;
282 
283     /* Get the SD */
284     Status = SampGetObjectAttribute(DbObject,
285                                     L"SecDesc",
286                                     NULL,
287                                     RelativeSd,
288                                     &RelativeSdSize);
289     if (!NT_SUCCESS(Status))
290         goto done;
291 
292     /* Build the new security descriptor */
293     Status = RtlSetSecurityObject(SecurityInformation,
294                                   (PSECURITY_DESCRIPTOR)SecurityDescriptor->SecurityDescriptor,
295                                   &RelativeSd,
296                                   Mapping,
297                                   TokenHandle);
298     if (!NT_SUCCESS(Status))
299     {
300         ERR("RtlSetSecurityObject failed (Status 0x%08lx)\n", Status);
301         goto done;
302     }
303 
304     /* Set the modified SD */
305     Status = SampSetObjectAttribute(DbObject,
306                                     L"SecDesc",
307                                     REG_BINARY,
308                                     RelativeSd,
309                                     RtlLengthSecurityDescriptor(RelativeSd));
310     if (!NT_SUCCESS(Status))
311     {
312         ERR("SampSetObjectAttribute failed (Status 0x%08lx)\n", Status);
313     }
314 
315 done:
316     if (TokenHandle != NULL)
317         NtClose(TokenHandle);
318 
319     if (RelativeSd != NULL)
320         RtlFreeHeap(RtlGetProcessHeap(), 0, RelativeSd);
321 
322     return Status;
323 }
324 
325 
326 /* Function 3 */
327 NTSTATUS
328 NTAPI
329 SamrQuerySecurityObject(IN SAMPR_HANDLE ObjectHandle,
330                         IN SECURITY_INFORMATION SecurityInformation,
331                         OUT PSAMPR_SR_SECURITY_DESCRIPTOR *SecurityDescriptor)
332 {
333     PSAM_DB_OBJECT SamObject;
334     PSAMPR_SR_SECURITY_DESCRIPTOR SdData = NULL;
335     PISECURITY_DESCRIPTOR_RELATIVE RelativeSd = NULL;
336     PSECURITY_DESCRIPTOR ResultSd = NULL;
337     ACCESS_MASK DesiredAccess = 0;
338     ULONG RelativeSdSize = 0;
339     ULONG ResultSdSize = 0;
340     NTSTATUS Status;
341 
342     TRACE("SamrQuerySecurityObject(%p %lx %p)\n",
343           ObjectHandle, SecurityInformation, SecurityDescriptor);
344 
345     *SecurityDescriptor = NULL;
346 
347     RtlAcquireResourceShared(&SampResource,
348                              TRUE);
349 
350     if (SecurityInformation & (DACL_SECURITY_INFORMATION |
351                                OWNER_SECURITY_INFORMATION |
352                                GROUP_SECURITY_INFORMATION))
353         DesiredAccess |= READ_CONTROL;
354 
355     if (SecurityInformation & SACL_SECURITY_INFORMATION)
356         DesiredAccess |= ACCESS_SYSTEM_SECURITY;
357 
358     /* Validate the server handle */
359     Status = SampValidateDbObject(ObjectHandle,
360                                   SamDbIgnoreObject,
361                                   DesiredAccess,
362                                   &SamObject);
363     if (!NT_SUCCESS(Status))
364         goto done;
365 
366     /* Get the size of the SD */
367     Status = SampGetObjectAttribute(SamObject,
368                                     L"SecDesc",
369                                     NULL,
370                                     NULL,
371                                     &RelativeSdSize);
372     if (!NT_SUCCESS(Status) && Status != STATUS_BUFFER_OVERFLOW)
373     {
374         TRACE("Status 0x%08lx\n", Status);
375         goto done;
376     }
377 
378     /* Allocate a buffer for the SD */
379     RelativeSd = midl_user_allocate(RelativeSdSize);
380     if (RelativeSd == NULL)
381     {
382         Status = STATUS_INSUFFICIENT_RESOURCES;
383         goto done;
384     }
385 
386     /* Get the SD */
387     Status = SampGetObjectAttribute(SamObject,
388                                     L"SecDesc",
389                                     NULL,
390                                     RelativeSd,
391                                     &RelativeSdSize);
392     if (!NT_SUCCESS(Status))
393     {
394         TRACE("Status 0x%08lx\n", Status);
395         goto done;
396     }
397 
398     /* Invalidate the SD information that was not requested */
399     if (!(SecurityInformation & OWNER_SECURITY_INFORMATION))
400         RelativeSd->Owner = 0;
401 
402     if (!(SecurityInformation & GROUP_SECURITY_INFORMATION))
403         RelativeSd->Group = 0;
404 
405     if (!(SecurityInformation & DACL_SECURITY_INFORMATION))
406         RelativeSd->Control &= ~SE_DACL_PRESENT;
407 
408     if (!(SecurityInformation & SACL_SECURITY_INFORMATION))
409         RelativeSd->Control &= ~SE_SACL_PRESENT;
410 
411     /* Calculate the required SD size */
412     Status = RtlMakeSelfRelativeSD(RelativeSd,
413                                    NULL,
414                                    &ResultSdSize);
415     if (Status != STATUS_BUFFER_TOO_SMALL)
416         goto done;
417 
418     /* Allocate a buffer for the new SD */
419     ResultSd = MIDL_user_allocate(ResultSdSize);
420     if (ResultSd == NULL)
421     {
422         Status = STATUS_INSUFFICIENT_RESOURCES;
423         goto done;
424     }
425 
426     /* Build the new SD */
427     Status = RtlMakeSelfRelativeSD(RelativeSd,
428                                    ResultSd,
429                                    &ResultSdSize);
430     if (!NT_SUCCESS(Status))
431         goto done;
432 
433     /* Allocate the SD data buffer */
434     SdData = midl_user_allocate(sizeof(SAMPR_SR_SECURITY_DESCRIPTOR));
435     if (SdData == NULL)
436     {
437         Status = STATUS_INSUFFICIENT_RESOURCES;
438         goto done;
439     }
440 
441     /* Fill the SD data buffer and return it to the caller */
442     SdData->Length = ResultSdSize;
443     SdData->SecurityDescriptor = (PBYTE)ResultSd;
444 
445     *SecurityDescriptor = SdData;
446 
447 done:
448     RtlReleaseResource(&SampResource);
449 
450     if (!NT_SUCCESS(Status))
451     {
452         if (ResultSd != NULL)
453             MIDL_user_free(ResultSd);
454     }
455 
456     if (RelativeSd != NULL)
457         MIDL_user_free(RelativeSd);
458 
459     return Status;
460 }
461 
462 
463 /* Function 4 */
464 NTSTATUS
465 NTAPI
466 SamrShutdownSamServer(IN SAMPR_HANDLE ServerHandle)
467 {
468     PSAM_DB_OBJECT ServerObject;
469     NTSTATUS Status;
470 
471     TRACE("SamrShutdownSamServer(%p)\n",
472           ServerHandle);
473 
474     RtlAcquireResourceShared(&SampResource,
475                              TRUE);
476 
477     /* Validate the server handle */
478     Status = SampValidateDbObject(ServerHandle,
479                                   SamDbServerObject,
480                                   SAM_SERVER_SHUTDOWN,
481                                   &ServerObject);
482 
483     RtlReleaseResource(&SampResource);
484 
485     if (!NT_SUCCESS(Status))
486         return Status;
487 
488     /* Shut the server down */
489     RpcMgmtStopServerListening(0);
490 
491     Status = SampShutdownDisplayCache();
492     if (!NT_SUCCESS(Status))
493     {
494         ERR("SampShutdownDisplayCache() failed (Status 0x%08lx)\n", Status);
495     }
496 
497     return STATUS_SUCCESS;
498 }
499 
500 
501 /* Function 5 */
502 NTSTATUS
503 NTAPI
504 SamrLookupDomainInSamServer(IN SAMPR_HANDLE ServerHandle,
505                             IN PRPC_UNICODE_STRING Name,
506                             OUT PRPC_SID *DomainId)
507 {
508     PSAM_DB_OBJECT ServerObject;
509     HANDLE DomainsKeyHandle = NULL;
510     HANDLE DomainKeyHandle = NULL;
511     WCHAR DomainKeyName[64];
512     ULONG Index;
513     WCHAR DomainNameString[MAX_COMPUTERNAME_LENGTH + 1];
514     UNICODE_STRING DomainName;
515     ULONG Length;
516     BOOL Found = FALSE;
517     NTSTATUS Status;
518 
519     TRACE("SamrLookupDomainInSamServer(%p %p %p)\n",
520           ServerHandle, Name, DomainId);
521 
522     RtlAcquireResourceShared(&SampResource,
523                              TRUE);
524 
525     /* Validate the server handle */
526     Status = SampValidateDbObject(ServerHandle,
527                                   SamDbServerObject,
528                                   SAM_SERVER_LOOKUP_DOMAIN,
529                                   &ServerObject);
530     if (!NT_SUCCESS(Status))
531         goto done;
532 
533     *DomainId = NULL;
534 
535     Status = SampRegOpenKey(ServerObject->KeyHandle,
536                             L"Domains",
537                             KEY_READ,
538                             &DomainsKeyHandle);
539     if (!NT_SUCCESS(Status))
540         goto done;
541 
542     Index = 0;
543     while (Found == FALSE)
544     {
545         Status = SampRegEnumerateSubKey(DomainsKeyHandle,
546                                         Index,
547                                         64,
548                                         DomainKeyName);
549         if (!NT_SUCCESS(Status))
550         {
551             if (Status == STATUS_NO_MORE_ENTRIES)
552                 Status = STATUS_NO_SUCH_DOMAIN;
553             break;
554         }
555 
556         TRACE("Domain key name: %S\n", DomainKeyName);
557 
558         Status = SampRegOpenKey(DomainsKeyHandle,
559                                 DomainKeyName,
560                                 KEY_READ,
561                                 &DomainKeyHandle);
562         if (NT_SUCCESS(Status))
563         {
564             Length = (MAX_COMPUTERNAME_LENGTH + 1) * sizeof(WCHAR);
565             Status = SampRegQueryValue(DomainKeyHandle,
566                                        L"Name",
567                                        NULL,
568                                        (PVOID)&DomainNameString,
569                                        &Length);
570             if (NT_SUCCESS(Status))
571             {
572                 TRACE("Domain name: %S\n", DomainNameString);
573 
574                 RtlInitUnicodeString(&DomainName,
575                                      DomainNameString);
576                 if (RtlEqualUnicodeString(&DomainName, (PUNICODE_STRING)Name, TRUE))
577                 {
578                    TRACE("Found it!\n");
579                    Found = TRUE;
580 
581                    Status = SampRegQueryValue(DomainKeyHandle,
582                                               L"SID",
583                                               NULL,
584                                               NULL,
585                                               &Length);
586                    if (NT_SUCCESS(Status))
587                    {
588                        *DomainId = midl_user_allocate(Length);
589 
590                        SampRegQueryValue(DomainKeyHandle,
591                                          L"SID",
592                                          NULL,
593                                          (PVOID)*DomainId,
594                                          &Length);
595 
596                        Status = STATUS_SUCCESS;
597                        break;
598                    }
599                 }
600             }
601 
602             SampRegCloseKey(&DomainKeyHandle);
603         }
604 
605         Index++;
606     }
607 
608 done:
609     SampRegCloseKey(&DomainKeyHandle);
610     SampRegCloseKey(&DomainsKeyHandle);
611 
612     RtlReleaseResource(&SampResource);
613 
614     return Status;
615 }
616 
617 
618 /* Function 6 */
619 NTSTATUS
620 NTAPI
621 SamrEnumerateDomainsInSamServer(IN SAMPR_HANDLE ServerHandle,
622                                 IN OUT unsigned long *EnumerationContext,
623                                 OUT PSAMPR_ENUMERATION_BUFFER *Buffer,
624                                 IN ULONG PreferedMaximumLength,
625                                 OUT PULONG CountReturned)
626 {
627     PSAM_DB_OBJECT ServerObject;
628     WCHAR DomainKeyName[64];
629     HANDLE DomainsKeyHandle = NULL;
630     HANDLE DomainKeyHandle = NULL;
631     ULONG EnumIndex;
632     ULONG EnumCount;
633     ULONG RequiredLength;
634     ULONG DataLength;
635     ULONG i;
636     PSAMPR_ENUMERATION_BUFFER EnumBuffer = NULL;
637     NTSTATUS Status;
638 
639     TRACE("SamrEnumerateDomainsInSamServer(%p %p %p %lu %p)\n",
640           ServerHandle, EnumerationContext, Buffer, PreferedMaximumLength,
641           CountReturned);
642 
643     RtlAcquireResourceShared(&SampResource,
644                              TRUE);
645 
646     /* Validate the server handle */
647     Status = SampValidateDbObject(ServerHandle,
648                                   SamDbServerObject,
649                                   SAM_SERVER_ENUMERATE_DOMAINS,
650                                   &ServerObject);
651     if (!NT_SUCCESS(Status))
652         goto done;
653 
654     Status = SampRegOpenKey(ServerObject->KeyHandle,
655                             L"Domains",
656                             KEY_READ,
657                             &DomainsKeyHandle);
658     if (!NT_SUCCESS(Status))
659         goto done;
660 
661     EnumIndex = *EnumerationContext;
662     EnumCount = 0;
663     RequiredLength = 0;
664 
665     while (TRUE)
666     {
667         Status = SampRegEnumerateSubKey(DomainsKeyHandle,
668                                         EnumIndex,
669                                         64 * sizeof(WCHAR),
670                                         DomainKeyName);
671         if (!NT_SUCCESS(Status))
672             break;
673 
674         TRACE("EnumIndex: %lu\n", EnumIndex);
675         TRACE("Domain key name: %S\n", DomainKeyName);
676 
677         Status = SampRegOpenKey(DomainsKeyHandle,
678                                 DomainKeyName,
679                                 KEY_READ,
680                                 &DomainKeyHandle);
681         TRACE("SampRegOpenKey returned %08lX\n", Status);
682         if (NT_SUCCESS(Status))
683         {
684             DataLength = 0;
685             Status = SampRegQueryValue(DomainKeyHandle,
686                                        L"Name",
687                                        NULL,
688                                        NULL,
689                                        &DataLength);
690             TRACE("SampRegQueryValue returned %08lX\n", Status);
691             if (NT_SUCCESS(Status))
692             {
693                 TRACE("Data length: %lu\n", DataLength);
694 
695                 if ((RequiredLength + DataLength + sizeof(UNICODE_STRING)) > PreferedMaximumLength)
696                     break;
697 
698                 RequiredLength += (DataLength + sizeof(UNICODE_STRING));
699                 EnumCount++;
700             }
701 
702             SampRegCloseKey(&DomainKeyHandle);
703         }
704 
705         EnumIndex++;
706     }
707 
708     TRACE("EnumCount: %lu\n", EnumCount);
709     TRACE("RequiredLength: %lu\n", RequiredLength);
710 
711     EnumBuffer = midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER));
712     if (EnumBuffer == NULL)
713     {
714         Status = STATUS_INSUFFICIENT_RESOURCES;
715         goto done;
716     }
717 
718     EnumBuffer->EntriesRead = EnumCount;
719     EnumBuffer->Buffer = midl_user_allocate(EnumCount * sizeof(SAMPR_RID_ENUMERATION));
720     if (EnumBuffer->Buffer == NULL)
721     {
722         Status = STATUS_INSUFFICIENT_RESOURCES;
723         goto done;
724     }
725 
726     EnumIndex = *EnumerationContext;
727     for (i = 0; i < EnumCount; i++, EnumIndex++)
728     {
729         Status = SampRegEnumerateSubKey(DomainsKeyHandle,
730                                         EnumIndex,
731                                         64 * sizeof(WCHAR),
732                                         DomainKeyName);
733         if (!NT_SUCCESS(Status))
734             break;
735 
736         TRACE("EnumIndex: %lu\n", EnumIndex);
737         TRACE("Domain key name: %S\n", DomainKeyName);
738 
739         Status = SampRegOpenKey(DomainsKeyHandle,
740                                 DomainKeyName,
741                                 KEY_READ,
742                                 &DomainKeyHandle);
743         TRACE("SampRegOpenKey returned %08lX\n", Status);
744         if (NT_SUCCESS(Status))
745         {
746             DataLength = 0;
747             Status = SampRegQueryValue(DomainKeyHandle,
748                                        L"Name",
749                                        NULL,
750                                        NULL,
751                                        &DataLength);
752             TRACE("SampRegQueryValue returned %08lX\n", Status);
753             if (NT_SUCCESS(Status))
754             {
755                 EnumBuffer->Buffer[i].RelativeId = 0;
756                 EnumBuffer->Buffer[i].Name.Length = (USHORT)DataLength - sizeof(WCHAR);
757                 EnumBuffer->Buffer[i].Name.MaximumLength = (USHORT)DataLength;
758                 EnumBuffer->Buffer[i].Name.Buffer = midl_user_allocate(DataLength);
759                 if (EnumBuffer->Buffer[i].Name.Buffer == NULL)
760                 {
761                     SampRegCloseKey(&DomainKeyHandle);
762                     Status = STATUS_INSUFFICIENT_RESOURCES;
763                     goto done;
764                 }
765 
766                 Status = SampRegQueryValue(DomainKeyHandle,
767                                            L"Name",
768                                            NULL,
769                                            EnumBuffer->Buffer[i].Name.Buffer,
770                                            &DataLength);
771                 TRACE("SampRegQueryValue returned %08lX\n", Status);
772                 if (NT_SUCCESS(Status))
773                 {
774                     TRACE("Domain name: %S\n", EnumBuffer->Buffer[i].Name.Buffer);
775                 }
776             }
777 
778             SampRegCloseKey(&DomainKeyHandle);
779 
780             if (!NT_SUCCESS(Status))
781                 goto done;
782         }
783     }
784 
785     if (NT_SUCCESS(Status))
786     {
787         *EnumerationContext += EnumCount;
788         *Buffer = EnumBuffer;
789         *CountReturned = EnumCount;
790     }
791 
792 done:
793     SampRegCloseKey(&DomainKeyHandle);
794     SampRegCloseKey(&DomainsKeyHandle);
795 
796     if (!NT_SUCCESS(Status))
797     {
798         *EnumerationContext = 0;
799         *Buffer = NULL;
800         *CountReturned = 0;
801 
802         if (EnumBuffer != NULL)
803         {
804             if (EnumBuffer->Buffer != NULL)
805             {
806                 if (EnumBuffer->EntriesRead != 0)
807                 {
808                     for (i = 0; i < EnumBuffer->EntriesRead; i++)
809                     {
810                         if (EnumBuffer->Buffer[i].Name.Buffer != NULL)
811                             midl_user_free(EnumBuffer->Buffer[i].Name.Buffer);
812                     }
813                 }
814 
815                 midl_user_free(EnumBuffer->Buffer);
816             }
817 
818             midl_user_free(EnumBuffer);
819         }
820     }
821 
822     RtlReleaseResource(&SampResource);
823 
824     return Status;
825 }
826 
827 
828 /* Function 7 */
829 NTSTATUS
830 NTAPI
831 SamrOpenDomain(IN SAMPR_HANDLE ServerHandle,
832                IN ACCESS_MASK DesiredAccess,
833                IN PRPC_SID DomainId,
834                OUT SAMPR_HANDLE *DomainHandle)
835 {
836     PSAM_DB_OBJECT ServerObject;
837     PSAM_DB_OBJECT DomainObject;
838     NTSTATUS Status;
839 
840     TRACE("SamrOpenDomain(%p %lx %p %p)\n",
841           ServerHandle, DesiredAccess, DomainId, DomainHandle);
842 
843     /* Map generic access rights */
844     RtlMapGenericMask(&DesiredAccess,
845                       &DomainMapping);
846 
847     RtlAcquireResourceShared(&SampResource,
848                              TRUE);
849 
850     /* Validate the server handle */
851     Status = SampValidateDbObject(ServerHandle,
852                                   SamDbServerObject,
853                                   SAM_SERVER_LOOKUP_DOMAIN,
854                                   &ServerObject);
855     if (!NT_SUCCESS(Status))
856         return Status;
857 
858     /* Validate the Domain SID */
859     if ((DomainId->Revision != SID_REVISION) ||
860         (DomainId->SubAuthorityCount > SID_MAX_SUB_AUTHORITIES) ||
861         (memcmp(&DomainId->IdentifierAuthority, &NtSidAuthority, sizeof(SID_IDENTIFIER_AUTHORITY)) != 0))
862         return STATUS_INVALID_PARAMETER;
863 
864     /* Open the domain object */
865     if ((DomainId->SubAuthorityCount == 1) &&
866         (DomainId->SubAuthority[0] == SECURITY_BUILTIN_DOMAIN_RID))
867     {
868         /* Builtin domain object */
869         TRACE("Opening the builtin domain object.\n");
870 
871         Status = SampOpenDbObject(ServerObject,
872                                   L"Domains",
873                                   L"Builtin",
874                                   0,
875                                   SamDbDomainObject,
876                                   DesiredAccess,
877                                   &DomainObject);
878     }
879     else if ((DomainId->SubAuthorityCount == 4) &&
880              (DomainId->SubAuthority[0] == SECURITY_NT_NON_UNIQUE))
881     {
882         /* Account domain object */
883         TRACE("Opening the account domain object.\n");
884 
885         /* FIXME: Check the account domain sub authorities!!! */
886 
887         Status = SampOpenDbObject(ServerObject,
888                                   L"Domains",
889                                   L"Account",
890                                   0,
891                                   SamDbDomainObject,
892                                   DesiredAccess,
893                                   &DomainObject);
894     }
895     else
896     {
897         /* No valid domain SID */
898         Status = STATUS_INVALID_PARAMETER;
899     }
900 
901     if (NT_SUCCESS(Status))
902         *DomainHandle = (SAMPR_HANDLE)DomainObject;
903 
904     RtlReleaseResource(&SampResource);
905 
906     TRACE("SamrOpenDomain done (Status 0x%08lx)\n", Status);
907 
908     return Status;
909 }
910 
911 
912 static NTSTATUS
913 SampQueryDomainPassword(PSAM_DB_OBJECT DomainObject,
914                         PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
915 {
916     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
917     SAM_DOMAIN_FIXED_DATA FixedData;
918     ULONG Length = 0;
919     NTSTATUS Status;
920 
921     *Buffer = NULL;
922 
923     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
924     if (InfoBuffer == NULL)
925         return STATUS_INSUFFICIENT_RESOURCES;
926 
927     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
928     Status = SampGetObjectAttribute(DomainObject,
929                                     L"F",
930                                     NULL,
931                                     (PVOID)&FixedData,
932                                     &Length);
933     if (!NT_SUCCESS(Status))
934         goto done;
935 
936     InfoBuffer->Password.MinPasswordLength = FixedData.MinPasswordLength;
937     InfoBuffer->Password.PasswordHistoryLength = FixedData.PasswordHistoryLength;
938     InfoBuffer->Password.PasswordProperties = FixedData.PasswordProperties;
939     InfoBuffer->Password.MaxPasswordAge.LowPart = FixedData.MaxPasswordAge.LowPart;
940     InfoBuffer->Password.MaxPasswordAge.HighPart = FixedData.MaxPasswordAge.HighPart;
941     InfoBuffer->Password.MinPasswordAge.LowPart = FixedData.MinPasswordAge.LowPart;
942     InfoBuffer->Password.MinPasswordAge.HighPart = FixedData.MinPasswordAge.HighPart;
943 
944     *Buffer = InfoBuffer;
945 
946 done:
947     if (!NT_SUCCESS(Status))
948     {
949         if (InfoBuffer != NULL)
950         {
951             midl_user_free(InfoBuffer);
952         }
953     }
954 
955     return Status;
956 }
957 
958 
959 static NTSTATUS
960 SampGetNumberOfAccounts(PSAM_DB_OBJECT DomainObject,
961                         LPCWSTR AccountType,
962                         PULONG Count)
963 {
964     HANDLE AccountKeyHandle = NULL;
965     HANDLE NamesKeyHandle = NULL;
966     NTSTATUS Status;
967 
968     *Count = 0;
969 
970     Status = SampRegOpenKey(DomainObject->KeyHandle,
971                             AccountType,
972                             KEY_READ,
973                             &AccountKeyHandle);
974     if (!NT_SUCCESS(Status))
975         return Status;
976 
977     Status = SampRegOpenKey(AccountKeyHandle,
978                             L"Names",
979                             KEY_READ,
980                             &NamesKeyHandle);
981     if (!NT_SUCCESS(Status))
982         goto done;
983 
984     Status = SampRegQueryKeyInfo(NamesKeyHandle,
985                                  NULL,
986                                  Count);
987 
988 done:
989     SampRegCloseKey(&NamesKeyHandle);
990     SampRegCloseKey(&AccountKeyHandle);
991 
992     return Status;
993 }
994 
995 
996 static NTSTATUS
997 SampQueryDomainGeneral(PSAM_DB_OBJECT DomainObject,
998                        PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
999 {
1000     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1001     SAM_DOMAIN_FIXED_DATA FixedData;
1002     ULONG Length = 0;
1003     NTSTATUS Status;
1004 
1005     *Buffer = NULL;
1006 
1007     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1008     if (InfoBuffer == NULL)
1009         return STATUS_INSUFFICIENT_RESOURCES;
1010 
1011     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1012     Status = SampGetObjectAttribute(DomainObject,
1013                                     L"F",
1014                                     NULL,
1015                                     (PVOID)&FixedData,
1016                                     &Length);
1017     if (!NT_SUCCESS(Status))
1018         goto done;
1019 
1020     InfoBuffer->General.ForceLogoff.LowPart = FixedData.ForceLogoff.LowPart;
1021     InfoBuffer->General.ForceLogoff.HighPart = FixedData.ForceLogoff.HighPart;
1022     InfoBuffer->General.DomainModifiedCount.LowPart = FixedData.DomainModifiedCount.LowPart;
1023     InfoBuffer->General.DomainModifiedCount.HighPart = FixedData.DomainModifiedCount.HighPart;
1024     InfoBuffer->General.DomainServerState = FixedData.DomainServerState;
1025     InfoBuffer->General.DomainServerRole = FixedData.DomainServerRole;
1026     InfoBuffer->General.UasCompatibilityRequired = FixedData.UasCompatibilityRequired;
1027 
1028     /* Get the OemInformation string */
1029     Status = SampGetObjectAttributeString(DomainObject,
1030                                           L"OemInformation",
1031                                           &InfoBuffer->General.OemInformation);
1032     if (!NT_SUCCESS(Status))
1033     {
1034         TRACE("Status 0x%08lx\n", Status);
1035         goto done;
1036     }
1037 
1038     /* Get the Name string */
1039     Status = SampGetObjectAttributeString(DomainObject,
1040                                           L"Name",
1041                                           &InfoBuffer->General.DomainName);
1042     if (!NT_SUCCESS(Status))
1043     {
1044         TRACE("Status 0x%08lx\n", Status);
1045         goto done;
1046     }
1047 
1048     /* Get the ReplicaSourceNodeName string */
1049     Status = SampGetObjectAttributeString(DomainObject,
1050                                           L"ReplicaSourceNodeName",
1051                                           &InfoBuffer->General.ReplicaSourceNodeName);
1052     if (!NT_SUCCESS(Status))
1053     {
1054         TRACE("Status 0x%08lx\n", Status);
1055         goto done;
1056     }
1057 
1058     /* Get the number of Users in the Domain */
1059     Status = SampGetNumberOfAccounts(DomainObject,
1060                                      L"Users",
1061                                      &InfoBuffer->General.UserCount);
1062     if (!NT_SUCCESS(Status))
1063     {
1064         TRACE("Status 0x%08lx\n", Status);
1065         goto done;
1066     }
1067 
1068     /* Get the number of Groups in the Domain */
1069     Status = SampGetNumberOfAccounts(DomainObject,
1070                                      L"Groups",
1071                                      &InfoBuffer->General.GroupCount);
1072     if (!NT_SUCCESS(Status))
1073     {
1074         TRACE("Status 0x%08lx\n", Status);
1075         goto done;
1076     }
1077 
1078     /* Get the number of Aliases in the Domain */
1079     Status = SampGetNumberOfAccounts(DomainObject,
1080                                      L"Aliases",
1081                                      &InfoBuffer->General.AliasCount);
1082     if (!NT_SUCCESS(Status))
1083     {
1084         TRACE("Status 0x%08lx\n", Status);
1085         goto done;
1086     }
1087 
1088     *Buffer = InfoBuffer;
1089 
1090 done:
1091     if (!NT_SUCCESS(Status))
1092     {
1093         if (InfoBuffer != NULL)
1094         {
1095             if (InfoBuffer->General.OemInformation.Buffer != NULL)
1096                 midl_user_free(InfoBuffer->General.OemInformation.Buffer);
1097 
1098             if (InfoBuffer->General.DomainName.Buffer != NULL)
1099                 midl_user_free(InfoBuffer->General.DomainName.Buffer);
1100 
1101             if (InfoBuffer->General.ReplicaSourceNodeName.Buffer != NULL)
1102                 midl_user_free(InfoBuffer->General.ReplicaSourceNodeName.Buffer);
1103 
1104             midl_user_free(InfoBuffer);
1105         }
1106     }
1107 
1108     return Status;
1109 }
1110 
1111 
1112 static NTSTATUS
1113 SampQueryDomainLogoff(PSAM_DB_OBJECT DomainObject,
1114                       PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1115 {
1116     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1117     SAM_DOMAIN_FIXED_DATA FixedData;
1118     ULONG Length = 0;
1119     NTSTATUS Status;
1120 
1121     *Buffer = NULL;
1122 
1123     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1124     if (InfoBuffer == NULL)
1125         return STATUS_INSUFFICIENT_RESOURCES;
1126 
1127     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1128     Status = SampGetObjectAttribute(DomainObject,
1129                                     L"F",
1130                                     NULL,
1131                                     (PVOID)&FixedData,
1132                                     &Length);
1133     if (!NT_SUCCESS(Status))
1134         goto done;
1135 
1136     InfoBuffer->Logoff.ForceLogoff.LowPart = FixedData.ForceLogoff.LowPart;
1137     InfoBuffer->Logoff.ForceLogoff.HighPart = FixedData.ForceLogoff.HighPart;
1138 
1139     *Buffer = InfoBuffer;
1140 
1141 done:
1142     if (!NT_SUCCESS(Status))
1143     {
1144         if (InfoBuffer != NULL)
1145         {
1146             midl_user_free(InfoBuffer);
1147         }
1148     }
1149 
1150     return Status;
1151 }
1152 
1153 
1154 static NTSTATUS
1155 SampQueryDomainOem(PSAM_DB_OBJECT DomainObject,
1156                    PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1157 {
1158     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1159     NTSTATUS Status;
1160 
1161     *Buffer = NULL;
1162 
1163     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1164     if (InfoBuffer == NULL)
1165         return STATUS_INSUFFICIENT_RESOURCES;
1166 
1167     /* Get the OemInformation string */
1168     Status = SampGetObjectAttributeString(DomainObject,
1169                                           L"OemInformation",
1170                                           &InfoBuffer->Oem.OemInformation);
1171     if (!NT_SUCCESS(Status))
1172     {
1173         TRACE("Status 0x%08lx\n", Status);
1174         goto done;
1175     }
1176 
1177     *Buffer = InfoBuffer;
1178 
1179 done:
1180     if (!NT_SUCCESS(Status))
1181     {
1182         if (InfoBuffer != NULL)
1183         {
1184             if (InfoBuffer->Oem.OemInformation.Buffer != NULL)
1185                 midl_user_free(InfoBuffer->Oem.OemInformation.Buffer);
1186 
1187             midl_user_free(InfoBuffer);
1188         }
1189     }
1190 
1191     return Status;
1192 }
1193 
1194 
1195 static NTSTATUS
1196 SampQueryDomainName(PSAM_DB_OBJECT DomainObject,
1197                     PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1198 {
1199     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1200     NTSTATUS Status;
1201 
1202     *Buffer = NULL;
1203 
1204     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1205     if (InfoBuffer == NULL)
1206         return STATUS_INSUFFICIENT_RESOURCES;
1207 
1208     /* Get the Name string */
1209     Status = SampGetObjectAttributeString(DomainObject,
1210                                           L"Name",
1211                                           &InfoBuffer->Name.DomainName);
1212     if (!NT_SUCCESS(Status))
1213     {
1214         TRACE("Status 0x%08lx\n", Status);
1215         goto done;
1216     }
1217 
1218     *Buffer = InfoBuffer;
1219 
1220 done:
1221     if (!NT_SUCCESS(Status))
1222     {
1223         if (InfoBuffer != NULL)
1224         {
1225             if (InfoBuffer->Name.DomainName.Buffer != NULL)
1226                 midl_user_free(InfoBuffer->Name.DomainName.Buffer);
1227 
1228             midl_user_free(InfoBuffer);
1229         }
1230     }
1231 
1232     return Status;
1233 }
1234 
1235 
1236 static NTSTATUS
1237 SampQueryDomainReplication(PSAM_DB_OBJECT DomainObject,
1238                            PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1239 {
1240     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1241     NTSTATUS Status;
1242 
1243     *Buffer = NULL;
1244 
1245     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1246     if (InfoBuffer == NULL)
1247         return STATUS_INSUFFICIENT_RESOURCES;
1248 
1249     /* Get the ReplicaSourceNodeName string */
1250     Status = SampGetObjectAttributeString(DomainObject,
1251                                           L"ReplicaSourceNodeName",
1252                                           &InfoBuffer->Replication.ReplicaSourceNodeName);
1253     if (!NT_SUCCESS(Status))
1254     {
1255         TRACE("Status 0x%08lx\n", Status);
1256         goto done;
1257     }
1258 
1259     *Buffer = InfoBuffer;
1260 
1261 done:
1262     if (!NT_SUCCESS(Status))
1263     {
1264         if (InfoBuffer != NULL)
1265         {
1266             if (InfoBuffer->Replication.ReplicaSourceNodeName.Buffer != NULL)
1267                 midl_user_free(InfoBuffer->Replication.ReplicaSourceNodeName.Buffer);
1268 
1269             midl_user_free(InfoBuffer);
1270         }
1271     }
1272 
1273     return Status;
1274 }
1275 
1276 
1277 static NTSTATUS
1278 SampQueryDomainServerRole(PSAM_DB_OBJECT DomainObject,
1279                           PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1280 {
1281     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1282     SAM_DOMAIN_FIXED_DATA FixedData;
1283     ULONG Length = 0;
1284     NTSTATUS Status;
1285 
1286     *Buffer = NULL;
1287 
1288     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1289     if (InfoBuffer == NULL)
1290         return STATUS_INSUFFICIENT_RESOURCES;
1291 
1292     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1293     Status = SampGetObjectAttribute(DomainObject,
1294                                     L"F",
1295                                     NULL,
1296                                     (PVOID)&FixedData,
1297                                     &Length);
1298     if (!NT_SUCCESS(Status))
1299         goto done;
1300 
1301     InfoBuffer->Role.DomainServerRole = FixedData.DomainServerRole;
1302 
1303     *Buffer = InfoBuffer;
1304 
1305 done:
1306     if (!NT_SUCCESS(Status))
1307     {
1308         if (InfoBuffer != NULL)
1309         {
1310             midl_user_free(InfoBuffer);
1311         }
1312     }
1313 
1314     return Status;
1315 }
1316 
1317 
1318 static NTSTATUS
1319 SampQueryDomainModified(PSAM_DB_OBJECT DomainObject,
1320                         PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1321 {
1322     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1323     SAM_DOMAIN_FIXED_DATA FixedData;
1324     ULONG Length = 0;
1325     NTSTATUS Status;
1326 
1327     *Buffer = NULL;
1328 
1329     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1330     if (InfoBuffer == NULL)
1331         return STATUS_INSUFFICIENT_RESOURCES;
1332 
1333     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1334     Status = SampGetObjectAttribute(DomainObject,
1335                                     L"F",
1336                                     NULL,
1337                                     (PVOID)&FixedData,
1338                                     &Length);
1339     if (!NT_SUCCESS(Status))
1340         goto done;
1341 
1342     InfoBuffer->Modified.DomainModifiedCount.LowPart = FixedData.DomainModifiedCount.LowPart;
1343     InfoBuffer->Modified.DomainModifiedCount.HighPart = FixedData.DomainModifiedCount.HighPart;
1344     InfoBuffer->Modified.CreationTime.LowPart = FixedData.CreationTime.LowPart;
1345     InfoBuffer->Modified.CreationTime.HighPart = FixedData.CreationTime.HighPart;
1346 
1347     *Buffer = InfoBuffer;
1348 
1349 done:
1350     if (!NT_SUCCESS(Status))
1351     {
1352         if (InfoBuffer != NULL)
1353         {
1354             midl_user_free(InfoBuffer);
1355         }
1356     }
1357 
1358     return Status;
1359 }
1360 
1361 
1362 static NTSTATUS
1363 SampQueryDomainState(PSAM_DB_OBJECT DomainObject,
1364                      PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1365 {
1366     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1367     SAM_DOMAIN_FIXED_DATA FixedData;
1368     ULONG Length = 0;
1369     NTSTATUS Status;
1370 
1371     *Buffer = NULL;
1372 
1373     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1374     if (InfoBuffer == NULL)
1375         return STATUS_INSUFFICIENT_RESOURCES;
1376 
1377     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1378     Status = SampGetObjectAttribute(DomainObject,
1379                                     L"F",
1380                                     NULL,
1381                                     (PVOID)&FixedData,
1382                                     &Length);
1383     if (!NT_SUCCESS(Status))
1384         goto done;
1385 
1386     InfoBuffer->State.DomainServerState = FixedData.DomainServerState;
1387 
1388     *Buffer = InfoBuffer;
1389 
1390 done:
1391     if (!NT_SUCCESS(Status))
1392     {
1393         if (InfoBuffer != NULL)
1394         {
1395             midl_user_free(InfoBuffer);
1396         }
1397     }
1398 
1399     return Status;
1400 }
1401 
1402 
1403 static NTSTATUS
1404 SampQueryDomainGeneral2(PSAM_DB_OBJECT DomainObject,
1405                         PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1406 {
1407     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1408     SAM_DOMAIN_FIXED_DATA FixedData;
1409     ULONG Length = 0;
1410     NTSTATUS Status;
1411 
1412     *Buffer = NULL;
1413 
1414     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1415     if (InfoBuffer == NULL)
1416         return STATUS_INSUFFICIENT_RESOURCES;
1417 
1418     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1419     Status = SampGetObjectAttribute(DomainObject,
1420                                     L"F",
1421                                     NULL,
1422                                     (PVOID)&FixedData,
1423                                     &Length);
1424     if (!NT_SUCCESS(Status))
1425         goto done;
1426 
1427     InfoBuffer->General2.I1.ForceLogoff.LowPart = FixedData.ForceLogoff.LowPart;
1428     InfoBuffer->General2.I1.ForceLogoff.HighPart = FixedData.ForceLogoff.HighPart;
1429     InfoBuffer->General2.I1.DomainModifiedCount.LowPart = FixedData.DomainModifiedCount.LowPart;
1430     InfoBuffer->General2.I1.DomainModifiedCount.HighPart = FixedData.DomainModifiedCount.HighPart;
1431     InfoBuffer->General2.I1.DomainServerState = FixedData.DomainServerState;
1432     InfoBuffer->General2.I1.DomainServerRole = FixedData.DomainServerRole;
1433     InfoBuffer->General2.I1.UasCompatibilityRequired = FixedData.UasCompatibilityRequired;
1434 
1435     InfoBuffer->General2.LockoutDuration = FixedData.LockoutDuration;
1436     InfoBuffer->General2.LockoutObservationWindow = FixedData.LockoutObservationWindow;
1437     InfoBuffer->General2.LockoutThreshold = FixedData.LockoutThreshold;
1438 
1439     /* Get the OemInformation string */
1440     Status = SampGetObjectAttributeString(DomainObject,
1441                                           L"OemInformation",
1442                                           &InfoBuffer->General2.I1.OemInformation);
1443     if (!NT_SUCCESS(Status))
1444     {
1445         TRACE("Status 0x%08lx\n", Status);
1446         goto done;
1447     }
1448 
1449     /* Get the Name string */
1450     Status = SampGetObjectAttributeString(DomainObject,
1451                                           L"Name",
1452                                           &InfoBuffer->General2.I1.DomainName);
1453     if (!NT_SUCCESS(Status))
1454     {
1455         TRACE("Status 0x%08lx\n", Status);
1456         goto done;
1457     }
1458 
1459     /* Get the ReplicaSourceNodeName string */
1460     Status = SampGetObjectAttributeString(DomainObject,
1461                                           L"ReplicaSourceNodeName",
1462                                           &InfoBuffer->General2.I1.ReplicaSourceNodeName);
1463     if (!NT_SUCCESS(Status))
1464     {
1465         TRACE("Status 0x%08lx\n", Status);
1466         goto done;
1467     }
1468 
1469     /* Get the number of Users in the Domain */
1470     Status = SampGetNumberOfAccounts(DomainObject,
1471                                      L"Users",
1472                                      &InfoBuffer->General2.I1.UserCount);
1473     if (!NT_SUCCESS(Status))
1474     {
1475         TRACE("Status 0x%08lx\n", Status);
1476         goto done;
1477     }
1478 
1479     /* Get the number of Groups in the Domain */
1480     Status = SampGetNumberOfAccounts(DomainObject,
1481                                      L"Groups",
1482                                      &InfoBuffer->General2.I1.GroupCount);
1483     if (!NT_SUCCESS(Status))
1484     {
1485         TRACE("Status 0x%08lx\n", Status);
1486         goto done;
1487     }
1488 
1489     /* Get the number of Aliases in the Domain */
1490     Status = SampGetNumberOfAccounts(DomainObject,
1491                                      L"Aliases",
1492                                      &InfoBuffer->General2.I1.AliasCount);
1493     if (!NT_SUCCESS(Status))
1494     {
1495         TRACE("Status 0x%08lx\n", Status);
1496         goto done;
1497     }
1498 
1499     *Buffer = InfoBuffer;
1500 
1501 done:
1502     if (!NT_SUCCESS(Status))
1503     {
1504         if (InfoBuffer != NULL)
1505         {
1506             if (InfoBuffer->General2.I1.OemInformation.Buffer != NULL)
1507                 midl_user_free(InfoBuffer->General2.I1.OemInformation.Buffer);
1508 
1509             if (InfoBuffer->General2.I1.DomainName.Buffer != NULL)
1510                 midl_user_free(InfoBuffer->General2.I1.DomainName.Buffer);
1511 
1512             if (InfoBuffer->General2.I1.ReplicaSourceNodeName.Buffer != NULL)
1513                 midl_user_free(InfoBuffer->General2.I1.ReplicaSourceNodeName.Buffer);
1514 
1515             midl_user_free(InfoBuffer);
1516         }
1517     }
1518 
1519     return Status;
1520 }
1521 
1522 
1523 static NTSTATUS
1524 SampQueryDomainLockout(PSAM_DB_OBJECT DomainObject,
1525                        PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1526 {
1527     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1528     SAM_DOMAIN_FIXED_DATA FixedData;
1529     ULONG Length = 0;
1530     NTSTATUS Status;
1531 
1532     *Buffer = NULL;
1533 
1534     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1535     if (InfoBuffer == NULL)
1536         return STATUS_INSUFFICIENT_RESOURCES;
1537 
1538     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1539     Status = SampGetObjectAttribute(DomainObject,
1540                                     L"F",
1541                                     NULL,
1542                                     (PVOID)&FixedData,
1543                                     &Length);
1544     if (!NT_SUCCESS(Status))
1545         goto done;
1546 
1547     InfoBuffer->Lockout.LockoutDuration = FixedData.LockoutDuration;
1548     InfoBuffer->Lockout.LockoutObservationWindow = FixedData.LockoutObservationWindow;
1549     InfoBuffer->Lockout.LockoutThreshold = FixedData.LockoutThreshold;
1550 
1551     *Buffer = InfoBuffer;
1552 
1553 done:
1554     if (!NT_SUCCESS(Status))
1555     {
1556         if (InfoBuffer != NULL)
1557         {
1558             midl_user_free(InfoBuffer);
1559         }
1560     }
1561 
1562     return Status;
1563 }
1564 
1565 
1566 static NTSTATUS
1567 SampQueryDomainModified2(PSAM_DB_OBJECT DomainObject,
1568                         PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1569 {
1570     PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer = NULL;
1571     SAM_DOMAIN_FIXED_DATA FixedData;
1572     ULONG Length = 0;
1573     NTSTATUS Status;
1574 
1575     *Buffer = NULL;
1576 
1577     InfoBuffer = midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER));
1578     if (InfoBuffer == NULL)
1579         return STATUS_INSUFFICIENT_RESOURCES;
1580 
1581     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1582     Status = SampGetObjectAttribute(DomainObject,
1583                                     L"F",
1584                                     NULL,
1585                                     (PVOID)&FixedData,
1586                                     &Length);
1587     if (!NT_SUCCESS(Status))
1588         goto done;
1589 
1590     InfoBuffer->Modified2.DomainModifiedCount.LowPart = FixedData.DomainModifiedCount.LowPart;
1591     InfoBuffer->Modified2.DomainModifiedCount.HighPart = FixedData.DomainModifiedCount.HighPart;
1592     InfoBuffer->Modified2.CreationTime.LowPart = FixedData.CreationTime.LowPart;
1593     InfoBuffer->Modified2.CreationTime.HighPart = FixedData.CreationTime.HighPart;
1594     InfoBuffer->Modified2.ModifiedCountAtLastPromotion.LowPart = FixedData.ModifiedCountAtLastPromotion.LowPart;
1595     InfoBuffer->Modified2.ModifiedCountAtLastPromotion.HighPart = FixedData.ModifiedCountAtLastPromotion.HighPart;
1596 
1597     *Buffer = InfoBuffer;
1598 
1599 done:
1600     if (!NT_SUCCESS(Status))
1601     {
1602         if (InfoBuffer != NULL)
1603         {
1604             midl_user_free(InfoBuffer);
1605         }
1606     }
1607 
1608     return Status;
1609 }
1610 
1611 
1612 /* Function 8 */
1613 NTSTATUS
1614 NTAPI
1615 SamrQueryInformationDomain(IN SAMPR_HANDLE DomainHandle,
1616                            IN DOMAIN_INFORMATION_CLASS DomainInformationClass,
1617                            OUT PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
1618 {
1619     TRACE("SamrQueryInformationDomain(%p %lu %p)\n",
1620           DomainHandle, DomainInformationClass, Buffer);
1621 
1622     return SamrQueryInformationDomain2(DomainHandle,
1623                                        DomainInformationClass,
1624                                        Buffer);
1625 }
1626 
1627 
1628 static NTSTATUS
1629 SampSetDomainPassword(PSAM_DB_OBJECT DomainObject,
1630                       PSAMPR_DOMAIN_INFO_BUFFER Buffer)
1631 {
1632     SAM_DOMAIN_FIXED_DATA FixedData;
1633     ULONG Length = 0;
1634     NTSTATUS Status;
1635 
1636     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1637     Status = SampGetObjectAttribute(DomainObject,
1638                                     L"F",
1639                                     NULL,
1640                                     (PVOID)&FixedData,
1641                                     &Length);
1642     if (!NT_SUCCESS(Status))
1643         goto done;
1644 
1645     FixedData.MinPasswordLength = Buffer->Password.MinPasswordLength;
1646     FixedData.PasswordHistoryLength = Buffer->Password.PasswordHistoryLength;
1647     FixedData.PasswordProperties = Buffer->Password.PasswordProperties;
1648     FixedData.MaxPasswordAge.LowPart = Buffer->Password.MaxPasswordAge.LowPart;
1649     FixedData.MaxPasswordAge.HighPart = Buffer->Password.MaxPasswordAge.HighPart;
1650     FixedData.MinPasswordAge.LowPart = Buffer->Password.MinPasswordAge.LowPart;
1651     FixedData.MinPasswordAge.HighPart = Buffer->Password.MinPasswordAge.HighPart;
1652 
1653     Status = SampSetObjectAttribute(DomainObject,
1654                                     L"F",
1655                                     REG_BINARY,
1656                                     &FixedData,
1657                                     Length);
1658 
1659 done:
1660     return Status;
1661 }
1662 
1663 
1664 static NTSTATUS
1665 SampSetDomainLogoff(PSAM_DB_OBJECT DomainObject,
1666                     PSAMPR_DOMAIN_INFO_BUFFER Buffer)
1667 {
1668     SAM_DOMAIN_FIXED_DATA FixedData;
1669     ULONG Length = 0;
1670     NTSTATUS Status;
1671 
1672     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1673     Status = SampGetObjectAttribute(DomainObject,
1674                                     L"F",
1675                                     NULL,
1676                                     (PVOID)&FixedData,
1677                                     &Length);
1678     if (!NT_SUCCESS(Status))
1679         goto done;
1680 
1681     FixedData.ForceLogoff.LowPart = Buffer->Logoff.ForceLogoff.LowPart;
1682     FixedData.ForceLogoff.HighPart = Buffer->Logoff.ForceLogoff.HighPart;
1683 
1684     Status = SampSetObjectAttribute(DomainObject,
1685                                     L"F",
1686                                     REG_BINARY,
1687                                     &FixedData,
1688                                     Length);
1689 
1690 done:
1691     return Status;
1692 }
1693 
1694 
1695 static NTSTATUS
1696 SampSetDomainServerRole(PSAM_DB_OBJECT DomainObject,
1697                         PSAMPR_DOMAIN_INFO_BUFFER Buffer)
1698 {
1699     SAM_DOMAIN_FIXED_DATA FixedData;
1700     ULONG Length = 0;
1701     NTSTATUS Status;
1702 
1703     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1704     Status = SampGetObjectAttribute(DomainObject,
1705                                     L"F",
1706                                     NULL,
1707                                     (PVOID)&FixedData,
1708                                     &Length);
1709     if (!NT_SUCCESS(Status))
1710         goto done;
1711 
1712     FixedData.DomainServerRole = Buffer->Role.DomainServerRole;
1713 
1714     Status = SampSetObjectAttribute(DomainObject,
1715                                     L"F",
1716                                     REG_BINARY,
1717                                     &FixedData,
1718                                     Length);
1719 
1720 done:
1721     return Status;
1722 }
1723 
1724 
1725 static NTSTATUS
1726 SampSetDomainState(PSAM_DB_OBJECT DomainObject,
1727                    PSAMPR_DOMAIN_INFO_BUFFER Buffer)
1728 {
1729     SAM_DOMAIN_FIXED_DATA FixedData;
1730     ULONG Length = 0;
1731     NTSTATUS Status;
1732 
1733     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1734     Status = SampGetObjectAttribute(DomainObject,
1735                                     L"F",
1736                                     NULL,
1737                                     (PVOID)&FixedData,
1738                                     &Length);
1739     if (!NT_SUCCESS(Status))
1740         goto done;
1741 
1742     FixedData.DomainServerState = Buffer->State.DomainServerState;
1743 
1744     Status = SampSetObjectAttribute(DomainObject,
1745                                     L"F",
1746                                     REG_BINARY,
1747                                     &FixedData,
1748                                     Length);
1749 
1750 done:
1751     return Status;
1752 }
1753 
1754 
1755 static NTSTATUS
1756 SampSetDomainLockout(PSAM_DB_OBJECT DomainObject,
1757                      PSAMPR_DOMAIN_INFO_BUFFER Buffer)
1758 {
1759     SAM_DOMAIN_FIXED_DATA FixedData;
1760     ULONG Length = 0;
1761     NTSTATUS Status;
1762 
1763     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
1764     Status = SampGetObjectAttribute(DomainObject,
1765                                     L"F",
1766                                     NULL,
1767                                     (PVOID)&FixedData,
1768                                     &Length);
1769     if (!NT_SUCCESS(Status))
1770         goto done;
1771 
1772     FixedData.LockoutDuration = Buffer->Lockout.LockoutDuration;
1773     FixedData.LockoutObservationWindow = Buffer->Lockout.LockoutObservationWindow;
1774     FixedData.LockoutThreshold = Buffer->Lockout.LockoutThreshold;
1775 
1776     Status = SampSetObjectAttribute(DomainObject,
1777                                     L"F",
1778                                     REG_BINARY,
1779                                     &FixedData,
1780                                     Length);
1781 
1782 done:
1783     return Status;
1784 }
1785 
1786 
1787 /* Function 9 */
1788 NTSTATUS
1789 NTAPI
1790 SamrSetInformationDomain(IN SAMPR_HANDLE DomainHandle,
1791                          IN DOMAIN_INFORMATION_CLASS DomainInformationClass,
1792                          IN PSAMPR_DOMAIN_INFO_BUFFER DomainInformation)
1793 {
1794     PSAM_DB_OBJECT DomainObject;
1795     ACCESS_MASK DesiredAccess;
1796     NTSTATUS Status;
1797 
1798     TRACE("SamrSetInformationDomain(%p %lu %p)\n",
1799           DomainHandle, DomainInformationClass, DomainInformation);
1800 
1801     switch (DomainInformationClass)
1802     {
1803         case DomainPasswordInformation:
1804         case DomainLockoutInformation:
1805             DesiredAccess = DOMAIN_WRITE_PASSWORD_PARAMS;
1806             break;
1807 
1808         case DomainLogoffInformation:
1809         case DomainOemInformation:
1810         case DomainNameInformation:
1811             DesiredAccess = DOMAIN_WRITE_OTHER_PARAMETERS;
1812             break;
1813 
1814         case DomainReplicationInformation:
1815         case DomainServerRoleInformation:
1816         case DomainStateInformation:
1817             DesiredAccess = DOMAIN_ADMINISTER_SERVER;
1818             break;
1819 
1820         default:
1821             return STATUS_INVALID_INFO_CLASS;
1822     }
1823 
1824     RtlAcquireResourceExclusive(&SampResource,
1825                                 TRUE);
1826 
1827     /* Validate the server handle */
1828     Status = SampValidateDbObject(DomainHandle,
1829                                   SamDbDomainObject,
1830                                   DesiredAccess,
1831                                   &DomainObject);
1832     if (!NT_SUCCESS(Status))
1833         goto done;
1834 
1835     switch (DomainInformationClass)
1836     {
1837         case DomainPasswordInformation:
1838             Status = SampSetDomainPassword(DomainObject,
1839                                            DomainInformation);
1840             break;
1841 
1842         case DomainLogoffInformation:
1843             Status = SampSetDomainLogoff(DomainObject,
1844                                          DomainInformation);
1845             break;
1846 
1847         case DomainOemInformation:
1848             Status = SampSetObjectAttributeString(DomainObject,
1849                                                   L"OemInformation",
1850                                                   &DomainInformation->Oem.OemInformation);
1851             break;
1852 
1853         case DomainNameInformation:
1854             Status = SampSetObjectAttributeString(DomainObject,
1855                                                   L"Name",
1856                                                   &DomainInformation->Name.DomainName);
1857             break;
1858 
1859         case DomainReplicationInformation:
1860             Status = SampSetObjectAttributeString(DomainObject,
1861                                                   L"ReplicaSourceNodeName",
1862                                                   &DomainInformation->Replication.ReplicaSourceNodeName);
1863             break;
1864 
1865         case DomainServerRoleInformation:
1866             Status = SampSetDomainServerRole(DomainObject,
1867                                              DomainInformation);
1868             break;
1869 
1870         case DomainStateInformation:
1871             Status = SampSetDomainState(DomainObject,
1872                                         DomainInformation);
1873             break;
1874 
1875         case DomainLockoutInformation:
1876             Status = SampSetDomainLockout(DomainObject,
1877                                           DomainInformation);
1878             break;
1879 
1880         default:
1881             Status = STATUS_NOT_IMPLEMENTED;
1882     }
1883 
1884 done:
1885     RtlReleaseResource(&SampResource);
1886 
1887     return Status;
1888 }
1889 
1890 
1891 /* Function 10 */
1892 NTSTATUS
1893 NTAPI
1894 SamrCreateGroupInDomain(IN SAMPR_HANDLE DomainHandle,
1895                         IN PRPC_UNICODE_STRING Name,
1896                         IN ACCESS_MASK DesiredAccess,
1897                         OUT SAMPR_HANDLE *GroupHandle,
1898                         OUT unsigned long *RelativeId)
1899 {
1900     SAM_DOMAIN_FIXED_DATA FixedDomainData;
1901     SAM_GROUP_FIXED_DATA FixedGroupData;
1902     PSAM_DB_OBJECT DomainObject;
1903     PSAM_DB_OBJECT GroupObject;
1904     PSECURITY_DESCRIPTOR Sd = NULL;
1905     ULONG SdSize = 0;
1906     ULONG ulSize;
1907     ULONG ulRid;
1908     WCHAR szRid[9];
1909     NTSTATUS Status;
1910 
1911     TRACE("SamrCreateGroupInDomain(%p %p %lx %p %p)\n",
1912           DomainHandle, Name, DesiredAccess, GroupHandle, RelativeId);
1913 
1914     /* Map generic access rights */
1915     RtlMapGenericMask(&DesiredAccess,
1916                       &GroupMapping);
1917 
1918     RtlAcquireResourceExclusive(&SampResource,
1919                                 TRUE);
1920 
1921     /* Validate the domain handle */
1922     Status = SampValidateDbObject(DomainHandle,
1923                                   SamDbDomainObject,
1924                                   DOMAIN_CREATE_GROUP,
1925                                   &DomainObject);
1926     if (!NT_SUCCESS(Status))
1927     {
1928         TRACE("failed with status 0x%08lx\n", Status);
1929         goto done;
1930     }
1931 
1932     /* Check the group account name */
1933     Status = SampCheckAccountName(Name, 256);
1934     if (!NT_SUCCESS(Status))
1935     {
1936         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
1937         goto done;
1938     }
1939 
1940     /* Check if the group name already exists in the domain */
1941     Status = SampCheckAccountNameInDomain(DomainObject,
1942                                           Name->Buffer);
1943     if (!NT_SUCCESS(Status))
1944     {
1945         TRACE("Group name \'%S\' already exists in domain (Status 0x%08lx)\n",
1946               Name->Buffer, Status);
1947         goto done;
1948     }
1949 
1950     /* Create the security descriptor */
1951     Status = SampCreateGroupSD(&Sd,
1952                                &SdSize);
1953     if (!NT_SUCCESS(Status))
1954     {
1955         TRACE("SampCreateGroupSD failed (Status 0x%08lx)\n", Status);
1956         goto done;
1957     }
1958 
1959     /* Get the fixed domain attributes */
1960     ulSize = sizeof(SAM_DOMAIN_FIXED_DATA);
1961     Status = SampGetObjectAttribute(DomainObject,
1962                                     L"F",
1963                                     NULL,
1964                                     (PVOID)&FixedDomainData,
1965                                     &ulSize);
1966     if (!NT_SUCCESS(Status))
1967     {
1968         TRACE("failed with status 0x%08lx\n", Status);
1969         goto done;
1970     }
1971 
1972     /* Increment the NextRid attribute */
1973     ulRid = FixedDomainData.NextRid;
1974     FixedDomainData.NextRid++;
1975 
1976     /* Store the fixed domain attributes */
1977     Status = SampSetObjectAttribute(DomainObject,
1978                                     L"F",
1979                                     REG_BINARY,
1980                                     &FixedDomainData,
1981                                     ulSize);
1982     if (!NT_SUCCESS(Status))
1983     {
1984         TRACE("failed with status 0x%08lx\n", Status);
1985         goto done;
1986     }
1987 
1988     TRACE("RID: %lx\n", ulRid);
1989 
1990     /* Convert the RID into a string (hex) */
1991     swprintf(szRid, L"%08lX", ulRid);
1992 
1993     /* Create the group object */
1994     Status = SampCreateDbObject(DomainObject,
1995                                 L"Groups",
1996                                 szRid,
1997                                 ulRid,
1998                                 SamDbGroupObject,
1999                                 DesiredAccess,
2000                                 &GroupObject);
2001     if (!NT_SUCCESS(Status))
2002     {
2003         TRACE("failed with status 0x%08lx\n", Status);
2004         goto done;
2005     }
2006 
2007     /* Add the account name of the user object */
2008     Status = SampSetAccountNameInDomain(DomainObject,
2009                                         L"Groups",
2010                                         Name->Buffer,
2011                                         ulRid);
2012     if (!NT_SUCCESS(Status))
2013     {
2014         TRACE("failed with status 0x%08lx\n", Status);
2015         goto done;
2016     }
2017 
2018     /* Initialize fixed user data */
2019     memset(&FixedGroupData, 0, sizeof(SAM_GROUP_FIXED_DATA));
2020     FixedGroupData.Version = 1;
2021     FixedGroupData.GroupId = ulRid;
2022 
2023     /* Set fixed user data attribute */
2024     Status = SampSetObjectAttribute(GroupObject,
2025                                     L"F",
2026                                     REG_BINARY,
2027                                     (LPVOID)&FixedGroupData,
2028                                     sizeof(SAM_GROUP_FIXED_DATA));
2029     if (!NT_SUCCESS(Status))
2030     {
2031         TRACE("failed with status 0x%08lx\n", Status);
2032         goto done;
2033     }
2034 
2035     /* Set the Name attribute */
2036     Status = SampSetObjectAttributeString(GroupObject,
2037                                           L"Name",
2038                                           Name);
2039     if (!NT_SUCCESS(Status))
2040     {
2041         TRACE("failed with status 0x%08lx\n", Status);
2042         goto done;
2043     }
2044 
2045     /* Set the AdminComment attribute */
2046     Status = SampSetObjectAttributeString(GroupObject,
2047                                           L"AdminComment",
2048                                           NULL);
2049     if (!NT_SUCCESS(Status))
2050     {
2051         TRACE("failed with status 0x%08lx\n", Status);
2052         goto done;
2053     }
2054 
2055     /* Set the SecDesc attribute*/
2056     Status = SampSetObjectAttribute(GroupObject,
2057                                     L"SecDesc",
2058                                     REG_BINARY,
2059                                     Sd,
2060                                     SdSize);
2061     if (!NT_SUCCESS(Status))
2062     {
2063         TRACE("failed with status 0x%08lx\n", Status);
2064         goto done;
2065     }
2066 
2067     if (NT_SUCCESS(Status))
2068     {
2069         *GroupHandle = (SAMPR_HANDLE)GroupObject;
2070         *RelativeId = ulRid;
2071     }
2072 
2073 done:
2074     if (Sd != NULL)
2075         RtlFreeHeap(RtlGetProcessHeap(), 0, Sd);
2076 
2077     RtlReleaseResource(&SampResource);
2078 
2079     TRACE("returns with status 0x%08lx\n", Status);
2080 
2081     return Status;
2082 }
2083 
2084 
2085 /* Function 11 */
2086 NTSTATUS
2087 NTAPI
2088 SamrEnumerateGroupsInDomain(IN SAMPR_HANDLE DomainHandle,
2089                             IN OUT unsigned long *EnumerationContext,
2090                             OUT PSAMPR_ENUMERATION_BUFFER *Buffer,
2091                             IN unsigned long PreferedMaximumLength,
2092                             OUT unsigned long *CountReturned)
2093 {
2094     PSAMPR_ENUMERATION_BUFFER EnumBuffer = NULL;
2095     PSAM_DB_OBJECT DomainObject;
2096     HANDLE GroupsKeyHandle = NULL;
2097     HANDLE NamesKeyHandle = NULL;
2098     WCHAR GroupName[64];
2099     ULONG EnumIndex;
2100     ULONG EnumCount = 0;
2101     ULONG RequiredLength = 0;
2102     ULONG NameLength;
2103     ULONG DataLength;
2104     ULONG Rid;
2105     ULONG i;
2106     BOOLEAN MoreEntries = FALSE;
2107     NTSTATUS Status;
2108 
2109     TRACE("SamrEnumerateUsersInDomain(%p %p %p %lu %p)\n",
2110           DomainHandle, EnumerationContext, Buffer,
2111           PreferedMaximumLength, CountReturned);
2112 
2113     RtlAcquireResourceShared(&SampResource,
2114                              TRUE);
2115 
2116     /* Validate the domain handle */
2117     Status = SampValidateDbObject(DomainHandle,
2118                                   SamDbDomainObject,
2119                                   DOMAIN_LIST_ACCOUNTS,
2120                                   &DomainObject);
2121     if (!NT_SUCCESS(Status))
2122         goto done;
2123 
2124     Status = SampRegOpenKey(DomainObject->KeyHandle,
2125                             L"Groups",
2126                             KEY_READ,
2127                             &GroupsKeyHandle);
2128     if (!NT_SUCCESS(Status))
2129         goto done;
2130 
2131     Status = SampRegOpenKey(GroupsKeyHandle,
2132                             L"Names",
2133                             KEY_READ,
2134                             &NamesKeyHandle);
2135     if (!NT_SUCCESS(Status))
2136         goto done;
2137 
2138     TRACE("Part 1\n");
2139 
2140     EnumIndex = *EnumerationContext;
2141 
2142     while (TRUE)
2143     {
2144         NameLength = 64 * sizeof(WCHAR);
2145         Status = SampRegEnumerateValue(NamesKeyHandle,
2146                                        EnumIndex,
2147                                        GroupName,
2148                                        &NameLength,
2149                                        NULL,
2150                                        NULL,
2151                                        NULL);
2152         if (!NT_SUCCESS(Status))
2153         {
2154             if (Status == STATUS_NO_MORE_ENTRIES)
2155                 Status = STATUS_SUCCESS;
2156             break;
2157         }
2158 
2159         TRACE("EnumIndex: %lu\n", EnumIndex);
2160         TRACE("Group name: %S\n", GroupName);
2161         TRACE("Name length: %lu\n", NameLength);
2162 
2163         if ((RequiredLength + NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION)) > PreferedMaximumLength)
2164         {
2165             MoreEntries = TRUE;
2166             break;
2167         }
2168 
2169         RequiredLength += (NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION));
2170         EnumCount++;
2171 
2172         EnumIndex++;
2173     }
2174 
2175     TRACE("EnumCount: %lu\n", EnumCount);
2176     TRACE("RequiredLength: %lu\n", RequiredLength);
2177 
2178     if (!NT_SUCCESS(Status))
2179         goto done;
2180 
2181     EnumBuffer = midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER));
2182     if (EnumBuffer == NULL)
2183     {
2184         Status = STATUS_INSUFFICIENT_RESOURCES;
2185         goto done;
2186     }
2187 
2188     EnumBuffer->EntriesRead = EnumCount;
2189     if (EnumCount == 0)
2190     {
2191         Status = STATUS_NO_MORE_ENTRIES;
2192         goto done;
2193     }
2194 
2195     EnumBuffer->Buffer = midl_user_allocate(EnumCount * sizeof(SAMPR_RID_ENUMERATION));
2196     if (EnumBuffer->Buffer == NULL)
2197     {
2198         Status = STATUS_INSUFFICIENT_RESOURCES;
2199         goto done;
2200     }
2201 
2202     TRACE("Part 2\n");
2203 
2204     EnumIndex = *EnumerationContext;
2205     for (i = 0; i < EnumCount; i++, EnumIndex++)
2206     {
2207         NameLength = 64 * sizeof(WCHAR);
2208         DataLength = sizeof(ULONG);
2209         Status = SampRegEnumerateValue(NamesKeyHandle,
2210                                        EnumIndex,
2211                                        GroupName,
2212                                        &NameLength,
2213                                        NULL,
2214                                        &Rid,
2215                                        &DataLength);
2216         if (!NT_SUCCESS(Status))
2217         {
2218             if (Status == STATUS_NO_MORE_ENTRIES)
2219                 Status = STATUS_SUCCESS;
2220             break;
2221         }
2222 
2223         TRACE("EnumIndex: %lu\n", EnumIndex);
2224         TRACE("Group name: %S\n", GroupName);
2225         TRACE("Name length: %lu\n", NameLength);
2226         TRACE("RID: %lu\n", Rid);
2227 
2228         EnumBuffer->Buffer[i].RelativeId = Rid;
2229 
2230         EnumBuffer->Buffer[i].Name.Length = (USHORT)NameLength;
2231         EnumBuffer->Buffer[i].Name.MaximumLength = (USHORT)(NameLength + sizeof(UNICODE_NULL));
2232 
2233 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
2234 #if 0
2235         EnumBuffer->Buffer[i].Name.Buffer = midl_user_allocate(EnumBuffer->Buffer[i].Name.MaximumLength);
2236         if (EnumBuffer->Buffer[i].Name.Buffer == NULL)
2237         {
2238             Status = STATUS_INSUFFICIENT_RESOURCES;
2239             goto done;
2240         }
2241 
2242         memcpy(EnumBuffer->Buffer[i].Name.Buffer,
2243                GroupName,
2244                EnumBuffer->Buffer[i].Name.Length);
2245 #endif
2246     }
2247 
2248 done:
2249     if (NT_SUCCESS(Status))
2250     {
2251         *EnumerationContext += EnumCount;
2252         *Buffer = EnumBuffer;
2253         *CountReturned = EnumCount;
2254     }
2255     else
2256     {
2257         *EnumerationContext = 0;
2258         *Buffer = NULL;
2259         *CountReturned = 0;
2260 
2261         if (EnumBuffer != NULL)
2262         {
2263             if (EnumBuffer->Buffer != NULL)
2264             {
2265                 if (EnumBuffer->EntriesRead != 0)
2266                 {
2267                     for (i = 0; i < EnumBuffer->EntriesRead; i++)
2268                     {
2269                         if (EnumBuffer->Buffer[i].Name.Buffer != NULL)
2270                             midl_user_free(EnumBuffer->Buffer[i].Name.Buffer);
2271                     }
2272                 }
2273 
2274                 midl_user_free(EnumBuffer->Buffer);
2275             }
2276 
2277             midl_user_free(EnumBuffer);
2278         }
2279     }
2280 
2281     SampRegCloseKey(&NamesKeyHandle);
2282     SampRegCloseKey(&GroupsKeyHandle);
2283 
2284     if ((Status == STATUS_SUCCESS) && (MoreEntries != FALSE))
2285         Status = STATUS_MORE_ENTRIES;
2286 
2287     RtlReleaseResource(&SampResource);
2288 
2289     return Status;
2290 }
2291 
2292 
2293 /* Function 12 */
2294 NTSTATUS
2295 NTAPI
2296 SamrCreateUserInDomain(IN SAMPR_HANDLE DomainHandle,
2297                        IN PRPC_UNICODE_STRING Name,
2298                        IN ACCESS_MASK DesiredAccess,
2299                        OUT SAMPR_HANDLE *UserHandle,
2300                        OUT unsigned long *RelativeId)
2301 {
2302     SAM_DOMAIN_FIXED_DATA FixedDomainData;
2303     SAM_USER_FIXED_DATA FixedUserData;
2304     PSAM_DB_OBJECT DomainObject;
2305     PSAM_DB_OBJECT UserObject;
2306     GROUP_MEMBERSHIP GroupMembership;
2307     UCHAR LogonHours[23];
2308     ULONG ulSize;
2309     ULONG ulRid;
2310     WCHAR szRid[9];
2311     PSECURITY_DESCRIPTOR Sd = NULL;
2312     ULONG SdSize = 0;
2313     PSID UserSid = NULL;
2314     NTSTATUS Status;
2315 
2316     TRACE("SamrCreateUserInDomain(%p %p %lx %p %p)\n",
2317           DomainHandle, Name, DesiredAccess, UserHandle, RelativeId);
2318 
2319     if (Name == NULL ||
2320         Name->Length == 0 ||
2321         Name->Buffer == NULL ||
2322         UserHandle == NULL ||
2323         RelativeId == NULL)
2324         return STATUS_INVALID_PARAMETER;
2325 
2326     /* Map generic access rights */
2327     RtlMapGenericMask(&DesiredAccess,
2328                       &UserMapping);
2329 
2330     RtlAcquireResourceExclusive(&SampResource,
2331                                 TRUE);
2332 
2333     /* Validate the domain handle */
2334     Status = SampValidateDbObject(DomainHandle,
2335                                   SamDbDomainObject,
2336                                   DOMAIN_CREATE_USER,
2337                                   &DomainObject);
2338     if (!NT_SUCCESS(Status))
2339     {
2340         TRACE("failed with status 0x%08lx\n", Status);
2341         goto done;
2342     }
2343 
2344     /* Check the user account name */
2345     Status = SampCheckAccountName(Name, 20);
2346     if (!NT_SUCCESS(Status))
2347     {
2348         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
2349         goto done;
2350     }
2351 
2352     /* Check if the user name already exists in the domain */
2353     Status = SampCheckAccountNameInDomain(DomainObject,
2354                                           Name->Buffer);
2355     if (!NT_SUCCESS(Status))
2356     {
2357         TRACE("User name \'%S\' already exists in domain (Status 0x%08lx)\n",
2358               Name->Buffer, Status);
2359         goto done;
2360     }
2361 
2362     /* Get the fixed domain attributes */
2363     ulSize = sizeof(SAM_DOMAIN_FIXED_DATA);
2364     Status = SampGetObjectAttribute(DomainObject,
2365                                     L"F",
2366                                     NULL,
2367                                     (PVOID)&FixedDomainData,
2368                                     &ulSize);
2369     if (!NT_SUCCESS(Status))
2370     {
2371         TRACE("failed with status 0x%08lx\n", Status);
2372         goto done;
2373     }
2374 
2375     /* Increment the NextRid attribute */
2376     ulRid = FixedDomainData.NextRid;
2377     FixedDomainData.NextRid++;
2378 
2379     TRACE("RID: %lx\n", ulRid);
2380 
2381     /* Create the user SID */
2382     Status = SampCreateAccountSid(DomainObject,
2383                                   ulRid,
2384                                   &UserSid);
2385     if (!NT_SUCCESS(Status))
2386     {
2387         TRACE("SampCreateAccountSid failed (Status 0x%08lx)\n", Status);
2388         goto done;
2389     }
2390 
2391     /* Create the security descriptor */
2392     Status = SampCreateUserSD(UserSid,
2393                               &Sd,
2394                               &SdSize);
2395     if (!NT_SUCCESS(Status))
2396     {
2397         TRACE("SampCreateUserSD failed (Status 0x%08lx)\n", Status);
2398         goto done;
2399     }
2400 
2401     /* Store the fixed domain attributes */
2402     Status = SampSetObjectAttribute(DomainObject,
2403                                     L"F",
2404                                     REG_BINARY,
2405                                     &FixedDomainData,
2406                                     ulSize);
2407     if (!NT_SUCCESS(Status))
2408     {
2409         TRACE("failed with status 0x%08lx\n", Status);
2410         goto done;
2411     }
2412 
2413     /* Convert the RID into a string (hex) */
2414     swprintf(szRid, L"%08lX", ulRid);
2415 
2416     /* Create the user object */
2417     Status = SampCreateDbObject(DomainObject,
2418                                 L"Users",
2419                                 szRid,
2420                                 ulRid,
2421                                 SamDbUserObject,
2422                                 DesiredAccess,
2423                                 &UserObject);
2424     if (!NT_SUCCESS(Status))
2425     {
2426         TRACE("failed with status 0x%08lx\n", Status);
2427         goto done;
2428     }
2429 
2430     /* Add the account name for the user object */
2431     Status = SampSetAccountNameInDomain(DomainObject,
2432                                         L"Users",
2433                                         Name->Buffer,
2434                                         ulRid);
2435     if (!NT_SUCCESS(Status))
2436     {
2437         TRACE("failed with status 0x%08lx\n", Status);
2438         goto done;
2439     }
2440 
2441     /* Initialize fixed user data */
2442     memset(&FixedUserData, 0, sizeof(SAM_USER_FIXED_DATA));
2443     FixedUserData.Version = 1;
2444     FixedUserData.Reserved = 0;
2445     FixedUserData.LastLogon.QuadPart = 0;
2446     FixedUserData.LastLogoff.QuadPart = 0;
2447     FixedUserData.PasswordLastSet.QuadPart = 0;
2448     FixedUserData.AccountExpires.QuadPart = MAXLONGLONG;
2449     FixedUserData.LastBadPasswordTime.QuadPart = 0;
2450     FixedUserData.UserId = ulRid;
2451     FixedUserData.PrimaryGroupId = DOMAIN_GROUP_RID_USERS;
2452     FixedUserData.UserAccountControl = USER_ACCOUNT_DISABLED |
2453                                        USER_PASSWORD_NOT_REQUIRED |
2454                                        USER_NORMAL_ACCOUNT;
2455     FixedUserData.CountryCode = 0;
2456     FixedUserData.CodePage = 0;
2457     FixedUserData.BadPasswordCount = 0;
2458     FixedUserData.LogonCount = 0;
2459     FixedUserData.AdminCount = 0;
2460     FixedUserData.OperatorCount = 0;
2461 
2462     /* Set fixed user data attribute */
2463     Status = SampSetObjectAttribute(UserObject,
2464                                     L"F",
2465                                     REG_BINARY,
2466                                     (LPVOID)&FixedUserData,
2467                                     sizeof(SAM_USER_FIXED_DATA));
2468     if (!NT_SUCCESS(Status))
2469     {
2470         TRACE("failed with status 0x%08lx\n", Status);
2471         goto done;
2472     }
2473 
2474     /* Set the Name attribute */
2475     Status = SampSetObjectAttributeString(UserObject,
2476                                           L"Name",
2477                                           Name);
2478     if (!NT_SUCCESS(Status))
2479     {
2480         TRACE("failed with status 0x%08lx\n", Status);
2481         goto done;
2482     }
2483 
2484     /* Set the FullName attribute */
2485     Status = SampSetObjectAttributeString(UserObject,
2486                                           L"FullName",
2487                                           NULL);
2488     if (!NT_SUCCESS(Status))
2489     {
2490         TRACE("failed with status 0x%08lx\n", Status);
2491         goto done;
2492     }
2493 
2494     /* Set the HomeDirectory attribute */
2495     Status = SampSetObjectAttributeString(UserObject,
2496                                           L"HomeDirectory",
2497                                           NULL);
2498     if (!NT_SUCCESS(Status))
2499     {
2500         TRACE("failed with status 0x%08lx\n", Status);
2501         goto done;
2502     }
2503 
2504     /* Set the HomeDirectoryDrive attribute */
2505     Status = SampSetObjectAttributeString(UserObject,
2506                                           L"HomeDirectoryDrive",
2507                                           NULL);
2508     if (!NT_SUCCESS(Status))
2509     {
2510         TRACE("failed with status 0x%08lx\n", Status);
2511         goto done;
2512     }
2513 
2514     /* Set the ScriptPath attribute */
2515     Status = SampSetObjectAttributeString(UserObject,
2516                                           L"ScriptPath",
2517                                           NULL);
2518     if (!NT_SUCCESS(Status))
2519     {
2520         TRACE("failed with status 0x%08lx\n", Status);
2521         goto done;
2522     }
2523 
2524     /* Set the ProfilePath attribute */
2525     Status = SampSetObjectAttributeString(UserObject,
2526                                           L"ProfilePath",
2527                                           NULL);
2528     if (!NT_SUCCESS(Status))
2529     {
2530         TRACE("failed with status 0x%08lx\n", Status);
2531         goto done;
2532     }
2533 
2534     /* Set the AdminComment attribute */
2535     Status = SampSetObjectAttributeString(UserObject,
2536                                           L"AdminComment",
2537                                           NULL);
2538     if (!NT_SUCCESS(Status))
2539     {
2540         TRACE("failed with status 0x%08lx\n", Status);
2541         goto done;
2542     }
2543 
2544     /* Set the UserComment attribute */
2545     Status = SampSetObjectAttributeString(UserObject,
2546                                           L"UserComment",
2547                                           NULL);
2548     if (!NT_SUCCESS(Status))
2549     {
2550         TRACE("failed with status 0x%08lx\n", Status);
2551         goto done;
2552     }
2553 
2554     /* Set the WorkStations attribute */
2555     Status = SampSetObjectAttributeString(UserObject,
2556                                           L"WorkStations",
2557                                           NULL);
2558     if (!NT_SUCCESS(Status))
2559     {
2560         TRACE("failed with status 0x%08lx\n", Status);
2561         goto done;
2562     }
2563 
2564     /* Set the Parameters attribute */
2565     Status = SampSetObjectAttributeString(UserObject,
2566                                           L"Parameters",
2567                                           NULL);
2568     if (!NT_SUCCESS(Status))
2569     {
2570         TRACE("failed with status 0x%08lx\n", Status);
2571         goto done;
2572     }
2573 
2574     /* Set LogonHours attribute*/
2575     *((PUSHORT)LogonHours) = 168;
2576     memset(&(LogonHours[2]), 0xff, 21);
2577 
2578     Status = SampSetObjectAttribute(UserObject,
2579                                     L"LogonHours",
2580                                     REG_BINARY,
2581                                     &LogonHours,
2582                                     sizeof(LogonHours));
2583     if (!NT_SUCCESS(Status))
2584     {
2585         TRACE("failed with status 0x%08lx\n", Status);
2586         goto done;
2587     }
2588 
2589     /* Set Groups attribute*/
2590     GroupMembership.RelativeId = DOMAIN_GROUP_RID_USERS;
2591     GroupMembership.Attributes = SE_GROUP_MANDATORY |
2592                                  SE_GROUP_ENABLED |
2593                                  SE_GROUP_ENABLED_BY_DEFAULT;
2594 
2595     Status = SampSetObjectAttribute(UserObject,
2596                                     L"Groups",
2597                                     REG_BINARY,
2598                                     &GroupMembership,
2599                                     sizeof(GROUP_MEMBERSHIP));
2600     if (!NT_SUCCESS(Status))
2601     {
2602         TRACE("failed with status 0x%08lx\n", Status);
2603         goto done;
2604     }
2605 
2606     /* Set LMPwd attribute*/
2607     Status = SampSetObjectAttribute(UserObject,
2608                                     L"LMPwd",
2609                                     REG_BINARY,
2610                                     &EmptyLmHash,
2611                                     sizeof(ENCRYPTED_LM_OWF_PASSWORD));
2612     if (!NT_SUCCESS(Status))
2613     {
2614         TRACE("failed with status 0x%08lx\n", Status);
2615         goto done;
2616     }
2617 
2618     /* Set NTPwd attribute*/
2619     Status = SampSetObjectAttribute(UserObject,
2620                                     L"NTPwd",
2621                                     REG_BINARY,
2622                                     &EmptyNtHash,
2623                                     sizeof(ENCRYPTED_NT_OWF_PASSWORD));
2624     if (!NT_SUCCESS(Status))
2625     {
2626         TRACE("failed with status 0x%08lx\n", Status);
2627         goto done;
2628     }
2629 
2630     /* Set LMPwdHistory attribute*/
2631     Status = SampSetObjectAttribute(UserObject,
2632                                     L"LMPwdHistory",
2633                                     REG_BINARY,
2634                                     NULL,
2635                                     0);
2636     if (!NT_SUCCESS(Status))
2637     {
2638         TRACE("failed with status 0x%08lx\n", Status);
2639         goto done;
2640     }
2641 
2642     /* Set NTPwdHistory attribute*/
2643     Status = SampSetObjectAttribute(UserObject,
2644                                     L"NTPwdHistory",
2645                                     REG_BINARY,
2646                                     NULL,
2647                                     0);
2648     if (!NT_SUCCESS(Status))
2649     {
2650         TRACE("failed with status 0x%08lx\n", Status);
2651         goto done;
2652     }
2653 
2654     /* Set the PrivateData attribute */
2655     Status = SampSetObjectAttributeString(UserObject,
2656                                           L"PrivateData",
2657                                           NULL);
2658     if (!NT_SUCCESS(Status))
2659     {
2660         TRACE("failed with status 0x%08lx\n", Status);
2661         goto done;
2662     }
2663 
2664     /* Set the SecDesc attribute*/
2665     Status = SampSetObjectAttribute(UserObject,
2666                                     L"SecDesc",
2667                                     REG_BINARY,
2668                                     Sd,
2669                                     SdSize);
2670     if (!NT_SUCCESS(Status))
2671     {
2672         TRACE("failed with status 0x%08lx\n", Status);
2673         goto done;
2674     }
2675 
2676     if (NT_SUCCESS(Status))
2677     {
2678         *UserHandle = (SAMPR_HANDLE)UserObject;
2679         *RelativeId = ulRid;
2680     }
2681 
2682 done:
2683     if (Sd != NULL)
2684         RtlFreeHeap(RtlGetProcessHeap(), 0, Sd);
2685 
2686     if (UserSid != NULL)
2687         RtlFreeHeap(RtlGetProcessHeap(), 0, UserSid);
2688 
2689     RtlReleaseResource(&SampResource);
2690 
2691     TRACE("returns with status 0x%08lx\n", Status);
2692 
2693     return Status;
2694 }
2695 
2696 
2697 /* Function 13 */
2698 NTSTATUS
2699 NTAPI
2700 SamrEnumerateUsersInDomain(IN SAMPR_HANDLE DomainHandle,
2701                            IN OUT unsigned long *EnumerationContext,
2702                            IN unsigned long UserAccountControl,
2703                            OUT PSAMPR_ENUMERATION_BUFFER *Buffer,
2704                            IN unsigned long PreferedMaximumLength,
2705                            OUT unsigned long *CountReturned)
2706 {
2707     PSAMPR_ENUMERATION_BUFFER EnumBuffer = NULL;
2708     PSAM_DB_OBJECT DomainObject;
2709     HANDLE UsersKeyHandle = NULL;
2710     HANDLE NamesKeyHandle = NULL;
2711     WCHAR UserName[64];
2712     ULONG EnumIndex;
2713     ULONG EnumCount = 0;
2714     ULONG RequiredLength = 0;
2715     ULONG NameLength;
2716     ULONG DataLength;
2717     ULONG Rid;
2718     ULONG i;
2719     BOOLEAN MoreEntries = FALSE;
2720     NTSTATUS Status;
2721 
2722     TRACE("SamrEnumerateUsersInDomain(%p %p %lx %p %lu %p)\n",
2723           DomainHandle, EnumerationContext, UserAccountControl, Buffer,
2724           PreferedMaximumLength, CountReturned);
2725 
2726     RtlAcquireResourceShared(&SampResource,
2727                              TRUE);
2728 
2729     /* Validate the domain handle */
2730     Status = SampValidateDbObject(DomainHandle,
2731                                   SamDbDomainObject,
2732                                   DOMAIN_LIST_ACCOUNTS,
2733                                   &DomainObject);
2734     if (!NT_SUCCESS(Status))
2735         goto done;
2736 
2737     Status = SampRegOpenKey(DomainObject->KeyHandle,
2738                             L"Users",
2739                             KEY_READ,
2740                             &UsersKeyHandle);
2741     if (!NT_SUCCESS(Status))
2742         goto done;
2743 
2744     Status = SampRegOpenKey(UsersKeyHandle,
2745                             L"Names",
2746                             KEY_READ,
2747                             &NamesKeyHandle);
2748     if (!NT_SUCCESS(Status))
2749         goto done;
2750 
2751     TRACE("Part 1\n");
2752 
2753     EnumIndex = *EnumerationContext;
2754 
2755     while (TRUE)
2756     {
2757         NameLength = 64 * sizeof(WCHAR);
2758         Status = SampRegEnumerateValue(NamesKeyHandle,
2759                                        EnumIndex,
2760                                        UserName,
2761                                        &NameLength,
2762                                        NULL,
2763                                        NULL,
2764                                        NULL);
2765         if (!NT_SUCCESS(Status))
2766         {
2767             if (Status == STATUS_NO_MORE_ENTRIES)
2768                 Status = STATUS_SUCCESS;
2769             break;
2770         }
2771 
2772         TRACE("EnumIndex: %lu\n", EnumIndex);
2773         TRACE("User name: %S\n", UserName);
2774         TRACE("Name length: %lu\n", NameLength);
2775 
2776         if ((RequiredLength + NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION)) > PreferedMaximumLength)
2777         {
2778             MoreEntries = TRUE;
2779             break;
2780         }
2781 
2782         RequiredLength += (NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION));
2783         EnumCount++;
2784 
2785         EnumIndex++;
2786     }
2787 
2788     TRACE("EnumCount: %lu\n", EnumCount);
2789     TRACE("RequiredLength: %lu\n", RequiredLength);
2790 
2791     if (!NT_SUCCESS(Status))
2792         goto done;
2793 
2794     EnumBuffer = midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER));
2795     if (EnumBuffer == NULL)
2796     {
2797         Status = STATUS_INSUFFICIENT_RESOURCES;
2798         goto done;
2799     }
2800 
2801     EnumBuffer->EntriesRead = EnumCount;
2802     if (EnumCount == 0)
2803     {
2804         Status = STATUS_NO_MORE_ENTRIES;
2805         goto done;
2806     }
2807 
2808     EnumBuffer->Buffer = midl_user_allocate(EnumCount * sizeof(SAMPR_RID_ENUMERATION));
2809     if (EnumBuffer->Buffer == NULL)
2810     {
2811         Status = STATUS_INSUFFICIENT_RESOURCES;
2812         goto done;
2813     }
2814 
2815     TRACE("Part 2\n");
2816 
2817     EnumIndex = *EnumerationContext;
2818     for (i = 0; i < EnumCount; i++, EnumIndex++)
2819     {
2820         NameLength = 64 * sizeof(WCHAR);
2821         DataLength = sizeof(ULONG);
2822         Status = SampRegEnumerateValue(NamesKeyHandle,
2823                                        EnumIndex,
2824                                        UserName,
2825                                        &NameLength,
2826                                        NULL,
2827                                        &Rid,
2828                                        &DataLength);
2829         if (!NT_SUCCESS(Status))
2830         {
2831             if (Status == STATUS_NO_MORE_ENTRIES)
2832                 Status = STATUS_SUCCESS;
2833             break;
2834         }
2835 
2836         TRACE("EnumIndex: %lu\n", EnumIndex);
2837         TRACE("User name: %S\n", UserName);
2838         TRACE("Name length: %lu\n", NameLength);
2839         TRACE("RID: %lu\n", Rid);
2840 
2841         EnumBuffer->Buffer[i].RelativeId = Rid;
2842 
2843         EnumBuffer->Buffer[i].Name.Length = (USHORT)NameLength;
2844         EnumBuffer->Buffer[i].Name.MaximumLength = (USHORT)(NameLength + sizeof(UNICODE_NULL));
2845 
2846 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
2847 #if 0
2848         EnumBuffer->Buffer[i].Name.Buffer = midl_user_allocate(EnumBuffer->Buffer[i].Name.MaximumLength);
2849         if (EnumBuffer->Buffer[i].Name.Buffer == NULL)
2850         {
2851             Status = STATUS_INSUFFICIENT_RESOURCES;
2852             goto done;
2853         }
2854 
2855         memcpy(EnumBuffer->Buffer[i].Name.Buffer,
2856                UserName,
2857                EnumBuffer->Buffer[i].Name.Length);
2858 #endif
2859     }
2860 
2861 done:
2862     if (NT_SUCCESS(Status))
2863     {
2864         *EnumerationContext += EnumCount;
2865         *Buffer = EnumBuffer;
2866         *CountReturned = EnumCount;
2867     }
2868     else
2869     {
2870         *EnumerationContext = 0;
2871         *Buffer = NULL;
2872         *CountReturned = 0;
2873 
2874         if (EnumBuffer != NULL)
2875         {
2876             if (EnumBuffer->Buffer != NULL)
2877             {
2878                 if (EnumBuffer->EntriesRead != 0)
2879                 {
2880                     for (i = 0; i < EnumBuffer->EntriesRead; i++)
2881                     {
2882                         if (EnumBuffer->Buffer[i].Name.Buffer != NULL)
2883                             midl_user_free(EnumBuffer->Buffer[i].Name.Buffer);
2884                     }
2885                 }
2886 
2887                 midl_user_free(EnumBuffer->Buffer);
2888             }
2889 
2890             midl_user_free(EnumBuffer);
2891         }
2892     }
2893 
2894     SampRegCloseKey(&NamesKeyHandle);
2895     SampRegCloseKey(&UsersKeyHandle);
2896 
2897     if ((Status == STATUS_SUCCESS) && (MoreEntries != FALSE))
2898         Status = STATUS_MORE_ENTRIES;
2899 
2900     RtlReleaseResource(&SampResource);
2901 
2902     return Status;
2903 }
2904 
2905 
2906 /* Function 14 */
2907 NTSTATUS
2908 NTAPI
2909 SamrCreateAliasInDomain(IN SAMPR_HANDLE DomainHandle,
2910                         IN PRPC_UNICODE_STRING AccountName,
2911                         IN ACCESS_MASK DesiredAccess,
2912                         OUT SAMPR_HANDLE *AliasHandle,
2913                         OUT unsigned long *RelativeId)
2914 {
2915     SAM_DOMAIN_FIXED_DATA FixedDomainData;
2916     PSAM_DB_OBJECT DomainObject;
2917     PSAM_DB_OBJECT AliasObject;
2918     PSECURITY_DESCRIPTOR Sd = NULL;
2919     ULONG SdSize = 0;
2920     ULONG ulSize;
2921     ULONG ulRid;
2922     WCHAR szRid[9];
2923     NTSTATUS Status;
2924 
2925     TRACE("SamrCreateAliasInDomain(%p %p %lx %p %p)\n",
2926           DomainHandle, AccountName, DesiredAccess, AliasHandle, RelativeId);
2927 
2928     /* Map generic access rights */
2929     RtlMapGenericMask(&DesiredAccess,
2930                       &AliasMapping);
2931 
2932     RtlAcquireResourceExclusive(&SampResource,
2933                                 TRUE);
2934 
2935     /* Validate the domain handle */
2936     Status = SampValidateDbObject(DomainHandle,
2937                                   SamDbDomainObject,
2938                                   DOMAIN_CREATE_ALIAS,
2939                                   &DomainObject);
2940     if (!NT_SUCCESS(Status))
2941     {
2942         TRACE("failed with status 0x%08lx\n", Status);
2943         goto done;
2944     }
2945 
2946     /* Check the alias account name */
2947     Status = SampCheckAccountName(AccountName, 256);
2948     if (!NT_SUCCESS(Status))
2949     {
2950         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
2951         goto done;
2952     }
2953 
2954     /* Check if the alias name already exists in the domain */
2955     Status = SampCheckAccountNameInDomain(DomainObject,
2956                                           AccountName->Buffer);
2957     if (!NT_SUCCESS(Status))
2958     {
2959         TRACE("Alias name \'%S\' already exists in domain (Status 0x%08lx)\n",
2960               AccountName->Buffer, Status);
2961         goto done;
2962     }
2963 
2964     /* Create the security descriptor */
2965     Status = SampCreateAliasSD(&Sd,
2966                                &SdSize);
2967     if (!NT_SUCCESS(Status))
2968     {
2969         TRACE("SampCreateAliasSD failed (Status 0x%08lx)\n", Status);
2970         goto done;
2971     }
2972 
2973     /* Get the fixed domain attributes */
2974     ulSize = sizeof(SAM_DOMAIN_FIXED_DATA);
2975     Status = SampGetObjectAttribute(DomainObject,
2976                                     L"F",
2977                                     NULL,
2978                                     (PVOID)&FixedDomainData,
2979                                     &ulSize);
2980     if (!NT_SUCCESS(Status))
2981     {
2982         TRACE("failed with status 0x%08lx\n", Status);
2983         goto done;
2984     }
2985 
2986     /* Increment the NextRid attribute */
2987     ulRid = FixedDomainData.NextRid;
2988     FixedDomainData.NextRid++;
2989 
2990     /* Store the fixed domain attributes */
2991     Status = SampSetObjectAttribute(DomainObject,
2992                            L"F",
2993                            REG_BINARY,
2994                            &FixedDomainData,
2995                            ulSize);
2996     if (!NT_SUCCESS(Status))
2997     {
2998         TRACE("failed with status 0x%08lx\n", Status);
2999         goto done;
3000     }
3001 
3002     TRACE("RID: %lx\n", ulRid);
3003 
3004     /* Convert the RID into a string (hex) */
3005     swprintf(szRid, L"%08lX", ulRid);
3006 
3007     /* Create the alias object */
3008     Status = SampCreateDbObject(DomainObject,
3009                                 L"Aliases",
3010                                 szRid,
3011                                 ulRid,
3012                                 SamDbAliasObject,
3013                                 DesiredAccess,
3014                                 &AliasObject);
3015     if (!NT_SUCCESS(Status))
3016     {
3017         TRACE("failed with status 0x%08lx\n", Status);
3018         goto done;
3019     }
3020 
3021     /* Add the account name for the alias object */
3022     Status = SampSetAccountNameInDomain(DomainObject,
3023                                         L"Aliases",
3024                                         AccountName->Buffer,
3025                                         ulRid);
3026     if (!NT_SUCCESS(Status))
3027     {
3028         TRACE("failed with status 0x%08lx\n", Status);
3029         goto done;
3030     }
3031 
3032     /* Set the Name attribute */
3033     Status = SampSetObjectAttributeString(AliasObject,
3034                                           L"Name",
3035                                           AccountName);
3036     if (!NT_SUCCESS(Status))
3037     {
3038         TRACE("failed with status 0x%08lx\n", Status);
3039         goto done;
3040     }
3041 
3042     /* Set the Description attribute */
3043     Status = SampSetObjectAttributeString(AliasObject,
3044                                           L"Description",
3045                                           NULL);
3046     if (!NT_SUCCESS(Status))
3047     {
3048         TRACE("failed with status 0x%08lx\n", Status);
3049         goto done;
3050     }
3051 
3052     /* Set the SecDesc attribute*/
3053     Status = SampSetObjectAttribute(AliasObject,
3054                                     L"SecDesc",
3055                                     REG_BINARY,
3056                                     Sd,
3057                                     SdSize);
3058     if (!NT_SUCCESS(Status))
3059     {
3060         TRACE("failed with status 0x%08lx\n", Status);
3061         goto done;
3062     }
3063 
3064     if (NT_SUCCESS(Status))
3065     {
3066         *AliasHandle = (SAMPR_HANDLE)AliasObject;
3067         *RelativeId = ulRid;
3068     }
3069 
3070 done:
3071     if (Sd != NULL)
3072         RtlFreeHeap(RtlGetProcessHeap(), 0, Sd);
3073 
3074     RtlReleaseResource(&SampResource);
3075 
3076     TRACE("returns with status 0x%08lx\n", Status);
3077 
3078     return Status;
3079 }
3080 
3081 
3082 /* Function 15 */
3083 NTSTATUS
3084 NTAPI
3085 SamrEnumerateAliasesInDomain(IN SAMPR_HANDLE DomainHandle,
3086                              IN OUT unsigned long *EnumerationContext,
3087                              OUT PSAMPR_ENUMERATION_BUFFER *Buffer,
3088                              IN unsigned long PreferedMaximumLength,
3089                              OUT unsigned long *CountReturned)
3090 {
3091     PSAMPR_ENUMERATION_BUFFER EnumBuffer = NULL;
3092     PSAM_DB_OBJECT DomainObject;
3093     HANDLE AliasesKeyHandle = NULL;
3094     HANDLE NamesKeyHandle = NULL;
3095     WCHAR AliasName[64];
3096     ULONG EnumIndex;
3097     ULONG EnumCount = 0;
3098     ULONG RequiredLength = 0;
3099     ULONG NameLength;
3100     ULONG DataLength;
3101     ULONG Rid;
3102     ULONG i;
3103     BOOLEAN MoreEntries = FALSE;
3104     NTSTATUS Status;
3105 
3106     TRACE("SamrEnumerateAliasesInDomain(%p %p %p %lu %p)\n",
3107           DomainHandle, EnumerationContext, Buffer,
3108           PreferedMaximumLength, CountReturned);
3109 
3110     RtlAcquireResourceShared(&SampResource,
3111                              TRUE);
3112 
3113     /* Validate the domain handle */
3114     Status = SampValidateDbObject(DomainHandle,
3115                                   SamDbDomainObject,
3116                                   DOMAIN_LIST_ACCOUNTS,
3117                                   &DomainObject);
3118     if (!NT_SUCCESS(Status))
3119         goto done;
3120 
3121     Status = SampRegOpenKey(DomainObject->KeyHandle,
3122                             L"Aliases",
3123                             KEY_READ,
3124                             &AliasesKeyHandle);
3125     if (!NT_SUCCESS(Status))
3126         goto done;
3127 
3128     Status = SampRegOpenKey(AliasesKeyHandle,
3129                             L"Names",
3130                             KEY_READ,
3131                             &NamesKeyHandle);
3132     if (!NT_SUCCESS(Status))
3133         goto done;
3134 
3135     TRACE("Part 1\n");
3136 
3137     EnumIndex = *EnumerationContext;
3138 
3139     while (TRUE)
3140     {
3141         NameLength = 64 * sizeof(WCHAR);
3142         Status = SampRegEnumerateValue(NamesKeyHandle,
3143                                        EnumIndex,
3144                                        AliasName,
3145                                        &NameLength,
3146                                        NULL,
3147                                        NULL,
3148                                        NULL);
3149         if (!NT_SUCCESS(Status))
3150         {
3151             if (Status == STATUS_NO_MORE_ENTRIES)
3152                 Status = STATUS_SUCCESS;
3153             break;
3154         }
3155 
3156         TRACE("EnumIndex: %lu\n", EnumIndex);
3157         TRACE("Alias name: %S\n", AliasName);
3158         TRACE("Name length: %lu\n", NameLength);
3159 
3160         if ((RequiredLength + NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION)) > PreferedMaximumLength)
3161         {
3162             MoreEntries = TRUE;
3163             break;
3164         }
3165 
3166         RequiredLength += (NameLength + sizeof(UNICODE_NULL) + sizeof(SAMPR_RID_ENUMERATION));
3167         EnumCount++;
3168 
3169         EnumIndex++;
3170     }
3171 
3172     TRACE("EnumCount: %lu\n", EnumCount);
3173     TRACE("RequiredLength: %lu\n", RequiredLength);
3174 
3175     if (!NT_SUCCESS(Status))
3176         goto done;
3177 
3178     EnumBuffer = midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER));
3179     if (EnumBuffer == NULL)
3180     {
3181         Status = STATUS_INSUFFICIENT_RESOURCES;
3182         goto done;
3183     }
3184 
3185     EnumBuffer->EntriesRead = EnumCount;
3186     if (EnumCount == 0)
3187     {
3188         Status = STATUS_NO_MORE_ENTRIES;
3189         goto done;
3190     }
3191 
3192     EnumBuffer->Buffer = midl_user_allocate(EnumCount * sizeof(SAMPR_RID_ENUMERATION));
3193     if (EnumBuffer->Buffer == NULL)
3194     {
3195         Status = STATUS_INSUFFICIENT_RESOURCES;
3196         goto done;
3197     }
3198 
3199     TRACE("Part 2\n");
3200 
3201     EnumIndex = *EnumerationContext;
3202     for (i = 0; i < EnumCount; i++, EnumIndex++)
3203     {
3204         NameLength = 64 * sizeof(WCHAR);
3205         DataLength = sizeof(ULONG);
3206         Status = SampRegEnumerateValue(NamesKeyHandle,
3207                                        EnumIndex,
3208                                        AliasName,
3209                                        &NameLength,
3210                                        NULL,
3211                                        &Rid,
3212                                        &DataLength);
3213         if (!NT_SUCCESS(Status))
3214         {
3215             if (Status == STATUS_NO_MORE_ENTRIES)
3216                 Status = STATUS_SUCCESS;
3217             break;
3218         }
3219 
3220         TRACE("EnumIndex: %lu\n", EnumIndex);
3221         TRACE("Alias name: %S\n", AliasName);
3222         TRACE("Name length: %lu\n", NameLength);
3223         TRACE("RID: %lu\n", Rid);
3224 
3225         EnumBuffer->Buffer[i].RelativeId = Rid;
3226 
3227         EnumBuffer->Buffer[i].Name.Length = (USHORT)NameLength;
3228         EnumBuffer->Buffer[i].Name.MaximumLength = (USHORT)(NameLength + sizeof(UNICODE_NULL));
3229 
3230 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
3231 #if 0
3232         EnumBuffer->Buffer[i].Name.Buffer = midl_user_allocate(EnumBuffer->Buffer[i].Name.MaximumLength);
3233         if (EnumBuffer->Buffer[i].Name.Buffer == NULL)
3234         {
3235             Status = STATUS_INSUFFICIENT_RESOURCES;
3236             goto done;
3237         }
3238 
3239         memcpy(EnumBuffer->Buffer[i].Name.Buffer,
3240                AliasName,
3241                EnumBuffer->Buffer[i].Name.Length);
3242 #endif
3243     }
3244 
3245 done:
3246     if (NT_SUCCESS(Status))
3247     {
3248         *EnumerationContext += EnumCount;
3249         *Buffer = EnumBuffer;
3250         *CountReturned = EnumCount;
3251     }
3252     else
3253     {
3254         *EnumerationContext = 0;
3255         *Buffer = NULL;
3256         *CountReturned = 0;
3257 
3258         if (EnumBuffer != NULL)
3259         {
3260             if (EnumBuffer->Buffer != NULL)
3261             {
3262                 if (EnumBuffer->EntriesRead != 0)
3263                 {
3264                     for (i = 0; i < EnumBuffer->EntriesRead; i++)
3265                     {
3266                         if (EnumBuffer->Buffer[i].Name.Buffer != NULL)
3267                             midl_user_free(EnumBuffer->Buffer[i].Name.Buffer);
3268                     }
3269                 }
3270 
3271                 midl_user_free(EnumBuffer->Buffer);
3272             }
3273 
3274             midl_user_free(EnumBuffer);
3275         }
3276     }
3277 
3278     SampRegCloseKey(&NamesKeyHandle);
3279     SampRegCloseKey(&AliasesKeyHandle);
3280 
3281     if ((Status == STATUS_SUCCESS) && (MoreEntries != FALSE))
3282         Status = STATUS_MORE_ENTRIES;
3283 
3284     RtlReleaseResource(&SampResource);
3285 
3286     return Status;
3287 }
3288 
3289 
3290 /* Function 16 */
3291 NTSTATUS
3292 NTAPI
3293 SamrGetAliasMembership(IN SAMPR_HANDLE DomainHandle,
3294                        IN PSAMPR_PSID_ARRAY SidArray,
3295                        OUT PSAMPR_ULONG_ARRAY Membership)
3296 {
3297     PSAM_DB_OBJECT DomainObject;
3298     HANDLE AliasesKeyHandle = NULL;
3299     HANDLE MembersKeyHandle = NULL;
3300     HANDLE MemberKeyHandle = NULL;
3301     LPWSTR MemberSidString = NULL;
3302     PULONG RidArray = NULL;
3303     ULONG MaxSidCount = 0;
3304     ULONG ValueCount;
3305     ULONG DataLength;
3306     ULONG i, j;
3307     ULONG RidIndex;
3308     NTSTATUS Status;
3309     WCHAR NameBuffer[9];
3310 
3311     TRACE("SamrGetAliasMembership(%p %p %p)\n",
3312           DomainHandle, SidArray, Membership);
3313 
3314     RtlAcquireResourceShared(&SampResource,
3315                              TRUE);
3316 
3317     /* Validate the domain handle */
3318     Status = SampValidateDbObject(DomainHandle,
3319                                   SamDbDomainObject,
3320                                   DOMAIN_GET_ALIAS_MEMBERSHIP,
3321                                   &DomainObject);
3322     if (!NT_SUCCESS(Status))
3323         goto done;
3324 
3325     Status = SampRegOpenKey(DomainObject->KeyHandle,
3326                             L"Aliases",
3327                             KEY_READ,
3328                             &AliasesKeyHandle);
3329     TRACE("SampRegOpenKey returned %08lX\n", Status);
3330     if (!NT_SUCCESS(Status))
3331         goto done;
3332 
3333     Status = SampRegOpenKey(AliasesKeyHandle,
3334                             L"Members",
3335                             KEY_READ,
3336                             &MembersKeyHandle);
3337     TRACE("SampRegOpenKey returned %08lX\n", Status);
3338 
3339     if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
3340     {
3341         Status = STATUS_SUCCESS;
3342         goto done;
3343     }
3344 
3345     if (!NT_SUCCESS(Status))
3346         goto done;
3347 
3348     for (i = 0; i < SidArray->Count; i++)
3349     {
3350         ConvertSidToStringSid(SidArray->Sids[i].SidPointer, &MemberSidString);
3351 TRACE("Open %S\n", MemberSidString);
3352 
3353         Status = SampRegOpenKey(MembersKeyHandle,
3354                                 MemberSidString,
3355                                 KEY_READ,
3356                                 &MemberKeyHandle);
3357         TRACE("SampRegOpenKey returned %08lX\n", Status);
3358         if (NT_SUCCESS(Status))
3359         {
3360             Status = SampRegQueryKeyInfo(MemberKeyHandle,
3361                                          NULL,
3362                                          &ValueCount);
3363             if (NT_SUCCESS(Status))
3364             {
3365                 TRACE("Found %lu values\n", ValueCount);
3366                 MaxSidCount += ValueCount;
3367             }
3368 
3369             SampRegCloseKey(&MemberKeyHandle);
3370         }
3371 
3372         if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
3373             Status = STATUS_SUCCESS;
3374 
3375         LocalFree(MemberSidString);
3376     }
3377 
3378     if (MaxSidCount == 0)
3379     {
3380         Status = STATUS_SUCCESS;
3381         goto done;
3382     }
3383 
3384     TRACE("Maximum sid count: %lu\n", MaxSidCount);
3385     RidArray = midl_user_allocate(MaxSidCount * sizeof(ULONG));
3386     if (RidArray == NULL)
3387     {
3388         Status = STATUS_INSUFFICIENT_RESOURCES;
3389         goto done;
3390     }
3391 
3392     RidIndex = 0;
3393     for (i = 0; i < SidArray->Count; i++)
3394     {
3395         ConvertSidToStringSid(SidArray->Sids[i].SidPointer, &MemberSidString);
3396 TRACE("Open %S\n", MemberSidString);
3397 
3398         Status = SampRegOpenKey(MembersKeyHandle,
3399                                 MemberSidString,
3400                                 KEY_READ,
3401                                 &MemberKeyHandle);
3402         TRACE("SampRegOpenKey returned %08lX\n", Status);
3403         if (NT_SUCCESS(Status))
3404         {
3405             Status = SampRegQueryKeyInfo(MemberKeyHandle,
3406                                          NULL,
3407                                          &ValueCount);
3408             if (NT_SUCCESS(Status))
3409             {
3410                 TRACE("Found %lu values\n", ValueCount);
3411 
3412                 for (j = 0; j < ValueCount; j++)
3413                 {
3414                     DataLength = 9 * sizeof(WCHAR);
3415                     Status = SampRegEnumerateValue(MemberKeyHandle,
3416                                                    j,
3417                                                    NameBuffer,
3418                                                    &DataLength,
3419                                                    NULL,
3420                                                    NULL,
3421                                                    NULL);
3422                     if (NT_SUCCESS(Status))
3423                     {
3424                         /* FIXME: Do not return each RID more than once. */
3425                         RidArray[RidIndex] = wcstoul(NameBuffer, NULL, 16);
3426                         RidIndex++;
3427                     }
3428                 }
3429             }
3430 
3431             SampRegCloseKey(&MemberKeyHandle);
3432         }
3433 
3434         if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
3435             Status = STATUS_SUCCESS;
3436 
3437         LocalFree(MemberSidString);
3438     }
3439 
3440 done:
3441     SampRegCloseKey(&MembersKeyHandle);
3442     SampRegCloseKey(&AliasesKeyHandle);
3443 
3444     if (NT_SUCCESS(Status))
3445     {
3446         Membership->Count = MaxSidCount;
3447         Membership->Element = RidArray;
3448     }
3449     else
3450     {
3451         if (RidArray != NULL)
3452             midl_user_free(RidArray);
3453     }
3454 
3455     RtlReleaseResource(&SampResource);
3456 
3457     return Status;
3458 }
3459 
3460 
3461 /* Function 17 */
3462 NTSTATUS
3463 NTAPI
3464 SamrLookupNamesInDomain(IN SAMPR_HANDLE DomainHandle,
3465                         IN ULONG Count,
3466                         IN RPC_UNICODE_STRING Names[],
3467                         OUT PSAMPR_ULONG_ARRAY RelativeIds,
3468                         OUT PSAMPR_ULONG_ARRAY Use)
3469 {
3470     PSAM_DB_OBJECT DomainObject;
3471     HANDLE AccountsKeyHandle = NULL;
3472     HANDLE NamesKeyHandle = NULL;
3473     ULONG MappedCount = 0;
3474     ULONG DataLength;
3475     ULONG i;
3476     ULONG RelativeId;
3477     NTSTATUS Status;
3478 
3479     TRACE("SamrLookupNamesInDomain(%p %lu %p %p %p)\n",
3480           DomainHandle, Count, Names, RelativeIds, Use);
3481 
3482     RtlAcquireResourceShared(&SampResource,
3483                              TRUE);
3484 
3485     /* Validate the domain handle */
3486     Status = SampValidateDbObject(DomainHandle,
3487                                   SamDbDomainObject,
3488                                   DOMAIN_LOOKUP,
3489                                   &DomainObject);
3490     if (!NT_SUCCESS(Status))
3491     {
3492         TRACE("failed with status 0x%08lx\n", Status);
3493         goto done;
3494     }
3495 
3496     RelativeIds->Count = 0;
3497     Use->Count = 0;
3498 
3499     if (Count == 0)
3500     {
3501         Status = STATUS_SUCCESS;
3502         goto done;
3503     }
3504 
3505     /* Allocate the relative IDs array */
3506     RelativeIds->Element = midl_user_allocate(Count * sizeof(ULONG));
3507     if (RelativeIds->Element == NULL)
3508     {
3509         Status = STATUS_INSUFFICIENT_RESOURCES;
3510         goto done;
3511     }
3512 
3513     /* Allocate the use array */
3514     Use->Element = midl_user_allocate(Count * sizeof(ULONG));
3515     if (Use->Element == NULL)
3516     {
3517         Status = STATUS_INSUFFICIENT_RESOURCES;
3518         goto done;
3519     }
3520 
3521     RelativeIds->Count = Count;
3522     Use->Count = Count;
3523 
3524     for (i = 0; i < Count; i++)
3525     {
3526         TRACE("Name: %S\n", Names[i].Buffer);
3527 
3528         RelativeId = 0;
3529 
3530         /* Lookup aliases */
3531         Status = SampRegOpenKey(DomainObject->KeyHandle,
3532                                 L"Aliases",
3533                                 KEY_READ,
3534                                 &AccountsKeyHandle);
3535         if (NT_SUCCESS(Status))
3536         {
3537             Status = SampRegOpenKey(AccountsKeyHandle,
3538                                     L"Names",
3539                                     KEY_READ,
3540                                     &NamesKeyHandle);
3541             if (NT_SUCCESS(Status))
3542             {
3543                 DataLength = sizeof(ULONG);
3544                 Status = SampRegQueryValue(NamesKeyHandle,
3545                                            Names[i].Buffer,
3546                                            NULL,
3547                                            &RelativeId,
3548                                            &DataLength);
3549 
3550                 SampRegCloseKey(&NamesKeyHandle);
3551             }
3552 
3553             SampRegCloseKey(&AccountsKeyHandle);
3554         }
3555 
3556         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3557             break;
3558 
3559         /* Return alias account */
3560         if (NT_SUCCESS(Status) && RelativeId != 0)
3561         {
3562             TRACE("Rid: %lu\n", RelativeId);
3563             RelativeIds->Element[i] = RelativeId;
3564             Use->Element[i] = SidTypeAlias;
3565             MappedCount++;
3566             continue;
3567         }
3568 
3569         /* Lookup groups */
3570         Status = SampRegOpenKey(DomainObject->KeyHandle,
3571                                 L"Groups",
3572                                 KEY_READ,
3573                                 &AccountsKeyHandle);
3574         if (NT_SUCCESS(Status))
3575         {
3576             Status = SampRegOpenKey(AccountsKeyHandle,
3577                                     L"Names",
3578                                     KEY_READ,
3579                                     &NamesKeyHandle);
3580             if (NT_SUCCESS(Status))
3581             {
3582                 DataLength = sizeof(ULONG);
3583                 Status = SampRegQueryValue(NamesKeyHandle,
3584                                            Names[i].Buffer,
3585                                            NULL,
3586                                            &RelativeId,
3587                                            &DataLength);
3588 
3589                 SampRegCloseKey(&NamesKeyHandle);
3590             }
3591 
3592             SampRegCloseKey(&AccountsKeyHandle);
3593         }
3594 
3595         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3596             break;
3597 
3598         /* Return group account */
3599         if (NT_SUCCESS(Status) && RelativeId != 0)
3600         {
3601             TRACE("Rid: %lu\n", RelativeId);
3602             RelativeIds->Element[i] = RelativeId;
3603             Use->Element[i] = SidTypeGroup;
3604             MappedCount++;
3605             continue;
3606         }
3607 
3608         /* Lookup users */
3609         Status = SampRegOpenKey(DomainObject->KeyHandle,
3610                                 L"Users",
3611                                 KEY_READ,
3612                                 &AccountsKeyHandle);
3613         if (NT_SUCCESS(Status))
3614         {
3615             Status = SampRegOpenKey(AccountsKeyHandle,
3616                                     L"Names",
3617                                     KEY_READ,
3618                                     &NamesKeyHandle);
3619             if (NT_SUCCESS(Status))
3620             {
3621                 DataLength = sizeof(ULONG);
3622                 Status = SampRegQueryValue(NamesKeyHandle,
3623                                            Names[i].Buffer,
3624                                            NULL,
3625                                            &RelativeId,
3626                                            &DataLength);
3627 
3628                 SampRegCloseKey(&NamesKeyHandle);
3629             }
3630 
3631             SampRegCloseKey(&AccountsKeyHandle);
3632         }
3633 
3634         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3635             break;
3636 
3637         /* Return user account */
3638         if (NT_SUCCESS(Status) && RelativeId != 0)
3639         {
3640             TRACE("Rid: %lu\n", RelativeId);
3641             RelativeIds->Element[i] = RelativeId;
3642             Use->Element[i] = SidTypeUser;
3643             MappedCount++;
3644             continue;
3645         }
3646 
3647         /* Return unknown account */
3648         RelativeIds->Element[i] = 0;
3649         Use->Element[i] = SidTypeUnknown;
3650     }
3651 
3652 done:
3653     if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
3654         Status = STATUS_SUCCESS;
3655 
3656     if (NT_SUCCESS(Status))
3657     {
3658         if (MappedCount == 0)
3659             Status = STATUS_NONE_MAPPED;
3660         else if (MappedCount < Count)
3661             Status = STATUS_SOME_NOT_MAPPED;
3662     }
3663     else
3664     {
3665         if (RelativeIds->Element != NULL)
3666         {
3667             midl_user_free(RelativeIds->Element);
3668             RelativeIds->Element = NULL;
3669         }
3670 
3671         RelativeIds->Count = 0;
3672 
3673         if (Use->Element != NULL)
3674         {
3675             midl_user_free(Use->Element);
3676             Use->Element = NULL;
3677         }
3678 
3679         Use->Count = 0;
3680     }
3681 
3682     RtlReleaseResource(&SampResource);
3683 
3684     TRACE("Returned Status %lx\n", Status);
3685 
3686     return Status;
3687 }
3688 
3689 
3690 /* Function 18 */
3691 NTSTATUS
3692 NTAPI
3693 SamrLookupIdsInDomain(IN SAMPR_HANDLE DomainHandle,
3694                       IN ULONG Count,
3695                       IN ULONG *RelativeIds,
3696                       OUT PSAMPR_RETURNED_USTRING_ARRAY Names,
3697                       OUT PSAMPR_ULONG_ARRAY Use)
3698 {
3699     PSAM_DB_OBJECT DomainObject;
3700     WCHAR RidString[9];
3701     HANDLE AccountsKeyHandle = NULL;
3702     HANDLE AccountKeyHandle = NULL;
3703     ULONG MappedCount = 0;
3704     ULONG DataLength;
3705     ULONG i;
3706     NTSTATUS Status;
3707 
3708     TRACE("SamrLookupIdsInDomain(%p %lu %p %p %p)\n",
3709           DomainHandle, Count, RelativeIds, Names, Use);
3710 
3711     RtlAcquireResourceShared(&SampResource,
3712                              TRUE);
3713 
3714     /* Validate the domain handle */
3715     Status = SampValidateDbObject(DomainHandle,
3716                                   SamDbDomainObject,
3717                                   DOMAIN_LOOKUP,
3718                                   &DomainObject);
3719     if (!NT_SUCCESS(Status))
3720     {
3721         TRACE("failed with status 0x%08lx\n", Status);
3722         goto done;
3723     }
3724 
3725     Names->Count = 0;
3726     Use->Count = 0;
3727 
3728     if (Count == 0)
3729     {
3730         Status = STATUS_SUCCESS;
3731         goto done;
3732     }
3733 
3734     /* Allocate the names array */
3735     Names->Element = midl_user_allocate(Count * sizeof(*Names->Element));
3736     if (Names->Element == NULL)
3737     {
3738         Status = STATUS_INSUFFICIENT_RESOURCES;
3739         goto done;
3740     }
3741 
3742     /* Allocate the use array */
3743     Use->Element = midl_user_allocate(Count * sizeof(*Use->Element));
3744     if (Use->Element == NULL)
3745     {
3746         Status = STATUS_INSUFFICIENT_RESOURCES;
3747         goto done;
3748     }
3749 
3750     Names->Count = Count;
3751     Use->Count = Count;
3752 
3753     for (i = 0; i < Count; i++)
3754     {
3755         TRACE("RID: %lu\n", RelativeIds[i]);
3756 
3757         swprintf(RidString, L"%08lx", RelativeIds[i]);
3758 
3759         /* Lookup aliases */
3760         Status = SampRegOpenKey(DomainObject->KeyHandle,
3761                                 L"Aliases",
3762                                 KEY_READ,
3763                                 &AccountsKeyHandle);
3764         if (NT_SUCCESS(Status))
3765         {
3766             Status = SampRegOpenKey(AccountsKeyHandle,
3767                                     RidString,
3768                                     KEY_READ,
3769                                     &AccountKeyHandle);
3770             if (NT_SUCCESS(Status))
3771             {
3772                 DataLength = 0;
3773                 Status = SampRegQueryValue(AccountKeyHandle,
3774                                            L"Name",
3775                                            NULL,
3776                                            NULL,
3777                                            &DataLength);
3778                 if (NT_SUCCESS(Status))
3779                 {
3780                     Names->Element[i].Buffer = midl_user_allocate(DataLength);
3781                     if (Names->Element[i].Buffer == NULL)
3782                         Status = STATUS_INSUFFICIENT_RESOURCES;
3783 
3784                     if (NT_SUCCESS(Status))
3785                     {
3786                         Names->Element[i].MaximumLength = (USHORT)DataLength;
3787                         Names->Element[i].Length = (USHORT)(DataLength - sizeof(WCHAR));
3788 
3789                         Status = SampRegQueryValue(AccountKeyHandle,
3790                                                    L"Name",
3791                                                    NULL,
3792                                                    Names->Element[i].Buffer,
3793                                                    &DataLength);
3794                     }
3795                 }
3796 
3797                 SampRegCloseKey(&AccountKeyHandle);
3798             }
3799 
3800             SampRegCloseKey(&AccountsKeyHandle);
3801         }
3802 
3803         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3804             break;
3805 
3806         /* Return alias account */
3807         if (NT_SUCCESS(Status) && Names->Element[i].Buffer != NULL)
3808         {
3809             TRACE("Name: %S\n", Names->Element[i].Buffer);
3810             Use->Element[i] = SidTypeAlias;
3811             MappedCount++;
3812             continue;
3813         }
3814 
3815         /* Lookup groups */
3816         Status = SampRegOpenKey(DomainObject->KeyHandle,
3817                                 L"Groups",
3818                                 KEY_READ,
3819                                 &AccountsKeyHandle);
3820         if (NT_SUCCESS(Status))
3821         {
3822             Status = SampRegOpenKey(AccountsKeyHandle,
3823                                     RidString,
3824                                     KEY_READ,
3825                                     &AccountKeyHandle);
3826             if (NT_SUCCESS(Status))
3827             {
3828                 DataLength = 0;
3829                 Status = SampRegQueryValue(AccountKeyHandle,
3830                                            L"Name",
3831                                            NULL,
3832                                            NULL,
3833                                            &DataLength);
3834                 if (NT_SUCCESS(Status))
3835                 {
3836                     Names->Element[i].Buffer = midl_user_allocate(DataLength);
3837                     if (Names->Element[i].Buffer == NULL)
3838                         Status = STATUS_INSUFFICIENT_RESOURCES;
3839 
3840                     if (NT_SUCCESS(Status))
3841                     {
3842                         Names->Element[i].MaximumLength = (USHORT)DataLength;
3843                         Names->Element[i].Length = (USHORT)(DataLength - sizeof(WCHAR));
3844 
3845                         Status = SampRegQueryValue(AccountKeyHandle,
3846                                                    L"Name",
3847                                                    NULL,
3848                                                    Names->Element[i].Buffer,
3849                                                    &DataLength);
3850                     }
3851                 }
3852 
3853                 SampRegCloseKey(&AccountKeyHandle);
3854             }
3855 
3856             SampRegCloseKey(&AccountsKeyHandle);
3857         }
3858 
3859         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3860             break;
3861 
3862         /* Return group account */
3863         if (NT_SUCCESS(Status) && Names->Element[i].Buffer != NULL)
3864         {
3865             TRACE("Name: %S\n", Names->Element[i].Buffer);
3866             Use->Element[i] = SidTypeGroup;
3867             MappedCount++;
3868             continue;
3869         }
3870 
3871         /* Lookup users */
3872         Status = SampRegOpenKey(DomainObject->KeyHandle,
3873                                 L"Users",
3874                                 KEY_READ,
3875                                 &AccountsKeyHandle);
3876         if (NT_SUCCESS(Status))
3877         {
3878             Status = SampRegOpenKey(AccountsKeyHandle,
3879                                     RidString,
3880                                     KEY_READ,
3881                                     &AccountKeyHandle);
3882             if (NT_SUCCESS(Status))
3883             {
3884                 DataLength = 0;
3885                 Status = SampRegQueryValue(AccountKeyHandle,
3886                                            L"Name",
3887                                            NULL,
3888                                            NULL,
3889                                            &DataLength);
3890                 if (NT_SUCCESS(Status))
3891                 {
3892                     TRACE("DataLength: %lu\n", DataLength);
3893 
3894                     Names->Element[i].Buffer = midl_user_allocate(DataLength);
3895                     if (Names->Element[i].Buffer == NULL)
3896                         Status = STATUS_INSUFFICIENT_RESOURCES;
3897 
3898                     if (NT_SUCCESS(Status))
3899                     {
3900                         Names->Element[i].MaximumLength = (USHORT)DataLength;
3901                         Names->Element[i].Length = (USHORT)(DataLength - sizeof(WCHAR));
3902 
3903                         Status = SampRegQueryValue(AccountKeyHandle,
3904                                                    L"Name",
3905                                                    NULL,
3906                                                    Names->Element[i].Buffer,
3907                                                    &DataLength);
3908                     }
3909                 }
3910 
3911                 SampRegCloseKey(&AccountKeyHandle);
3912             }
3913 
3914             SampRegCloseKey(&AccountsKeyHandle);
3915         }
3916 
3917         if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
3918             break;
3919 
3920         /* Return user account */
3921         if (NT_SUCCESS(Status) && Names->Element[i].Buffer != NULL)
3922         {
3923             TRACE("Name: %S\n", Names->Element[i].Buffer);
3924             Use->Element[i] = SidTypeUser;
3925             MappedCount++;
3926             continue;
3927         }
3928 
3929         /* Return unknown account */
3930         Use->Element[i] = SidTypeUnknown;
3931     }
3932 
3933 done:
3934     if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
3935         Status = STATUS_SUCCESS;
3936 
3937     if (NT_SUCCESS(Status))
3938     {
3939         if (MappedCount == 0)
3940             Status = STATUS_NONE_MAPPED;
3941         else if (MappedCount < Count)
3942             Status = STATUS_SOME_NOT_MAPPED;
3943     }
3944     else
3945     {
3946         if (Names->Element != NULL)
3947         {
3948             for (i = 0; i < Count; i++)
3949             {
3950                 if (Names->Element[i].Buffer != NULL)
3951                     midl_user_free(Names->Element[i].Buffer);
3952             }
3953 
3954             midl_user_free(Names->Element);
3955             Names->Element = NULL;
3956         }
3957 
3958         Names->Count = 0;
3959 
3960         if (Use->Element != NULL)
3961         {
3962             midl_user_free(Use->Element);
3963             Use->Element = NULL;
3964         }
3965 
3966         Use->Count = 0;
3967     }
3968 
3969     RtlReleaseResource(&SampResource);
3970 
3971     return Status;
3972 }
3973 
3974 
3975 /* Function 19 */
3976 NTSTATUS
3977 NTAPI
3978 SamrOpenGroup(IN SAMPR_HANDLE DomainHandle,
3979               IN ACCESS_MASK DesiredAccess,
3980               IN unsigned long GroupId,
3981               OUT SAMPR_HANDLE *GroupHandle)
3982 {
3983     PSAM_DB_OBJECT DomainObject;
3984     PSAM_DB_OBJECT GroupObject;
3985     WCHAR szRid[9];
3986     NTSTATUS Status;
3987 
3988     TRACE("SamrOpenGroup(%p %lx %lx %p)\n",
3989           DomainHandle, DesiredAccess, GroupId, GroupHandle);
3990 
3991     /* Map generic access rights */
3992     RtlMapGenericMask(&DesiredAccess,
3993                       &GroupMapping);
3994 
3995     RtlAcquireResourceShared(&SampResource,
3996                              TRUE);
3997 
3998     /* Validate the domain handle */
3999     Status = SampValidateDbObject(DomainHandle,
4000                                   SamDbDomainObject,
4001                                   DOMAIN_LOOKUP,
4002                                   &DomainObject);
4003     if (!NT_SUCCESS(Status))
4004     {
4005         TRACE("failed with status 0x%08lx\n", Status);
4006         goto done;
4007     }
4008 
4009     /* Convert the RID into a string (hex) */
4010     swprintf(szRid, L"%08lX", GroupId);
4011 
4012     /* Create the group object */
4013     Status = SampOpenDbObject(DomainObject,
4014                               L"Groups",
4015                               szRid,
4016                               GroupId,
4017                               SamDbGroupObject,
4018                               DesiredAccess,
4019                               &GroupObject);
4020     if (!NT_SUCCESS(Status))
4021     {
4022         TRACE("failed with status 0x%08lx\n", Status);
4023         goto done;
4024     }
4025 
4026     *GroupHandle = (SAMPR_HANDLE)GroupObject;
4027 
4028 done:
4029     RtlReleaseResource(&SampResource);
4030 
4031     return Status;
4032 }
4033 
4034 
4035 static NTSTATUS
4036 SampQueryGroupGeneral(PSAM_DB_OBJECT GroupObject,
4037                       PSAMPR_GROUP_INFO_BUFFER *Buffer)
4038 {
4039     PSAMPR_GROUP_INFO_BUFFER InfoBuffer = NULL;
4040     SAM_GROUP_FIXED_DATA FixedData;
4041     ULONG MembersLength = 0;
4042     ULONG Length = 0;
4043     NTSTATUS Status;
4044 
4045     *Buffer = NULL;
4046 
4047     InfoBuffer = midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER));
4048     if (InfoBuffer == NULL)
4049         return STATUS_INSUFFICIENT_RESOURCES;
4050 
4051     Status = SampGetObjectAttributeString(GroupObject,
4052                                           L"Name",
4053                                           &InfoBuffer->General.Name);
4054     if (!NT_SUCCESS(Status))
4055     {
4056         TRACE("Status 0x%08lx\n", Status);
4057         goto done;
4058     }
4059 
4060     Status = SampGetObjectAttributeString(GroupObject,
4061                                           L"AdminComment",
4062                                           &InfoBuffer->General.AdminComment);
4063     if (!NT_SUCCESS(Status))
4064     {
4065         TRACE("Status 0x%08lx\n", Status);
4066         goto done;
4067     }
4068 
4069     Length = sizeof(SAM_GROUP_FIXED_DATA);
4070     Status = SampGetObjectAttribute(GroupObject,
4071                                     L"F",
4072                                     NULL,
4073                                     (PVOID)&FixedData,
4074                                     &Length);
4075     if (!NT_SUCCESS(Status))
4076     {
4077         TRACE("Status 0x%08lx\n", Status);
4078         goto done;
4079     }
4080 
4081     InfoBuffer->General.Attributes = FixedData.Attributes;
4082 
4083     Status = SampGetObjectAttribute(GroupObject,
4084                                     L"Members",
4085                                     NULL,
4086                                     NULL,
4087                                     &MembersLength);
4088     if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_NOT_FOUND)
4089     {
4090         TRACE("Status 0x%08lx\n", Status);
4091         goto done;
4092     }
4093 
4094     if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
4095     {
4096         InfoBuffer->General.MemberCount = 0;
4097         Status = STATUS_SUCCESS;
4098     }
4099     else
4100     {
4101         InfoBuffer->General.MemberCount = MembersLength / sizeof(ULONG);
4102     }
4103 
4104     *Buffer = InfoBuffer;
4105 
4106 done:
4107     if (!NT_SUCCESS(Status))
4108     {
4109         if (InfoBuffer != NULL)
4110         {
4111             if (InfoBuffer->General.Name.Buffer != NULL)
4112                 midl_user_free(InfoBuffer->General.Name.Buffer);
4113 
4114             if (InfoBuffer->General.AdminComment.Buffer != NULL)
4115                 midl_user_free(InfoBuffer->General.AdminComment.Buffer);
4116 
4117             midl_user_free(InfoBuffer);
4118         }
4119     }
4120 
4121     return Status;
4122 }
4123 
4124 
4125 static NTSTATUS
4126 SampQueryGroupName(PSAM_DB_OBJECT GroupObject,
4127                    PSAMPR_GROUP_INFO_BUFFER *Buffer)
4128 {
4129     PSAMPR_GROUP_INFO_BUFFER InfoBuffer = NULL;
4130     NTSTATUS Status;
4131 
4132     *Buffer = NULL;
4133 
4134     InfoBuffer = midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER));
4135     if (InfoBuffer == NULL)
4136         return STATUS_INSUFFICIENT_RESOURCES;
4137 
4138     Status = SampGetObjectAttributeString(GroupObject,
4139                                           L"Name",
4140                                           &InfoBuffer->Name.Name);
4141     if (!NT_SUCCESS(Status))
4142     {
4143         TRACE("Status 0x%08lx\n", Status);
4144         goto done;
4145     }
4146 
4147     *Buffer = InfoBuffer;
4148 
4149 done:
4150     if (!NT_SUCCESS(Status))
4151     {
4152         if (InfoBuffer != NULL)
4153         {
4154             if (InfoBuffer->Name.Name.Buffer != NULL)
4155                 midl_user_free(InfoBuffer->Name.Name.Buffer);
4156 
4157             midl_user_free(InfoBuffer);
4158         }
4159     }
4160 
4161     return Status;
4162 }
4163 
4164 
4165 static NTSTATUS
4166 SampQueryGroupAttribute(PSAM_DB_OBJECT GroupObject,
4167                         PSAMPR_GROUP_INFO_BUFFER *Buffer)
4168 {
4169     PSAMPR_GROUP_INFO_BUFFER InfoBuffer = NULL;
4170     SAM_GROUP_FIXED_DATA FixedData;
4171     ULONG Length = 0;
4172     NTSTATUS Status;
4173 
4174     *Buffer = NULL;
4175 
4176     InfoBuffer = midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER));
4177     if (InfoBuffer == NULL)
4178         return STATUS_INSUFFICIENT_RESOURCES;
4179 
4180     Length = sizeof(SAM_GROUP_FIXED_DATA);
4181     Status = SampGetObjectAttribute(GroupObject,
4182                                     L"F",
4183                                     NULL,
4184                                     (PVOID)&FixedData,
4185                                     &Length);
4186     if (!NT_SUCCESS(Status))
4187     {
4188         TRACE("Status 0x%08lx\n", Status);
4189         goto done;
4190     }
4191 
4192     InfoBuffer->Attribute.Attributes = FixedData.Attributes;
4193 
4194     *Buffer = InfoBuffer;
4195 
4196 done:
4197     if (!NT_SUCCESS(Status))
4198     {
4199         if (InfoBuffer != NULL)
4200         {
4201             midl_user_free(InfoBuffer);
4202         }
4203     }
4204 
4205     return Status;
4206 }
4207 
4208 
4209 static NTSTATUS
4210 SampQueryGroupAdminComment(PSAM_DB_OBJECT GroupObject,
4211                            PSAMPR_GROUP_INFO_BUFFER *Buffer)
4212 {
4213     PSAMPR_GROUP_INFO_BUFFER InfoBuffer = NULL;
4214     NTSTATUS Status;
4215 
4216     *Buffer = NULL;
4217 
4218     InfoBuffer = midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER));
4219     if (InfoBuffer == NULL)
4220         return STATUS_INSUFFICIENT_RESOURCES;
4221 
4222     Status = SampGetObjectAttributeString(GroupObject,
4223                                           L"AdminComment",
4224                                           &InfoBuffer->AdminComment.AdminComment);
4225     if (!NT_SUCCESS(Status))
4226     {
4227         TRACE("Status 0x%08lx\n", Status);
4228         goto done;
4229     }
4230 
4231     *Buffer = InfoBuffer;
4232 
4233 done:
4234     if (!NT_SUCCESS(Status))
4235     {
4236         if (InfoBuffer != NULL)
4237         {
4238             if (InfoBuffer->AdminComment.AdminComment.Buffer != NULL)
4239                 midl_user_free(InfoBuffer->AdminComment.AdminComment.Buffer);
4240 
4241             midl_user_free(InfoBuffer);
4242         }
4243     }
4244 
4245     return Status;
4246 }
4247 
4248 
4249 /* Function 20 */
4250 NTSTATUS
4251 NTAPI
4252 SamrQueryInformationGroup(IN SAMPR_HANDLE GroupHandle,
4253                           IN GROUP_INFORMATION_CLASS GroupInformationClass,
4254                           OUT PSAMPR_GROUP_INFO_BUFFER *Buffer)
4255 {
4256     PSAM_DB_OBJECT GroupObject;
4257     NTSTATUS Status;
4258 
4259     TRACE("SamrQueryInformationGroup(%p %lu %p)\n",
4260           GroupHandle, GroupInformationClass, Buffer);
4261 
4262     RtlAcquireResourceShared(&SampResource,
4263                              TRUE);
4264 
4265     /* Validate the group handle */
4266     Status = SampValidateDbObject(GroupHandle,
4267                                   SamDbGroupObject,
4268                                   GROUP_READ_INFORMATION,
4269                                   &GroupObject);
4270     if (!NT_SUCCESS(Status))
4271         goto done;
4272 
4273     switch (GroupInformationClass)
4274     {
4275         case GroupGeneralInformation:
4276             Status = SampQueryGroupGeneral(GroupObject,
4277                                            Buffer);
4278             break;
4279 
4280         case GroupNameInformation:
4281             Status = SampQueryGroupName(GroupObject,
4282                                         Buffer);
4283             break;
4284 
4285         case GroupAttributeInformation:
4286             Status = SampQueryGroupAttribute(GroupObject,
4287                                              Buffer);
4288             break;
4289 
4290         case GroupAdminCommentInformation:
4291             Status = SampQueryGroupAdminComment(GroupObject,
4292                                                 Buffer);
4293             break;
4294 
4295         default:
4296             Status = STATUS_INVALID_INFO_CLASS;
4297             break;
4298     }
4299 
4300 done:
4301     RtlReleaseResource(&SampResource);
4302 
4303     return Status;
4304 }
4305 
4306 
4307 static NTSTATUS
4308 SampSetGroupName(PSAM_DB_OBJECT GroupObject,
4309                  PSAMPR_GROUP_INFO_BUFFER Buffer)
4310 {
4311     UNICODE_STRING OldGroupName = {0, 0, NULL};
4312     UNICODE_STRING NewGroupName;
4313     NTSTATUS Status;
4314 
4315     Status = SampGetObjectAttributeString(GroupObject,
4316                                           L"Name",
4317                                           (PRPC_UNICODE_STRING)&OldGroupName);
4318     if (!NT_SUCCESS(Status))
4319     {
4320         TRACE("SampGetObjectAttributeString failed (Status 0x%08lx)\n", Status);
4321         goto done;
4322     }
4323 
4324     /* Check the new account name */
4325     Status = SampCheckAccountName(&Buffer->Name.Name, 256);
4326     if (!NT_SUCCESS(Status))
4327     {
4328         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
4329         return Status;
4330     }
4331 
4332     NewGroupName.Length = Buffer->Name.Name.Length;
4333     NewGroupName.MaximumLength = Buffer->Name.Name.MaximumLength;
4334     NewGroupName.Buffer = Buffer->Name.Name.Buffer;
4335 
4336     if (!RtlEqualUnicodeString(&OldGroupName, &NewGroupName, TRUE))
4337     {
4338         Status = SampCheckAccountNameInDomain(GroupObject->ParentObject,
4339                                               NewGroupName.Buffer);
4340         if (!NT_SUCCESS(Status))
4341         {
4342             TRACE("Group name \'%S\' already exists in domain (Status 0x%08lx)\n",
4343                   NewGroupName.Buffer, Status);
4344             goto done;
4345         }
4346     }
4347 
4348     Status = SampSetAccountNameInDomain(GroupObject->ParentObject,
4349                                         L"Groups",
4350                                         NewGroupName.Buffer,
4351                                         GroupObject->RelativeId);
4352     if (!NT_SUCCESS(Status))
4353     {
4354         TRACE("SampSetAccountNameInDomain failed (Status 0x%08lx)\n", Status);
4355         goto done;
4356     }
4357 
4358     Status = SampRemoveAccountNameFromDomain(GroupObject->ParentObject,
4359                                              L"Groups",
4360                                              OldGroupName.Buffer);
4361     if (!NT_SUCCESS(Status))
4362     {
4363         TRACE("SampRemoveAccountNameFromDomain failed (Status 0x%08lx)\n", Status);
4364         goto done;
4365     }
4366 
4367     Status = SampSetObjectAttributeString(GroupObject,
4368                                           L"Name",
4369                                           (PRPC_UNICODE_STRING)&NewGroupName);
4370     if (!NT_SUCCESS(Status))
4371     {
4372         TRACE("SampSetObjectAttribute failed (Status 0x%08lx)\n", Status);
4373     }
4374 
4375 done:
4376     if (OldGroupName.Buffer != NULL)
4377         midl_user_free(OldGroupName.Buffer);
4378 
4379     return Status;
4380 }
4381 
4382 
4383 static NTSTATUS
4384 SampSetGroupAttribute(PSAM_DB_OBJECT GroupObject,
4385                       PSAMPR_GROUP_INFO_BUFFER Buffer)
4386 {
4387     SAM_GROUP_FIXED_DATA FixedData;
4388     ULONG Length = 0;
4389     NTSTATUS Status;
4390 
4391     Length = sizeof(SAM_GROUP_FIXED_DATA);
4392     Status = SampGetObjectAttribute(GroupObject,
4393                                     L"F",
4394                                     NULL,
4395                                     (PVOID)&FixedData,
4396                                     &Length);
4397     if (!NT_SUCCESS(Status))
4398         goto done;
4399 
4400     FixedData.Attributes = Buffer->Attribute.Attributes;
4401 
4402     Status = SampSetObjectAttribute(GroupObject,
4403                                     L"F",
4404                                     REG_BINARY,
4405                                     &FixedData,
4406                                     Length);
4407 
4408 done:
4409     return Status;
4410 }
4411 
4412 
4413 /* Function 21 */
4414 NTSTATUS
4415 NTAPI
4416 SamrSetInformationGroup(IN SAMPR_HANDLE GroupHandle,
4417                         IN GROUP_INFORMATION_CLASS GroupInformationClass,
4418                         IN PSAMPR_GROUP_INFO_BUFFER Buffer)
4419 {
4420     PSAM_DB_OBJECT GroupObject;
4421     NTSTATUS Status;
4422 
4423     TRACE("SamrSetInformationGroup(%p %lu %p)\n",
4424           GroupHandle, GroupInformationClass, Buffer);
4425 
4426     RtlAcquireResourceExclusive(&SampResource,
4427                                 TRUE);
4428 
4429     /* Validate the group handle */
4430     Status = SampValidateDbObject(GroupHandle,
4431                                   SamDbGroupObject,
4432                                   GROUP_WRITE_ACCOUNT,
4433                                   &GroupObject);
4434     if (!NT_SUCCESS(Status))
4435         goto done;
4436 
4437     switch (GroupInformationClass)
4438     {
4439         case GroupNameInformation:
4440             Status = SampSetGroupName(GroupObject,
4441                                       Buffer);
4442             break;
4443 
4444         case GroupAttributeInformation:
4445             Status = SampSetGroupAttribute(GroupObject,
4446                                            Buffer);
4447             break;
4448 
4449         case GroupAdminCommentInformation:
4450             Status = SampSetObjectAttributeString(GroupObject,
4451                                                   L"AdminComment",
4452                                                   &Buffer->AdminComment.AdminComment);
4453             break;
4454 
4455         default:
4456             Status = STATUS_INVALID_INFO_CLASS;
4457             break;
4458     }
4459 
4460 done:
4461     RtlReleaseResource(&SampResource);
4462 
4463     return Status;
4464 }
4465 
4466 
4467 /* Function 22 */
4468 NTSTATUS
4469 NTAPI
4470 SamrAddMemberToGroup(IN SAMPR_HANDLE GroupHandle,
4471                      IN unsigned long MemberId,
4472                      IN unsigned long Attributes)
4473 {
4474     PSAM_DB_OBJECT GroupObject;
4475     PSAM_DB_OBJECT UserObject = NULL;
4476     NTSTATUS Status;
4477 
4478     TRACE("SamrAddMemberToGroup(%p %lu %lx)\n",
4479           GroupHandle, MemberId, Attributes);
4480 
4481     RtlAcquireResourceExclusive(&SampResource,
4482                                 TRUE);
4483 
4484     /* Validate the group handle */
4485     Status = SampValidateDbObject(GroupHandle,
4486                                   SamDbGroupObject,
4487                                   GROUP_ADD_MEMBER,
4488                                   &GroupObject);
4489     if (!NT_SUCCESS(Status))
4490         goto done;
4491 
4492     /* Open the user object in the same domain */
4493     Status = SampOpenUserObject(GroupObject->ParentObject,
4494                                 MemberId,
4495                                 0,
4496                                 &UserObject);
4497     if (!NT_SUCCESS(Status))
4498     {
4499         TRACE("SampOpenUserObject() failed (Status 0x%08lx)\n", Status);
4500         goto done;
4501     }
4502 
4503     /* Add group membership to the user object */
4504     Status = SampAddGroupMembershipToUser(UserObject,
4505                                           GroupObject->RelativeId,
4506                                           Attributes);
4507     if (!NT_SUCCESS(Status))
4508     {
4509         TRACE("SampAddGroupMembershipToUser() failed (Status 0x%08lx)\n", Status);
4510         goto done;
4511     }
4512 
4513     /* Add the member to the group object */
4514     Status = SampAddMemberToGroup(GroupObject,
4515                                   MemberId);
4516     if (!NT_SUCCESS(Status))
4517     {
4518         TRACE("SampAddMemberToGroup() failed (Status 0x%08lx)\n", Status);
4519     }
4520 
4521 done:
4522     if (UserObject)
4523         SampCloseDbObject(UserObject);
4524 
4525     RtlReleaseResource(&SampResource);
4526 
4527     return Status;
4528 }
4529 
4530 
4531 /* Function 23 */
4532 NTSTATUS
4533 NTAPI
4534 SamrDeleteGroup(IN OUT SAMPR_HANDLE *GroupHandle)
4535 {
4536     PSAM_DB_OBJECT GroupObject;
4537     ULONG Length = 0;
4538     NTSTATUS Status;
4539 
4540     TRACE("SamrDeleteGroup(%p)\n", GroupHandle);
4541 
4542     RtlAcquireResourceExclusive(&SampResource,
4543                                 TRUE);
4544 
4545     /* Validate the group handle */
4546     Status = SampValidateDbObject(*GroupHandle,
4547                                   SamDbGroupObject,
4548                                   DELETE,
4549                                   &GroupObject);
4550     if (!NT_SUCCESS(Status))
4551     {
4552         TRACE("SampValidateDbObject() failed (Status 0x%08lx)\n", Status);
4553         goto done;
4554     }
4555 
4556     /* Fail, if the group is built-in */
4557     if (GroupObject->RelativeId < 1000)
4558     {
4559         TRACE("You can not delete a special account!\n");
4560         Status = STATUS_SPECIAL_ACCOUNT;
4561         goto done;
4562     }
4563 
4564     /* Get the length of the Members attribute */
4565     SampGetObjectAttribute(GroupObject,
4566                            L"Members",
4567                            NULL,
4568                            NULL,
4569                            &Length);
4570 
4571     /* Fail, if the group has members */
4572     if (Length != 0)
4573     {
4574         TRACE("There are still members in the group!\n");
4575         Status = STATUS_MEMBER_IN_GROUP;
4576         goto done;
4577     }
4578 
4579     /* FIXME: Remove the group from all aliases */
4580 
4581     /* Delete the group from the database */
4582     Status = SampDeleteAccountDbObject(GroupObject);
4583     if (!NT_SUCCESS(Status))
4584     {
4585         TRACE("SampDeleteAccountDbObject() failed (Status 0x%08lx)\n", Status);
4586         goto done;
4587     }
4588 
4589     /* Invalidate the handle */
4590     *GroupHandle = NULL;
4591 
4592 done:
4593     RtlReleaseResource(&SampResource);
4594 
4595     return Status;
4596 }
4597 
4598 
4599 /* Function 24 */
4600 NTSTATUS
4601 NTAPI
4602 SamrRemoveMemberFromGroup(IN SAMPR_HANDLE GroupHandle,
4603                           IN unsigned long MemberId)
4604 {
4605     PSAM_DB_OBJECT GroupObject;
4606     PSAM_DB_OBJECT UserObject = NULL;
4607     NTSTATUS Status;
4608 
4609     TRACE("SamrRemoveMemberFromGroup(%p %lu)\n",
4610           GroupHandle, MemberId);
4611 
4612     RtlAcquireResourceExclusive(&SampResource,
4613                                 TRUE);
4614 
4615     /* Validate the group handle */
4616     Status = SampValidateDbObject(GroupHandle,
4617                                   SamDbGroupObject,
4618                                   GROUP_REMOVE_MEMBER,
4619                                   &GroupObject);
4620     if (!NT_SUCCESS(Status))
4621         goto done;
4622 
4623     /* Open the user object in the same domain */
4624     Status = SampOpenUserObject(GroupObject->ParentObject,
4625                                 MemberId,
4626                                 0,
4627                                 &UserObject);
4628     if (!NT_SUCCESS(Status))
4629     {
4630         ERR("SampOpenUserObject() failed (Status 0x%08lx)\n", Status);
4631         goto done;
4632     }
4633 
4634     /* Remove group membership from the user object */
4635     Status = SampRemoveGroupMembershipFromUser(UserObject,
4636                                                GroupObject->RelativeId);
4637     if (!NT_SUCCESS(Status))
4638     {
4639         ERR("SampAddGroupMembershipToUser() failed (Status 0x%08lx)\n", Status);
4640         goto done;
4641     }
4642 
4643     /* Remove the member from the group object */
4644     Status = SampRemoveMemberFromGroup(GroupObject,
4645                                        MemberId);
4646     if (!NT_SUCCESS(Status))
4647     {
4648         ERR("SampRemoveMemberFromGroup() failed (Status 0x%08lx)\n", Status);
4649     }
4650 
4651 done:
4652     if (UserObject)
4653         SampCloseDbObject(UserObject);
4654 
4655     RtlReleaseResource(&SampResource);
4656 
4657     return Status;
4658 }
4659 
4660 
4661 /* Function 25 */
4662 NTSTATUS
4663 NTAPI
4664 SamrGetMembersInGroup(IN SAMPR_HANDLE GroupHandle,
4665                       OUT PSAMPR_GET_MEMBERS_BUFFER *Members)
4666 {
4667     PSAMPR_GET_MEMBERS_BUFFER MembersBuffer = NULL;
4668     PSAM_DB_OBJECT GroupObject;
4669     ULONG Length = 0;
4670     ULONG i;
4671     NTSTATUS Status;
4672 
4673     TRACE("SamrGetMembersInGroup(%p %p)\n",
4674           GroupHandle, Members);
4675 
4676     RtlAcquireResourceShared(&SampResource,
4677                              TRUE);
4678 
4679     /* Validate the group handle */
4680     Status = SampValidateDbObject(GroupHandle,
4681                                   SamDbGroupObject,
4682                                   GROUP_LIST_MEMBERS,
4683                                   &GroupObject);
4684     if (!NT_SUCCESS(Status))
4685         goto done;
4686 
4687     MembersBuffer = midl_user_allocate(sizeof(SAMPR_GET_MEMBERS_BUFFER));
4688     if (MembersBuffer == NULL)
4689     {
4690         Status = STATUS_INSUFFICIENT_RESOURCES;
4691         goto done;
4692     }
4693 
4694     SampGetObjectAttribute(GroupObject,
4695                            L"Members",
4696                            NULL,
4697                            NULL,
4698                            &Length);
4699 
4700     if (Length == 0)
4701     {
4702         MembersBuffer->MemberCount = 0;
4703         MembersBuffer->Members = NULL;
4704         MembersBuffer->Attributes = NULL;
4705 
4706         *Members = MembersBuffer;
4707 
4708         Status = STATUS_SUCCESS;
4709         goto done;
4710     }
4711 
4712     MembersBuffer->Members = midl_user_allocate(Length);
4713     if (MembersBuffer->Members == NULL)
4714     {
4715         Status = STATUS_INSUFFICIENT_RESOURCES;
4716         goto done;
4717     }
4718 
4719     MembersBuffer->Attributes = midl_user_allocate(Length);
4720     if (MembersBuffer->Attributes == NULL)
4721     {
4722         Status = STATUS_INSUFFICIENT_RESOURCES;
4723         goto done;
4724     }
4725 
4726     Status = SampGetObjectAttribute(GroupObject,
4727                                     L"Members",
4728                                     NULL,
4729                                     MembersBuffer->Members,
4730                                     &Length);
4731     if (!NT_SUCCESS(Status))
4732     {
4733         TRACE("SampGetObjectAttributes() failed (Status 0x%08lx)\n", Status);
4734         goto done;
4735     }
4736 
4737     MembersBuffer->MemberCount = Length / sizeof(ULONG);
4738 
4739     for (i = 0; i < MembersBuffer->MemberCount; i++)
4740     {
4741         Status = SampGetUserGroupAttributes(GroupObject->ParentObject,
4742                                             MembersBuffer->Members[i],
4743                                             GroupObject->RelativeId,
4744                                             &(MembersBuffer->Attributes[i]));
4745         if (!NT_SUCCESS(Status))
4746         {
4747             TRACE("SampGetUserGroupAttributes() failed (Status 0x%08lx)\n", Status);
4748             goto done;
4749         }
4750     }
4751 
4752     *Members = MembersBuffer;
4753 
4754 done:
4755     if (!NT_SUCCESS(Status))
4756     {
4757         if (MembersBuffer != NULL)
4758         {
4759             if (MembersBuffer->Members != NULL)
4760                 midl_user_free(MembersBuffer->Members);
4761 
4762             if (MembersBuffer->Attributes != NULL)
4763                 midl_user_free(MembersBuffer->Attributes);
4764 
4765             midl_user_free(MembersBuffer);
4766         }
4767     }
4768 
4769     RtlReleaseResource(&SampResource);
4770 
4771     return Status;
4772 }
4773 
4774 
4775 /* Function 26 */
4776 NTSTATUS
4777 NTAPI
4778 SamrSetMemberAttributesOfGroup(IN SAMPR_HANDLE GroupHandle,
4779                                IN unsigned long MemberId,
4780                                IN unsigned long Attributes)
4781 {
4782     PSAM_DB_OBJECT GroupObject;
4783     NTSTATUS Status;
4784 
4785     TRACE("SamrSetMemberAttributesOfGroup(%p %lu %lx)\n",
4786           GroupHandle, MemberId, Attributes);
4787 
4788     RtlAcquireResourceExclusive(&SampResource,
4789                                 TRUE);
4790 
4791     /* Validate the group handle */
4792     Status = SampValidateDbObject(GroupHandle,
4793                                   SamDbGroupObject,
4794                                   GROUP_ADD_MEMBER,
4795                                   &GroupObject);
4796     if (!NT_SUCCESS(Status))
4797     {
4798         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
4799         goto done;
4800     }
4801 
4802     Status = SampSetUserGroupAttributes(GroupObject->ParentObject,
4803                                         MemberId,
4804                                         GroupObject->RelativeId,
4805                                         Attributes);
4806     if (!NT_SUCCESS(Status))
4807     {
4808         TRACE("SampSetUserGroupAttributes failed with status 0x%08lx\n", Status);
4809     }
4810 
4811 done:
4812     RtlReleaseResource(&SampResource);
4813 
4814     return Status;
4815 }
4816 
4817 
4818 /* Function 27 */
4819 NTSTATUS
4820 NTAPI
4821 SamrOpenAlias(IN SAMPR_HANDLE DomainHandle,
4822               IN ACCESS_MASK DesiredAccess,
4823               IN ULONG AliasId,
4824               OUT SAMPR_HANDLE *AliasHandle)
4825 {
4826     PSAM_DB_OBJECT DomainObject;
4827     PSAM_DB_OBJECT AliasObject;
4828     WCHAR szRid[9];
4829     NTSTATUS Status;
4830 
4831     TRACE("SamrOpenAlias(%p %lx %lx %p)\n",
4832           DomainHandle, DesiredAccess, AliasId, AliasHandle);
4833 
4834     /* Map generic access rights */
4835     RtlMapGenericMask(&DesiredAccess,
4836                       &AliasMapping);
4837 
4838     RtlAcquireResourceShared(&SampResource,
4839                              TRUE);
4840 
4841     /* Validate the domain handle */
4842     Status = SampValidateDbObject(DomainHandle,
4843                                   SamDbDomainObject,
4844                                   DOMAIN_LOOKUP,
4845                                   &DomainObject);
4846     if (!NT_SUCCESS(Status))
4847     {
4848         TRACE("failed with status 0x%08lx\n", Status);
4849         goto done;
4850     }
4851 
4852     /* Convert the RID into a string (hex) */
4853     swprintf(szRid, L"%08lX", AliasId);
4854 
4855     /* Create the alias object */
4856     Status = SampOpenDbObject(DomainObject,
4857                               L"Aliases",
4858                               szRid,
4859                               AliasId,
4860                               SamDbAliasObject,
4861                               DesiredAccess,
4862                               &AliasObject);
4863     if (!NT_SUCCESS(Status))
4864     {
4865         TRACE("failed with status 0x%08lx\n", Status);
4866         goto done;
4867     }
4868 
4869     *AliasHandle = (SAMPR_HANDLE)AliasObject;
4870 
4871 done:
4872     RtlReleaseResource(&SampResource);
4873 
4874     return Status;
4875 }
4876 
4877 
4878 static NTSTATUS
4879 SampQueryAliasGeneral(PSAM_DB_OBJECT AliasObject,
4880                       PSAMPR_ALIAS_INFO_BUFFER *Buffer)
4881 {
4882     PSAMPR_ALIAS_INFO_BUFFER InfoBuffer = NULL;
4883     HANDLE MembersKeyHandle = NULL;
4884     NTSTATUS Status;
4885 
4886     *Buffer = NULL;
4887 
4888     InfoBuffer = midl_user_allocate(sizeof(SAMPR_ALIAS_INFO_BUFFER));
4889     if (InfoBuffer == NULL)
4890         return STATUS_INSUFFICIENT_RESOURCES;
4891 
4892     Status = SampGetObjectAttributeString(AliasObject,
4893                                           L"Name",
4894                                           &InfoBuffer->General.Name);
4895     if (!NT_SUCCESS(Status))
4896     {
4897         TRACE("Status 0x%08lx\n", Status);
4898         goto done;
4899     }
4900 
4901     Status = SampGetObjectAttributeString(AliasObject,
4902                                           L"Description",
4903                                           &InfoBuffer->General.AdminComment);
4904     if (!NT_SUCCESS(Status))
4905     {
4906         TRACE("Status 0x%08lx\n", Status);
4907         goto done;
4908     }
4909 
4910     /* Open the Members subkey */
4911     Status = SampRegOpenKey(AliasObject->KeyHandle,
4912                             L"Members",
4913                             KEY_READ,
4914                             &MembersKeyHandle);
4915     if (NT_SUCCESS(Status))
4916     {
4917         /* Retrieve the number of members of the alias */
4918         Status = SampRegQueryKeyInfo(MembersKeyHandle,
4919                                      NULL,
4920                                      &InfoBuffer->General.MemberCount);
4921         if (!NT_SUCCESS(Status))
4922         {
4923             TRACE("Status 0x%08lx\n", Status);
4924             goto done;
4925         }
4926     }
4927     else if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
4928     {
4929         InfoBuffer->General.MemberCount = 0;
4930         Status = STATUS_SUCCESS;
4931     }
4932     else
4933     {
4934         TRACE("Status 0x%08lx\n", Status);
4935         goto done;
4936     }
4937 
4938     *Buffer = InfoBuffer;
4939 
4940 done:
4941     SampRegCloseKey(&MembersKeyHandle);
4942 
4943     if (!NT_SUCCESS(Status))
4944     {
4945         if (InfoBuffer != NULL)
4946         {
4947             if (InfoBuffer->General.Name.Buffer != NULL)
4948                 midl_user_free(InfoBuffer->General.Name.Buffer);
4949 
4950             if (InfoBuffer->General.AdminComment.Buffer != NULL)
4951                 midl_user_free(InfoBuffer->General.AdminComment.Buffer);
4952 
4953             midl_user_free(InfoBuffer);
4954         }
4955     }
4956 
4957     return Status;
4958 }
4959 
4960 
4961 static NTSTATUS
4962 SampQueryAliasName(PSAM_DB_OBJECT AliasObject,
4963                    PSAMPR_ALIAS_INFO_BUFFER *Buffer)
4964 {
4965     PSAMPR_ALIAS_INFO_BUFFER InfoBuffer = NULL;
4966     NTSTATUS Status;
4967 
4968     *Buffer = NULL;
4969 
4970     InfoBuffer = midl_user_allocate(sizeof(SAMPR_ALIAS_INFO_BUFFER));
4971     if (InfoBuffer == NULL)
4972         return STATUS_INSUFFICIENT_RESOURCES;
4973 
4974     Status = SampGetObjectAttributeString(AliasObject,
4975                                           L"Name",
4976                                           &InfoBuffer->Name.Name);
4977     if (!NT_SUCCESS(Status))
4978     {
4979         TRACE("Status 0x%08lx\n", Status);
4980         goto done;
4981     }
4982 
4983     *Buffer = InfoBuffer;
4984 
4985 done:
4986     if (!NT_SUCCESS(Status))
4987     {
4988         if (InfoBuffer != NULL)
4989         {
4990             if (InfoBuffer->Name.Name.Buffer != NULL)
4991                 midl_user_free(InfoBuffer->Name.Name.Buffer);
4992 
4993             midl_user_free(InfoBuffer);
4994         }
4995     }
4996 
4997     return Status;
4998 }
4999 
5000 
5001 static NTSTATUS
5002 SampQueryAliasAdminComment(PSAM_DB_OBJECT AliasObject,
5003                            PSAMPR_ALIAS_INFO_BUFFER *Buffer)
5004 {
5005     PSAMPR_ALIAS_INFO_BUFFER InfoBuffer = NULL;
5006     NTSTATUS Status;
5007 
5008     *Buffer = NULL;
5009 
5010     InfoBuffer = midl_user_allocate(sizeof(SAMPR_ALIAS_INFO_BUFFER));
5011     if (InfoBuffer == NULL)
5012         return STATUS_INSUFFICIENT_RESOURCES;
5013 
5014     Status = SampGetObjectAttributeString(AliasObject,
5015                                           L"Description",
5016                                           &InfoBuffer->AdminComment.AdminComment);
5017     if (!NT_SUCCESS(Status))
5018     {
5019         TRACE("Status 0x%08lx\n", Status);
5020         goto done;
5021     }
5022 
5023     *Buffer = InfoBuffer;
5024 
5025 done:
5026     if (!NT_SUCCESS(Status))
5027     {
5028         if (InfoBuffer != NULL)
5029         {
5030             if (InfoBuffer->AdminComment.AdminComment.Buffer != NULL)
5031                 midl_user_free(InfoBuffer->AdminComment.AdminComment.Buffer);
5032 
5033             midl_user_free(InfoBuffer);
5034         }
5035     }
5036 
5037     return Status;
5038 }
5039 
5040 
5041 /* Function 28 */
5042 NTSTATUS
5043 NTAPI
5044 SamrQueryInformationAlias(IN SAMPR_HANDLE AliasHandle,
5045                           IN ALIAS_INFORMATION_CLASS AliasInformationClass,
5046                           OUT PSAMPR_ALIAS_INFO_BUFFER *Buffer)
5047 {
5048     PSAM_DB_OBJECT AliasObject;
5049     NTSTATUS Status;
5050 
5051     TRACE("SamrQueryInformationAlias(%p %lu %p)\n",
5052           AliasHandle, AliasInformationClass, Buffer);
5053 
5054     RtlAcquireResourceShared(&SampResource,
5055                              TRUE);
5056 
5057     /* Validate the alias handle */
5058     Status = SampValidateDbObject(AliasHandle,
5059                                   SamDbAliasObject,
5060                                   ALIAS_READ_INFORMATION,
5061                                   &AliasObject);
5062     if (!NT_SUCCESS(Status))
5063         goto done;
5064 
5065     switch (AliasInformationClass)
5066     {
5067         case AliasGeneralInformation:
5068             Status = SampQueryAliasGeneral(AliasObject,
5069                                            Buffer);
5070             break;
5071 
5072         case AliasNameInformation:
5073             Status = SampQueryAliasName(AliasObject,
5074                                         Buffer);
5075             break;
5076 
5077         case AliasAdminCommentInformation:
5078             Status = SampQueryAliasAdminComment(AliasObject,
5079                                                 Buffer);
5080             break;
5081 
5082         default:
5083             Status = STATUS_INVALID_INFO_CLASS;
5084             break;
5085     }
5086 
5087 done:
5088     RtlReleaseResource(&SampResource);
5089 
5090     return Status;
5091 }
5092 
5093 
5094 static NTSTATUS
5095 SampSetAliasName(PSAM_DB_OBJECT AliasObject,
5096                  PSAMPR_ALIAS_INFO_BUFFER Buffer)
5097 {
5098     UNICODE_STRING OldAliasName = {0, 0, NULL};
5099     UNICODE_STRING NewAliasName;
5100     NTSTATUS Status;
5101 
5102     Status = SampGetObjectAttributeString(AliasObject,
5103                                           L"Name",
5104                                           (PRPC_UNICODE_STRING)&OldAliasName);
5105     if (!NT_SUCCESS(Status))
5106     {
5107         TRACE("SampGetObjectAttributeString failed (Status 0x%08lx)\n", Status);
5108         goto done;
5109     }
5110 
5111     /* Check the new account name */
5112     Status = SampCheckAccountName(&Buffer->Name.Name, 256);
5113     if (!NT_SUCCESS(Status))
5114     {
5115         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
5116         return Status;
5117     }
5118 
5119     NewAliasName.Length = Buffer->Name.Name.Length;
5120     NewAliasName.MaximumLength = Buffer->Name.Name.MaximumLength;
5121     NewAliasName.Buffer = Buffer->Name.Name.Buffer;
5122 
5123     if (!RtlEqualUnicodeString(&OldAliasName, &NewAliasName, TRUE))
5124     {
5125         Status = SampCheckAccountNameInDomain(AliasObject->ParentObject,
5126                                               NewAliasName.Buffer);
5127         if (!NT_SUCCESS(Status))
5128         {
5129             TRACE("Alias name \'%S\' already exists in domain (Status 0x%08lx)\n",
5130                   NewAliasName.Buffer, Status);
5131             goto done;
5132         }
5133     }
5134 
5135     Status = SampSetAccountNameInDomain(AliasObject->ParentObject,
5136                                         L"Aliases",
5137                                         NewAliasName.Buffer,
5138                                         AliasObject->RelativeId);
5139     if (!NT_SUCCESS(Status))
5140     {
5141         TRACE("SampSetAccountNameInDomain failed (Status 0x%08lx)\n", Status);
5142         goto done;
5143     }
5144 
5145     Status = SampRemoveAccountNameFromDomain(AliasObject->ParentObject,
5146                                              L"Aliases",
5147                                              OldAliasName.Buffer);
5148     if (!NT_SUCCESS(Status))
5149     {
5150         TRACE("SampRemoveAccountNameFromDomain failed (Status 0x%08lx)\n", Status);
5151         goto done;
5152     }
5153 
5154     Status = SampSetObjectAttributeString(AliasObject,
5155                                           L"Name",
5156                                           (PRPC_UNICODE_STRING)&NewAliasName);
5157     if (!NT_SUCCESS(Status))
5158     {
5159         TRACE("SampSetObjectAttribute failed (Status 0x%08lx)\n", Status);
5160     }
5161 
5162 done:
5163     if (OldAliasName.Buffer != NULL)
5164         midl_user_free(OldAliasName.Buffer);
5165 
5166     return Status;
5167 }
5168 
5169 
5170 /* Function 29 */
5171 NTSTATUS
5172 NTAPI
5173 SamrSetInformationAlias(IN SAMPR_HANDLE AliasHandle,
5174                         IN ALIAS_INFORMATION_CLASS AliasInformationClass,
5175                         IN PSAMPR_ALIAS_INFO_BUFFER Buffer)
5176 {
5177     PSAM_DB_OBJECT AliasObject;
5178     NTSTATUS Status;
5179 
5180     TRACE("SamrSetInformationAlias(%p %lu %p)\n",
5181           AliasHandle, AliasInformationClass, Buffer);
5182 
5183     RtlAcquireResourceExclusive(&SampResource,
5184                                 TRUE);
5185 
5186     /* Validate the alias handle */
5187     Status = SampValidateDbObject(AliasHandle,
5188                                   SamDbAliasObject,
5189                                   ALIAS_WRITE_ACCOUNT,
5190                                   &AliasObject);
5191     if (!NT_SUCCESS(Status))
5192         goto done;
5193 
5194     switch (AliasInformationClass)
5195     {
5196         case AliasNameInformation:
5197             Status = SampSetAliasName(AliasObject,
5198                                       Buffer);
5199             break;
5200 
5201         case AliasAdminCommentInformation:
5202             Status = SampSetObjectAttributeString(AliasObject,
5203                                                   L"Description",
5204                                                   &Buffer->AdminComment.AdminComment);
5205             break;
5206 
5207         default:
5208             Status = STATUS_INVALID_INFO_CLASS;
5209             break;
5210     }
5211 
5212 done:
5213     RtlReleaseResource(&SampResource);
5214 
5215     return Status;
5216 }
5217 
5218 
5219 /* Function 30 */
5220 NTSTATUS
5221 NTAPI
5222 SamrDeleteAlias(IN OUT SAMPR_HANDLE *AliasHandle)
5223 {
5224     PSAM_DB_OBJECT AliasObject;
5225     NTSTATUS Status;
5226 
5227     TRACE("SamrDeleteAlias(%p)\n", AliasHandle);
5228 
5229     RtlAcquireResourceExclusive(&SampResource,
5230                                 TRUE);
5231 
5232     /* Validate the alias handle */
5233     Status = SampValidateDbObject(*AliasHandle,
5234                                   SamDbAliasObject,
5235                                   DELETE,
5236                                   &AliasObject);
5237     if (!NT_SUCCESS(Status))
5238     {
5239         TRACE("SampValidateDbObject failed (Status 0x%08lx)\n", Status);
5240         goto done;
5241     }
5242 
5243     /* Fail, if the alias is built-in */
5244     if (AliasObject->RelativeId < 1000)
5245     {
5246         TRACE("You can not delete a special account!\n");
5247         Status = STATUS_SPECIAL_ACCOUNT;
5248         goto done;
5249     }
5250 
5251     /* Remove all members from the alias */
5252     Status = SampRemoveAllMembersFromAlias(AliasObject);
5253     if (!NT_SUCCESS(Status))
5254     {
5255         TRACE("SampRemoveAllMembersFromAlias() failed (Status 0x%08lx)\n", Status);
5256         goto done;
5257     }
5258 
5259     /* Delete the alias from the database */
5260     Status = SampDeleteAccountDbObject(AliasObject);
5261     if (!NT_SUCCESS(Status))
5262     {
5263         TRACE("SampDeleteAccountDbObject() failed (Status 0x%08lx)\n", Status);
5264         goto done;
5265     }
5266 
5267     /* Invalidate the handle */
5268     *AliasHandle = NULL;
5269 
5270 done:
5271     RtlReleaseResource(&SampResource);
5272 
5273     return Status;
5274 }
5275 
5276 
5277 /* Function 31 */
5278 NTSTATUS
5279 NTAPI
5280 SamrAddMemberToAlias(IN SAMPR_HANDLE AliasHandle,
5281                      IN PRPC_SID MemberId)
5282 {
5283     PSAM_DB_OBJECT AliasObject;
5284     NTSTATUS Status;
5285 
5286     TRACE("SamrAddMemberToAlias(%p %p)\n",
5287           AliasHandle, MemberId);
5288 
5289     RtlAcquireResourceExclusive(&SampResource,
5290                                 TRUE);
5291 
5292     /* Validate the alias handle */
5293     Status = SampValidateDbObject(AliasHandle,
5294                                   SamDbAliasObject,
5295                                   ALIAS_ADD_MEMBER,
5296                                   &AliasObject);
5297     if (!NT_SUCCESS(Status))
5298     {
5299         TRACE("failed with status 0x%08lx\n", Status);
5300         goto done;
5301     }
5302 
5303     Status = SampAddMemberToAlias(AliasObject,
5304                                   MemberId);
5305     if (!NT_SUCCESS(Status))
5306     {
5307         TRACE("failed with status 0x%08lx\n", Status);
5308     }
5309 
5310 done:
5311     RtlReleaseResource(&SampResource);
5312 
5313     return Status;
5314 }
5315 
5316 
5317 /* Function 32 */
5318 NTSTATUS
5319 NTAPI
5320 SamrRemoveMemberFromAlias(IN SAMPR_HANDLE AliasHandle,
5321                           IN PRPC_SID MemberId)
5322 {
5323     PSAM_DB_OBJECT AliasObject;
5324     NTSTATUS Status;
5325 
5326     TRACE("SamrRemoveMemberFromAlias(%p %p)\n",
5327           AliasHandle, MemberId);
5328 
5329     RtlAcquireResourceExclusive(&SampResource,
5330                                 TRUE);
5331 
5332     /* Validate the alias handle */
5333     Status = SampValidateDbObject(AliasHandle,
5334                                   SamDbAliasObject,
5335                                   ALIAS_REMOVE_MEMBER,
5336                                   &AliasObject);
5337     if (!NT_SUCCESS(Status))
5338     {
5339         TRACE("failed with status 0x%08lx\n", Status);
5340         goto done;
5341     }
5342 
5343     Status = SampRemoveMemberFromAlias(AliasObject,
5344                                        MemberId);
5345     if (!NT_SUCCESS(Status))
5346     {
5347         TRACE("failed with status 0x%08lx\n", Status);
5348     }
5349 
5350 done:
5351     RtlReleaseResource(&SampResource);
5352 
5353     return Status;
5354 }
5355 
5356 
5357 /* Function 33 */
5358 NTSTATUS
5359 NTAPI
5360 SamrGetMembersInAlias(IN SAMPR_HANDLE AliasHandle,
5361                       OUT PSAMPR_PSID_ARRAY_OUT Members)
5362 {
5363     PSAM_DB_OBJECT AliasObject;
5364     PSAMPR_SID_INFORMATION MemberArray = NULL;
5365     ULONG MemberCount = 0;
5366     ULONG Index;
5367     NTSTATUS Status;
5368 
5369     TRACE("SamrGetMembersInAlias(%p %p %p)\n",
5370           AliasHandle, Members);
5371 
5372     RtlAcquireResourceShared(&SampResource,
5373                              TRUE);
5374 
5375     /* Validate the alias handle */
5376     Status = SampValidateDbObject(AliasHandle,
5377                                   SamDbAliasObject,
5378                                   ALIAS_LIST_MEMBERS,
5379                                   &AliasObject);
5380     if (!NT_SUCCESS(Status))
5381     {
5382         ERR("failed with status 0x%08lx\n", Status);
5383         goto done;
5384     }
5385 
5386     Status = SampGetMembersInAlias(AliasObject,
5387                                    &MemberCount,
5388                                    &MemberArray);
5389 
5390     /* Return the number of members and the member array */
5391     if (NT_SUCCESS(Status))
5392     {
5393         Members->Count = MemberCount;
5394         Members->Sids = MemberArray;
5395     }
5396 
5397 done:
5398     /* Clean up the members array and the SID buffers if something failed */
5399     if (!NT_SUCCESS(Status))
5400     {
5401         if (MemberArray != NULL)
5402         {
5403             for (Index = 0; Index < MemberCount; Index++)
5404             {
5405                 if (MemberArray[Index].SidPointer != NULL)
5406                     midl_user_free(MemberArray[Index].SidPointer);
5407             }
5408 
5409             midl_user_free(MemberArray);
5410         }
5411     }
5412 
5413     RtlReleaseResource(&SampResource);
5414 
5415     return Status;
5416 }
5417 
5418 
5419 /* Function 34 */
5420 NTSTATUS
5421 NTAPI
5422 SamrOpenUser(IN SAMPR_HANDLE DomainHandle,
5423              IN ACCESS_MASK DesiredAccess,
5424              IN unsigned long UserId,
5425              OUT SAMPR_HANDLE *UserHandle)
5426 {
5427     PSAM_DB_OBJECT DomainObject;
5428     PSAM_DB_OBJECT UserObject;
5429     WCHAR szRid[9];
5430     NTSTATUS Status;
5431 
5432     TRACE("SamrOpenUser(%p %lx %lx %p)\n",
5433           DomainHandle, DesiredAccess, UserId, UserHandle);
5434 
5435     /* Map generic access rights */
5436     RtlMapGenericMask(&DesiredAccess,
5437                       &UserMapping);
5438 
5439     RtlAcquireResourceShared(&SampResource,
5440                              TRUE);
5441 
5442     /* Validate the domain handle */
5443     Status = SampValidateDbObject(DomainHandle,
5444                                   SamDbDomainObject,
5445                                   DOMAIN_LOOKUP,
5446                                   &DomainObject);
5447     if (!NT_SUCCESS(Status))
5448     {
5449         TRACE("failed with status 0x%08lx\n", Status);
5450         goto done;
5451     }
5452 
5453     /* Convert the RID into a string (hex) */
5454     swprintf(szRid, L"%08lX", UserId);
5455 
5456     /* Create the user object */
5457     Status = SampOpenDbObject(DomainObject,
5458                               L"Users",
5459                               szRid,
5460                               UserId,
5461                               SamDbUserObject,
5462                               DesiredAccess,
5463                               &UserObject);
5464     if (!NT_SUCCESS(Status))
5465     {
5466         TRACE("failed with status 0x%08lx\n", Status);
5467         goto done;
5468     }
5469 
5470     *UserHandle = (SAMPR_HANDLE)UserObject;
5471 
5472 done:
5473     RtlReleaseResource(&SampResource);
5474 
5475     return Status;
5476 }
5477 
5478 
5479 /* Function 35 */
5480 NTSTATUS
5481 NTAPI
5482 SamrDeleteUser(IN OUT SAMPR_HANDLE *UserHandle)
5483 {
5484     PSAM_DB_OBJECT UserObject;
5485     NTSTATUS Status;
5486 
5487     TRACE("SamrDeleteUser(%p)\n", UserHandle);
5488 
5489     RtlAcquireResourceExclusive(&SampResource,
5490                                 TRUE);
5491 
5492     /* Validate the user handle */
5493     Status = SampValidateDbObject(*UserHandle,
5494                                   SamDbUserObject,
5495                                   DELETE,
5496                                   &UserObject);
5497     if (!NT_SUCCESS(Status))
5498     {
5499         TRACE("SampValidateDbObject() failed (Status 0x%08lx)\n", Status);
5500         goto done;
5501     }
5502 
5503     /* Fail, if the user is built-in */
5504     if (UserObject->RelativeId < 1000)
5505     {
5506         TRACE("You can not delete a special account!\n");
5507         Status = STATUS_SPECIAL_ACCOUNT;
5508         goto done;
5509     }
5510 
5511     /* Remove the user from all groups */
5512     Status = SampRemoveUserFromAllGroups(UserObject);
5513     if (!NT_SUCCESS(Status))
5514     {
5515         TRACE("SampRemoveUserFromAllGroups() failed (Status 0x%08lx)\n", Status);
5516         goto done;
5517     }
5518 
5519     /* Remove the user from all aliases */
5520     Status = SampRemoveUserFromAllAliases(UserObject);
5521     if (!NT_SUCCESS(Status))
5522     {
5523         TRACE("SampRemoveUserFromAllAliases() failed (Status 0x%08lx)\n", Status);
5524         goto done;
5525     }
5526 
5527     /* Delete the user from the database */
5528     Status = SampDeleteAccountDbObject(UserObject);
5529     if (!NT_SUCCESS(Status))
5530     {
5531         TRACE("SampDeleteAccountDbObject() failed (Status 0x%08lx)\n", Status);
5532         goto done;
5533     }
5534 
5535     /* Invalidate the handle */
5536     *UserHandle = NULL;
5537 
5538 done:
5539     RtlReleaseResource(&SampResource);
5540 
5541     return Status;
5542 }
5543 
5544 
5545 static
5546 NTSTATUS
5547 SampQueryUserGeneral(PSAM_DB_OBJECT UserObject,
5548                      PSAMPR_USER_INFO_BUFFER *Buffer)
5549 {
5550     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
5551     SAM_USER_FIXED_DATA FixedData;
5552     ULONG Length = 0;
5553     NTSTATUS Status;
5554 
5555     *Buffer = NULL;
5556 
5557     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
5558     if (InfoBuffer == NULL)
5559         return STATUS_INSUFFICIENT_RESOURCES;
5560 
5561     Length = sizeof(SAM_USER_FIXED_DATA);
5562     Status = SampGetObjectAttribute(UserObject,
5563                                     L"F",
5564                                     NULL,
5565                                     (PVOID)&FixedData,
5566                                     &Length);
5567     if (!NT_SUCCESS(Status))
5568         goto done;
5569 
5570     InfoBuffer->General.PrimaryGroupId = FixedData.PrimaryGroupId;
5571 
5572     /* Get the Name string */
5573     Status = SampGetObjectAttributeString(UserObject,
5574                                           L"Name",
5575                                           &InfoBuffer->General.UserName);
5576     if (!NT_SUCCESS(Status))
5577     {
5578         TRACE("Status 0x%08lx\n", Status);
5579         goto done;
5580     }
5581 
5582     /* Get the FullName string */
5583     Status = SampGetObjectAttributeString(UserObject,
5584                                           L"FullName",
5585                                           &InfoBuffer->General.FullName);
5586     if (!NT_SUCCESS(Status))
5587     {
5588         TRACE("Status 0x%08lx\n", Status);
5589         goto done;
5590     }
5591 
5592     /* Get the AdminComment string */
5593     Status = SampGetObjectAttributeString(UserObject,
5594                                           L"AdminComment",
5595                                           &InfoBuffer->General.AdminComment);
5596     if (!NT_SUCCESS(Status))
5597     {
5598         TRACE("Status 0x%08lx\n", Status);
5599         goto done;
5600     }
5601 
5602     /* Get the UserComment string */
5603     Status = SampGetObjectAttributeString(UserObject,
5604                                           L"UserComment",
5605                                           &InfoBuffer->General.UserComment);
5606     if (!NT_SUCCESS(Status))
5607     {
5608         TRACE("Status 0x%08lx\n", Status);
5609         goto done;
5610     }
5611 
5612     *Buffer = InfoBuffer;
5613 
5614 done:
5615     if (!NT_SUCCESS(Status))
5616     {
5617         if (InfoBuffer != NULL)
5618         {
5619             if (InfoBuffer->General.UserName.Buffer != NULL)
5620                 midl_user_free(InfoBuffer->General.UserName.Buffer);
5621 
5622             if (InfoBuffer->General.FullName.Buffer != NULL)
5623                 midl_user_free(InfoBuffer->General.FullName.Buffer);
5624 
5625             if (InfoBuffer->General.AdminComment.Buffer != NULL)
5626                 midl_user_free(InfoBuffer->General.AdminComment.Buffer);
5627 
5628             if (InfoBuffer->General.UserComment.Buffer != NULL)
5629                 midl_user_free(InfoBuffer->General.UserComment.Buffer);
5630 
5631             midl_user_free(InfoBuffer);
5632         }
5633     }
5634 
5635     return Status;
5636 }
5637 
5638 
5639 static
5640 NTSTATUS
5641 SampQueryUserPreferences(PSAM_DB_OBJECT UserObject,
5642                          PSAMPR_USER_INFO_BUFFER *Buffer)
5643 {
5644     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
5645     SAM_USER_FIXED_DATA FixedData;
5646     ULONG Length = 0;
5647     NTSTATUS Status;
5648 
5649     *Buffer = NULL;
5650 
5651     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
5652     if (InfoBuffer == NULL)
5653         return STATUS_INSUFFICIENT_RESOURCES;
5654 
5655     Length = sizeof(SAM_USER_FIXED_DATA);
5656     Status = SampGetObjectAttribute(UserObject,
5657                                     L"F",
5658                                     NULL,
5659                                     (PVOID)&FixedData,
5660                                     &Length);
5661     if (!NT_SUCCESS(Status))
5662         goto done;
5663 
5664     InfoBuffer->Preferences.CountryCode = FixedData.CountryCode;
5665     InfoBuffer->Preferences.CodePage = FixedData.CodePage;
5666 
5667     /* Get the UserComment string */
5668     Status = SampGetObjectAttributeString(UserObject,
5669                                           L"UserComment",
5670                                           &InfoBuffer->Preferences.UserComment);
5671     if (!NT_SUCCESS(Status))
5672     {
5673         TRACE("Status 0x%08lx\n", Status);
5674         goto done;
5675     }
5676 
5677     *Buffer = InfoBuffer;
5678 
5679 done:
5680     if (!NT_SUCCESS(Status))
5681     {
5682         if (InfoBuffer != NULL)
5683         {
5684             if (InfoBuffer->Preferences.UserComment.Buffer != NULL)
5685                 midl_user_free(InfoBuffer->Preferences.UserComment.Buffer);
5686 
5687             midl_user_free(InfoBuffer);
5688         }
5689     }
5690 
5691     return Status;
5692 }
5693 
5694 
5695 static
5696 NTSTATUS
5697 SampQueryUserLogon(PSAM_DB_OBJECT UserObject,
5698                    PSAMPR_USER_INFO_BUFFER *Buffer)
5699 {
5700     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
5701     SAM_DOMAIN_FIXED_DATA DomainFixedData;
5702     SAM_USER_FIXED_DATA FixedData;
5703     LARGE_INTEGER PasswordCanChange;
5704     LARGE_INTEGER PasswordMustChange;
5705     ULONG Length = 0;
5706     NTSTATUS Status;
5707 
5708     *Buffer = NULL;
5709 
5710     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
5711     if (InfoBuffer == NULL)
5712         return STATUS_INSUFFICIENT_RESOURCES;
5713 
5714     /* Get the fixed size domain data */
5715     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
5716     Status = SampGetObjectAttribute(UserObject->ParentObject,
5717                                     L"F",
5718                                     NULL,
5719                                     (PVOID)&DomainFixedData,
5720                                     &Length);
5721     if (!NT_SUCCESS(Status))
5722         goto done;
5723 
5724     /* Get the fixed size user data */
5725     Length = sizeof(SAM_USER_FIXED_DATA);
5726     Status = SampGetObjectAttribute(UserObject,
5727                                     L"F",
5728                                     NULL,
5729                                     (PVOID)&FixedData,
5730                                     &Length);
5731     if (!NT_SUCCESS(Status))
5732         goto done;
5733 
5734     InfoBuffer->Logon.UserId = FixedData.UserId;
5735     InfoBuffer->Logon.PrimaryGroupId = FixedData.PrimaryGroupId;
5736     InfoBuffer->Logon.LastLogon.LowPart = FixedData.LastLogon.LowPart;
5737     InfoBuffer->Logon.LastLogon.HighPart = FixedData.LastLogon.HighPart;
5738     InfoBuffer->Logon.LastLogoff.LowPart = FixedData.LastLogoff.LowPart;
5739     InfoBuffer->Logon.LastLogoff.HighPart = FixedData.LastLogoff.HighPart;
5740     InfoBuffer->Logon.PasswordLastSet.LowPart = FixedData.PasswordLastSet.LowPart;
5741     InfoBuffer->Logon.PasswordLastSet.HighPart = FixedData.PasswordLastSet.HighPart;
5742     InfoBuffer->Logon.BadPasswordCount = FixedData.BadPasswordCount;
5743     InfoBuffer->Logon.LogonCount = FixedData.LogonCount;
5744     InfoBuffer->Logon.UserAccountControl = FixedData.UserAccountControl;
5745 
5746     PasswordCanChange = SampAddRelativeTimeToTime(FixedData.PasswordLastSet,
5747                                                   DomainFixedData.MinPasswordAge);
5748     InfoBuffer->Logon.PasswordCanChange.LowPart = PasswordCanChange.LowPart;
5749     InfoBuffer->Logon.PasswordCanChange.HighPart = PasswordCanChange.HighPart;
5750 
5751     PasswordMustChange = SampAddRelativeTimeToTime(FixedData.PasswordLastSet,
5752                                                    DomainFixedData.MaxPasswordAge);
5753     InfoBuffer->Logon.PasswordMustChange.LowPart = PasswordMustChange.LowPart;
5754     InfoBuffer->Logon.PasswordMustChange.HighPart = PasswordMustChange.HighPart;
5755 
5756     /* Get the Name string */
5757     Status = SampGetObjectAttributeString(UserObject,
5758                                           L"Name",
5759                                           &InfoBuffer->Logon.UserName);
5760     if (!NT_SUCCESS(Status))
5761     {
5762         TRACE("Status 0x%08lx\n", Status);
5763         goto done;
5764     }
5765 
5766     /* Get the FullName string */
5767     Status = SampGetObjectAttributeString(UserObject,
5768                                           L"FullName",
5769                                           &InfoBuffer->Logon.FullName);
5770     if (!NT_SUCCESS(Status))
5771     {
5772         TRACE("Status 0x%08lx\n", Status);
5773         goto done;
5774     }
5775 
5776     /* Get the HomeDirectory string */
5777     Status = SampGetObjectAttributeString(UserObject,
5778                                           L"HomeDirectory",
5779                                           &InfoBuffer->Logon.HomeDirectory);
5780     if (!NT_SUCCESS(Status))
5781     {
5782         TRACE("Status 0x%08lx\n", Status);
5783         goto done;
5784     }
5785 
5786     /* Get the HomeDirectoryDrive string */
5787     Status = SampGetObjectAttributeString(UserObject,
5788                                           L"HomeDirectoryDrive",
5789                                           &InfoBuffer->Logon.HomeDirectoryDrive);
5790     if (!NT_SUCCESS(Status))
5791     {
5792         TRACE("Status 0x%08lx\n", Status);
5793         goto done;
5794     }
5795 
5796     /* Get the ScriptPath string */
5797     Status = SampGetObjectAttributeString(UserObject,
5798                                           L"ScriptPath",
5799                                           &InfoBuffer->Logon.ScriptPath);
5800     if (!NT_SUCCESS(Status))
5801     {
5802         TRACE("Status 0x%08lx\n", Status);
5803         goto done;
5804     }
5805 
5806     /* Get the ProfilePath string */
5807     Status = SampGetObjectAttributeString(UserObject,
5808                                           L"ProfilePath",
5809                                           &InfoBuffer->Logon.ProfilePath);
5810     if (!NT_SUCCESS(Status))
5811     {
5812         TRACE("Status 0x%08lx\n", Status);
5813         goto done;
5814     }
5815 
5816     /* Get the WorkStations string */
5817     Status = SampGetObjectAttributeString(UserObject,
5818                                           L"WorkStations",
5819                                           &InfoBuffer->Logon.WorkStations);
5820     if (!NT_SUCCESS(Status))
5821     {
5822         TRACE("Status 0x%08lx\n", Status);
5823         goto done;
5824     }
5825 
5826     /* Get the LogonHours attribute */
5827     Status = SampGetLogonHoursAttribute(UserObject,
5828                                        &InfoBuffer->Logon.LogonHours);
5829     if (!NT_SUCCESS(Status))
5830     {
5831         TRACE("Status 0x%08lx\n", Status);
5832         goto done;
5833     }
5834 
5835     *Buffer = InfoBuffer;
5836 
5837 done:
5838     if (!NT_SUCCESS(Status))
5839     {
5840         if (InfoBuffer != NULL)
5841         {
5842             if (InfoBuffer->Logon.UserName.Buffer != NULL)
5843                 midl_user_free(InfoBuffer->Logon.UserName.Buffer);
5844 
5845             if (InfoBuffer->Logon.FullName.Buffer != NULL)
5846                 midl_user_free(InfoBuffer->Logon.FullName.Buffer);
5847 
5848             if (InfoBuffer->Logon.HomeDirectory.Buffer != NULL)
5849                 midl_user_free(InfoBuffer->Logon.HomeDirectory.Buffer);
5850 
5851             if (InfoBuffer->Logon.HomeDirectoryDrive.Buffer != NULL)
5852                 midl_user_free(InfoBuffer->Logon.HomeDirectoryDrive.Buffer);
5853 
5854             if (InfoBuffer->Logon.ScriptPath.Buffer != NULL)
5855                 midl_user_free(InfoBuffer->Logon.ScriptPath.Buffer);
5856 
5857             if (InfoBuffer->Logon.ProfilePath.Buffer != NULL)
5858                 midl_user_free(InfoBuffer->Logon.ProfilePath.Buffer);
5859 
5860             if (InfoBuffer->Logon.WorkStations.Buffer != NULL)
5861                 midl_user_free(InfoBuffer->Logon.WorkStations.Buffer);
5862 
5863             if (InfoBuffer->Logon.LogonHours.LogonHours != NULL)
5864                 midl_user_free(InfoBuffer->Logon.LogonHours.LogonHours);
5865 
5866             midl_user_free(InfoBuffer);
5867         }
5868     }
5869 
5870     return Status;
5871 }
5872 
5873 
5874 static
5875 NTSTATUS
5876 SampQueryUserAccount(PSAM_DB_OBJECT UserObject,
5877                      PSAMPR_USER_INFO_BUFFER *Buffer)
5878 {
5879     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
5880     SAM_USER_FIXED_DATA FixedData;
5881     ULONG Length = 0;
5882     NTSTATUS Status;
5883 
5884     *Buffer = NULL;
5885 
5886     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
5887     if (InfoBuffer == NULL)
5888         return STATUS_INSUFFICIENT_RESOURCES;
5889 
5890     Length = sizeof(SAM_USER_FIXED_DATA);
5891     Status = SampGetObjectAttribute(UserObject,
5892                                     L"F",
5893                                     NULL,
5894                                     (PVOID)&FixedData,
5895                                     &Length);
5896     if (!NT_SUCCESS(Status))
5897         goto done;
5898 
5899     InfoBuffer->Account.UserId = FixedData.UserId;
5900     InfoBuffer->Account.PrimaryGroupId = FixedData.PrimaryGroupId;
5901     InfoBuffer->Account.LastLogon.LowPart = FixedData.LastLogon.LowPart;
5902     InfoBuffer->Account.LastLogon.HighPart = FixedData.LastLogon.HighPart;
5903     InfoBuffer->Account.LastLogoff.LowPart = FixedData.LastLogoff.LowPart;
5904     InfoBuffer->Account.LastLogoff.HighPart = FixedData.LastLogoff.HighPart;
5905     InfoBuffer->Account.PasswordLastSet.LowPart = FixedData.PasswordLastSet.LowPart;
5906     InfoBuffer->Account.PasswordLastSet.HighPart = FixedData.PasswordLastSet.HighPart;
5907     InfoBuffer->Account.AccountExpires.LowPart = FixedData.AccountExpires.LowPart;
5908     InfoBuffer->Account.AccountExpires.HighPart = FixedData.AccountExpires.HighPart;
5909     InfoBuffer->Account.BadPasswordCount = FixedData.BadPasswordCount;
5910     InfoBuffer->Account.LogonCount = FixedData.LogonCount;
5911     InfoBuffer->Account.UserAccountControl = FixedData.UserAccountControl;
5912 
5913     /* Get the Name string */
5914     Status = SampGetObjectAttributeString(UserObject,
5915                                           L"Name",
5916                                           &InfoBuffer->Account.UserName);
5917     if (!NT_SUCCESS(Status))
5918     {
5919         TRACE("Status 0x%08lx\n", Status);
5920         goto done;
5921     }
5922 
5923     /* Get the FullName string */
5924     Status = SampGetObjectAttributeString(UserObject,
5925                                           L"FullName",
5926                                           &InfoBuffer->Account.FullName);
5927     if (!NT_SUCCESS(Status))
5928     {
5929         TRACE("Status 0x%08lx\n", Status);
5930         goto done;
5931     }
5932 
5933     /* Get the HomeDirectory string */
5934     Status = SampGetObjectAttributeString(UserObject,
5935                                           L"HomeDirectory",
5936                                           &InfoBuffer->Account.HomeDirectory);
5937     if (!NT_SUCCESS(Status))
5938     {
5939         TRACE("Status 0x%08lx\n", Status);
5940         goto done;
5941     }
5942 
5943     /* Get the HomeDirectoryDrive string */
5944     Status = SampGetObjectAttributeString(UserObject,
5945                                           L"HomeDirectoryDrive",
5946                                           &InfoBuffer->Account.HomeDirectoryDrive);
5947     if (!NT_SUCCESS(Status))
5948     {
5949         TRACE("Status 0x%08lx\n", Status);
5950         goto done;
5951     }
5952 
5953     /* Get the ScriptPath string */
5954     Status = SampGetObjectAttributeString(UserObject,
5955                                           L"ScriptPath",
5956                                           &InfoBuffer->Account.ScriptPath);
5957     if (!NT_SUCCESS(Status))
5958     {
5959         TRACE("Status 0x%08lx\n", Status);
5960         goto done;
5961     }
5962 
5963     /* Get the ProfilePath string */
5964     Status = SampGetObjectAttributeString(UserObject,
5965                                           L"ProfilePath",
5966                                           &InfoBuffer->Account.ProfilePath);
5967     if (!NT_SUCCESS(Status))
5968     {
5969         TRACE("Status 0x%08lx\n", Status);
5970         goto done;
5971     }
5972 
5973     /* Get the AdminComment string */
5974     Status = SampGetObjectAttributeString(UserObject,
5975                                           L"AdminComment",
5976                                           &InfoBuffer->Account.AdminComment);
5977     if (!NT_SUCCESS(Status))
5978     {
5979         TRACE("Status 0x%08lx\n", Status);
5980         goto done;
5981     }
5982 
5983     /* Get the WorkStations string */
5984     Status = SampGetObjectAttributeString(UserObject,
5985                                           L"WorkStations",
5986                                           &InfoBuffer->Account.WorkStations);
5987     if (!NT_SUCCESS(Status))
5988     {
5989         TRACE("Status 0x%08lx\n", Status);
5990         goto done;
5991     }
5992 
5993     /* Get the LogonHours attribute */
5994     Status = SampGetLogonHoursAttribute(UserObject,
5995                                        &InfoBuffer->Account.LogonHours);
5996     if (!NT_SUCCESS(Status))
5997     {
5998         TRACE("Status 0x%08lx\n", Status);
5999         goto done;
6000     }
6001 
6002     *Buffer = InfoBuffer;
6003 
6004 done:
6005     if (!NT_SUCCESS(Status))
6006     {
6007         if (InfoBuffer != NULL)
6008         {
6009             if (InfoBuffer->Account.UserName.Buffer != NULL)
6010                 midl_user_free(InfoBuffer->Account.UserName.Buffer);
6011 
6012             if (InfoBuffer->Account.FullName.Buffer != NULL)
6013                 midl_user_free(InfoBuffer->Account.FullName.Buffer);
6014 
6015             if (InfoBuffer->Account.HomeDirectory.Buffer != NULL)
6016                 midl_user_free(InfoBuffer->Account.HomeDirectory.Buffer);
6017 
6018             if (InfoBuffer->Account.HomeDirectoryDrive.Buffer != NULL)
6019                 midl_user_free(InfoBuffer->Account.HomeDirectoryDrive.Buffer);
6020 
6021             if (InfoBuffer->Account.ScriptPath.Buffer != NULL)
6022                 midl_user_free(InfoBuffer->Account.ScriptPath.Buffer);
6023 
6024             if (InfoBuffer->Account.ProfilePath.Buffer != NULL)
6025                 midl_user_free(InfoBuffer->Account.ProfilePath.Buffer);
6026 
6027             if (InfoBuffer->Account.AdminComment.Buffer != NULL)
6028                 midl_user_free(InfoBuffer->Account.AdminComment.Buffer);
6029 
6030             if (InfoBuffer->Account.WorkStations.Buffer != NULL)
6031                 midl_user_free(InfoBuffer->Account.WorkStations.Buffer);
6032 
6033             if (InfoBuffer->Account.LogonHours.LogonHours != NULL)
6034                 midl_user_free(InfoBuffer->Account.LogonHours.LogonHours);
6035 
6036             midl_user_free(InfoBuffer);
6037         }
6038     }
6039 
6040     return Status;
6041 }
6042 
6043 
6044 static
6045 NTSTATUS
6046 SampQueryUserLogonHours(PSAM_DB_OBJECT UserObject,
6047                         PSAMPR_USER_INFO_BUFFER *Buffer)
6048 {
6049     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6050     NTSTATUS Status;
6051 
6052     TRACE("(%p %p)\n", UserObject, Buffer);
6053 
6054     *Buffer = NULL;
6055 
6056     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6057     if (InfoBuffer == NULL)
6058     {
6059         TRACE("Failed to allocate InfoBuffer!\n");
6060         return STATUS_INSUFFICIENT_RESOURCES;
6061     }
6062 
6063     Status = SampGetLogonHoursAttribute(UserObject,
6064                                         &InfoBuffer->LogonHours.LogonHours);
6065     if (!NT_SUCCESS(Status))
6066     {
6067         TRACE("SampGetLogonHoursAttribute failed (Status 0x%08lx)\n", Status);
6068         goto done;
6069     }
6070 
6071     *Buffer = InfoBuffer;
6072 
6073 done:
6074     if (!NT_SUCCESS(Status))
6075     {
6076         if (InfoBuffer != NULL)
6077         {
6078             if (InfoBuffer->LogonHours.LogonHours.LogonHours != NULL)
6079                 midl_user_free(InfoBuffer->LogonHours.LogonHours.LogonHours);
6080 
6081             midl_user_free(InfoBuffer);
6082         }
6083     }
6084 
6085     return Status;
6086 }
6087 
6088 
6089 static
6090 NTSTATUS
6091 SampQueryUserName(PSAM_DB_OBJECT UserObject,
6092                   PSAMPR_USER_INFO_BUFFER *Buffer)
6093 {
6094     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6095     NTSTATUS Status;
6096 
6097     *Buffer = NULL;
6098 
6099     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6100     if (InfoBuffer == NULL)
6101         return STATUS_INSUFFICIENT_RESOURCES;
6102 
6103     /* Get the Name string */
6104     Status = SampGetObjectAttributeString(UserObject,
6105                                           L"Name",
6106                                           &InfoBuffer->Name.UserName);
6107     if (!NT_SUCCESS(Status))
6108     {
6109         TRACE("Status 0x%08lx\n", Status);
6110         goto done;
6111     }
6112 
6113     /* Get the FullName string */
6114     Status = SampGetObjectAttributeString(UserObject,
6115                                           L"FullName",
6116                                           &InfoBuffer->Name.FullName);
6117     if (!NT_SUCCESS(Status))
6118     {
6119         TRACE("Status 0x%08lx\n", Status);
6120         goto done;
6121     }
6122 
6123     *Buffer = InfoBuffer;
6124 
6125 done:
6126     if (!NT_SUCCESS(Status))
6127     {
6128         if (InfoBuffer != NULL)
6129         {
6130             if (InfoBuffer->Name.UserName.Buffer != NULL)
6131                 midl_user_free(InfoBuffer->Name.UserName.Buffer);
6132 
6133             if (InfoBuffer->Name.FullName.Buffer != NULL)
6134                 midl_user_free(InfoBuffer->Name.FullName.Buffer);
6135 
6136             midl_user_free(InfoBuffer);
6137         }
6138     }
6139 
6140     return Status;
6141 }
6142 
6143 
6144 static NTSTATUS
6145 SampQueryUserAccountName(PSAM_DB_OBJECT UserObject,
6146                          PSAMPR_USER_INFO_BUFFER *Buffer)
6147 {
6148     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6149     NTSTATUS Status;
6150 
6151     *Buffer = NULL;
6152 
6153     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6154     if (InfoBuffer == NULL)
6155         return STATUS_INSUFFICIENT_RESOURCES;
6156 
6157     /* Get the Name string */
6158     Status = SampGetObjectAttributeString(UserObject,
6159                                           L"Name",
6160                                           &InfoBuffer->AccountName.UserName);
6161     if (!NT_SUCCESS(Status))
6162     {
6163         TRACE("Status 0x%08lx\n", Status);
6164         goto done;
6165     }
6166 
6167     *Buffer = InfoBuffer;
6168 
6169 done:
6170     if (!NT_SUCCESS(Status))
6171     {
6172         if (InfoBuffer != NULL)
6173         {
6174             if (InfoBuffer->AccountName.UserName.Buffer != NULL)
6175                 midl_user_free(InfoBuffer->AccountName.UserName.Buffer);
6176 
6177             midl_user_free(InfoBuffer);
6178         }
6179     }
6180 
6181     return Status;
6182 }
6183 
6184 
6185 static NTSTATUS
6186 SampQueryUserFullName(PSAM_DB_OBJECT UserObject,
6187                       PSAMPR_USER_INFO_BUFFER *Buffer)
6188 {
6189     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6190     NTSTATUS Status;
6191 
6192     *Buffer = NULL;
6193 
6194     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6195     if (InfoBuffer == NULL)
6196         return STATUS_INSUFFICIENT_RESOURCES;
6197 
6198     /* Get the FullName string */
6199     Status = SampGetObjectAttributeString(UserObject,
6200                                           L"FullName",
6201                                           &InfoBuffer->FullName.FullName);
6202     if (!NT_SUCCESS(Status))
6203     {
6204         TRACE("Status 0x%08lx\n", Status);
6205         goto done;
6206     }
6207 
6208     *Buffer = InfoBuffer;
6209 
6210 done:
6211     if (!NT_SUCCESS(Status))
6212     {
6213         if (InfoBuffer != NULL)
6214         {
6215             if (InfoBuffer->FullName.FullName.Buffer != NULL)
6216                 midl_user_free(InfoBuffer->FullName.FullName.Buffer);
6217 
6218             midl_user_free(InfoBuffer);
6219         }
6220     }
6221 
6222     return Status;
6223 }
6224 
6225 
6226 static
6227 NTSTATUS
6228 SampQueryUserPrimaryGroup(PSAM_DB_OBJECT UserObject,
6229                           PSAMPR_USER_INFO_BUFFER *Buffer)
6230 {
6231     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6232     SAM_USER_FIXED_DATA FixedData;
6233     ULONG Length = 0;
6234     NTSTATUS Status;
6235 
6236     *Buffer = NULL;
6237 
6238     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6239     if (InfoBuffer == NULL)
6240         return STATUS_INSUFFICIENT_RESOURCES;
6241 
6242     Length = sizeof(SAM_USER_FIXED_DATA);
6243     Status = SampGetObjectAttribute(UserObject,
6244                                     L"F",
6245                                     NULL,
6246                                     (PVOID)&FixedData,
6247                                     &Length);
6248     if (!NT_SUCCESS(Status))
6249         goto done;
6250 
6251     InfoBuffer->PrimaryGroup.PrimaryGroupId = FixedData.PrimaryGroupId;
6252 
6253     *Buffer = InfoBuffer;
6254 
6255 done:
6256     if (!NT_SUCCESS(Status))
6257     {
6258         if (InfoBuffer != NULL)
6259         {
6260             midl_user_free(InfoBuffer);
6261         }
6262     }
6263 
6264     return Status;
6265 }
6266 
6267 
6268 static NTSTATUS
6269 SampQueryUserHome(PSAM_DB_OBJECT UserObject,
6270                   PSAMPR_USER_INFO_BUFFER *Buffer)
6271 {
6272     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6273     NTSTATUS Status;
6274 
6275     *Buffer = NULL;
6276 
6277     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6278     if (InfoBuffer == NULL)
6279         return STATUS_INSUFFICIENT_RESOURCES;
6280 
6281     /* Get the HomeDirectory string */
6282     Status = SampGetObjectAttributeString(UserObject,
6283                                           L"HomeDirectory",
6284                                           &InfoBuffer->Home.HomeDirectory);
6285     if (!NT_SUCCESS(Status))
6286     {
6287         TRACE("Status 0x%08lx\n", Status);
6288         goto done;
6289     }
6290 
6291     /* Get the HomeDirectoryDrive string */
6292     Status = SampGetObjectAttributeString(UserObject,
6293                                           L"HomeDirectoryDrive",
6294                                           &InfoBuffer->Home.HomeDirectoryDrive);
6295     if (!NT_SUCCESS(Status))
6296     {
6297         TRACE("Status 0x%08lx\n", Status);
6298         goto done;
6299     }
6300 
6301     *Buffer = InfoBuffer;
6302 
6303 done:
6304     if (!NT_SUCCESS(Status))
6305     {
6306         if (InfoBuffer != NULL)
6307         {
6308             if (InfoBuffer->Home.HomeDirectory.Buffer != NULL)
6309                 midl_user_free(InfoBuffer->Home.HomeDirectory.Buffer);
6310 
6311             if (InfoBuffer->Home.HomeDirectoryDrive.Buffer != NULL)
6312                 midl_user_free(InfoBuffer->Home.HomeDirectoryDrive.Buffer);
6313 
6314             midl_user_free(InfoBuffer);
6315         }
6316     }
6317 
6318     return Status;
6319 }
6320 
6321 
6322 static NTSTATUS
6323 SampQueryUserScript(PSAM_DB_OBJECT UserObject,
6324                     PSAMPR_USER_INFO_BUFFER *Buffer)
6325 {
6326     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6327     NTSTATUS Status;
6328 
6329     *Buffer = NULL;
6330 
6331     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6332     if (InfoBuffer == NULL)
6333         return STATUS_INSUFFICIENT_RESOURCES;
6334 
6335     /* Get the ScriptPath string */
6336     Status = SampGetObjectAttributeString(UserObject,
6337                                           L"ScriptPath",
6338                                           &InfoBuffer->Script.ScriptPath);
6339     if (!NT_SUCCESS(Status))
6340     {
6341         TRACE("Status 0x%08lx\n", Status);
6342         goto done;
6343     }
6344 
6345     *Buffer = InfoBuffer;
6346 
6347 done:
6348     if (!NT_SUCCESS(Status))
6349     {
6350         if (InfoBuffer != NULL)
6351         {
6352             if (InfoBuffer->Script.ScriptPath.Buffer != NULL)
6353                 midl_user_free(InfoBuffer->Script.ScriptPath.Buffer);
6354 
6355             midl_user_free(InfoBuffer);
6356         }
6357     }
6358 
6359     return Status;
6360 }
6361 
6362 
6363 static NTSTATUS
6364 SampQueryUserProfile(PSAM_DB_OBJECT UserObject,
6365                      PSAMPR_USER_INFO_BUFFER *Buffer)
6366 {
6367     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6368     NTSTATUS Status;
6369 
6370     *Buffer = NULL;
6371 
6372     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6373     if (InfoBuffer == NULL)
6374         return STATUS_INSUFFICIENT_RESOURCES;
6375 
6376     /* Get the ProfilePath string */
6377     Status = SampGetObjectAttributeString(UserObject,
6378                                           L"ProfilePath",
6379                                           &InfoBuffer->Profile.ProfilePath);
6380     if (!NT_SUCCESS(Status))
6381     {
6382         TRACE("Status 0x%08lx\n", Status);
6383         goto done;
6384     }
6385 
6386     *Buffer = InfoBuffer;
6387 
6388 done:
6389     if (!NT_SUCCESS(Status))
6390     {
6391         if (InfoBuffer != NULL)
6392         {
6393             if (InfoBuffer->Profile.ProfilePath.Buffer != NULL)
6394                 midl_user_free(InfoBuffer->Profile.ProfilePath.Buffer);
6395 
6396             midl_user_free(InfoBuffer);
6397         }
6398     }
6399 
6400     return Status;
6401 }
6402 
6403 
6404 static NTSTATUS
6405 SampQueryUserAdminComment(PSAM_DB_OBJECT UserObject,
6406                           PSAMPR_USER_INFO_BUFFER *Buffer)
6407 {
6408     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6409     NTSTATUS Status;
6410 
6411     *Buffer = NULL;
6412 
6413     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6414     if (InfoBuffer == NULL)
6415         return STATUS_INSUFFICIENT_RESOURCES;
6416 
6417     /* Get the AdminComment string */
6418     Status = SampGetObjectAttributeString(UserObject,
6419                                           L"AdminComment",
6420                                           &InfoBuffer->AdminComment.AdminComment);
6421     if (!NT_SUCCESS(Status))
6422     {
6423         TRACE("Status 0x%08lx\n", Status);
6424         goto done;
6425     }
6426 
6427     *Buffer = InfoBuffer;
6428 
6429 done:
6430     if (!NT_SUCCESS(Status))
6431     {
6432         if (InfoBuffer != NULL)
6433         {
6434             if (InfoBuffer->AdminComment.AdminComment.Buffer != NULL)
6435                 midl_user_free(InfoBuffer->AdminComment.AdminComment.Buffer);
6436 
6437             midl_user_free(InfoBuffer);
6438         }
6439     }
6440 
6441     return Status;
6442 }
6443 
6444 
6445 static NTSTATUS
6446 SampQueryUserWorkStations(PSAM_DB_OBJECT UserObject,
6447                           PSAMPR_USER_INFO_BUFFER *Buffer)
6448 {
6449     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6450     NTSTATUS Status;
6451 
6452     *Buffer = NULL;
6453 
6454     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6455     if (InfoBuffer == NULL)
6456         return STATUS_INSUFFICIENT_RESOURCES;
6457 
6458     /* Get the WorkStations string */
6459     Status = SampGetObjectAttributeString(UserObject,
6460                                           L"WorkStations",
6461                                           &InfoBuffer->WorkStations.WorkStations);
6462     if (!NT_SUCCESS(Status))
6463     {
6464         TRACE("Status 0x%08lx\n", Status);
6465         goto done;
6466     }
6467 
6468     *Buffer = InfoBuffer;
6469 
6470 done:
6471     if (!NT_SUCCESS(Status))
6472     {
6473         if (InfoBuffer != NULL)
6474         {
6475             if (InfoBuffer->WorkStations.WorkStations.Buffer != NULL)
6476                 midl_user_free(InfoBuffer->WorkStations.WorkStations.Buffer);
6477 
6478             midl_user_free(InfoBuffer);
6479         }
6480     }
6481 
6482     return Status;
6483 }
6484 
6485 
6486 static
6487 NTSTATUS
6488 SampQueryUserControl(PSAM_DB_OBJECT UserObject,
6489                      PSAMPR_USER_INFO_BUFFER *Buffer)
6490 {
6491     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6492     SAM_USER_FIXED_DATA FixedData;
6493     ULONG Length = 0;
6494     NTSTATUS Status;
6495 
6496     *Buffer = NULL;
6497 
6498     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6499     if (InfoBuffer == NULL)
6500         return STATUS_INSUFFICIENT_RESOURCES;
6501 
6502     Length = sizeof(SAM_USER_FIXED_DATA);
6503     Status = SampGetObjectAttribute(UserObject,
6504                                     L"F",
6505                                     NULL,
6506                                     (PVOID)&FixedData,
6507                                     &Length);
6508     if (!NT_SUCCESS(Status))
6509         goto done;
6510 
6511     InfoBuffer->Control.UserAccountControl = FixedData.UserAccountControl;
6512 
6513     *Buffer = InfoBuffer;
6514 
6515 done:
6516     if (!NT_SUCCESS(Status))
6517     {
6518         if (InfoBuffer != NULL)
6519         {
6520             midl_user_free(InfoBuffer);
6521         }
6522     }
6523 
6524     return Status;
6525 }
6526 
6527 
6528 static
6529 NTSTATUS
6530 SampQueryUserExpires(PSAM_DB_OBJECT UserObject,
6531                      PSAMPR_USER_INFO_BUFFER *Buffer)
6532 {
6533     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6534     SAM_USER_FIXED_DATA FixedData;
6535     ULONG Length = 0;
6536     NTSTATUS Status;
6537 
6538     *Buffer = NULL;
6539 
6540     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6541     if (InfoBuffer == NULL)
6542         return STATUS_INSUFFICIENT_RESOURCES;
6543 
6544     Length = sizeof(SAM_USER_FIXED_DATA);
6545     Status = SampGetObjectAttribute(UserObject,
6546                                     L"F",
6547                                     NULL,
6548                                     (PVOID)&FixedData,
6549                                     &Length);
6550     if (!NT_SUCCESS(Status))
6551         goto done;
6552 
6553     InfoBuffer->Expires.AccountExpires.LowPart = FixedData.AccountExpires.LowPart;
6554     InfoBuffer->Expires.AccountExpires.HighPart = FixedData.AccountExpires.HighPart;
6555 
6556     *Buffer = InfoBuffer;
6557 
6558 done:
6559     if (!NT_SUCCESS(Status))
6560     {
6561         if (InfoBuffer != NULL)
6562         {
6563             midl_user_free(InfoBuffer);
6564         }
6565     }
6566 
6567     return Status;
6568 }
6569 
6570 
6571 static
6572 NTSTATUS
6573 SampQueryUserInternal1(PSAM_DB_OBJECT UserObject,
6574                        PSAMPR_USER_INFO_BUFFER *Buffer)
6575 {
6576     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6577     ULONG Length = 0;
6578     NTSTATUS Status = STATUS_SUCCESS;
6579 
6580     /* Fail, if the caller is not a trusted caller */
6581     if (UserObject->Trusted == FALSE)
6582         return STATUS_INVALID_INFO_CLASS;
6583 
6584     *Buffer = NULL;
6585 
6586     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6587     if (InfoBuffer == NULL)
6588         return STATUS_INSUFFICIENT_RESOURCES;
6589 
6590     InfoBuffer->Internal1.LmPasswordPresent = FALSE;
6591     InfoBuffer->Internal1.NtPasswordPresent = FALSE;
6592 
6593     /* Get the NT password */
6594     Length = 0;
6595     SampGetObjectAttribute(UserObject,
6596                            L"NTPwd",
6597                            NULL,
6598                            NULL,
6599                            &Length);
6600 
6601     if (Length == sizeof(ENCRYPTED_NT_OWF_PASSWORD))
6602     {
6603         Status = SampGetObjectAttribute(UserObject,
6604                                         L"NTPwd",
6605                                         NULL,
6606                                         (PVOID)&InfoBuffer->Internal1.EncryptedNtOwfPassword,
6607                                         &Length);
6608         if (!NT_SUCCESS(Status))
6609             goto done;
6610 
6611         if (memcmp(&InfoBuffer->Internal1.EncryptedNtOwfPassword,
6612                    &EmptyNtHash,
6613                    sizeof(ENCRYPTED_NT_OWF_PASSWORD)))
6614             InfoBuffer->Internal1.NtPasswordPresent = TRUE;
6615     }
6616 
6617 
6618     /* Get the LM password */
6619     Length = 0;
6620     SampGetObjectAttribute(UserObject,
6621                            L"LMPwd",
6622                            NULL,
6623                            NULL,
6624                            &Length);
6625 
6626     if (Length == sizeof(ENCRYPTED_LM_OWF_PASSWORD))
6627     {
6628         Status = SampGetObjectAttribute(UserObject,
6629                                         L"LMPwd",
6630                                         NULL,
6631                                         (PVOID)&InfoBuffer->Internal1.EncryptedLmOwfPassword,
6632                                         &Length);
6633         if (!NT_SUCCESS(Status))
6634             goto done;
6635 
6636         if (memcmp(&InfoBuffer->Internal1.EncryptedLmOwfPassword,
6637                    &EmptyLmHash,
6638                    sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
6639             InfoBuffer->Internal1.LmPasswordPresent = TRUE;
6640     }
6641 
6642     InfoBuffer->Internal1.PasswordExpired = FALSE;
6643 
6644     *Buffer = InfoBuffer;
6645 
6646 done:
6647     if (!NT_SUCCESS(Status))
6648     {
6649         if (InfoBuffer != NULL)
6650         {
6651             midl_user_free(InfoBuffer);
6652         }
6653     }
6654 
6655     return Status;
6656 }
6657 
6658 
6659 static
6660 NTSTATUS
6661 SampQueryUserInternal2(PSAM_DB_OBJECT UserObject,
6662                        PSAMPR_USER_INFO_BUFFER *Buffer)
6663 {
6664     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6665     SAM_USER_FIXED_DATA FixedData;
6666     ULONG Length = 0;
6667     NTSTATUS Status;
6668 
6669     *Buffer = NULL;
6670 
6671     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6672     if (InfoBuffer == NULL)
6673         return STATUS_INSUFFICIENT_RESOURCES;
6674 
6675     Length = sizeof(SAM_USER_FIXED_DATA);
6676     Status = SampGetObjectAttribute(UserObject,
6677                                     L"F",
6678                                     NULL,
6679                                     (PVOID)&FixedData,
6680                                     &Length);
6681     if (!NT_SUCCESS(Status))
6682         goto done;
6683 
6684     InfoBuffer->Internal2.Flags = 0;
6685     InfoBuffer->Internal2.LastLogon.LowPart = FixedData.LastLogon.LowPart;
6686     InfoBuffer->Internal2.LastLogon.HighPart = FixedData.LastLogon.HighPart;
6687     InfoBuffer->Internal2.LastLogoff.LowPart = FixedData.LastLogoff.LowPart;
6688     InfoBuffer->Internal2.LastLogoff.HighPart = FixedData.LastLogoff.HighPart;
6689     InfoBuffer->Internal2.BadPasswordCount = FixedData.BadPasswordCount;
6690     InfoBuffer->Internal2.LogonCount = FixedData.LogonCount;
6691 
6692     *Buffer = InfoBuffer;
6693 
6694 done:
6695     if (!NT_SUCCESS(Status))
6696     {
6697         if (InfoBuffer != NULL)
6698         {
6699             midl_user_free(InfoBuffer);
6700         }
6701     }
6702 
6703     return Status;
6704 }
6705 
6706 
6707 static NTSTATUS
6708 SampQueryUserParameters(PSAM_DB_OBJECT UserObject,
6709                         PSAMPR_USER_INFO_BUFFER *Buffer)
6710 {
6711     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6712     NTSTATUS Status;
6713 
6714     *Buffer = NULL;
6715 
6716     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6717     if (InfoBuffer == NULL)
6718         return STATUS_INSUFFICIENT_RESOURCES;
6719 
6720     /* Get the Parameters string */
6721     Status = SampGetObjectAttributeString(UserObject,
6722                                           L"Parameters",
6723                                           &InfoBuffer->Parameters.Parameters);
6724     if (!NT_SUCCESS(Status))
6725     {
6726         TRACE("Status 0x%08lx\n", Status);
6727         goto done;
6728     }
6729 
6730     *Buffer = InfoBuffer;
6731 
6732 done:
6733     if (!NT_SUCCESS(Status))
6734     {
6735         if (InfoBuffer != NULL)
6736         {
6737             if (InfoBuffer->Parameters.Parameters.Buffer != NULL)
6738                 midl_user_free(InfoBuffer->Parameters.Parameters.Buffer);
6739 
6740             midl_user_free(InfoBuffer);
6741         }
6742     }
6743 
6744     return Status;
6745 }
6746 
6747 
6748 static NTSTATUS
6749 SampQueryUserAll(PSAM_DB_OBJECT UserObject,
6750                  PSAMPR_USER_INFO_BUFFER *Buffer)
6751 {
6752     PSAMPR_USER_INFO_BUFFER InfoBuffer = NULL;
6753     SAM_DOMAIN_FIXED_DATA DomainFixedData;
6754     SAM_USER_FIXED_DATA FixedData;
6755     LARGE_INTEGER PasswordCanChange;
6756     LARGE_INTEGER PasswordMustChange;
6757     ULONG Length = 0;
6758     NTSTATUS Status;
6759 
6760     *Buffer = NULL;
6761 
6762     InfoBuffer = midl_user_allocate(sizeof(SAMPR_USER_INFO_BUFFER));
6763     if (InfoBuffer == NULL)
6764         return STATUS_INSUFFICIENT_RESOURCES;
6765 
6766     /* Get the fixed size domain data */
6767     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
6768     Status = SampGetObjectAttribute(UserObject->ParentObject,
6769                                     L"F",
6770                                     NULL,
6771                                     (PVOID)&DomainFixedData,
6772                                     &Length);
6773     if (!NT_SUCCESS(Status))
6774         goto done;
6775 
6776     /* Get the fixed size user data */
6777     Length = sizeof(SAM_USER_FIXED_DATA);
6778     Status = SampGetObjectAttribute(UserObject,
6779                                     L"F",
6780                                     NULL,
6781                                     (PVOID)&FixedData,
6782                                     &Length);
6783     if (!NT_SUCCESS(Status))
6784         goto done;
6785 
6786     /* Set the fields to be returned */
6787     if (UserObject->Trusted)
6788     {
6789         InfoBuffer->All.WhichFields = USER_ALL_READ_GENERAL_MASK |
6790                                       USER_ALL_READ_LOGON_MASK |
6791                                       USER_ALL_READ_ACCOUNT_MASK |
6792                                       USER_ALL_READ_PREFERENCES_MASK |
6793                                       USER_ALL_READ_TRUSTED_MASK;
6794     }
6795     else
6796     {
6797         InfoBuffer->All.WhichFields = 0;
6798 
6799         if (UserObject->Access & USER_READ_GENERAL)
6800             InfoBuffer->All.WhichFields |= USER_ALL_READ_GENERAL_MASK;
6801 
6802         if (UserObject->Access & USER_READ_LOGON)
6803             InfoBuffer->All.WhichFields |= USER_ALL_READ_LOGON_MASK;
6804 
6805         if (UserObject->Access & USER_READ_ACCOUNT)
6806             InfoBuffer->All.WhichFields |= USER_ALL_READ_ACCOUNT_MASK;
6807 
6808         if (UserObject->Access & USER_READ_PREFERENCES)
6809             InfoBuffer->All.WhichFields |= USER_ALL_READ_PREFERENCES_MASK;
6810     }
6811 
6812     /* Fail, if no fields are to be returned */
6813     if (InfoBuffer->All.WhichFields == 0)
6814     {
6815         Status = STATUS_ACCESS_DENIED;
6816         goto done;
6817     }
6818 
6819     /* Get the UserName attribute */
6820     if (InfoBuffer->All.WhichFields & USER_ALL_USERNAME)
6821     {
6822         Status = SampGetObjectAttributeString(UserObject,
6823                                               L"Name",
6824                                               &InfoBuffer->All.UserName);
6825         if (!NT_SUCCESS(Status))
6826         {
6827             TRACE("Status 0x%08lx\n", Status);
6828             goto done;
6829         }
6830     }
6831 
6832     /* Get the FullName attribute */
6833     if (InfoBuffer->All.WhichFields & USER_ALL_FULLNAME)
6834     {
6835         Status = SampGetObjectAttributeString(UserObject,
6836                                               L"FullName",
6837                                               &InfoBuffer->All.FullName);
6838         if (!NT_SUCCESS(Status))
6839         {
6840             TRACE("Status 0x%08lx\n", Status);
6841             goto done;
6842         }
6843     }
6844 
6845     /* Get the UserId attribute */
6846     if (InfoBuffer->All.WhichFields & USER_ALL_USERID)
6847     {
6848         InfoBuffer->All.UserId = FixedData.UserId;
6849     }
6850 
6851     /* Get the PrimaryGroupId attribute */
6852     if (InfoBuffer->All.WhichFields & USER_ALL_PRIMARYGROUPID)
6853     {
6854         InfoBuffer->All.PrimaryGroupId = FixedData.PrimaryGroupId;
6855     }
6856 
6857     /* Get the AdminComment attribute */
6858     if (InfoBuffer->All.WhichFields & USER_ALL_ADMINCOMMENT)
6859     {
6860         Status = SampGetObjectAttributeString(UserObject,
6861                                               L"AdminComment",
6862                                               &InfoBuffer->All.AdminComment);
6863         if (!NT_SUCCESS(Status))
6864         {
6865             TRACE("Status 0x%08lx\n", Status);
6866             goto done;
6867         }
6868     }
6869 
6870     /* Get the UserComment attribute */
6871     if (InfoBuffer->All.WhichFields & USER_ALL_USERCOMMENT)
6872     {
6873         Status = SampGetObjectAttributeString(UserObject,
6874                                               L"UserComment",
6875                                               &InfoBuffer->All.UserComment);
6876         if (!NT_SUCCESS(Status))
6877         {
6878             TRACE("Status 0x%08lx\n", Status);
6879             goto done;
6880         }
6881     }
6882 
6883     /* Get the HomeDirectory attribute */
6884     if (InfoBuffer->All.WhichFields & USER_ALL_HOMEDIRECTORY)
6885     {
6886         Status = SampGetObjectAttributeString(UserObject,
6887                                               L"HomeDirectory",
6888                                               &InfoBuffer->All.HomeDirectory);
6889         if (!NT_SUCCESS(Status))
6890         {
6891             TRACE("Status 0x%08lx\n", Status);
6892             goto done;
6893         }
6894     }
6895 
6896     /* Get the HomeDirectoryDrive attribute */
6897     if (InfoBuffer->All.WhichFields & USER_ALL_HOMEDIRECTORYDRIVE)
6898     {
6899         Status = SampGetObjectAttributeString(UserObject,
6900                                               L"HomeDirectoryDrive",
6901                                               &InfoBuffer->Home.HomeDirectoryDrive);
6902         if (!NT_SUCCESS(Status))
6903         {
6904             TRACE("Status 0x%08lx\n", Status);
6905             goto done;
6906         }
6907     }
6908 
6909     /* Get the ScriptPath attribute */
6910     if (InfoBuffer->All.WhichFields & USER_ALL_SCRIPTPATH)
6911     {
6912         Status = SampGetObjectAttributeString(UserObject,
6913                                               L"ScriptPath",
6914                                               &InfoBuffer->All.ScriptPath);
6915         if (!NT_SUCCESS(Status))
6916         {
6917             TRACE("Status 0x%08lx\n", Status);
6918             goto done;
6919         }
6920     }
6921 
6922     /* Get the ProfilePath attribute */
6923     if (InfoBuffer->All.WhichFields & USER_ALL_PROFILEPATH)
6924     {
6925         Status = SampGetObjectAttributeString(UserObject,
6926                                               L"ProfilePath",
6927                                               &InfoBuffer->All.ProfilePath);
6928         if (!NT_SUCCESS(Status))
6929         {
6930             TRACE("Status 0x%08lx\n", Status);
6931             goto done;
6932         }
6933     }
6934 
6935     /* Get the WorkStations attribute */
6936     if (InfoBuffer->All.WhichFields & USER_ALL_WORKSTATIONS)
6937     {
6938         Status = SampGetObjectAttributeString(UserObject,
6939                                               L"WorkStations",
6940                                               &InfoBuffer->All.WorkStations);
6941         if (!NT_SUCCESS(Status))
6942         {
6943             TRACE("Status 0x%08lx\n", Status);
6944             goto done;
6945         }
6946     }
6947 
6948     /* Get the LastLogon attribute */
6949     if (InfoBuffer->All.WhichFields & USER_ALL_LASTLOGON)
6950     {
6951         InfoBuffer->All.LastLogon.LowPart = FixedData.LastLogon.LowPart;
6952         InfoBuffer->All.LastLogon.HighPart = FixedData.LastLogon.HighPart;
6953     }
6954 
6955     /* Get the LastLogoff attribute */
6956     if (InfoBuffer->All.WhichFields & USER_ALL_LASTLOGOFF)
6957     {
6958         InfoBuffer->All.LastLogoff.LowPart = FixedData.LastLogoff.LowPart;
6959         InfoBuffer->All.LastLogoff.HighPart = FixedData.LastLogoff.HighPart;
6960     }
6961 
6962     /* Get the LogonHours attribute */
6963     if (InfoBuffer->All.WhichFields & USER_ALL_LOGONHOURS)
6964     {
6965         Status = SampGetLogonHoursAttribute(UserObject,
6966                                             &InfoBuffer->All.LogonHours);
6967         if (!NT_SUCCESS(Status))
6968         {
6969             TRACE("Status 0x%08lx\n", Status);
6970             goto done;
6971         }
6972     }
6973 
6974     /* Get the BadPasswordCount attribute */
6975     if (InfoBuffer->All.WhichFields & USER_ALL_BADPASSWORDCOUNT)
6976     {
6977         InfoBuffer->All.BadPasswordCount = FixedData.BadPasswordCount;
6978     }
6979 
6980     /* Get the LogonCount attribute */
6981     if (InfoBuffer->All.WhichFields & USER_ALL_LOGONCOUNT)
6982     {
6983         InfoBuffer->All.LogonCount = FixedData.LogonCount;
6984     }
6985 
6986     /* Get the PasswordCanChange attribute */
6987     if (InfoBuffer->All.WhichFields & USER_ALL_PASSWORDCANCHANGE)
6988     {
6989         PasswordCanChange = SampAddRelativeTimeToTime(FixedData.PasswordLastSet,
6990                                                       DomainFixedData.MinPasswordAge);
6991         InfoBuffer->All.PasswordCanChange.LowPart = PasswordCanChange.LowPart;
6992         InfoBuffer->All.PasswordCanChange.HighPart = PasswordCanChange.HighPart;
6993     }
6994 
6995     /* Get the PasswordMustChange attribute */
6996     if (InfoBuffer->All.WhichFields & USER_ALL_PASSWORDMUSTCHANGE)
6997     {
6998         PasswordMustChange = SampAddRelativeTimeToTime(FixedData.PasswordLastSet,
6999                                                        DomainFixedData.MaxPasswordAge);
7000         InfoBuffer->All.PasswordMustChange.LowPart = PasswordMustChange.LowPart;
7001         InfoBuffer->All.PasswordMustChange.HighPart = PasswordMustChange.HighPart;
7002     }
7003 
7004     /* Get the PasswordLastSet attribute */
7005     if (InfoBuffer->All.WhichFields & USER_ALL_PASSWORDLASTSET)
7006     {
7007         InfoBuffer->All.PasswordLastSet.LowPart = FixedData.PasswordLastSet.LowPart;
7008         InfoBuffer->All.PasswordLastSet.HighPart = FixedData.PasswordLastSet.HighPart;
7009     }
7010 
7011     /* Get the AccountExpires attribute */
7012     if (InfoBuffer->All.WhichFields & USER_ALL_ACCOUNTEXPIRES)
7013     {
7014         InfoBuffer->All.AccountExpires.LowPart = FixedData.AccountExpires.LowPart;
7015         InfoBuffer->All.AccountExpires.HighPart = FixedData.AccountExpires.HighPart;
7016     }
7017 
7018     /* Get the UserAccountControl attribute */
7019     if (InfoBuffer->All.WhichFields & USER_ALL_USERACCOUNTCONTROL)
7020     {
7021         InfoBuffer->All.UserAccountControl = FixedData.UserAccountControl;
7022     }
7023 
7024     /* Get the Parameters attribute */
7025     if (InfoBuffer->All.WhichFields & USER_ALL_PARAMETERS)
7026     {
7027         Status = SampGetObjectAttributeString(UserObject,
7028                                               L"Parameters",
7029                                               &InfoBuffer->All.Parameters);
7030         if (!NT_SUCCESS(Status))
7031         {
7032             TRACE("Status 0x%08lx\n", Status);
7033             goto done;
7034         }
7035     }
7036 
7037     /* Get the CountryCode attribute */
7038     if (InfoBuffer->All.WhichFields & USER_ALL_COUNTRYCODE)
7039     {
7040         InfoBuffer->All.CountryCode = FixedData.CountryCode;
7041     }
7042 
7043     /* Get the CodePage attribute */
7044     if (InfoBuffer->All.WhichFields & USER_ALL_CODEPAGE)
7045     {
7046         InfoBuffer->All.CodePage = FixedData.CodePage;
7047     }
7048 
7049     /* Get the LmPassword and NtPassword attributes */
7050     if (InfoBuffer->All.WhichFields & (USER_ALL_NTPASSWORDPRESENT | USER_ALL_LMPASSWORDPRESENT))
7051     {
7052         InfoBuffer->All.LmPasswordPresent = FALSE;
7053         InfoBuffer->All.NtPasswordPresent = FALSE;
7054 
7055         /* Get the NT password */
7056         Length = 0;
7057         SampGetObjectAttribute(UserObject,
7058                                L"NTPwd",
7059                                NULL,
7060                                NULL,
7061                                &Length);
7062 
7063         if (Length == sizeof(ENCRYPTED_NT_OWF_PASSWORD))
7064         {
7065             InfoBuffer->All.NtOwfPassword.Buffer = midl_user_allocate(sizeof(ENCRYPTED_NT_OWF_PASSWORD));
7066             if (InfoBuffer->All.NtOwfPassword.Buffer == NULL)
7067             {
7068                 Status = STATUS_INSUFFICIENT_RESOURCES;
7069                 goto done;
7070             }
7071 
7072             InfoBuffer->All.NtOwfPassword.Length = sizeof(ENCRYPTED_NT_OWF_PASSWORD);
7073             InfoBuffer->All.NtOwfPassword.MaximumLength = sizeof(ENCRYPTED_NT_OWF_PASSWORD);
7074 
7075             Status = SampGetObjectAttribute(UserObject,
7076                                             L"NTPwd",
7077                                             NULL,
7078                                             (PVOID)InfoBuffer->All.NtOwfPassword.Buffer,
7079                                             &Length);
7080             if (!NT_SUCCESS(Status))
7081                 goto done;
7082 
7083             if (memcmp(InfoBuffer->All.NtOwfPassword.Buffer,
7084                        &EmptyNtHash,
7085                        sizeof(ENCRYPTED_NT_OWF_PASSWORD)))
7086                 InfoBuffer->All.NtPasswordPresent = TRUE;
7087         }
7088 
7089         /* Get the LM password */
7090         Length = 0;
7091         SampGetObjectAttribute(UserObject,
7092                                L"LMPwd",
7093                                NULL,
7094                                NULL,
7095                                &Length);
7096 
7097         if (Length == sizeof(ENCRYPTED_LM_OWF_PASSWORD))
7098         {
7099             InfoBuffer->All.LmOwfPassword.Buffer = midl_user_allocate(sizeof(ENCRYPTED_LM_OWF_PASSWORD));
7100             if (InfoBuffer->All.LmOwfPassword.Buffer == NULL)
7101             {
7102                 Status = STATUS_INSUFFICIENT_RESOURCES;
7103                 goto done;
7104             }
7105 
7106             InfoBuffer->All.LmOwfPassword.Length = sizeof(ENCRYPTED_LM_OWF_PASSWORD);
7107             InfoBuffer->All.LmOwfPassword.MaximumLength = sizeof(ENCRYPTED_LM_OWF_PASSWORD);
7108 
7109             Status = SampGetObjectAttribute(UserObject,
7110                                             L"LMPwd",
7111                                             NULL,
7112                                             (PVOID)InfoBuffer->All.LmOwfPassword.Buffer,
7113                                             &Length);
7114             if (!NT_SUCCESS(Status))
7115                 goto done;
7116 
7117             if (memcmp(InfoBuffer->All.LmOwfPassword.Buffer,
7118                        &EmptyLmHash,
7119                        sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
7120                 InfoBuffer->All.LmPasswordPresent = TRUE;
7121         }
7122     }
7123 
7124     if (InfoBuffer->All.WhichFields & USER_ALL_PRIVATEDATA)
7125     {
7126         Status = SampGetObjectAttributeString(UserObject,
7127                                               L"PrivateData",
7128                                               &InfoBuffer->All.PrivateData);
7129         if (!NT_SUCCESS(Status))
7130         {
7131             TRACE("Status 0x%08lx\n", Status);
7132             goto done;
7133         }
7134     }
7135 
7136     if (InfoBuffer->All.WhichFields & USER_ALL_PASSWORDEXPIRED)
7137     {
7138         /* FIXME */
7139     }
7140 
7141     if (InfoBuffer->All.WhichFields & USER_ALL_SECURITYDESCRIPTOR)
7142     {
7143         Length = 0;
7144         SampGetObjectAttribute(UserObject,
7145                                L"SecDesc",
7146                                NULL,
7147                                NULL,
7148                                &Length);
7149 
7150         if (Length > 0)
7151         {
7152             InfoBuffer->All.SecurityDescriptor.SecurityDescriptor = midl_user_allocate(Length);
7153             if (InfoBuffer->All.SecurityDescriptor.SecurityDescriptor == NULL)
7154             {
7155                 Status = STATUS_INSUFFICIENT_RESOURCES;
7156                 goto done;
7157             }
7158 
7159             InfoBuffer->All.SecurityDescriptor.Length = Length;
7160 
7161             Status = SampGetObjectAttribute(UserObject,
7162                                             L"SecDesc",
7163                                             NULL,
7164                                             (PVOID)InfoBuffer->All.SecurityDescriptor.SecurityDescriptor,
7165                                             &Length);
7166             if (!NT_SUCCESS(Status))
7167                 goto done;
7168         }
7169     }
7170 
7171     *Buffer = InfoBuffer;
7172 
7173 done:
7174     if (!NT_SUCCESS(Status))
7175     {
7176         if (InfoBuffer != NULL)
7177         {
7178             if (InfoBuffer->All.UserName.Buffer != NULL)
7179                 midl_user_free(InfoBuffer->All.UserName.Buffer);
7180 
7181             if (InfoBuffer->All.FullName.Buffer != NULL)
7182                 midl_user_free(InfoBuffer->All.FullName.Buffer);
7183 
7184             if (InfoBuffer->All.AdminComment.Buffer != NULL)
7185                 midl_user_free(InfoBuffer->All.AdminComment.Buffer);
7186 
7187             if (InfoBuffer->All.UserComment.Buffer != NULL)
7188                 midl_user_free(InfoBuffer->All.UserComment.Buffer);
7189 
7190             if (InfoBuffer->All.HomeDirectory.Buffer != NULL)
7191                 midl_user_free(InfoBuffer->All.HomeDirectory.Buffer);
7192 
7193             if (InfoBuffer->All.HomeDirectoryDrive.Buffer != NULL)
7194                 midl_user_free(InfoBuffer->All.HomeDirectoryDrive.Buffer);
7195 
7196             if (InfoBuffer->All.ScriptPath.Buffer != NULL)
7197                 midl_user_free(InfoBuffer->All.ScriptPath.Buffer);
7198 
7199             if (InfoBuffer->All.ProfilePath.Buffer != NULL)
7200                 midl_user_free(InfoBuffer->All.ProfilePath.Buffer);
7201 
7202             if (InfoBuffer->All.WorkStations.Buffer != NULL)
7203                 midl_user_free(InfoBuffer->All.WorkStations.Buffer);
7204 
7205             if (InfoBuffer->All.LogonHours.LogonHours != NULL)
7206                 midl_user_free(InfoBuffer->All.LogonHours.LogonHours);
7207 
7208             if (InfoBuffer->All.Parameters.Buffer != NULL)
7209                 midl_user_free(InfoBuffer->All.Parameters.Buffer);
7210 
7211             if (InfoBuffer->All.LmOwfPassword.Buffer != NULL)
7212                 midl_user_free(InfoBuffer->All.LmOwfPassword.Buffer);
7213 
7214             if (InfoBuffer->All.NtOwfPassword.Buffer != NULL)
7215                 midl_user_free(InfoBuffer->All.NtOwfPassword.Buffer);
7216 
7217             if (InfoBuffer->All.PrivateData.Buffer != NULL)
7218                 midl_user_free(InfoBuffer->All.PrivateData.Buffer);
7219 
7220             if (InfoBuffer->All.SecurityDescriptor.SecurityDescriptor != NULL)
7221                 midl_user_free(InfoBuffer->All.SecurityDescriptor.SecurityDescriptor);
7222 
7223             midl_user_free(InfoBuffer);
7224         }
7225     }
7226 
7227     return Status;
7228 }
7229 
7230 
7231 /* Function 36 */
7232 NTSTATUS
7233 NTAPI
7234 SamrQueryInformationUser(IN SAMPR_HANDLE UserHandle,
7235                          IN USER_INFORMATION_CLASS UserInformationClass,
7236                          OUT PSAMPR_USER_INFO_BUFFER *Buffer)
7237 {
7238     TRACE("SamrQueryInformationUser(%p %lu %p)\n",
7239           UserHandle, UserInformationClass, Buffer);
7240 
7241     return SamrQueryInformationUser2(UserHandle,
7242                                      UserInformationClass,
7243                                      Buffer);
7244 }
7245 
7246 
7247 static NTSTATUS
7248 SampSetUserName(PSAM_DB_OBJECT UserObject,
7249                 PRPC_UNICODE_STRING NewUserName)
7250 {
7251     UNICODE_STRING OldUserName = {0, 0, NULL};
7252     NTSTATUS Status;
7253 
7254     /* Check the account name */
7255     Status = SampCheckAccountName(NewUserName, 20);
7256     if (!NT_SUCCESS(Status))
7257     {
7258         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
7259         return Status;
7260     }
7261 
7262     Status = SampGetObjectAttributeString(UserObject,
7263                                           L"Name",
7264                                           (PRPC_UNICODE_STRING)&OldUserName);
7265     if (!NT_SUCCESS(Status))
7266     {
7267         TRACE("SampGetObjectAttributeString failed (Status 0x%08lx)\n", Status);
7268         goto done;
7269     }
7270 
7271     if (!RtlEqualUnicodeString(&OldUserName, (PCUNICODE_STRING)NewUserName, TRUE))
7272     {
7273         Status = SampCheckAccountNameInDomain(UserObject->ParentObject,
7274                                               NewUserName->Buffer);
7275         if (!NT_SUCCESS(Status))
7276         {
7277             TRACE("User name \'%S\' already exists in domain (Status 0x%08lx)\n",
7278                   NewUserName->Buffer, Status);
7279             goto done;
7280         }
7281     }
7282 
7283     Status = SampSetAccountNameInDomain(UserObject->ParentObject,
7284                                         L"Users",
7285                                         NewUserName->Buffer,
7286                                         UserObject->RelativeId);
7287     if (!NT_SUCCESS(Status))
7288     {
7289         TRACE("SampSetAccountNameInDomain failed (Status 0x%08lx)\n", Status);
7290         goto done;
7291     }
7292 
7293     Status = SampRemoveAccountNameFromDomain(UserObject->ParentObject,
7294                                              L"Users",
7295                                              OldUserName.Buffer);
7296     if (!NT_SUCCESS(Status))
7297     {
7298         TRACE("SampRemoveAccountNameFromDomain failed (Status 0x%08lx)\n", Status);
7299         goto done;
7300     }
7301 
7302     Status = SampSetObjectAttributeString(UserObject,
7303                                           L"Name",
7304                                           NewUserName);
7305     if (!NT_SUCCESS(Status))
7306     {
7307         TRACE("SampSetObjectAttribute failed (Status 0x%08lx)\n", Status);
7308     }
7309 
7310 done:
7311     if (OldUserName.Buffer != NULL)
7312         midl_user_free(OldUserName.Buffer);
7313 
7314     return Status;
7315 }
7316 
7317 
7318 static NTSTATUS
7319 SampSetUserGeneral(PSAM_DB_OBJECT UserObject,
7320                    PSAMPR_USER_INFO_BUFFER Buffer)
7321 {
7322     SAM_USER_FIXED_DATA FixedData;
7323     ULONG Length = 0;
7324     NTSTATUS Status;
7325 
7326     Length = sizeof(SAM_USER_FIXED_DATA);
7327     Status = SampGetObjectAttribute(UserObject,
7328                                     L"F",
7329                                     NULL,
7330                                     (PVOID)&FixedData,
7331                                     &Length);
7332     if (!NT_SUCCESS(Status))
7333         goto done;
7334 
7335     FixedData.PrimaryGroupId = Buffer->General.PrimaryGroupId;
7336 
7337     Status = SampSetObjectAttribute(UserObject,
7338                                     L"F",
7339                                     REG_BINARY,
7340                                     &FixedData,
7341                                     Length);
7342     if (!NT_SUCCESS(Status))
7343         goto done;
7344 
7345     Status = SampSetUserName(UserObject,
7346                              &Buffer->General.UserName);
7347     if (!NT_SUCCESS(Status))
7348         goto done;
7349 
7350     Status = SampSetObjectAttributeString(UserObject,
7351                                           L"FullName",
7352                                           &Buffer->General.FullName);
7353     if (!NT_SUCCESS(Status))
7354         goto done;
7355 
7356     Status = SampSetObjectAttributeString(UserObject,
7357                                           L"AdminComment",
7358                                           &Buffer->General.AdminComment);
7359     if (!NT_SUCCESS(Status))
7360         goto done;
7361 
7362     Status = SampSetObjectAttributeString(UserObject,
7363                                           L"UserComment",
7364                                           &Buffer->General.UserComment);
7365 
7366 done:
7367     return Status;
7368 }
7369 
7370 
7371 static NTSTATUS
7372 SampSetUserPreferences(PSAM_DB_OBJECT UserObject,
7373                        PSAMPR_USER_INFO_BUFFER Buffer)
7374 {
7375     SAM_USER_FIXED_DATA FixedData;
7376     ULONG Length = 0;
7377     NTSTATUS Status;
7378 
7379     Length = sizeof(SAM_USER_FIXED_DATA);
7380     Status = SampGetObjectAttribute(UserObject,
7381                                     L"F",
7382                                     NULL,
7383                                     (PVOID)&FixedData,
7384                                     &Length);
7385     if (!NT_SUCCESS(Status))
7386         goto done;
7387 
7388     FixedData.CountryCode = Buffer->Preferences.CountryCode;
7389     FixedData.CodePage = Buffer->Preferences.CodePage;
7390 
7391     Status = SampSetObjectAttribute(UserObject,
7392                                     L"F",
7393                                     REG_BINARY,
7394                                     &FixedData,
7395                                     Length);
7396     if (!NT_SUCCESS(Status))
7397         goto done;
7398 
7399     Status = SampSetObjectAttributeString(UserObject,
7400                                           L"UserComment",
7401                                           &Buffer->Preferences.UserComment);
7402 
7403 done:
7404     return Status;
7405 }
7406 
7407 
7408 static NTSTATUS
7409 SampSetUserPrimaryGroup(PSAM_DB_OBJECT UserObject,
7410                         PSAMPR_USER_INFO_BUFFER Buffer)
7411 {
7412     SAM_USER_FIXED_DATA FixedData;
7413     ULONG Length = 0;
7414     NTSTATUS Status;
7415 
7416     Length = sizeof(SAM_USER_FIXED_DATA);
7417     Status = SampGetObjectAttribute(UserObject,
7418                                     L"F",
7419                                     NULL,
7420                                     (PVOID)&FixedData,
7421                                     &Length);
7422     if (!NT_SUCCESS(Status))
7423         goto done;
7424 
7425     FixedData.PrimaryGroupId = Buffer->PrimaryGroup.PrimaryGroupId;
7426 
7427     Status = SampSetObjectAttribute(UserObject,
7428                                     L"F",
7429                                     REG_BINARY,
7430                                     &FixedData,
7431                                     Length);
7432 
7433 done:
7434     return Status;
7435 }
7436 
7437 
7438 static NTSTATUS
7439 SampSetUserControl(PSAM_DB_OBJECT UserObject,
7440                    PSAMPR_USER_INFO_BUFFER Buffer)
7441 {
7442     SAM_USER_FIXED_DATA FixedData;
7443     ULONG Length = 0;
7444     NTSTATUS Status;
7445 
7446     Length = sizeof(SAM_USER_FIXED_DATA);
7447     Status = SampGetObjectAttribute(UserObject,
7448                                     L"F",
7449                                     NULL,
7450                                     (PVOID)&FixedData,
7451                                     &Length);
7452     if (!NT_SUCCESS(Status))
7453         goto done;
7454 
7455     FixedData.UserAccountControl = Buffer->Control.UserAccountControl;
7456 
7457     Status = SampSetObjectAttribute(UserObject,
7458                                     L"F",
7459                                     REG_BINARY,
7460                                     &FixedData,
7461                                     Length);
7462 
7463 done:
7464     return Status;
7465 }
7466 
7467 
7468 static NTSTATUS
7469 SampSetUserExpires(PSAM_DB_OBJECT UserObject,
7470                    PSAMPR_USER_INFO_BUFFER Buffer)
7471 {
7472     SAM_USER_FIXED_DATA FixedData;
7473     ULONG Length = 0;
7474     NTSTATUS Status;
7475 
7476     Length = sizeof(SAM_USER_FIXED_DATA);
7477     Status = SampGetObjectAttribute(UserObject,
7478                                     L"F",
7479                                     NULL,
7480                                     (PVOID)&FixedData,
7481                                     &Length);
7482     if (!NT_SUCCESS(Status))
7483         goto done;
7484 
7485     FixedData.AccountExpires.LowPart = Buffer->Expires.AccountExpires.LowPart;
7486     FixedData.AccountExpires.HighPart = Buffer->Expires.AccountExpires.HighPart;
7487 
7488     Status = SampSetObjectAttribute(UserObject,
7489                                     L"F",
7490                                     REG_BINARY,
7491                                     &FixedData,
7492                                     Length);
7493 
7494 done:
7495     return Status;
7496 }
7497 
7498 
7499 static NTSTATUS
7500 SampSetUserInternal1(PSAM_DB_OBJECT UserObject,
7501                      PSAMPR_USER_INFO_BUFFER Buffer)
7502 {
7503     SAM_USER_FIXED_DATA FixedData;
7504     ULONG Length = 0;
7505     NTSTATUS Status = STATUS_SUCCESS;
7506 
7507     /* FIXME: Decrypt NT password */
7508     /* FIXME: Decrypt LM password */
7509 
7510     Status = SampSetUserPassword(UserObject,
7511                                  &Buffer->Internal1.EncryptedNtOwfPassword,
7512                                  Buffer->Internal1.NtPasswordPresent,
7513                                  &Buffer->Internal1.EncryptedLmOwfPassword,
7514                                  Buffer->Internal1.LmPasswordPresent);
7515     if (!NT_SUCCESS(Status))
7516         goto done;
7517 
7518     /* Get the fixed user attributes */
7519     Length = sizeof(SAM_USER_FIXED_DATA);
7520     Status = SampGetObjectAttribute(UserObject,
7521                                     L"F",
7522                                     NULL,
7523                                     (PVOID)&FixedData,
7524                                     &Length);
7525     if (!NT_SUCCESS(Status))
7526         goto done;
7527 
7528     if (Buffer->Internal1.PasswordExpired)
7529     {
7530         /* The password was last set ages ago */
7531         FixedData.PasswordLastSet.LowPart = 0;
7532         FixedData.PasswordLastSet.HighPart = 0;
7533     }
7534     else
7535     {
7536         /* The password was last set right now */
7537         Status = NtQuerySystemTime(&FixedData.PasswordLastSet);
7538         if (!NT_SUCCESS(Status))
7539             goto done;
7540     }
7541 
7542     /* Set the fixed user attributes */
7543     Status = SampSetObjectAttribute(UserObject,
7544                                     L"F",
7545                                     REG_BINARY,
7546                                     &FixedData,
7547                                     Length);
7548 
7549 done:
7550     return Status;
7551 }
7552 
7553 
7554 static NTSTATUS
7555 SampSetUserInternal2(PSAM_DB_OBJECT UserObject,
7556                      PSAMPR_USER_INFO_BUFFER Buffer)
7557 {
7558     SAM_USER_FIXED_DATA FixedData;
7559     ULONG Length = 0;
7560     NTSTATUS Status = STATUS_SUCCESS;
7561 
7562     /* Get the fixed user attributes */
7563     Length = sizeof(SAM_USER_FIXED_DATA);
7564     Status = SampGetObjectAttribute(UserObject,
7565                                     L"F",
7566                                     NULL,
7567                                     (PVOID)&FixedData,
7568                                     &Length);
7569     if (!NT_SUCCESS(Status))
7570         goto done;
7571 
7572     if ((Buffer->Internal2.Flags & USER_LOGON_SUCCESS) &&
7573         ((Buffer->Internal2.Flags & ~USER_LOGON_SUCCESS) == 0))
7574     {
7575         /* Update the LastLogon time */
7576         Status = NtQuerySystemTime(&FixedData.LastLogon);
7577         if (!NT_SUCCESS(Status))
7578             goto done;
7579 
7580         FixedData.LogonCount++;
7581         FixedData.BadPasswordCount = 0;
7582     }
7583 
7584     if ((Buffer->Internal2.Flags & USER_LOGON_BAD_PASSWORD) &&
7585         ((Buffer->Internal2.Flags & ~USER_LOGON_BAD_PASSWORD) == 0))
7586     {
7587         /* Update the LastBadPasswordTime */
7588         Status = NtQuerySystemTime(&FixedData.LastBadPasswordTime);
7589         if (!NT_SUCCESS(Status))
7590             goto done;
7591 
7592         FixedData.BadPasswordCount++;
7593     }
7594 
7595     /* Set the fixed user attributes */
7596     Status = SampSetObjectAttribute(UserObject,
7597                                     L"F",
7598                                     REG_BINARY,
7599                                     &FixedData,
7600                                     Length);
7601 
7602 done:
7603     return Status;
7604 }
7605 
7606 
7607 static NTSTATUS
7608 SampSetUserAll(PSAM_DB_OBJECT UserObject,
7609                PSAMPR_USER_INFO_BUFFER Buffer)
7610 {
7611     SAM_USER_FIXED_DATA FixedData;
7612     ULONG Length = 0;
7613     ULONG WhichFields;
7614     PENCRYPTED_NT_OWF_PASSWORD NtPassword = NULL;
7615     PENCRYPTED_LM_OWF_PASSWORD LmPassword = NULL;
7616     BOOLEAN NtPasswordPresent = FALSE;
7617     BOOLEAN LmPasswordPresent = FALSE;
7618     BOOLEAN WriteFixedData = FALSE;
7619     NTSTATUS Status = STATUS_SUCCESS;
7620 
7621     WhichFields = Buffer->All.WhichFields;
7622 
7623     /* Get the fixed size attributes */
7624     Length = sizeof(SAM_USER_FIXED_DATA);
7625     Status = SampGetObjectAttribute(UserObject,
7626                                     L"F",
7627                                     NULL,
7628                                     (PVOID)&FixedData,
7629                                     &Length);
7630     if (!NT_SUCCESS(Status))
7631         goto done;
7632 
7633     if (WhichFields & USER_ALL_USERNAME)
7634     {
7635         Status = SampSetUserName(UserObject,
7636                                  &Buffer->All.UserName);
7637         if (!NT_SUCCESS(Status))
7638             goto done;
7639     }
7640 
7641     if (WhichFields & USER_ALL_FULLNAME)
7642     {
7643         Status = SampSetObjectAttributeString(UserObject,
7644                                               L"FullName",
7645                                               &Buffer->All.FullName);
7646         if (!NT_SUCCESS(Status))
7647             goto done;
7648     }
7649 
7650     if (WhichFields & USER_ALL_ADMINCOMMENT)
7651     {
7652         Status = SampSetObjectAttributeString(UserObject,
7653                                               L"AdminComment",
7654                                               &Buffer->All.AdminComment);
7655         if (!NT_SUCCESS(Status))
7656             goto done;
7657     }
7658 
7659     if (WhichFields & USER_ALL_USERCOMMENT)
7660     {
7661         Status = SampSetObjectAttributeString(UserObject,
7662                                               L"UserComment",
7663                                               &Buffer->All.UserComment);
7664         if (!NT_SUCCESS(Status))
7665             goto done;
7666     }
7667 
7668     if (WhichFields & USER_ALL_HOMEDIRECTORY)
7669     {
7670         Status = SampSetObjectAttributeString(UserObject,
7671                                               L"HomeDirectory",
7672                                               &Buffer->All.HomeDirectory);
7673         if (!NT_SUCCESS(Status))
7674             goto done;
7675     }
7676 
7677     if (WhichFields & USER_ALL_HOMEDIRECTORYDRIVE)
7678     {
7679         Status = SampSetObjectAttributeString(UserObject,
7680                                               L"HomeDirectoryDrive",
7681                                               &Buffer->All.HomeDirectoryDrive);
7682         if (!NT_SUCCESS(Status))
7683             goto done;
7684     }
7685 
7686     if (WhichFields & USER_ALL_SCRIPTPATH)
7687     {
7688         Status = SampSetObjectAttributeString(UserObject,
7689                                               L"ScriptPath",
7690                                               &Buffer->All.ScriptPath);
7691         if (!NT_SUCCESS(Status))
7692             goto done;
7693     }
7694 
7695     if (WhichFields & USER_ALL_PROFILEPATH)
7696     {
7697         Status = SampSetObjectAttributeString(UserObject,
7698                                               L"ProfilePath",
7699                                               &Buffer->All.ProfilePath);
7700         if (!NT_SUCCESS(Status))
7701             goto done;
7702     }
7703 
7704     if (WhichFields & USER_ALL_WORKSTATIONS)
7705     {
7706         Status = SampSetObjectAttributeString(UserObject,
7707                                               L"WorkStations",
7708                                               &Buffer->All.WorkStations);
7709         if (!NT_SUCCESS(Status))
7710             goto done;
7711     }
7712 
7713     if (WhichFields & USER_ALL_PARAMETERS)
7714     {
7715         Status = SampSetObjectAttributeString(UserObject,
7716                                               L"Parameters",
7717                                               &Buffer->All.Parameters);
7718         if (!NT_SUCCESS(Status))
7719             goto done;
7720     }
7721 
7722     if (WhichFields & USER_ALL_LOGONHOURS)
7723     {
7724         Status = SampSetLogonHoursAttribute(UserObject,
7725                                            &Buffer->All.LogonHours);
7726         if (!NT_SUCCESS(Status))
7727             goto done;
7728     }
7729 
7730     if (WhichFields & USER_ALL_PRIMARYGROUPID)
7731     {
7732         FixedData.PrimaryGroupId = Buffer->All.PrimaryGroupId;
7733         WriteFixedData = TRUE;
7734     }
7735 
7736     if (WhichFields & USER_ALL_ACCOUNTEXPIRES)
7737     {
7738         FixedData.AccountExpires.LowPart = Buffer->All.AccountExpires.LowPart;
7739         FixedData.AccountExpires.HighPart = Buffer->All.AccountExpires.HighPart;
7740         WriteFixedData = TRUE;
7741     }
7742 
7743     if (WhichFields & USER_ALL_USERACCOUNTCONTROL)
7744     {
7745         FixedData.UserAccountControl = Buffer->All.UserAccountControl;
7746         WriteFixedData = TRUE;
7747     }
7748 
7749     if (WhichFields & USER_ALL_COUNTRYCODE)
7750     {
7751         FixedData.CountryCode = Buffer->All.CountryCode;
7752         WriteFixedData = TRUE;
7753     }
7754 
7755     if (WhichFields & USER_ALL_CODEPAGE)
7756     {
7757         FixedData.CodePage = Buffer->All.CodePage;
7758         WriteFixedData = TRUE;
7759     }
7760 
7761     if (WhichFields & (USER_ALL_NTPASSWORDPRESENT |
7762                        USER_ALL_LMPASSWORDPRESENT))
7763     {
7764         if (WhichFields & USER_ALL_NTPASSWORDPRESENT)
7765         {
7766             NtPassword = (PENCRYPTED_NT_OWF_PASSWORD)Buffer->All.NtOwfPassword.Buffer;
7767             NtPasswordPresent = Buffer->All.NtPasswordPresent;
7768         }
7769 
7770         if (WhichFields & USER_ALL_LMPASSWORDPRESENT)
7771         {
7772             LmPassword = (PENCRYPTED_LM_OWF_PASSWORD)Buffer->All.LmOwfPassword.Buffer;
7773             LmPasswordPresent = Buffer->All.LmPasswordPresent;
7774         }
7775 
7776         Status = SampSetUserPassword(UserObject,
7777                                      NtPassword,
7778                                      NtPasswordPresent,
7779                                      LmPassword,
7780                                      LmPasswordPresent);
7781         if (!NT_SUCCESS(Status))
7782             goto done;
7783 
7784         /* The password has just been set */
7785         Status = NtQuerySystemTime(&FixedData.PasswordLastSet);
7786         if (!NT_SUCCESS(Status))
7787             goto done;
7788 
7789         WriteFixedData = TRUE;
7790     }
7791 
7792     if (WhichFields & USER_ALL_PRIVATEDATA)
7793     {
7794         Status = SampSetObjectAttributeString(UserObject,
7795                                               L"PrivateData",
7796                                               &Buffer->All.PrivateData);
7797         if (!NT_SUCCESS(Status))
7798             goto done;
7799     }
7800 
7801     if (WhichFields & USER_ALL_PASSWORDEXPIRED)
7802     {
7803         if (Buffer->All.PasswordExpired)
7804         {
7805             /* The password was last set ages ago */
7806             FixedData.PasswordLastSet.LowPart = 0;
7807             FixedData.PasswordLastSet.HighPart = 0;
7808         }
7809         else
7810         {
7811             /* The password was last set right now */
7812             Status = NtQuerySystemTime(&FixedData.PasswordLastSet);
7813             if (!NT_SUCCESS(Status))
7814                 goto done;
7815         }
7816 
7817         WriteFixedData = TRUE;
7818     }
7819 
7820     if (WhichFields & USER_ALL_SECURITYDESCRIPTOR)
7821     {
7822         Status = SampSetObjectAttribute(UserObject,
7823                                         L"SecDesc",
7824                                         REG_BINARY,
7825                                         Buffer->All.SecurityDescriptor.SecurityDescriptor,
7826                                         Buffer->All.SecurityDescriptor.Length);
7827     }
7828 
7829     if (WriteFixedData != FALSE)
7830     {
7831         Status = SampSetObjectAttribute(UserObject,
7832                                         L"F",
7833                                         REG_BINARY,
7834                                         &FixedData,
7835                                         Length);
7836         if (!NT_SUCCESS(Status))
7837             goto done;
7838     }
7839 
7840 done:
7841     return Status;
7842 }
7843 
7844 
7845 /* Function 37 */
7846 NTSTATUS
7847 NTAPI
7848 SamrSetInformationUser(IN SAMPR_HANDLE UserHandle,
7849                        IN USER_INFORMATION_CLASS UserInformationClass,
7850                        IN PSAMPR_USER_INFO_BUFFER Buffer)
7851 {
7852     TRACE("SamrSetInformationUser(%p %lu %p)\n",
7853           UserHandle, UserInformationClass, Buffer);
7854 
7855     return SamrSetInformationUser2(UserHandle,
7856                                    UserInformationClass,
7857                                    Buffer);
7858 }
7859 
7860 
7861 /* Function 38 */
7862 NTSTATUS
7863 NTAPI
7864 SamrChangePasswordUser(IN SAMPR_HANDLE UserHandle,
7865                        IN unsigned char LmPresent,
7866                        IN PENCRYPTED_LM_OWF_PASSWORD OldLmEncryptedWithNewLm,
7867                        IN PENCRYPTED_LM_OWF_PASSWORD NewLmEncryptedWithOldLm,
7868                        IN unsigned char NtPresent,
7869                        IN PENCRYPTED_NT_OWF_PASSWORD OldNtEncryptedWithNewNt,
7870                        IN PENCRYPTED_NT_OWF_PASSWORD NewNtEncryptedWithOldNt,
7871                        IN unsigned char NtCrossEncryptionPresent,
7872                        IN PENCRYPTED_NT_OWF_PASSWORD NewNtEncryptedWithNewLm,
7873                        IN unsigned char LmCrossEncryptionPresent,
7874                        IN PENCRYPTED_LM_OWF_PASSWORD NewLmEncryptedWithNewNt)
7875 {
7876     ENCRYPTED_LM_OWF_PASSWORD StoredLmPassword;
7877     ENCRYPTED_NT_OWF_PASSWORD StoredNtPassword;
7878     LM_OWF_PASSWORD OldLmPassword;
7879     LM_OWF_PASSWORD NewLmPassword;
7880     NT_OWF_PASSWORD OldNtPassword;
7881     NT_OWF_PASSWORD NewNtPassword;
7882     BOOLEAN StoredLmPresent = FALSE;
7883     BOOLEAN StoredNtPresent = FALSE;
7884     BOOLEAN StoredLmEmpty = TRUE;
7885     BOOLEAN StoredNtEmpty = TRUE;
7886     PSAM_DB_OBJECT UserObject;
7887     ULONG Length;
7888     SAM_USER_FIXED_DATA UserFixedData;
7889     SAM_DOMAIN_FIXED_DATA DomainFixedData;
7890     LARGE_INTEGER SystemTime;
7891     NTSTATUS Status;
7892 
7893     DBG_UNREFERENCED_LOCAL_VARIABLE(StoredLmPresent);
7894     DBG_UNREFERENCED_LOCAL_VARIABLE(StoredNtPresent);
7895     DBG_UNREFERENCED_LOCAL_VARIABLE(StoredLmEmpty);
7896 
7897     TRACE("SamrChangePasswordUser(%p %u %p %p %u %p %p %u %p %u %p)\n",
7898           UserHandle, LmPresent, OldLmEncryptedWithNewLm, NewLmEncryptedWithOldLm,
7899           NtPresent, OldNtEncryptedWithNewNt, NewNtEncryptedWithOldNt, NtCrossEncryptionPresent,
7900           NewNtEncryptedWithNewLm, LmCrossEncryptionPresent, NewLmEncryptedWithNewNt);
7901 
7902     RtlAcquireResourceExclusive(&SampResource,
7903                                 TRUE);
7904 
7905     /* Validate the user handle */
7906     Status = SampValidateDbObject(UserHandle,
7907                                   SamDbUserObject,
7908                                   USER_CHANGE_PASSWORD,
7909                                   &UserObject);
7910     if (!NT_SUCCESS(Status))
7911     {
7912         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
7913         goto done;
7914     }
7915 
7916     /* Get the current time */
7917     Status = NtQuerySystemTime(&SystemTime);
7918     if (!NT_SUCCESS(Status))
7919     {
7920         TRACE("NtQuerySystemTime failed (Status 0x%08lx)\n", Status);
7921         goto done;
7922     }
7923 
7924     /* Retrieve the LM password */
7925     Length = sizeof(ENCRYPTED_LM_OWF_PASSWORD);
7926     Status = SampGetObjectAttribute(UserObject,
7927                                     L"LMPwd",
7928                                     NULL,
7929                                     &StoredLmPassword,
7930                                     &Length);
7931     if (NT_SUCCESS(Status))
7932     {
7933         if (Length == sizeof(ENCRYPTED_LM_OWF_PASSWORD))
7934         {
7935             StoredLmPresent = TRUE;
7936             if (!RtlEqualMemory(&StoredLmPassword,
7937                                 &EmptyLmHash,
7938                                 sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
7939                 StoredLmEmpty = FALSE;
7940         }
7941     }
7942 
7943     /* Retrieve the NT password */
7944     Length = sizeof(ENCRYPTED_NT_OWF_PASSWORD);
7945     Status = SampGetObjectAttribute(UserObject,
7946                                     L"NTPwd",
7947                                     NULL,
7948                                     &StoredNtPassword,
7949                                     &Length);
7950     if (NT_SUCCESS(Status))
7951     {
7952         if (Length == sizeof(ENCRYPTED_NT_OWF_PASSWORD))
7953         {
7954             StoredNtPresent = TRUE;
7955             if (!RtlEqualMemory(&StoredNtPassword,
7956                                 &EmptyNtHash,
7957                                 sizeof(ENCRYPTED_NT_OWF_PASSWORD)))
7958                 StoredNtEmpty = FALSE;
7959         }
7960     }
7961 
7962     /* Retrieve the fixed size user data */
7963     Length = sizeof(SAM_USER_FIXED_DATA);
7964     Status = SampGetObjectAttribute(UserObject,
7965                                     L"F",
7966                                     NULL,
7967                                     &UserFixedData,
7968                                     &Length);
7969     if (!NT_SUCCESS(Status))
7970     {
7971         TRACE("SampGetObjectAttribute failed to retrieve the fixed user data (Status 0x%08lx)\n", Status);
7972         goto done;
7973     }
7974 
7975     /* Check if we can change the password at this time */
7976     if ((StoredLmEmpty == FALSE) || (StoredNtEmpty == FALSE))
7977     {
7978         /* Get fixed domain data */
7979         Length = sizeof(SAM_DOMAIN_FIXED_DATA);
7980         Status = SampGetObjectAttribute(UserObject->ParentObject,
7981                                         L"F",
7982                                         NULL,
7983                                         &DomainFixedData,
7984                                         &Length);
7985         if (!NT_SUCCESS(Status))
7986         {
7987             TRACE("SampGetObjectAttribute failed to retrieve the fixed domain data (Status 0x%08lx)\n", Status);
7988             goto done;
7989         }
7990 
7991         if (DomainFixedData.MinPasswordAge.QuadPart > 0)
7992         {
7993             if (SystemTime.QuadPart < (UserFixedData.PasswordLastSet.QuadPart + DomainFixedData.MinPasswordAge.QuadPart))
7994             {
7995                 Status = STATUS_ACCOUNT_RESTRICTION;
7996                 goto done;
7997             }
7998         }
7999     }
8000 
8001     /* Decrypt the LM passwords, if present */
8002     if (LmPresent)
8003     {
8004         Status = SystemFunction013((const BYTE *)NewLmEncryptedWithOldLm,
8005                                    (const BYTE *)&StoredLmPassword,
8006                                    (LPBYTE)&NewLmPassword);
8007         if (!NT_SUCCESS(Status))
8008         {
8009             TRACE("SystemFunction013 failed (Status 0x%08lx)\n", Status);
8010             goto done;
8011         }
8012 
8013         Status = SystemFunction013((const BYTE *)OldLmEncryptedWithNewLm,
8014                                    (const BYTE *)&NewLmPassword,
8015                                    (LPBYTE)&OldLmPassword);
8016         if (!NT_SUCCESS(Status))
8017         {
8018             TRACE("SystemFunction013 failed (Status 0x%08lx)\n", Status);
8019             goto done;
8020         }
8021     }
8022 
8023     /* Decrypt the NT passwords, if present */
8024     if (NtPresent)
8025     {
8026         Status = SystemFunction013((const BYTE *)NewNtEncryptedWithOldNt,
8027                                    (const BYTE *)&StoredNtPassword,
8028                                    (LPBYTE)&NewNtPassword);
8029         if (!NT_SUCCESS(Status))
8030         {
8031             TRACE("SystemFunction013 failed (Status 0x%08lx)\n", Status);
8032             goto done;
8033         }
8034 
8035         Status = SystemFunction013((const BYTE *)OldNtEncryptedWithNewNt,
8036                                    (const BYTE *)&NewNtPassword,
8037                                    (LPBYTE)&OldNtPassword);
8038         if (!NT_SUCCESS(Status))
8039         {
8040             TRACE("SystemFunction013 failed (Status 0x%08lx)\n", Status);
8041             goto done;
8042         }
8043     }
8044 
8045     /* Check if the old passwords match the stored ones */
8046     if (NtPresent)
8047     {
8048         if (LmPresent)
8049         {
8050             if (!RtlEqualMemory(&StoredLmPassword,
8051                                 &OldLmPassword,
8052                                 sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
8053             {
8054                 TRACE("Old LM Password does not match!\n");
8055                 Status = STATUS_WRONG_PASSWORD;
8056             }
8057             else
8058             {
8059                 if (!RtlEqualMemory(&StoredNtPassword,
8060                                     &OldNtPassword,
8061                                     sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
8062                 {
8063                     TRACE("Old NT Password does not match!\n");
8064                     Status = STATUS_WRONG_PASSWORD;
8065                 }
8066             }
8067         }
8068         else
8069         {
8070             if (!RtlEqualMemory(&StoredNtPassword,
8071                                 &OldNtPassword,
8072                                 sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
8073             {
8074                 TRACE("Old NT Password does not match!\n");
8075                 Status = STATUS_WRONG_PASSWORD;
8076             }
8077         }
8078     }
8079     else
8080     {
8081         if (LmPresent)
8082         {
8083             if (!RtlEqualMemory(&StoredLmPassword,
8084                                 &OldLmPassword,
8085                                 sizeof(ENCRYPTED_LM_OWF_PASSWORD)))
8086             {
8087                 TRACE("Old LM Password does not match!\n");
8088                 Status = STATUS_WRONG_PASSWORD;
8089             }
8090         }
8091         else
8092         {
8093             Status = STATUS_INVALID_PARAMETER;
8094         }
8095     }
8096 
8097     /* Store the new password hashes */
8098     if (NT_SUCCESS(Status))
8099     {
8100         Status = SampSetUserPassword(UserObject,
8101                                      &NewNtPassword,
8102                                      NtPresent,
8103                                      &NewLmPassword,
8104                                      LmPresent);
8105         if (NT_SUCCESS(Status))
8106         {
8107             /* Update PasswordLastSet */
8108             UserFixedData.PasswordLastSet.QuadPart = SystemTime.QuadPart;
8109 
8110             /* Set the fixed size user data */
8111             Length = sizeof(SAM_USER_FIXED_DATA);
8112             Status = SampSetObjectAttribute(UserObject,
8113                                             L"F",
8114                                             REG_BINARY,
8115                                             &UserFixedData,
8116                                             Length);
8117         }
8118     }
8119 
8120     if (Status == STATUS_WRONG_PASSWORD)
8121     {
8122         /* Update BadPasswordCount and LastBadPasswordTime */
8123         UserFixedData.BadPasswordCount++;
8124         UserFixedData.LastBadPasswordTime.QuadPart = SystemTime.QuadPart;
8125 
8126         /* Set the fixed size user data */
8127         Length = sizeof(SAM_USER_FIXED_DATA);
8128         Status = SampSetObjectAttribute(UserObject,
8129                                         L"F",
8130                                         REG_BINARY,
8131                                         &UserFixedData,
8132                                         Length);
8133     }
8134 
8135 done:
8136     RtlReleaseResource(&SampResource);
8137 
8138     return Status;
8139 }
8140 
8141 
8142 /* Function 39 */
8143 NTSTATUS
8144 NTAPI
8145 SamrGetGroupsForUser(IN SAMPR_HANDLE UserHandle,
8146                      OUT PSAMPR_GET_GROUPS_BUFFER *Groups)
8147 {
8148     PSAMPR_GET_GROUPS_BUFFER GroupsBuffer = NULL;
8149     PSAM_DB_OBJECT UserObject;
8150     ULONG Length = 0;
8151     NTSTATUS Status;
8152 
8153     TRACE("SamrGetGroupsForUser(%p %p)\n",
8154           UserHandle, Groups);
8155 
8156     RtlAcquireResourceShared(&SampResource,
8157                              TRUE);
8158 
8159     /* Validate the user handle */
8160     Status = SampValidateDbObject(UserHandle,
8161                                   SamDbUserObject,
8162                                   USER_LIST_GROUPS,
8163                                   &UserObject);
8164     if (!NT_SUCCESS(Status))
8165     {
8166         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
8167         goto done;
8168     }
8169 
8170     /* Allocate the groups buffer */
8171     GroupsBuffer = midl_user_allocate(sizeof(SAMPR_GET_GROUPS_BUFFER));
8172     if (GroupsBuffer == NULL)
8173     {
8174         Status = STATUS_INSUFFICIENT_RESOURCES;
8175         goto done;
8176     }
8177 
8178     /*
8179      * Get the size of the Groups attribute.
8180      * Do not check the status code because in case of an error
8181      * Length will be 0. And that is all we need.
8182      */
8183     SampGetObjectAttribute(UserObject,
8184                            L"Groups",
8185                            NULL,
8186                            NULL,
8187                            &Length);
8188 
8189     /* If there is no Groups attribute, return a groups buffer without an array */
8190     if (Length == 0)
8191     {
8192         GroupsBuffer->MembershipCount = 0;
8193         GroupsBuffer->Groups = NULL;
8194 
8195         *Groups = GroupsBuffer;
8196 
8197         Status = STATUS_SUCCESS;
8198         goto done;
8199     }
8200 
8201     /* Allocate a buffer for the Groups attribute */
8202     GroupsBuffer->Groups = midl_user_allocate(Length);
8203     if (GroupsBuffer->Groups == NULL)
8204     {
8205         Status = STATUS_INSUFFICIENT_RESOURCES;
8206         goto done;
8207     }
8208 
8209     /* Retrieve the Grous attribute */
8210     Status = SampGetObjectAttribute(UserObject,
8211                                     L"Groups",
8212                                     NULL,
8213                                     GroupsBuffer->Groups,
8214                                     &Length);
8215     if (!NT_SUCCESS(Status))
8216     {
8217         TRACE("SampGetObjectAttribute failed with status 0x%08lx\n", Status);
8218         goto done;
8219     }
8220 
8221     /* Calculate the membership count */
8222     GroupsBuffer->MembershipCount = Length / sizeof(GROUP_MEMBERSHIP);
8223 
8224     /* Return the groups buffer to the caller */
8225     *Groups = GroupsBuffer;
8226 
8227 done:
8228     if (!NT_SUCCESS(Status))
8229     {
8230         if (GroupsBuffer != NULL)
8231         {
8232             if (GroupsBuffer->Groups != NULL)
8233                 midl_user_free(GroupsBuffer->Groups);
8234 
8235             midl_user_free(GroupsBuffer);
8236         }
8237     }
8238 
8239     RtlReleaseResource(&SampResource);
8240 
8241     return Status;
8242 }
8243 
8244 
8245 /* Function 40 */
8246 NTSTATUS
8247 NTAPI
8248 SamrQueryDisplayInformation(IN SAMPR_HANDLE DomainHandle,
8249                             IN DOMAIN_DISPLAY_INFORMATION DisplayInformationClass,
8250                             IN unsigned long Index,
8251                             IN unsigned long EntryCount,
8252                             IN unsigned long PreferredMaximumLength,
8253                             OUT unsigned long *TotalAvailable,
8254                             OUT unsigned long *TotalReturned,
8255                             OUT PSAMPR_DISPLAY_INFO_BUFFER Buffer)
8256 {
8257     TRACE("SamrQueryDisplayInformation(%p %lu %lu %lu %lu %p %p %p)\n",
8258           DomainHandle, DisplayInformationClass, Index,
8259           EntryCount, PreferredMaximumLength, TotalAvailable,
8260           TotalReturned, Buffer);
8261 
8262     return SamrQueryDisplayInformation3(DomainHandle,
8263                                         DisplayInformationClass,
8264                                         Index,
8265                                         EntryCount,
8266                                         PreferredMaximumLength,
8267                                         TotalAvailable,
8268                                         TotalReturned,
8269                                         Buffer);
8270 }
8271 
8272 
8273 /* Function 41 */
8274 NTSTATUS
8275 NTAPI
8276 SamrGetDisplayEnumerationIndex(IN SAMPR_HANDLE DomainHandle,
8277                                IN DOMAIN_DISPLAY_INFORMATION DisplayInformationClass,
8278                                IN PRPC_UNICODE_STRING Prefix,
8279                                OUT unsigned long *Index)
8280 {
8281     TRACE("SamrGetDisplayEnumerationIndex(%p %lu %p %p)\n",
8282           DomainHandle, DisplayInformationClass, Prefix, Index);
8283 
8284     return SamrGetDisplayEnumerationIndex2(DomainHandle,
8285                                            DisplayInformationClass,
8286                                            Prefix,
8287                                            Index);
8288 }
8289 
8290 
8291 /* Function 42 */
8292 NTSTATUS
8293 NTAPI
8294 SamrTestPrivateFunctionsDomain(IN SAMPR_HANDLE DomainHandle)
8295 {
8296     UNIMPLEMENTED;
8297     return STATUS_NOT_IMPLEMENTED;
8298 }
8299 
8300 
8301 /* Function 43 */
8302 NTSTATUS
8303 NTAPI
8304 SamrTestPrivateFunctionsUser(IN SAMPR_HANDLE UserHandle)
8305 {
8306     UNIMPLEMENTED;
8307     return STATUS_NOT_IMPLEMENTED;
8308 }
8309 
8310 
8311 /* Function 44 */
8312 NTSTATUS
8313 NTAPI
8314 SamrGetUserDomainPasswordInformation(IN SAMPR_HANDLE UserHandle,
8315                                      OUT PUSER_DOMAIN_PASSWORD_INFORMATION PasswordInformation)
8316 {
8317     SAM_DOMAIN_FIXED_DATA DomainFixedData;
8318     SAM_USER_FIXED_DATA UserFixedData;
8319     PSAM_DB_OBJECT DomainObject;
8320     PSAM_DB_OBJECT UserObject;
8321     ULONG Length = 0;
8322     NTSTATUS Status;
8323 
8324     TRACE("SamrGetUserDomainPasswordInformation(%p %p)\n",
8325           UserHandle, PasswordInformation);
8326 
8327     RtlAcquireResourceShared(&SampResource,
8328                              TRUE);
8329 
8330     /* Validate the user handle */
8331     Status = SampValidateDbObject(UserHandle,
8332                                   SamDbUserObject,
8333                                   0,
8334                                   &UserObject);
8335     if (!NT_SUCCESS(Status))
8336     {
8337         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
8338         goto done;
8339     }
8340 
8341     /* Validate the domain object */
8342     Status = SampValidateDbObject((SAMPR_HANDLE)UserObject->ParentObject,
8343                                   SamDbDomainObject,
8344                                   DOMAIN_READ_PASSWORD_PARAMETERS,
8345                                   &DomainObject);
8346     if (!NT_SUCCESS(Status))
8347     {
8348         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
8349         goto done;
8350     }
8351 
8352     /* Get fixed user data */
8353     Length = sizeof(SAM_USER_FIXED_DATA);
8354     Status = SampGetObjectAttribute(UserObject,
8355                                     L"F",
8356                                     NULL,
8357                                     (PVOID)&UserFixedData,
8358                                     &Length);
8359     if (!NT_SUCCESS(Status))
8360     {
8361         TRACE("SampGetObjectAttribute failed with status 0x%08lx\n", Status);
8362         goto done;
8363     }
8364 
8365     if ((UserObject->RelativeId == DOMAIN_USER_RID_KRBTGT) ||
8366         (UserFixedData.UserAccountControl & (USER_INTERDOMAIN_TRUST_ACCOUNT |
8367                                              USER_WORKSTATION_TRUST_ACCOUNT |
8368                                              USER_SERVER_TRUST_ACCOUNT)))
8369     {
8370         PasswordInformation->MinPasswordLength = 0;
8371         PasswordInformation->PasswordProperties = 0;
8372     }
8373     else
8374     {
8375         /* Get fixed domain data */
8376         Length = sizeof(SAM_DOMAIN_FIXED_DATA);
8377         Status = SampGetObjectAttribute(DomainObject,
8378                                         L"F",
8379                                         NULL,
8380                                         (PVOID)&DomainFixedData,
8381                                         &Length);
8382         if (!NT_SUCCESS(Status))
8383         {
8384             TRACE("SampGetObjectAttribute failed with status 0x%08lx\n", Status);
8385             goto done;
8386         }
8387 
8388         PasswordInformation->MinPasswordLength = DomainFixedData.MinPasswordLength;
8389         PasswordInformation->PasswordProperties = DomainFixedData.PasswordProperties;
8390     }
8391 
8392 done:
8393     RtlReleaseResource(&SampResource);
8394 
8395     return STATUS_SUCCESS;
8396 }
8397 
8398 
8399 /* Function 45 */
8400 NTSTATUS
8401 NTAPI
8402 SamrRemoveMemberFromForeignDomain(IN SAMPR_HANDLE DomainHandle,
8403                                   IN PRPC_SID MemberSid)
8404 {
8405     PSAM_DB_OBJECT DomainObject;
8406     ULONG Rid = 0;
8407     NTSTATUS Status;
8408 
8409     TRACE("SamrRemoveMemberFromForeignDomain(%p %p)\n",
8410           DomainHandle, MemberSid);
8411 
8412     RtlAcquireResourceExclusive(&SampResource,
8413                                 TRUE);
8414 
8415     /* Validate the domain object */
8416     Status = SampValidateDbObject(DomainHandle,
8417                                   SamDbDomainObject,
8418                                   DOMAIN_LOOKUP,
8419                                   &DomainObject);
8420     if (!NT_SUCCESS(Status))
8421     {
8422         TRACE("SampValidateDbObject failed with status 0x%08lx\n", Status);
8423         goto done;
8424     }
8425 
8426     /* Retrieve the RID from the MemberSID */
8427     Status = SampGetRidFromSid((PSID)MemberSid,
8428                                &Rid);
8429     if (!NT_SUCCESS(Status))
8430     {
8431         TRACE("SampGetRidFromSid failed with status 0x%08lx\n", Status);
8432         goto done;
8433     }
8434 
8435     /* Fail, if the RID represents a special account */
8436     if (Rid < 1000)
8437     {
8438         TRACE("Cannot remove a special account (RID: %lu)\n", Rid);
8439         Status = STATUS_SPECIAL_ACCOUNT;
8440         goto done;
8441     }
8442 
8443     /* Remove the member from all aliases in the domain */
8444     Status = SampRemoveMemberFromAllAliases(DomainObject,
8445                                             MemberSid);
8446     if (!NT_SUCCESS(Status))
8447     {
8448         TRACE("SampRemoveMemberFromAllAliases failed with status 0x%08lx\n", Status);
8449     }
8450 
8451 done:
8452     RtlReleaseResource(&SampResource);
8453 
8454     return Status;
8455 }
8456 
8457 
8458 /* Function 46 */
8459 NTSTATUS
8460 NTAPI
8461 SamrQueryInformationDomain2(IN SAMPR_HANDLE DomainHandle,
8462                             IN DOMAIN_INFORMATION_CLASS DomainInformationClass,
8463                             OUT PSAMPR_DOMAIN_INFO_BUFFER *Buffer)
8464 {
8465     PSAM_DB_OBJECT DomainObject;
8466     ACCESS_MASK DesiredAccess;
8467     NTSTATUS Status;
8468 
8469     TRACE("SamrQueryInformationDomain2(%p %lu %p)\n",
8470           DomainHandle, DomainInformationClass, Buffer);
8471 
8472     switch (DomainInformationClass)
8473     {
8474         case DomainPasswordInformation:
8475         case DomainLockoutInformation:
8476             DesiredAccess = DOMAIN_READ_PASSWORD_PARAMETERS;
8477             break;
8478 
8479         case DomainGeneralInformation:
8480         case DomainLogoffInformation:
8481         case DomainOemInformation:
8482         case DomainNameInformation:
8483         case DomainReplicationInformation:
8484         case DomainServerRoleInformation:
8485         case DomainModifiedInformation:
8486         case DomainStateInformation:
8487         case DomainModifiedInformation2:
8488             DesiredAccess = DOMAIN_READ_OTHER_PARAMETERS;
8489             break;
8490 
8491         case DomainGeneralInformation2:
8492             DesiredAccess = DOMAIN_READ_PASSWORD_PARAMETERS |
8493                             DOMAIN_READ_OTHER_PARAMETERS;
8494             break;
8495 
8496         default:
8497             return STATUS_INVALID_INFO_CLASS;
8498     }
8499 
8500     RtlAcquireResourceShared(&SampResource,
8501                              TRUE);
8502 
8503     /* Validate the server handle */
8504     Status = SampValidateDbObject(DomainHandle,
8505                                   SamDbDomainObject,
8506                                   DesiredAccess,
8507                                   &DomainObject);
8508     if (!NT_SUCCESS(Status))
8509         goto done;
8510 
8511     switch (DomainInformationClass)
8512     {
8513         case DomainPasswordInformation:
8514             Status = SampQueryDomainPassword(DomainObject,
8515                                              Buffer);
8516             break;
8517 
8518         case DomainGeneralInformation:
8519             Status = SampQueryDomainGeneral(DomainObject,
8520                                             Buffer);
8521             break;
8522 
8523         case DomainLogoffInformation:
8524             Status = SampQueryDomainLogoff(DomainObject,
8525                                            Buffer);
8526             break;
8527 
8528         case DomainOemInformation:
8529             Status = SampQueryDomainOem(DomainObject,
8530                                         Buffer);
8531             break;
8532 
8533         case DomainNameInformation:
8534             Status = SampQueryDomainName(DomainObject,
8535                                          Buffer);
8536             break;
8537 
8538         case DomainReplicationInformation:
8539             Status = SampQueryDomainReplication(DomainObject,
8540                                                 Buffer);
8541             break;
8542 
8543         case DomainServerRoleInformation:
8544             Status = SampQueryDomainServerRole(DomainObject,
8545                                                Buffer);
8546             break;
8547 
8548         case DomainModifiedInformation:
8549             Status = SampQueryDomainModified(DomainObject,
8550                                              Buffer);
8551             break;
8552 
8553         case DomainStateInformation:
8554             Status = SampQueryDomainState(DomainObject,
8555                                           Buffer);
8556             break;
8557 
8558         case DomainGeneralInformation2:
8559             Status = SampQueryDomainGeneral2(DomainObject,
8560                                              Buffer);
8561             break;
8562 
8563         case DomainLockoutInformation:
8564             Status = SampQueryDomainLockout(DomainObject,
8565                                             Buffer);
8566             break;
8567 
8568         case DomainModifiedInformation2:
8569             Status = SampQueryDomainModified2(DomainObject,
8570                                               Buffer);
8571             break;
8572 
8573         default:
8574             Status = STATUS_NOT_IMPLEMENTED;
8575     }
8576 
8577 done:
8578     RtlReleaseResource(&SampResource);
8579 
8580     return Status;
8581 }
8582 
8583 
8584 /* Function 47 */
8585 NTSTATUS
8586 NTAPI
8587 SamrQueryInformationUser2(IN SAMPR_HANDLE UserHandle,
8588                           IN USER_INFORMATION_CLASS UserInformationClass,
8589                           OUT PSAMPR_USER_INFO_BUFFER *Buffer)
8590 {
8591     PSAM_DB_OBJECT UserObject;
8592     ACCESS_MASK DesiredAccess;
8593     NTSTATUS Status;
8594 
8595     TRACE("SamrQueryInformationUser2(%p %lu %p)\n",
8596           UserHandle, UserInformationClass, Buffer);
8597 
8598     switch (UserInformationClass)
8599     {
8600         case UserGeneralInformation:
8601         case UserNameInformation:
8602         case UserAccountNameInformation:
8603         case UserFullNameInformation:
8604         case UserPrimaryGroupInformation:
8605         case UserAdminCommentInformation:
8606             DesiredAccess = USER_READ_GENERAL;
8607             break;
8608 
8609         case UserLogonHoursInformation:
8610         case UserHomeInformation:
8611         case UserScriptInformation:
8612         case UserProfileInformation:
8613         case UserWorkStationsInformation:
8614             DesiredAccess = USER_READ_LOGON;
8615             break;
8616 
8617         case UserControlInformation:
8618         case UserExpiresInformation:
8619         case UserParametersInformation:
8620             DesiredAccess = USER_READ_ACCOUNT;
8621             break;
8622 
8623         case UserPreferencesInformation:
8624             DesiredAccess = USER_READ_GENERAL |
8625                             USER_READ_PREFERENCES;
8626             break;
8627 
8628         case UserLogonInformation:
8629         case UserAccountInformation:
8630             DesiredAccess = USER_READ_GENERAL |
8631                             USER_READ_PREFERENCES |
8632                             USER_READ_LOGON |
8633                             USER_READ_ACCOUNT;
8634             break;
8635 
8636         case UserInternal1Information:
8637         case UserInternal2Information:
8638         case UserAllInformation:
8639             DesiredAccess = 0;
8640             break;
8641 
8642         default:
8643             return STATUS_INVALID_INFO_CLASS;
8644     }
8645 
8646     RtlAcquireResourceShared(&SampResource,
8647                              TRUE);
8648 
8649     /* Validate the domain handle */
8650     Status = SampValidateDbObject(UserHandle,
8651                                   SamDbUserObject,
8652                                   DesiredAccess,
8653                                   &UserObject);
8654     if (!NT_SUCCESS(Status))
8655     {
8656         TRACE("failed with status 0x%08lx\n", Status);
8657         goto done;
8658     }
8659 
8660     switch (UserInformationClass)
8661     {
8662         case UserGeneralInformation:
8663             Status = SampQueryUserGeneral(UserObject,
8664                                           Buffer);
8665             break;
8666 
8667         case UserPreferencesInformation:
8668             Status = SampQueryUserPreferences(UserObject,
8669                                               Buffer);
8670             break;
8671 
8672         case UserLogonInformation:
8673             Status = SampQueryUserLogon(UserObject,
8674                                         Buffer);
8675             break;
8676 
8677         case UserLogonHoursInformation:
8678             Status = SampQueryUserLogonHours(UserObject,
8679                                              Buffer);
8680             break;
8681 
8682         case UserAccountInformation:
8683             Status = SampQueryUserAccount(UserObject,
8684                                           Buffer);
8685             break;
8686 
8687         case UserNameInformation:
8688             Status = SampQueryUserName(UserObject,
8689                                        Buffer);
8690             break;
8691 
8692         case UserAccountNameInformation:
8693             Status = SampQueryUserAccountName(UserObject,
8694                                               Buffer);
8695             break;
8696 
8697         case UserFullNameInformation:
8698             Status = SampQueryUserFullName(UserObject,
8699                                            Buffer);
8700             break;
8701 
8702         case UserPrimaryGroupInformation:
8703             Status = SampQueryUserPrimaryGroup(UserObject,
8704                                                Buffer);
8705             break;
8706 
8707         case UserHomeInformation:
8708             Status = SampQueryUserHome(UserObject,
8709                                        Buffer);
8710             break;
8711 
8712         case UserScriptInformation:
8713             Status = SampQueryUserScript(UserObject,
8714                                          Buffer);
8715             break;
8716 
8717         case UserProfileInformation:
8718             Status = SampQueryUserProfile(UserObject,
8719                                           Buffer);
8720             break;
8721 
8722         case UserAdminCommentInformation:
8723             Status = SampQueryUserAdminComment(UserObject,
8724                                                Buffer);
8725             break;
8726 
8727         case UserWorkStationsInformation:
8728             Status = SampQueryUserWorkStations(UserObject,
8729                                                Buffer);
8730             break;
8731 
8732         case UserControlInformation:
8733             Status = SampQueryUserControl(UserObject,
8734                                           Buffer);
8735             break;
8736 
8737         case UserExpiresInformation:
8738             Status = SampQueryUserExpires(UserObject,
8739                                           Buffer);
8740             break;
8741 
8742         case UserInternal1Information:
8743             Status = SampQueryUserInternal1(UserObject,
8744                                             Buffer);
8745             break;
8746 
8747         case UserInternal2Information:
8748             Status = SampQueryUserInternal2(UserObject,
8749                                             Buffer);
8750             break;
8751 
8752         case UserParametersInformation:
8753             Status = SampQueryUserParameters(UserObject,
8754                                              Buffer);
8755             break;
8756 
8757         case UserAllInformation:
8758             Status = SampQueryUserAll(UserObject,
8759                                       Buffer);
8760             break;
8761 
8762 //        case UserInternal4Information:
8763 //        case UserInternal5Information:
8764 //        case UserInternal4InformationNew:
8765 //        case UserInternal5InformationNew:
8766 
8767         default:
8768             Status = STATUS_INVALID_INFO_CLASS;
8769     }
8770 
8771 done:
8772     RtlReleaseResource(&SampResource);
8773 
8774     return Status;
8775 }
8776 
8777 
8778 /* Function 48 */
8779 NTSTATUS
8780 NTAPI
8781 SamrQueryDisplayInformation2(IN SAMPR_HANDLE DomainHandle,
8782                              IN DOMAIN_DISPLAY_INFORMATION DisplayInformationClass,
8783                              IN unsigned long Index,
8784                              IN unsigned long EntryCount,
8785                              IN unsigned long PreferredMaximumLength,
8786                              OUT unsigned long *TotalAvailable,
8787                              OUT unsigned long *TotalReturned,
8788                              OUT PSAMPR_DISPLAY_INFO_BUFFER Buffer)
8789 {
8790     TRACE("SamrQueryDisplayInformation2(%p %lu %lu %lu %lu %p %p %p)\n",
8791           DomainHandle, DisplayInformationClass, Index,
8792           EntryCount, PreferredMaximumLength, TotalAvailable,
8793           TotalReturned, Buffer);
8794 
8795     return SamrQueryDisplayInformation3(DomainHandle,
8796                                         DisplayInformationClass,
8797                                         Index,
8798                                         EntryCount,
8799                                         PreferredMaximumLength,
8800                                         TotalAvailable,
8801                                         TotalReturned,
8802                                         Buffer);
8803 }
8804 
8805 
8806 /* Function 49 */
8807 NTSTATUS
8808 NTAPI
8809 SamrGetDisplayEnumerationIndex2(IN SAMPR_HANDLE DomainHandle,
8810                                 IN DOMAIN_DISPLAY_INFORMATION DisplayInformationClass,
8811                                 IN PRPC_UNICODE_STRING Prefix,
8812                                 OUT unsigned long *Index)
8813 {
8814     TRACE("SamrGetDisplayEnumerationIndex2(%p %lu %p %p)\n",
8815           DomainHandle, DisplayInformationClass, Prefix, Index);
8816 
8817     UNIMPLEMENTED;
8818     return STATUS_NOT_IMPLEMENTED;
8819 }
8820 
8821 
8822 /* Function 50 */
8823 NTSTATUS
8824 NTAPI
8825 SamrCreateUser2InDomain(IN SAMPR_HANDLE DomainHandle,
8826                         IN PRPC_UNICODE_STRING Name,
8827                         IN unsigned long AccountType,
8828                         IN ACCESS_MASK DesiredAccess,
8829                         OUT SAMPR_HANDLE *UserHandle,
8830                         OUT unsigned long *GrantedAccess,
8831                         OUT unsigned long *RelativeId)
8832 {
8833     SAM_DOMAIN_FIXED_DATA FixedDomainData;
8834     SAM_USER_FIXED_DATA FixedUserData;
8835     PSAM_DB_OBJECT DomainObject;
8836     PSAM_DB_OBJECT UserObject;
8837     GROUP_MEMBERSHIP GroupMembership;
8838     UCHAR LogonHours[23];
8839     ULONG ulSize;
8840     ULONG ulRid;
8841     WCHAR szRid[9];
8842     PSECURITY_DESCRIPTOR Sd = NULL;
8843     ULONG SdSize = 0;
8844     PSID UserSid = NULL;
8845     NTSTATUS Status;
8846 
8847     TRACE("SamrCreateUserInDomain(%p %p %lx %p %p)\n",
8848           DomainHandle, Name, DesiredAccess, UserHandle, RelativeId);
8849 
8850     if (Name == NULL ||
8851         Name->Length == 0 ||
8852         Name->Buffer == NULL ||
8853         UserHandle == NULL ||
8854         RelativeId == NULL)
8855         return STATUS_INVALID_PARAMETER;
8856 
8857     /* Check for valid account type */
8858     if (AccountType != USER_NORMAL_ACCOUNT &&
8859         AccountType != USER_WORKSTATION_TRUST_ACCOUNT &&
8860         AccountType != USER_INTERDOMAIN_TRUST_ACCOUNT &&
8861         AccountType != USER_SERVER_TRUST_ACCOUNT &&
8862         AccountType != USER_TEMP_DUPLICATE_ACCOUNT)
8863         return STATUS_INVALID_PARAMETER;
8864 
8865     /* Map generic access rights */
8866     RtlMapGenericMask(&DesiredAccess,
8867                       &UserMapping);
8868 
8869     RtlAcquireResourceExclusive(&SampResource,
8870                                 TRUE);
8871 
8872     /* Validate the domain handle */
8873     Status = SampValidateDbObject(DomainHandle,
8874                                   SamDbDomainObject,
8875                                   DOMAIN_CREATE_USER,
8876                                   &DomainObject);
8877     if (!NT_SUCCESS(Status))
8878     {
8879         TRACE("failed with status 0x%08lx\n", Status);
8880         goto done;
8881     }
8882 
8883     /* Check the user account name */
8884     Status = SampCheckAccountName(Name, 20);
8885     if (!NT_SUCCESS(Status))
8886     {
8887         TRACE("SampCheckAccountName failed (Status 0x%08lx)\n", Status);
8888         goto done;
8889     }
8890 
8891     /* Check if the user name already exists in the domain */
8892     Status = SampCheckAccountNameInDomain(DomainObject,
8893                                           Name->Buffer);
8894     if (!NT_SUCCESS(Status))
8895     {
8896         TRACE("User name \'%S\' already exists in domain (Status 0x%08lx)\n",
8897               Name->Buffer, Status);
8898         goto done;
8899     }
8900 
8901     /* Get the fixed domain attributes */
8902     ulSize = sizeof(SAM_DOMAIN_FIXED_DATA);
8903     Status = SampGetObjectAttribute(DomainObject,
8904                                     L"F",
8905                                     NULL,
8906                                     (PVOID)&FixedDomainData,
8907                                     &ulSize);
8908     if (!NT_SUCCESS(Status))
8909     {
8910         TRACE("failed with status 0x%08lx\n", Status);
8911         goto done;
8912     }
8913 
8914     /* Increment the NextRid attribute */
8915     ulRid = FixedDomainData.NextRid;
8916     FixedDomainData.NextRid++;
8917 
8918     TRACE("RID: %lx\n", ulRid);
8919 
8920     /* Create the user SID */
8921     Status = SampCreateAccountSid(DomainObject,
8922                                   ulRid,
8923                                   &UserSid);
8924     if (!NT_SUCCESS(Status))
8925     {
8926         TRACE("SampCreateAccountSid failed (Status 0x%08lx)\n", Status);
8927         goto done;
8928     }
8929 
8930     /* Create the security descriptor */
8931     Status = SampCreateUserSD(UserSid,
8932                               &Sd,
8933                               &SdSize);
8934     if (!NT_SUCCESS(Status))
8935     {
8936         TRACE("SampCreateUserSD failed (Status 0x%08lx)\n", Status);
8937         goto done;
8938     }
8939 
8940     /* Store the fixed domain attributes */
8941     Status = SampSetObjectAttribute(DomainObject,
8942                                     L"F",
8943                                     REG_BINARY,
8944                                     &FixedDomainData,
8945                                     ulSize);
8946     if (!NT_SUCCESS(Status))
8947     {
8948         TRACE("failed with status 0x%08lx\n", Status);
8949         goto done;
8950     }
8951 
8952     /* Convert the RID into a string (hex) */
8953     swprintf(szRid, L"%08lX", ulRid);
8954 
8955     /* Create the user object */
8956     Status = SampCreateDbObject(DomainObject,
8957                                 L"Users",
8958                                 szRid,
8959                                 ulRid,
8960                                 SamDbUserObject,
8961                                 DesiredAccess,
8962                                 &UserObject);
8963     if (!NT_SUCCESS(Status))
8964     {
8965         TRACE("failed with status 0x%08lx\n", Status);
8966         goto done;
8967     }
8968 
8969     /* Add the account name for the user object */
8970     Status = SampSetAccountNameInDomain(DomainObject,
8971                                         L"Users",
8972                                         Name->Buffer,
8973                                         ulRid);
8974     if (!NT_SUCCESS(Status))
8975     {
8976         TRACE("failed with status 0x%08lx\n", Status);
8977         goto done;
8978     }
8979 
8980     /* Initialize fixed user data */
8981     FixedUserData.Version = 1;
8982     FixedUserData.Reserved = 0;
8983     FixedUserData.LastLogon.QuadPart = 0;
8984     FixedUserData.LastLogoff.QuadPart = 0;
8985     FixedUserData.PasswordLastSet.QuadPart = 0;
8986     FixedUserData.AccountExpires.LowPart = MAXULONG;
8987     FixedUserData.AccountExpires.HighPart = MAXLONG;
8988     FixedUserData.LastBadPasswordTime.QuadPart = 0;
8989     FixedUserData.UserId = ulRid;
8990     FixedUserData.PrimaryGroupId = DOMAIN_GROUP_RID_USERS;
8991     FixedUserData.UserAccountControl = USER_ACCOUNT_DISABLED |
8992                                        USER_PASSWORD_NOT_REQUIRED |
8993                                        AccountType;
8994     FixedUserData.CountryCode = 0;
8995     FixedUserData.CodePage = 0;
8996     FixedUserData.BadPasswordCount = 0;
8997     FixedUserData.LogonCount = 0;
8998     FixedUserData.AdminCount = 0;
8999     FixedUserData.OperatorCount = 0;
9000 
9001     /* Set fixed user data attribute */
9002     Status = SampSetObjectAttribute(UserObject,
9003                                     L"F",
9004                                     REG_BINARY,
9005                                     (LPVOID)&FixedUserData,
9006                                     sizeof(SAM_USER_FIXED_DATA));
9007     if (!NT_SUCCESS(Status))
9008     {
9009         TRACE("failed with status 0x%08lx\n", Status);
9010         goto done;
9011     }
9012 
9013     /* Set the Name attribute */
9014     Status = SampSetObjectAttributeString(UserObject,
9015                                           L"Name",
9016                                           Name);
9017     if (!NT_SUCCESS(Status))
9018     {
9019         TRACE("failed with status 0x%08lx\n", Status);
9020         goto done;
9021     }
9022 
9023     /* Set the FullName attribute */
9024     Status = SampSetObjectAttributeString(UserObject,
9025                                           L"FullName",
9026                                           NULL);
9027     if (!NT_SUCCESS(Status))
9028     {
9029         TRACE("failed with status 0x%08lx\n", Status);
9030         goto done;
9031     }
9032 
9033     /* Set the HomeDirectory attribute */
9034     Status = SampSetObjectAttributeString(UserObject,
9035                                           L"HomeDirectory",
9036                                           NULL);
9037     if (!NT_SUCCESS(Status))
9038     {
9039         TRACE("failed with status 0x%08lx\n", Status);
9040         goto done;
9041     }
9042 
9043     /* Set the HomeDirectoryDrive attribute */
9044     Status = SampSetObjectAttributeString(UserObject,
9045                                           L"HomeDirectoryDrive",
9046                                           NULL);
9047     if (!NT_SUCCESS(Status))
9048     {
9049         TRACE("failed with status 0x%08lx\n", Status);
9050         goto done;
9051     }
9052 
9053     /* Set the ScriptPath attribute */
9054     Status = SampSetObjectAttributeString(UserObject,
9055                                           L"ScriptPath",
9056                                           NULL);
9057     if (!NT_SUCCESS(Status))
9058     {
9059         TRACE("failed with status 0x%08lx\n", Status);
9060         goto done;
9061     }
9062 
9063     /* Set the ProfilePath attribute */
9064     Status = SampSetObjectAttributeString(UserObject,
9065                                           L"ProfilePath",
9066                                           NULL);
9067     if (!NT_SUCCESS(Status))
9068     {
9069         TRACE("failed with status 0x%08lx\n", Status);
9070         goto done;
9071     }
9072 
9073     /* Set the AdminComment attribute */
9074     Status = SampSetObjectAttributeString(UserObject,
9075                                           L"AdminComment",
9076                                           NULL);
9077     if (!NT_SUCCESS(Status))
9078     {
9079         TRACE("failed with status 0x%08lx\n", Status);
9080         goto done;
9081     }
9082 
9083     /* Set the UserComment attribute */
9084     Status = SampSetObjectAttributeString(UserObject,
9085                                           L"UserComment",
9086                                           NULL);
9087     if (!NT_SUCCESS(Status))
9088     {
9089         TRACE("failed with status 0x%08lx\n", Status);
9090         goto done;
9091     }
9092 
9093     /* Set the WorkStations attribute */
9094     Status = SampSetObjectAttributeString(UserObject,
9095                                           L"WorkStations",
9096                                           NULL);
9097     if (!NT_SUCCESS(Status))
9098     {
9099         TRACE("failed with status 0x%08lx\n", Status);
9100         goto done;
9101     }
9102 
9103     /* Set the Parameters attribute */
9104     Status = SampSetObjectAttributeString(UserObject,
9105                                           L"Parameters",
9106                                           NULL);
9107     if (!NT_SUCCESS(Status))
9108     {
9109         TRACE("failed with status 0x%08lx\n", Status);
9110         goto done;
9111     }
9112 
9113     /* Set LogonHours attribute*/
9114     *((PUSHORT)LogonHours) = 168;
9115     memset(&(LogonHours[2]), 0xff, 21);
9116 
9117     Status = SampSetObjectAttribute(UserObject,
9118                                     L"LogonHours",
9119                                     REG_BINARY,
9120                                     &LogonHours,
9121                                     sizeof(LogonHours));
9122     if (!NT_SUCCESS(Status))
9123     {
9124         TRACE("failed with status 0x%08lx\n", Status);
9125         goto done;
9126     }
9127 
9128     /* Set Groups attribute*/
9129     GroupMembership.RelativeId = DOMAIN_GROUP_RID_USERS;
9130     GroupMembership.Attributes = SE_GROUP_MANDATORY |
9131                                  SE_GROUP_ENABLED |
9132                                  SE_GROUP_ENABLED_BY_DEFAULT;
9133 
9134     Status = SampSetObjectAttribute(UserObject,
9135                                     L"Groups",
9136                                     REG_BINARY,
9137                                     &GroupMembership,
9138                                     sizeof(GROUP_MEMBERSHIP));
9139     if (!NT_SUCCESS(Status))
9140     {
9141         TRACE("failed with status 0x%08lx\n", Status);
9142         goto done;
9143     }
9144 
9145     /* Set LMPwd attribute*/
9146     Status = SampSetObjectAttribute(UserObject,
9147                                     L"LMPwd",
9148                                     REG_BINARY,
9149                                     NULL,
9150                                     0);
9151     if (!NT_SUCCESS(Status))
9152     {
9153         TRACE("failed with status 0x%08lx\n", Status);
9154         goto done;
9155     }
9156 
9157     /* Set NTPwd attribute*/
9158     Status = SampSetObjectAttribute(UserObject,
9159                                     L"NTPwd",
9160                                     REG_BINARY,
9161                                     NULL,
9162                                     0);
9163     if (!NT_SUCCESS(Status))
9164     {
9165         TRACE("failed with status 0x%08lx\n", Status);
9166         goto done;
9167     }
9168 
9169     /* Set LMPwdHistory attribute*/
9170     Status = SampSetObjectAttribute(UserObject,
9171                                     L"LMPwdHistory",
9172                                     REG_BINARY,
9173                                     NULL,
9174                                     0);
9175     if (!NT_SUCCESS(Status))
9176     {
9177         TRACE("failed with status 0x%08lx\n", Status);
9178         goto done;
9179     }
9180 
9181     /* Set NTPwdHistory attribute*/
9182     Status = SampSetObjectAttribute(UserObject,
9183                                     L"NTPwdHistory",
9184                                     REG_BINARY,
9185                                     NULL,
9186                                     0);
9187     if (!NT_SUCCESS(Status))
9188     {
9189         TRACE("failed with status 0x%08lx\n", Status);
9190         goto done;
9191     }
9192 
9193     /* Set the PrivateData attribute */
9194     Status = SampSetObjectAttributeString(UserObject,
9195                                           L"PrivateData",
9196                                           NULL);
9197     if (!NT_SUCCESS(Status))
9198     {
9199         TRACE("failed with status 0x%08lx\n", Status);
9200         goto done;
9201     }
9202 
9203     /* Set the SecDesc attribute*/
9204     Status = SampSetObjectAttribute(UserObject,
9205                                     L"SecDesc",
9206                                     REG_BINARY,
9207                                     Sd,
9208                                     SdSize);
9209     if (!NT_SUCCESS(Status))
9210     {
9211         TRACE("failed with status 0x%08lx\n", Status);
9212         goto done;
9213     }
9214 
9215     if (NT_SUCCESS(Status))
9216     {
9217         *UserHandle = (SAMPR_HANDLE)UserObject;
9218         *RelativeId = ulRid;
9219         *GrantedAccess = UserObject->Access;
9220     }
9221 
9222 done:
9223     if (Sd != NULL)
9224         RtlFreeHeap(RtlGetProcessHeap(), 0, Sd);
9225 
9226     if (UserSid != NULL)
9227         RtlFreeHeap(RtlGetProcessHeap(), 0, UserSid);
9228 
9229     RtlReleaseResource(&SampResource);
9230 
9231     TRACE("returns with status 0x%08lx\n", Status);
9232 
9233     return Status;
9234 }
9235 
9236 
9237 /* Function 51 */
9238 NTSTATUS
9239 NTAPI
9240 SamrQueryDisplayInformation3(IN SAMPR_HANDLE DomainHandle,
9241                              IN DOMAIN_DISPLAY_INFORMATION DisplayInformationClass,
9242                              IN unsigned long Index,
9243                              IN unsigned long EntryCount,
9244                              IN unsigned long PreferredMaximumLength,
9245                              OUT unsigned long *TotalAvailable,
9246                              OUT unsigned long *TotalReturned,
9247                              OUT PSAMPR_DISPLAY_INFO_BUFFER Buffer)
9248 {
9249     PSAM_DB_OBJECT DomainObject;
9250     NTSTATUS Status;
9251 
9252     FIXME("SamrQueryDisplayInformation3(%p %lu %lu %lu %lu %p %p %p)\n",
9253           DomainHandle, DisplayInformationClass, Index,
9254           EntryCount, PreferredMaximumLength, TotalAvailable,
9255           TotalReturned, Buffer);
9256 
9257     RtlAcquireResourceShared(&SampResource,
9258                              TRUE);
9259 
9260     /* Validate the domain handle */
9261     Status = SampValidateDbObject(DomainHandle,
9262                                   SamDbDomainObject,
9263                                   DOMAIN_LIST_ACCOUNTS,
9264                                   &DomainObject);
9265     if (!NT_SUCCESS(Status))
9266     {
9267         ERR("SampValidateDbObject() failed (Status 0x%08lx)\n", Status);
9268         goto done;
9269     }
9270 
9271     Status = SampFillDisplayCache(DomainObject,
9272                                   DisplayInformationClass);
9273     if (!NT_SUCCESS(Status))
9274     {
9275         ERR("SampFillDisplayCache() failed (Status 0x%08lx)\n", Status);
9276         goto done;
9277     }
9278 
9279 done:
9280     TRACE("returns with status 0x%08lx\n", Status);
9281 
9282     RtlReleaseResource(&SampResource);
9283 
9284 //    return Status;
9285     return STATUS_NOT_IMPLEMENTED;
9286 }
9287 
9288 
9289 /* Function 52 */
9290 NTSTATUS
9291 NTAPI
9292 SamrAddMultipleMembersToAlias(IN SAMPR_HANDLE AliasHandle,
9293                               IN PSAMPR_PSID_ARRAY MembersBuffer)
9294 {
9295     ULONG i;
9296     NTSTATUS Status = STATUS_SUCCESS;
9297 
9298     TRACE("SamrAddMultipleMembersToAlias(%p %p)\n",
9299           AliasHandle, MembersBuffer);
9300 
9301     for (i = 0; i < MembersBuffer->Count; i++)
9302     {
9303         Status = SamrAddMemberToAlias(AliasHandle,
9304                                       ((PSID *)MembersBuffer->Sids)[i]);
9305 
9306         if (Status == STATUS_MEMBER_IN_ALIAS)
9307             Status = STATUS_SUCCESS;
9308 
9309         if (!NT_SUCCESS(Status))
9310             break;
9311     }
9312 
9313     return Status;
9314 }
9315 
9316 
9317 /* Function 53 */
9318 NTSTATUS
9319 NTAPI
9320 SamrRemoveMultipleMembersFromAlias(IN SAMPR_HANDLE AliasHandle,
9321                                    IN PSAMPR_PSID_ARRAY MembersBuffer)
9322 {
9323     ULONG i;
9324     NTSTATUS Status = STATUS_SUCCESS;
9325 
9326     TRACE("SamrRemoveMultipleMembersFromAlias(%p %p)\n",
9327           AliasHandle, MembersBuffer);
9328 
9329     for (i = 0; i < MembersBuffer->Count; i++)
9330     {
9331         Status = SamrRemoveMemberFromAlias(AliasHandle,
9332                                            ((PSID *)MembersBuffer->Sids)[i]);
9333 
9334         if (Status == STATUS_MEMBER_IN_ALIAS)
9335             Status = STATUS_SUCCESS;
9336 
9337         if (!NT_SUCCESS(Status))
9338             break;
9339     }
9340 
9341     return Status;
9342 }
9343 
9344 
9345 /* Function 54 */
9346 NTSTATUS
9347 NTAPI
9348 SamrOemChangePasswordUser2(IN handle_t BindingHandle,
9349                            IN PRPC_STRING ServerName,
9350                            IN PRPC_STRING UserName,
9351                            IN PSAMPR_ENCRYPTED_USER_PASSWORD NewPasswordEncryptedWithOldLm,
9352                            IN PENCRYPTED_LM_OWF_PASSWORD OldLmOwfPasswordEncryptedWithNewLm)
9353 {
9354     UNIMPLEMENTED;
9355     return STATUS_NOT_IMPLEMENTED;
9356 }
9357 
9358 /* Function 55 */
9359 NTSTATUS
9360 NTAPI
9361 SamrUnicodeChangePasswordUser2(IN handle_t BindingHandle,
9362                                IN PRPC_UNICODE_STRING ServerName,
9363                                IN PRPC_UNICODE_STRING UserName,
9364                                IN PSAMPR_ENCRYPTED_USER_PASSWORD NewPasswordEncryptedWithOldNt,
9365                                IN PENCRYPTED_NT_OWF_PASSWORD OldNtOwfPasswordEncryptedWithNewNt,
9366                                IN unsigned char LmPresent,
9367                                IN PSAMPR_ENCRYPTED_USER_PASSWORD NewPasswordEncryptedWithOldLm,
9368                                IN PENCRYPTED_LM_OWF_PASSWORD OldLmOwfPasswordEncryptedWithNewNt)
9369 {
9370     UNIMPLEMENTED;
9371     return STATUS_NOT_IMPLEMENTED;
9372 }
9373 
9374 
9375 /* Function 56 */
9376 NTSTATUS
9377 NTAPI
9378 SamrGetDomainPasswordInformation(IN handle_t BindingHandle,
9379                                  IN PRPC_UNICODE_STRING Unused,
9380                                  OUT PUSER_DOMAIN_PASSWORD_INFORMATION PasswordInformation)
9381 {
9382     SAMPR_HANDLE ServerHandle = NULL;
9383     PSAM_DB_OBJECT DomainObject = NULL;
9384     SAM_DOMAIN_FIXED_DATA FixedData;
9385     ULONG Length;
9386     NTSTATUS Status;
9387 
9388     TRACE("SamrGetDomainPasswordInformation(%p %p %p)\n",
9389           BindingHandle, Unused, PasswordInformation);
9390 
9391     Status = SamrConnect(NULL,
9392                          &ServerHandle,
9393                          SAM_SERVER_LOOKUP_DOMAIN);
9394     if (!NT_SUCCESS(Status))
9395     {
9396         TRACE("SamrConnect() failed (Status 0x%08lx)\n", Status);
9397         goto done;
9398     }
9399 
9400     Status = SampOpenDbObject((PSAM_DB_OBJECT)ServerHandle,
9401                               L"Domains",
9402                               L"Account",
9403                               0,
9404                               SamDbDomainObject,
9405                               DOMAIN_READ_PASSWORD_PARAMETERS,
9406                               &DomainObject);
9407     if (!NT_SUCCESS(Status))
9408     {
9409         TRACE("SampOpenDbObject() failed (Status 0x%08lx)\n", Status);
9410         goto done;
9411     }
9412 
9413     Length = sizeof(SAM_DOMAIN_FIXED_DATA);
9414     Status = SampGetObjectAttribute(DomainObject,
9415                                     L"F",
9416                                     NULL,
9417                                     &FixedData,
9418                                     &Length);
9419     if (!NT_SUCCESS(Status))
9420     {
9421         TRACE("SampGetObjectAttribute() failed (Status 0x%08lx)\n", Status);
9422         goto done;
9423     }
9424 
9425     PasswordInformation->MinPasswordLength = FixedData.MinPasswordLength;
9426     PasswordInformation->PasswordProperties = FixedData.PasswordProperties;
9427 
9428 done:
9429     if (DomainObject != NULL)
9430         SampCloseDbObject(DomainObject);
9431 
9432     if (ServerHandle != NULL)
9433         SamrCloseHandle(&ServerHandle);
9434 
9435     return Status;
9436 }
9437 
9438 
9439 /* Function 57 */
9440 NTSTATUS
9441 NTAPI
9442 SamrConnect2(IN PSAMPR_SERVER_NAME ServerName,
9443              OUT SAMPR_HANDLE *ServerHandle,
9444              IN ACCESS_MASK DesiredAccess)
9445 {
9446     SAMPR_REVISION_INFO InRevisionInfo, OutRevisionInfo;
9447     ULONG OutVersion;
9448 
9449     TRACE("SamrConnect2(%p %p %lx)\n",
9450           ServerName, ServerHandle, DesiredAccess);
9451 
9452     InRevisionInfo.V1.Revision = 1;
9453     InRevisionInfo.V1.SupportedFeatures = 0;
9454 
9455     return SamrConnect5(ServerName,
9456                         DesiredAccess,
9457                         1,
9458                         &InRevisionInfo,
9459                         &OutVersion,
9460                         &OutRevisionInfo,
9461                         ServerHandle);
9462 }
9463 
9464 
9465 /* Function 58 */
9466 NTSTATUS
9467 NTAPI
9468 SamrSetInformationUser2(IN SAMPR_HANDLE UserHandle,
9469                         IN USER_INFORMATION_CLASS UserInformationClass,
9470                         IN PSAMPR_USER_INFO_BUFFER Buffer)
9471 {
9472     PSAM_DB_OBJECT UserObject;
9473     ACCESS_MASK DesiredAccess;
9474     NTSTATUS Status;
9475 
9476     TRACE("SamrSetInformationUser2(%p %lu %p)\n",
9477           UserHandle, UserInformationClass, Buffer);
9478 
9479     switch (UserInformationClass)
9480     {
9481         case UserLogonHoursInformation:
9482         case UserNameInformation:
9483         case UserAccountNameInformation:
9484         case UserFullNameInformation:
9485         case UserPrimaryGroupInformation:
9486         case UserHomeInformation:
9487         case UserScriptInformation:
9488         case UserProfileInformation:
9489         case UserAdminCommentInformation:
9490         case UserWorkStationsInformation:
9491         case UserControlInformation:
9492         case UserExpiresInformation:
9493         case UserParametersInformation:
9494             DesiredAccess = USER_WRITE_ACCOUNT;
9495             break;
9496 
9497         case UserGeneralInformation:
9498             DesiredAccess = USER_WRITE_ACCOUNT |
9499                             USER_WRITE_PREFERENCES;
9500             break;
9501 
9502         case UserPreferencesInformation:
9503             DesiredAccess = USER_WRITE_PREFERENCES;
9504             break;
9505 
9506         case UserSetPasswordInformation:
9507         case UserInternal1Information:
9508             DesiredAccess = USER_FORCE_PASSWORD_CHANGE;
9509             break;
9510 
9511         case UserAllInformation:
9512         case UserInternal2Information:
9513             DesiredAccess = 0; /* FIXME */
9514             break;
9515 
9516         default:
9517             return STATUS_INVALID_INFO_CLASS;
9518     }
9519 
9520     RtlAcquireResourceExclusive(&SampResource,
9521                                 TRUE);
9522 
9523     /* Validate the domain handle */
9524     Status = SampValidateDbObject(UserHandle,
9525                                   SamDbUserObject,
9526                                   DesiredAccess,
9527                                   &UserObject);
9528     if (!NT_SUCCESS(Status))
9529     {
9530         TRACE("failed with status 0x%08lx\n", Status);
9531         goto done;
9532     }
9533 
9534     switch (UserInformationClass)
9535     {
9536         case UserGeneralInformation:
9537             Status = SampSetUserGeneral(UserObject,
9538                                         Buffer);
9539             break;
9540 
9541         case UserPreferencesInformation:
9542             Status = SampSetUserPreferences(UserObject,
9543                                             Buffer);
9544             break;
9545 
9546         case UserLogonHoursInformation:
9547             Status = SampSetLogonHoursAttribute(UserObject,
9548                                                &Buffer->LogonHours.LogonHours);
9549             break;
9550 
9551         case UserNameInformation:
9552             Status = SampSetUserName(UserObject,
9553                                      &Buffer->Name.UserName);
9554             if (!NT_SUCCESS(Status))
9555                 break;
9556 
9557             Status = SampSetObjectAttributeString(UserObject,
9558                                                   L"FullName",
9559                                                   &Buffer->Name.FullName);
9560             break;
9561 
9562         case UserAccountNameInformation:
9563             Status = SampSetUserName(UserObject,
9564                                      &Buffer->AccountName.UserName);
9565             break;
9566 
9567         case UserFullNameInformation:
9568             Status = SampSetObjectAttributeString(UserObject,
9569                                                   L"FullName",
9570                                                   &Buffer->FullName.FullName);
9571             break;
9572 
9573         case UserPrimaryGroupInformation:
9574             Status = SampSetUserPrimaryGroup(UserObject,
9575                                              Buffer);
9576             break;
9577 
9578         case UserHomeInformation:
9579             Status = SampSetObjectAttributeString(UserObject,
9580                                                   L"HomeDirectory",
9581                                                   &Buffer->Home.HomeDirectory);
9582             if (!NT_SUCCESS(Status))
9583                 break;
9584 
9585             Status = SampSetObjectAttributeString(UserObject,
9586                                                   L"HomeDirectoryDrive",
9587                                                   &Buffer->Home.HomeDirectoryDrive);
9588             break;
9589 
9590         case UserScriptInformation:
9591             Status = SampSetObjectAttributeString(UserObject,
9592                                                   L"ScriptPath",
9593                                                   &Buffer->Script.ScriptPath);
9594             break;
9595 
9596         case UserProfileInformation:
9597             Status = SampSetObjectAttributeString(UserObject,
9598                                                   L"ProfilePath",
9599                                                   &Buffer->Profile.ProfilePath);
9600             break;
9601 
9602         case UserAdminCommentInformation:
9603             Status = SampSetObjectAttributeString(UserObject,
9604                                                   L"AdminComment",
9605                                                   &Buffer->AdminComment.AdminComment);
9606             break;
9607 
9608         case UserWorkStationsInformation:
9609             Status = SampSetObjectAttributeString(UserObject,
9610                                                   L"WorkStations",
9611                                                   &Buffer->WorkStations.WorkStations);
9612             break;
9613 
9614         case UserSetPasswordInformation:
9615             TRACE("Password: %S\n", Buffer->SetPassword.Password.Buffer);
9616             TRACE("PasswordExpired: %d\n", Buffer->SetPassword.PasswordExpired);
9617 
9618             Status = SampSetObjectAttributeString(UserObject,
9619                                                   L"Password",
9620                                                   &Buffer->SetPassword.Password);
9621             break;
9622 
9623         case UserControlInformation:
9624             Status = SampSetUserControl(UserObject,
9625                                         Buffer);
9626             break;
9627 
9628         case UserExpiresInformation:
9629             Status = SampSetUserExpires(UserObject,
9630                                         Buffer);
9631             break;
9632 
9633         case UserInternal1Information:
9634             Status = SampSetUserInternal1(UserObject,
9635                                           Buffer);
9636             break;
9637 
9638         case UserInternal2Information:
9639             Status = SampSetUserInternal2(UserObject,
9640                                           Buffer);
9641             break;
9642 
9643         case UserParametersInformation:
9644             Status = SampSetObjectAttributeString(UserObject,
9645                                                   L"Parameters",
9646                                                   &Buffer->Parameters.Parameters);
9647             break;
9648 
9649         case UserAllInformation:
9650             Status = SampSetUserAll(UserObject,
9651                                     Buffer);
9652             break;
9653 
9654 //        case UserInternal4Information:
9655 //        case UserInternal5Information:
9656 //        case UserInternal4InformationNew:
9657 //        case UserInternal5InformationNew:
9658 
9659         default:
9660             Status = STATUS_INVALID_INFO_CLASS;
9661     }
9662 
9663 done:
9664     RtlReleaseResource(&SampResource);
9665 
9666     return Status;
9667 }
9668 
9669 
9670 /* Function 59 */
9671 NTSTATUS
9672 NTAPI
9673 SamrSetBootKeyInformation(IN handle_t BindingHandle) /* FIXME */
9674 {
9675     UNIMPLEMENTED;
9676     return STATUS_NOT_IMPLEMENTED;
9677 }
9678 
9679 /* Function 60 */
9680 NTSTATUS
9681 NTAPI
9682 SamrGetBootKeyInformation(IN handle_t BindingHandle) /* FIXME */
9683 {
9684     UNIMPLEMENTED;
9685     return STATUS_NOT_IMPLEMENTED;
9686 }
9687 
9688 /* Function 61 */
9689 NTSTATUS
9690 NTAPI
9691 SamrConnect3(IN handle_t BindingHandle) /* FIXME */
9692 {
9693     UNIMPLEMENTED;
9694     return STATUS_NOT_IMPLEMENTED;
9695 }
9696 
9697 
9698 /* Function 62 */
9699 NTSTATUS
9700 NTAPI
9701 SamrConnect4(IN PSAMPR_SERVER_NAME ServerName,
9702              OUT SAMPR_HANDLE *ServerHandle,
9703              IN unsigned long ClientRevision,
9704              IN ACCESS_MASK DesiredAccess)
9705 {
9706     SAMPR_REVISION_INFO InRevisionInfo, OutRevisionInfo;
9707     ULONG OutVersion;
9708 
9709     TRACE("SamrConnect4(%p %p %lu 0x%lx)\n",
9710           ServerName, ServerHandle, ClientRevision, DesiredAccess);
9711 
9712     InRevisionInfo.V1.Revision = 2;
9713     InRevisionInfo.V1.SupportedFeatures = 0;
9714 
9715     return SamrConnect5(ServerName,
9716                         DesiredAccess,
9717                         1,
9718                         &InRevisionInfo,
9719                         &OutVersion,
9720                         &OutRevisionInfo,
9721                         ServerHandle);
9722 }
9723 
9724 
9725 /* Function 63 */
9726 NTSTATUS
9727 NTAPI
9728 SamrUnicodeChangePasswordUser3(IN handle_t BindingHandle) /* FIXME */
9729 {
9730     UNIMPLEMENTED;
9731     return STATUS_NOT_IMPLEMENTED;
9732 }
9733 
9734 
9735 /* Function 64 */
9736 NTSTATUS
9737 NTAPI
9738 SamrConnect5(IN PSAMPR_SERVER_NAME ServerName,
9739              IN ACCESS_MASK DesiredAccess,
9740              IN unsigned long InVersion,
9741              IN SAMPR_REVISION_INFO *InRevisionInfo,
9742              OUT unsigned long *OutVersion,
9743              OUT SAMPR_REVISION_INFO *OutRevisionInfo,
9744              OUT SAMPR_HANDLE *ServerHandle)
9745 {
9746     PSAM_DB_OBJECT ServerObject;
9747     NTSTATUS Status;
9748 
9749     TRACE("SamrConnect5(%p 0x%lx %lu %p %p %p %p)\n",
9750           ServerName, DesiredAccess, InVersion, InRevisionInfo,
9751           OutVersion, OutRevisionInfo, ServerHandle);
9752 
9753     if (InVersion != 1)
9754         return STATUS_NOT_SUPPORTED;
9755 
9756     RtlAcquireResourceShared(&SampResource,
9757                              TRUE);
9758 
9759     /* Map generic access rights */
9760     RtlMapGenericMask(&DesiredAccess,
9761                       &ServerMapping);
9762 
9763     /* Open the Server Object */
9764     Status = SampOpenDbObject(NULL,
9765                               NULL,
9766                               L"SAM",
9767                               0,
9768                               SamDbServerObject,
9769                               DesiredAccess,
9770                               &ServerObject);
9771     if (NT_SUCCESS(Status))
9772     {
9773         *OutVersion = 1;
9774 
9775         OutRevisionInfo->V1.Revision = 3;
9776         OutRevisionInfo->V1.SupportedFeatures = 0;
9777 
9778         *ServerHandle = (SAMPR_HANDLE)ServerObject;
9779     }
9780 
9781     RtlReleaseResource(&SampResource);
9782 
9783     TRACE("SamrConnect5 done (Status 0x%08lx)\n", Status);
9784 
9785     return Status;
9786 }
9787 
9788 
9789 /* Function 65 */
9790 NTSTATUS
9791 NTAPI
9792 SamrRidToSid(IN SAMPR_HANDLE ObjectHandle,
9793              IN unsigned long Rid,
9794              OUT PRPC_SID *Sid)
9795 {
9796     UNIMPLEMENTED;
9797     return STATUS_NOT_IMPLEMENTED;
9798 }
9799 
9800 /* Function 66 */
9801 NTSTATUS
9802 NTAPI
9803 SamrSetDSRMPassword(IN handle_t BindingHandle,
9804                     IN PRPC_UNICODE_STRING Unused,
9805                     IN unsigned long UserId,
9806                     IN PENCRYPTED_NT_OWF_PASSWORD EncryptedNtOwfPassword)
9807 {
9808     UNIMPLEMENTED;
9809     return STATUS_NOT_IMPLEMENTED;
9810 }
9811 
9812 /* Function 67 */
9813 NTSTATUS
9814 NTAPI
9815 SamrValidatePassword(IN handle_t Handle,
9816                      IN PASSWORD_POLICY_VALIDATION_TYPE ValidationType,
9817                      IN PSAM_VALIDATE_INPUT_ARG InputArg,
9818                      OUT PSAM_VALIDATE_OUTPUT_ARG *OutputArg)
9819 {
9820     UNIMPLEMENTED;
9821     return STATUS_NOT_IMPLEMENTED;
9822 }
9823 
9824 /* EOF */
9825