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
7 * PROGRAMMERS: Eric Kohl
10 /* INCLUDES ******************************************************************/
14 WINE_DEFAULT_DEBUG_CHANNEL(samsrv
);
16 /* GLOBALS *******************************************************************/
18 static SID_IDENTIFIER_AUTHORITY NtSidAuthority
= {SECURITY_NT_AUTHORITY
};
20 static GENERIC_MAPPING ServerMapping
=
28 static GENERIC_MAPPING DomainMapping
=
36 static GENERIC_MAPPING AliasMapping
=
44 static GENERIC_MAPPING GroupMapping
=
52 static GENERIC_MAPPING UserMapping
=
60 PGENERIC_MAPPING pServerMapping
= &ServerMapping
;
63 /* FUNCTIONS *****************************************************************/
67 SampAddRelativeTimeToTime(IN LARGE_INTEGER AbsoluteTime
,
68 IN LARGE_INTEGER RelativeTime
)
70 LARGE_INTEGER NewTime
;
72 NewTime
.QuadPart
= AbsoluteTime
.QuadPart
- RelativeTime
.QuadPart
;
74 if (NewTime
.QuadPart
< 0)
82 SampStartRpcServer(VOID
)
86 TRACE("SampStartRpcServer() called\n");
88 Status
= RpcServerUseProtseqEpW(L
"ncacn_np",
92 if (Status
!= RPC_S_OK
)
94 WARN("RpcServerUseProtseqEpW() failed (Status %lx)\n", Status
);
98 Status
= RpcServerRegisterIf(samr_v1_0_s_ifspec
,
101 if (Status
!= RPC_S_OK
)
103 WARN("RpcServerRegisterIf() failed (Status %lx)\n", Status
);
107 Status
= RpcServerListen(1, 20, TRUE
);
108 if (Status
!= RPC_S_OK
)
110 WARN("RpcServerListen() failed (Status %lx)\n", Status
);
114 TRACE("SampStartRpcServer() done\n");
118 void __RPC_FAR
* __RPC_USER
midl_user_allocate(SIZE_T len
)
120 return HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, len
);
124 void __RPC_USER
midl_user_free(void __RPC_FAR
* ptr
)
126 HeapFree(GetProcessHeap(), 0, ptr
);
130 void __RPC_USER
SAMPR_HANDLE_rundown(SAMPR_HANDLE hHandle
)
138 SamrConnect(IN PSAMPR_SERVER_NAME ServerName
,
139 OUT SAMPR_HANDLE
*ServerHandle
,
140 IN ACCESS_MASK DesiredAccess
)
142 PSAM_DB_OBJECT ServerObject
;
145 TRACE("SamrConnect(%p %p %lx)\n",
146 ServerName
, ServerHandle
, DesiredAccess
);
148 /* Map generic access rights */
149 RtlMapGenericMask(&DesiredAccess
,
152 /* Open the Server Object */
153 Status
= SampOpenDbObject(NULL
,
160 if (NT_SUCCESS(Status
))
161 *ServerHandle
= (SAMPR_HANDLE
)ServerObject
;
163 TRACE("SamrConnect done (Status 0x%08lx)\n", Status
);
172 SamrCloseHandle(IN OUT SAMPR_HANDLE
*SamHandle
)
174 PSAM_DB_OBJECT DbObject
;
175 NTSTATUS Status
= STATUS_SUCCESS
;
177 TRACE("SamrCloseHandle(%p)\n", SamHandle
);
179 Status
= SampValidateDbObject(*SamHandle
,
183 if (Status
== STATUS_SUCCESS
)
185 Status
= SampCloseDbObject(DbObject
);
189 TRACE("SamrCloseHandle done (Status 0x%08lx)\n", Status
);
198 SamrSetSecurityObject(IN SAMPR_HANDLE ObjectHandle
,
199 IN SECURITY_INFORMATION SecurityInformation
,
200 IN PSAMPR_SR_SECURITY_DESCRIPTOR SecurityDescriptor
)
203 return STATUS_NOT_IMPLEMENTED
;
210 SamrQuerySecurityObject(IN SAMPR_HANDLE ObjectHandle
,
211 IN SECURITY_INFORMATION SecurityInformation
,
212 OUT PSAMPR_SR_SECURITY_DESCRIPTOR
*SecurityDescriptor
)
215 return STATUS_NOT_IMPLEMENTED
;
222 SamrShutdownSamServer(IN SAMPR_HANDLE ServerHandle
)
224 PSAM_DB_OBJECT ServerObject
;
227 TRACE("(%p)\n", ServerHandle
);
229 /* Validate the server handle */
230 Status
= SampValidateDbObject(ServerHandle
,
234 if (!NT_SUCCESS(Status
))
237 /* Shut the server down */
238 RpcMgmtStopServerListening(0);
240 return STATUS_SUCCESS
;
247 SamrLookupDomainInSamServer(IN SAMPR_HANDLE ServerHandle
,
248 IN PRPC_UNICODE_STRING Name
,
249 OUT PRPC_SID
*DomainId
)
251 PSAM_DB_OBJECT ServerObject
;
252 HANDLE DomainsKeyHandle
= NULL
;
253 HANDLE DomainKeyHandle
= NULL
;
254 WCHAR DomainKeyName
[64];
256 WCHAR DomainNameString
[MAX_COMPUTERNAME_LENGTH
+ 1];
257 UNICODE_STRING DomainName
;
262 TRACE("SamrLookupDomainInSamServer(%p %p %p)\n",
263 ServerHandle
, Name
, DomainId
);
265 /* Validate the server handle */
266 Status
= SampValidateDbObject(ServerHandle
,
268 SAM_SERVER_LOOKUP_DOMAIN
,
270 if (!NT_SUCCESS(Status
))
275 Status
= SampRegOpenKey(ServerObject
->KeyHandle
,
279 if (!NT_SUCCESS(Status
))
283 while (Found
== FALSE
)
285 Status
= SampRegEnumerateSubKey(DomainsKeyHandle
,
289 if (!NT_SUCCESS(Status
))
291 if (Status
== STATUS_NO_MORE_ENTRIES
)
292 Status
= STATUS_NO_SUCH_DOMAIN
;
296 TRACE("Domain key name: %S\n", DomainKeyName
);
298 Status
= SampRegOpenKey(DomainsKeyHandle
,
302 if (NT_SUCCESS(Status
))
304 Length
= (MAX_COMPUTERNAME_LENGTH
+ 1) * sizeof(WCHAR
);
305 Status
= SampRegQueryValue(DomainKeyHandle
,
308 (PVOID
)&DomainNameString
,
310 if (NT_SUCCESS(Status
))
312 TRACE("Domain name: %S\n", DomainNameString
);
314 RtlInitUnicodeString(&DomainName
,
316 if (RtlEqualUnicodeString(&DomainName
, (PUNICODE_STRING
)Name
, TRUE
))
318 TRACE("Found it!\n");
321 Status
= SampRegQueryValue(DomainKeyHandle
,
326 if (NT_SUCCESS(Status
))
328 *DomainId
= midl_user_allocate(Length
);
330 SampRegQueryValue(DomainKeyHandle
,
336 Status
= STATUS_SUCCESS
;
342 NtClose(DomainKeyHandle
);
348 NtClose(DomainsKeyHandle
);
357 SamrEnumerateDomainsInSamServer(IN SAMPR_HANDLE ServerHandle
,
358 IN OUT
unsigned long *EnumerationContext
,
359 OUT PSAMPR_ENUMERATION_BUFFER
*Buffer
,
360 IN ULONG PreferedMaximumLength
,
361 OUT PULONG CountReturned
)
363 PSAM_DB_OBJECT ServerObject
;
364 WCHAR DomainKeyName
[64];
365 HANDLE DomainsKeyHandle
;
366 HANDLE DomainKeyHandle
;
369 ULONG RequiredLength
;
372 PSAMPR_ENUMERATION_BUFFER EnumBuffer
= NULL
;
375 TRACE("SamrEnumerateDomainsInSamServer(%p %p %p %lu %p)\n",
376 ServerHandle
, EnumerationContext
, Buffer
, PreferedMaximumLength
,
379 /* Validate the server handle */
380 Status
= SampValidateDbObject(ServerHandle
,
382 SAM_SERVER_ENUMERATE_DOMAINS
,
384 if (!NT_SUCCESS(Status
))
387 Status
= SampRegOpenKey(ServerObject
->KeyHandle
,
391 if (!NT_SUCCESS(Status
))
394 EnumIndex
= *EnumerationContext
;
400 Status
= SampRegEnumerateSubKey(DomainsKeyHandle
,
404 if (!NT_SUCCESS(Status
))
407 TRACE("EnumIndex: %lu\n", EnumIndex
);
408 TRACE("Domain key name: %S\n", DomainKeyName
);
410 Status
= SampRegOpenKey(DomainsKeyHandle
,
414 TRACE("SampRegOpenKey returned %08lX\n", Status
);
415 if (NT_SUCCESS(Status
))
418 Status
= SampRegQueryValue(DomainKeyHandle
,
423 TRACE("SampRegQueryValue returned %08lX\n", Status
);
424 if (NT_SUCCESS(Status
))
426 TRACE("Data length: %lu\n", DataLength
);
428 if ((RequiredLength
+ DataLength
+ sizeof(UNICODE_STRING
)) > PreferedMaximumLength
)
431 RequiredLength
+= (DataLength
+ sizeof(UNICODE_STRING
));
435 NtClose(DomainKeyHandle
);
441 TRACE("EnumCount: %lu\n", EnumCount
);
442 TRACE("RequiredLength: %lu\n", RequiredLength
);
444 EnumBuffer
= midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER
));
445 if (EnumBuffer
== NULL
)
447 Status
= STATUS_INSUFFICIENT_RESOURCES
;
451 EnumBuffer
->EntriesRead
= EnumCount
;
452 EnumBuffer
->Buffer
= midl_user_allocate(EnumCount
* sizeof(SAMPR_RID_ENUMERATION
));
453 if (EnumBuffer
->Buffer
== NULL
)
455 Status
= STATUS_INSUFFICIENT_RESOURCES
;
459 EnumIndex
= *EnumerationContext
;
460 for (i
= 0; i
< EnumCount
; i
++, EnumIndex
++)
462 Status
= SampRegEnumerateSubKey(DomainsKeyHandle
,
466 if (!NT_SUCCESS(Status
))
469 TRACE("EnumIndex: %lu\n", EnumIndex
);
470 TRACE("Domain key name: %S\n", DomainKeyName
);
472 Status
= SampRegOpenKey(DomainsKeyHandle
,
476 TRACE("SampRegOpenKey returned %08lX\n", Status
);
477 if (NT_SUCCESS(Status
))
480 Status
= SampRegQueryValue(DomainKeyHandle
,
485 TRACE("SampRegQueryValue returned %08lX\n", Status
);
486 if (NT_SUCCESS(Status
))
488 EnumBuffer
->Buffer
[i
].RelativeId
= 0;
489 EnumBuffer
->Buffer
[i
].Name
.Length
= (USHORT
)DataLength
- sizeof(WCHAR
);
490 EnumBuffer
->Buffer
[i
].Name
.MaximumLength
= (USHORT
)DataLength
;
491 EnumBuffer
->Buffer
[i
].Name
.Buffer
= midl_user_allocate(DataLength
);
492 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
== NULL
)
494 NtClose(DomainKeyHandle
);
495 Status
= STATUS_INSUFFICIENT_RESOURCES
;
499 Status
= SampRegQueryValue(DomainKeyHandle
,
502 EnumBuffer
->Buffer
[i
].Name
.Buffer
,
504 TRACE("SampRegQueryValue returned %08lX\n", Status
);
505 if (NT_SUCCESS(Status
))
507 TRACE("Domain name: %S\n", EnumBuffer
->Buffer
[i
].Name
.Buffer
);
511 NtClose(DomainKeyHandle
);
513 if (!NT_SUCCESS(Status
))
518 if (NT_SUCCESS(Status
))
520 *EnumerationContext
+= EnumCount
;
521 *Buffer
= EnumBuffer
;
522 *CountReturned
= EnumCount
;
526 if (!NT_SUCCESS(Status
))
528 *EnumerationContext
= 0;
532 if (EnumBuffer
!= NULL
)
534 if (EnumBuffer
->Buffer
!= NULL
)
536 if (EnumBuffer
->EntriesRead
!= 0)
538 for (i
= 0; i
< EnumBuffer
->EntriesRead
; i
++)
540 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
!= NULL
)
541 midl_user_free(EnumBuffer
->Buffer
[i
].Name
.Buffer
);
545 midl_user_free(EnumBuffer
->Buffer
);
548 midl_user_free(EnumBuffer
);
552 NtClose(DomainsKeyHandle
);
561 SamrOpenDomain(IN SAMPR_HANDLE ServerHandle
,
562 IN ACCESS_MASK DesiredAccess
,
563 IN PRPC_SID DomainId
,
564 OUT SAMPR_HANDLE
*DomainHandle
)
566 PSAM_DB_OBJECT ServerObject
;
567 PSAM_DB_OBJECT DomainObject
;
570 TRACE("SamrOpenDomain(%p %lx %p %p)\n",
571 ServerHandle
, DesiredAccess
, DomainId
, DomainHandle
);
573 /* Map generic access rights */
574 RtlMapGenericMask(&DesiredAccess
,
577 /* Validate the server handle */
578 Status
= SampValidateDbObject(ServerHandle
,
580 SAM_SERVER_LOOKUP_DOMAIN
,
582 if (!NT_SUCCESS(Status
))
585 /* Validate the Domain SID */
586 if ((DomainId
->Revision
!= SID_REVISION
) ||
587 (DomainId
->SubAuthorityCount
> SID_MAX_SUB_AUTHORITIES
) ||
588 (memcmp(&DomainId
->IdentifierAuthority
, &NtSidAuthority
, sizeof(SID_IDENTIFIER_AUTHORITY
)) != 0))
589 return STATUS_INVALID_PARAMETER
;
591 /* Open the domain object */
592 if ((DomainId
->SubAuthorityCount
== 1) &&
593 (DomainId
->SubAuthority
[0] == SECURITY_BUILTIN_DOMAIN_RID
))
595 /* Builtin domain object */
596 TRACE("Opening the builtin domain object.\n");
598 Status
= SampOpenDbObject(ServerObject
,
606 else if ((DomainId
->SubAuthorityCount
== 4) &&
607 (DomainId
->SubAuthority
[0] == SECURITY_NT_NON_UNIQUE
))
609 /* Account domain object */
610 TRACE("Opening the account domain object.\n");
612 /* FIXME: Check the account domain sub authorities!!! */
614 Status
= SampOpenDbObject(ServerObject
,
624 /* No vaild domain SID */
625 Status
= STATUS_INVALID_PARAMETER
;
628 if (NT_SUCCESS(Status
))
629 *DomainHandle
= (SAMPR_HANDLE
)DomainObject
;
631 TRACE("SamrOpenDomain done (Status 0x%08lx)\n", Status
);
638 SampQueryDomainPassword(PSAM_DB_OBJECT DomainObject
,
639 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
641 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
642 SAM_DOMAIN_FIXED_DATA FixedData
;
648 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
649 if (InfoBuffer
== NULL
)
650 return STATUS_INSUFFICIENT_RESOURCES
;
652 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
653 Status
= SampGetObjectAttribute(DomainObject
,
658 if (!NT_SUCCESS(Status
))
661 InfoBuffer
->Password
.MinPasswordLength
= FixedData
.MinPasswordLength
;
662 InfoBuffer
->Password
.PasswordHistoryLength
= FixedData
.PasswordHistoryLength
;
663 InfoBuffer
->Password
.PasswordProperties
= FixedData
.PasswordProperties
;
664 InfoBuffer
->Password
.MaxPasswordAge
.LowPart
= FixedData
.MaxPasswordAge
.LowPart
;
665 InfoBuffer
->Password
.MaxPasswordAge
.HighPart
= FixedData
.MaxPasswordAge
.HighPart
;
666 InfoBuffer
->Password
.MinPasswordAge
.LowPart
= FixedData
.MinPasswordAge
.LowPart
;
667 InfoBuffer
->Password
.MinPasswordAge
.HighPart
= FixedData
.MinPasswordAge
.HighPart
;
669 *Buffer
= InfoBuffer
;
672 if (!NT_SUCCESS(Status
))
674 if (InfoBuffer
!= NULL
)
676 midl_user_free(InfoBuffer
);
685 SampGetNumberOfAccounts(PSAM_DB_OBJECT DomainObject
,
689 HANDLE AccountKeyHandle
= NULL
;
690 HANDLE NamesKeyHandle
= NULL
;
695 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
699 if (!NT_SUCCESS(Status
))
702 Status
= SampRegOpenKey(AccountKeyHandle
,
706 if (!NT_SUCCESS(Status
))
709 Status
= SampRegQueryKeyInfo(NamesKeyHandle
,
713 if (NamesKeyHandle
!= NULL
)
714 SampRegCloseKey(NamesKeyHandle
);
716 if (AccountKeyHandle
!= NULL
)
717 SampRegCloseKey(AccountKeyHandle
);
724 SampQueryDomainGeneral(PSAM_DB_OBJECT DomainObject
,
725 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
727 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
728 SAM_DOMAIN_FIXED_DATA FixedData
;
734 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
735 if (InfoBuffer
== NULL
)
736 return STATUS_INSUFFICIENT_RESOURCES
;
738 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
739 Status
= SampGetObjectAttribute(DomainObject
,
744 if (!NT_SUCCESS(Status
))
747 InfoBuffer
->General
.ForceLogoff
.LowPart
= FixedData
.ForceLogoff
.LowPart
;
748 InfoBuffer
->General
.ForceLogoff
.HighPart
= FixedData
.ForceLogoff
.HighPart
;
749 InfoBuffer
->General
.DomainModifiedCount
.LowPart
= FixedData
.DomainModifiedCount
.LowPart
;
750 InfoBuffer
->General
.DomainModifiedCount
.HighPart
= FixedData
.DomainModifiedCount
.HighPart
;
751 InfoBuffer
->General
.DomainServerState
= FixedData
.DomainServerState
;
752 InfoBuffer
->General
.DomainServerRole
= FixedData
.DomainServerRole
;
753 InfoBuffer
->General
.UasCompatibilityRequired
= FixedData
.UasCompatibilityRequired
;
755 /* Get the OemInformation string */
756 Status
= SampGetObjectAttributeString(DomainObject
,
758 &InfoBuffer
->General
.OemInformation
);
759 if (!NT_SUCCESS(Status
))
761 TRACE("Status 0x%08lx\n", Status
);
765 /* Get the Name string */
766 Status
= SampGetObjectAttributeString(DomainObject
,
768 &InfoBuffer
->General
.DomainName
);
769 if (!NT_SUCCESS(Status
))
771 TRACE("Status 0x%08lx\n", Status
);
775 /* Get the ReplicaSourceNodeName string */
776 Status
= SampGetObjectAttributeString(DomainObject
,
777 L
"ReplicaSourceNodeName",
778 &InfoBuffer
->General
.ReplicaSourceNodeName
);
779 if (!NT_SUCCESS(Status
))
781 TRACE("Status 0x%08lx\n", Status
);
785 /* Get the number of Users in the Domain */
786 Status
= SampGetNumberOfAccounts(DomainObject
,
788 &InfoBuffer
->General
.UserCount
);
789 if (!NT_SUCCESS(Status
))
791 TRACE("Status 0x%08lx\n", Status
);
795 /* Get the number of Groups in the Domain */
796 Status
= SampGetNumberOfAccounts(DomainObject
,
798 &InfoBuffer
->General
.GroupCount
);
799 if (!NT_SUCCESS(Status
))
801 TRACE("Status 0x%08lx\n", Status
);
805 /* Get the number of Aliases in the Domain */
806 Status
= SampGetNumberOfAccounts(DomainObject
,
808 &InfoBuffer
->General
.AliasCount
);
809 if (!NT_SUCCESS(Status
))
811 TRACE("Status 0x%08lx\n", Status
);
815 *Buffer
= InfoBuffer
;
818 if (!NT_SUCCESS(Status
))
820 if (InfoBuffer
!= NULL
)
822 if (InfoBuffer
->General
.OemInformation
.Buffer
!= NULL
)
823 midl_user_free(InfoBuffer
->General
.OemInformation
.Buffer
);
825 if (InfoBuffer
->General
.DomainName
.Buffer
!= NULL
)
826 midl_user_free(InfoBuffer
->General
.DomainName
.Buffer
);
828 if (InfoBuffer
->General
.ReplicaSourceNodeName
.Buffer
!= NULL
)
829 midl_user_free(InfoBuffer
->General
.ReplicaSourceNodeName
.Buffer
);
831 midl_user_free(InfoBuffer
);
840 SampQueryDomainLogoff(PSAM_DB_OBJECT DomainObject
,
841 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
843 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
844 SAM_DOMAIN_FIXED_DATA FixedData
;
850 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
851 if (InfoBuffer
== NULL
)
852 return STATUS_INSUFFICIENT_RESOURCES
;
854 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
855 Status
= SampGetObjectAttribute(DomainObject
,
860 if (!NT_SUCCESS(Status
))
863 InfoBuffer
->Logoff
.ForceLogoff
.LowPart
= FixedData
.ForceLogoff
.LowPart
;
864 InfoBuffer
->Logoff
.ForceLogoff
.HighPart
= FixedData
.ForceLogoff
.HighPart
;
866 *Buffer
= InfoBuffer
;
869 if (!NT_SUCCESS(Status
))
871 if (InfoBuffer
!= NULL
)
873 midl_user_free(InfoBuffer
);
882 SampQueryDomainOem(PSAM_DB_OBJECT DomainObject
,
883 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
885 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
890 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
891 if (InfoBuffer
== NULL
)
892 return STATUS_INSUFFICIENT_RESOURCES
;
894 /* Get the OemInformation string */
895 Status
= SampGetObjectAttributeString(DomainObject
,
897 &InfoBuffer
->Oem
.OemInformation
);
898 if (!NT_SUCCESS(Status
))
900 TRACE("Status 0x%08lx\n", Status
);
904 *Buffer
= InfoBuffer
;
907 if (!NT_SUCCESS(Status
))
909 if (InfoBuffer
!= NULL
)
911 if (InfoBuffer
->Oem
.OemInformation
.Buffer
!= NULL
)
912 midl_user_free(InfoBuffer
->Oem
.OemInformation
.Buffer
);
914 midl_user_free(InfoBuffer
);
923 SampQueryDomainName(PSAM_DB_OBJECT DomainObject
,
924 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
926 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
931 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
932 if (InfoBuffer
== NULL
)
933 return STATUS_INSUFFICIENT_RESOURCES
;
935 /* Get the Name string */
936 Status
= SampGetObjectAttributeString(DomainObject
,
938 &InfoBuffer
->Name
.DomainName
);
939 if (!NT_SUCCESS(Status
))
941 TRACE("Status 0x%08lx\n", Status
);
945 *Buffer
= InfoBuffer
;
948 if (!NT_SUCCESS(Status
))
950 if (InfoBuffer
!= NULL
)
952 if (InfoBuffer
->Name
.DomainName
.Buffer
!= NULL
)
953 midl_user_free(InfoBuffer
->Name
.DomainName
.Buffer
);
955 midl_user_free(InfoBuffer
);
964 SampQueryDomainReplication(PSAM_DB_OBJECT DomainObject
,
965 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
967 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
972 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
973 if (InfoBuffer
== NULL
)
974 return STATUS_INSUFFICIENT_RESOURCES
;
976 /* Get the ReplicaSourceNodeName string */
977 Status
= SampGetObjectAttributeString(DomainObject
,
978 L
"ReplicaSourceNodeName",
979 &InfoBuffer
->Replication
.ReplicaSourceNodeName
);
980 if (!NT_SUCCESS(Status
))
982 TRACE("Status 0x%08lx\n", Status
);
986 *Buffer
= InfoBuffer
;
989 if (!NT_SUCCESS(Status
))
991 if (InfoBuffer
!= NULL
)
993 if (InfoBuffer
->Replication
.ReplicaSourceNodeName
.Buffer
!= NULL
)
994 midl_user_free(InfoBuffer
->Replication
.ReplicaSourceNodeName
.Buffer
);
996 midl_user_free(InfoBuffer
);
1005 SampQueryDomainServerRole(PSAM_DB_OBJECT DomainObject
,
1006 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1008 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1009 SAM_DOMAIN_FIXED_DATA FixedData
;
1015 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1016 if (InfoBuffer
== NULL
)
1017 return STATUS_INSUFFICIENT_RESOURCES
;
1019 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1020 Status
= SampGetObjectAttribute(DomainObject
,
1025 if (!NT_SUCCESS(Status
))
1028 InfoBuffer
->Role
.DomainServerRole
= FixedData
.DomainServerRole
;
1030 *Buffer
= InfoBuffer
;
1033 if (!NT_SUCCESS(Status
))
1035 if (InfoBuffer
!= NULL
)
1037 midl_user_free(InfoBuffer
);
1046 SampQueryDomainModified(PSAM_DB_OBJECT DomainObject
,
1047 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1049 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1050 SAM_DOMAIN_FIXED_DATA FixedData
;
1056 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1057 if (InfoBuffer
== NULL
)
1058 return STATUS_INSUFFICIENT_RESOURCES
;
1060 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1061 Status
= SampGetObjectAttribute(DomainObject
,
1066 if (!NT_SUCCESS(Status
))
1069 InfoBuffer
->Modified
.DomainModifiedCount
.LowPart
= FixedData
.DomainModifiedCount
.LowPart
;
1070 InfoBuffer
->Modified
.DomainModifiedCount
.HighPart
= FixedData
.DomainModifiedCount
.HighPart
;
1071 InfoBuffer
->Modified
.CreationTime
.LowPart
= FixedData
.CreationTime
.LowPart
;
1072 InfoBuffer
->Modified
.CreationTime
.HighPart
= FixedData
.CreationTime
.HighPart
;
1074 *Buffer
= InfoBuffer
;
1077 if (!NT_SUCCESS(Status
))
1079 if (InfoBuffer
!= NULL
)
1081 midl_user_free(InfoBuffer
);
1090 SampQueryDomainState(PSAM_DB_OBJECT DomainObject
,
1091 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1093 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1094 SAM_DOMAIN_FIXED_DATA FixedData
;
1100 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1101 if (InfoBuffer
== NULL
)
1102 return STATUS_INSUFFICIENT_RESOURCES
;
1104 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1105 Status
= SampGetObjectAttribute(DomainObject
,
1110 if (!NT_SUCCESS(Status
))
1113 InfoBuffer
->State
.DomainServerState
= FixedData
.DomainServerState
;
1115 *Buffer
= InfoBuffer
;
1118 if (!NT_SUCCESS(Status
))
1120 if (InfoBuffer
!= NULL
)
1122 midl_user_free(InfoBuffer
);
1131 SampQueryDomainGeneral2(PSAM_DB_OBJECT DomainObject
,
1132 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1134 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1135 SAM_DOMAIN_FIXED_DATA FixedData
;
1141 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1142 if (InfoBuffer
== NULL
)
1143 return STATUS_INSUFFICIENT_RESOURCES
;
1145 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1146 Status
= SampGetObjectAttribute(DomainObject
,
1151 if (!NT_SUCCESS(Status
))
1154 InfoBuffer
->General2
.I1
.ForceLogoff
.LowPart
= FixedData
.ForceLogoff
.LowPart
;
1155 InfoBuffer
->General2
.I1
.ForceLogoff
.HighPart
= FixedData
.ForceLogoff
.HighPart
;
1156 InfoBuffer
->General2
.I1
.DomainModifiedCount
.LowPart
= FixedData
.DomainModifiedCount
.LowPart
;
1157 InfoBuffer
->General2
.I1
.DomainModifiedCount
.HighPart
= FixedData
.DomainModifiedCount
.HighPart
;
1158 InfoBuffer
->General2
.I1
.DomainServerState
= FixedData
.DomainServerState
;
1159 InfoBuffer
->General2
.I1
.DomainServerRole
= FixedData
.DomainServerRole
;
1160 InfoBuffer
->General2
.I1
.UasCompatibilityRequired
= FixedData
.UasCompatibilityRequired
;
1162 InfoBuffer
->General2
.LockoutDuration
= FixedData
.LockoutDuration
;
1163 InfoBuffer
->General2
.LockoutObservationWindow
= FixedData
.LockoutObservationWindow
;
1164 InfoBuffer
->General2
.LockoutThreshold
= FixedData
.LockoutThreshold
;
1166 /* Get the OemInformation string */
1167 Status
= SampGetObjectAttributeString(DomainObject
,
1169 &InfoBuffer
->General2
.I1
.OemInformation
);
1170 if (!NT_SUCCESS(Status
))
1172 TRACE("Status 0x%08lx\n", Status
);
1176 /* Get the Name string */
1177 Status
= SampGetObjectAttributeString(DomainObject
,
1179 &InfoBuffer
->General2
.I1
.DomainName
);
1180 if (!NT_SUCCESS(Status
))
1182 TRACE("Status 0x%08lx\n", Status
);
1186 /* Get the ReplicaSourceNodeName string */
1187 Status
= SampGetObjectAttributeString(DomainObject
,
1188 L
"ReplicaSourceNodeName",
1189 &InfoBuffer
->General2
.I1
.ReplicaSourceNodeName
);
1190 if (!NT_SUCCESS(Status
))
1192 TRACE("Status 0x%08lx\n", Status
);
1196 /* Get the number of Users in the Domain */
1197 Status
= SampGetNumberOfAccounts(DomainObject
,
1199 &InfoBuffer
->General2
.I1
.UserCount
);
1200 if (!NT_SUCCESS(Status
))
1202 TRACE("Status 0x%08lx\n", Status
);
1206 /* Get the number of Groups in the Domain */
1207 Status
= SampGetNumberOfAccounts(DomainObject
,
1209 &InfoBuffer
->General2
.I1
.GroupCount
);
1210 if (!NT_SUCCESS(Status
))
1212 TRACE("Status 0x%08lx\n", Status
);
1216 /* Get the number of Aliases in the Domain */
1217 Status
= SampGetNumberOfAccounts(DomainObject
,
1219 &InfoBuffer
->General2
.I1
.AliasCount
);
1220 if (!NT_SUCCESS(Status
))
1222 TRACE("Status 0x%08lx\n", Status
);
1226 *Buffer
= InfoBuffer
;
1229 if (!NT_SUCCESS(Status
))
1231 if (InfoBuffer
!= NULL
)
1233 if (InfoBuffer
->General2
.I1
.OemInformation
.Buffer
!= NULL
)
1234 midl_user_free(InfoBuffer
->General2
.I1
.OemInformation
.Buffer
);
1236 if (InfoBuffer
->General2
.I1
.DomainName
.Buffer
!= NULL
)
1237 midl_user_free(InfoBuffer
->General2
.I1
.DomainName
.Buffer
);
1239 if (InfoBuffer
->General2
.I1
.ReplicaSourceNodeName
.Buffer
!= NULL
)
1240 midl_user_free(InfoBuffer
->General2
.I1
.ReplicaSourceNodeName
.Buffer
);
1242 midl_user_free(InfoBuffer
);
1251 SampQueryDomainLockout(PSAM_DB_OBJECT DomainObject
,
1252 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1254 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1255 SAM_DOMAIN_FIXED_DATA FixedData
;
1261 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1262 if (InfoBuffer
== NULL
)
1263 return STATUS_INSUFFICIENT_RESOURCES
;
1265 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1266 Status
= SampGetObjectAttribute(DomainObject
,
1271 if (!NT_SUCCESS(Status
))
1274 InfoBuffer
->Lockout
.LockoutDuration
= FixedData
.LockoutDuration
;
1275 InfoBuffer
->Lockout
.LockoutObservationWindow
= FixedData
.LockoutObservationWindow
;
1276 InfoBuffer
->Lockout
.LockoutThreshold
= FixedData
.LockoutThreshold
;
1278 *Buffer
= InfoBuffer
;
1281 if (!NT_SUCCESS(Status
))
1283 if (InfoBuffer
!= NULL
)
1285 midl_user_free(InfoBuffer
);
1294 SampQueryDomainModified2(PSAM_DB_OBJECT DomainObject
,
1295 PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1297 PSAMPR_DOMAIN_INFO_BUFFER InfoBuffer
= NULL
;
1298 SAM_DOMAIN_FIXED_DATA FixedData
;
1304 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_DOMAIN_INFO_BUFFER
));
1305 if (InfoBuffer
== NULL
)
1306 return STATUS_INSUFFICIENT_RESOURCES
;
1308 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1309 Status
= SampGetObjectAttribute(DomainObject
,
1314 if (!NT_SUCCESS(Status
))
1317 InfoBuffer
->Modified2
.DomainModifiedCount
.LowPart
= FixedData
.DomainModifiedCount
.LowPart
;
1318 InfoBuffer
->Modified2
.DomainModifiedCount
.HighPart
= FixedData
.DomainModifiedCount
.HighPart
;
1319 InfoBuffer
->Modified2
.CreationTime
.LowPart
= FixedData
.CreationTime
.LowPart
;
1320 InfoBuffer
->Modified2
.CreationTime
.HighPart
= FixedData
.CreationTime
.HighPart
;
1321 InfoBuffer
->Modified2
.ModifiedCountAtLastPromotion
.LowPart
= FixedData
.ModifiedCountAtLastPromotion
.LowPart
;
1322 InfoBuffer
->Modified2
.ModifiedCountAtLastPromotion
.HighPart
= FixedData
.ModifiedCountAtLastPromotion
.HighPart
;
1324 *Buffer
= InfoBuffer
;
1327 if (!NT_SUCCESS(Status
))
1329 if (InfoBuffer
!= NULL
)
1331 midl_user_free(InfoBuffer
);
1342 SamrQueryInformationDomain(IN SAMPR_HANDLE DomainHandle
,
1343 IN DOMAIN_INFORMATION_CLASS DomainInformationClass
,
1344 OUT PSAMPR_DOMAIN_INFO_BUFFER
*Buffer
)
1346 PSAM_DB_OBJECT DomainObject
;
1347 ACCESS_MASK DesiredAccess
;
1350 TRACE("SamrQueryInformationDomain(%p %lu %p)\n",
1351 DomainHandle
, DomainInformationClass
, Buffer
);
1353 switch (DomainInformationClass
)
1355 case DomainPasswordInformation
:
1356 case DomainLockoutInformation
:
1357 DesiredAccess
= DOMAIN_READ_PASSWORD_PARAMETERS
;
1360 case DomainGeneralInformation
:
1361 case DomainLogoffInformation
:
1362 case DomainOemInformation
:
1363 case DomainNameInformation
:
1364 case DomainReplicationInformation
:
1365 case DomainServerRoleInformation
:
1366 case DomainModifiedInformation
:
1367 case DomainStateInformation
:
1368 case DomainModifiedInformation2
:
1369 DesiredAccess
= DOMAIN_READ_OTHER_PARAMETERS
;
1372 case DomainGeneralInformation2
:
1373 DesiredAccess
= DOMAIN_READ_PASSWORD_PARAMETERS
|
1374 DOMAIN_READ_OTHER_PARAMETERS
;
1378 return STATUS_INVALID_INFO_CLASS
;
1381 /* Validate the server handle */
1382 Status
= SampValidateDbObject(DomainHandle
,
1386 if (!NT_SUCCESS(Status
))
1389 switch (DomainInformationClass
)
1391 case DomainPasswordInformation
:
1392 Status
= SampQueryDomainPassword(DomainObject
,
1396 case DomainGeneralInformation
:
1397 Status
= SampQueryDomainGeneral(DomainObject
,
1401 case DomainLogoffInformation
:
1402 Status
= SampQueryDomainLogoff(DomainObject
,
1406 case DomainOemInformation
:
1407 Status
= SampQueryDomainOem(DomainObject
,
1411 case DomainNameInformation
:
1412 Status
= SampQueryDomainName(DomainObject
,
1416 case DomainReplicationInformation
:
1417 Status
= SampQueryDomainReplication(DomainObject
,
1421 case DomainServerRoleInformation
:
1422 Status
= SampQueryDomainServerRole(DomainObject
,
1426 case DomainModifiedInformation
:
1427 Status
= SampQueryDomainModified(DomainObject
,
1431 case DomainStateInformation
:
1432 Status
= SampQueryDomainState(DomainObject
,
1436 case DomainGeneralInformation2
:
1437 Status
= SampQueryDomainGeneral2(DomainObject
,
1441 case DomainLockoutInformation
:
1442 Status
= SampQueryDomainLockout(DomainObject
,
1446 case DomainModifiedInformation2
:
1447 Status
= SampQueryDomainModified2(DomainObject
,
1452 Status
= STATUS_NOT_IMPLEMENTED
;
1460 SampSetDomainPassword(PSAM_DB_OBJECT DomainObject
,
1461 PSAMPR_DOMAIN_INFO_BUFFER Buffer
)
1463 SAM_DOMAIN_FIXED_DATA FixedData
;
1467 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1468 Status
= SampGetObjectAttribute(DomainObject
,
1473 if (!NT_SUCCESS(Status
))
1476 FixedData
.MinPasswordLength
= Buffer
->Password
.MinPasswordLength
;
1477 FixedData
.PasswordHistoryLength
= Buffer
->Password
.PasswordHistoryLength
;
1478 FixedData
.PasswordProperties
= Buffer
->Password
.PasswordProperties
;
1479 FixedData
.MaxPasswordAge
.LowPart
= Buffer
->Password
.MaxPasswordAge
.LowPart
;
1480 FixedData
.MaxPasswordAge
.HighPart
= Buffer
->Password
.MaxPasswordAge
.HighPart
;
1481 FixedData
.MinPasswordAge
.LowPart
= Buffer
->Password
.MinPasswordAge
.LowPart
;
1482 FixedData
.MinPasswordAge
.HighPart
= Buffer
->Password
.MinPasswordAge
.HighPart
;
1484 Status
= SampSetObjectAttribute(DomainObject
,
1496 SampSetDomainLogoff(PSAM_DB_OBJECT DomainObject
,
1497 PSAMPR_DOMAIN_INFO_BUFFER Buffer
)
1499 SAM_DOMAIN_FIXED_DATA FixedData
;
1503 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1504 Status
= SampGetObjectAttribute(DomainObject
,
1509 if (!NT_SUCCESS(Status
))
1512 FixedData
.ForceLogoff
.LowPart
= Buffer
->Logoff
.ForceLogoff
.LowPart
;
1513 FixedData
.ForceLogoff
.HighPart
= Buffer
->Logoff
.ForceLogoff
.HighPart
;
1515 Status
= SampSetObjectAttribute(DomainObject
,
1527 SampSetDomainServerRole(PSAM_DB_OBJECT DomainObject
,
1528 PSAMPR_DOMAIN_INFO_BUFFER Buffer
)
1530 SAM_DOMAIN_FIXED_DATA FixedData
;
1534 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1535 Status
= SampGetObjectAttribute(DomainObject
,
1540 if (!NT_SUCCESS(Status
))
1543 FixedData
.DomainServerRole
= Buffer
->Role
.DomainServerRole
;
1545 Status
= SampSetObjectAttribute(DomainObject
,
1557 SampSetDomainState(PSAM_DB_OBJECT DomainObject
,
1558 PSAMPR_DOMAIN_INFO_BUFFER Buffer
)
1560 SAM_DOMAIN_FIXED_DATA FixedData
;
1564 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1565 Status
= SampGetObjectAttribute(DomainObject
,
1570 if (!NT_SUCCESS(Status
))
1573 FixedData
.DomainServerState
= Buffer
->State
.DomainServerState
;
1575 Status
= SampSetObjectAttribute(DomainObject
,
1587 SampSetDomainLockout(PSAM_DB_OBJECT DomainObject
,
1588 PSAMPR_DOMAIN_INFO_BUFFER Buffer
)
1590 SAM_DOMAIN_FIXED_DATA FixedData
;
1594 Length
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1595 Status
= SampGetObjectAttribute(DomainObject
,
1600 if (!NT_SUCCESS(Status
))
1603 FixedData
.LockoutDuration
= Buffer
->Lockout
.LockoutDuration
;
1604 FixedData
.LockoutObservationWindow
= Buffer
->Lockout
.LockoutObservationWindow
;
1605 FixedData
.LockoutThreshold
= Buffer
->Lockout
.LockoutThreshold
;
1607 Status
= SampSetObjectAttribute(DomainObject
,
1621 SamrSetInformationDomain(IN SAMPR_HANDLE DomainHandle
,
1622 IN DOMAIN_INFORMATION_CLASS DomainInformationClass
,
1623 IN PSAMPR_DOMAIN_INFO_BUFFER DomainInformation
)
1625 PSAM_DB_OBJECT DomainObject
;
1626 ACCESS_MASK DesiredAccess
;
1629 TRACE("SamrSetInformationDomain(%p %lu %p)\n",
1630 DomainHandle
, DomainInformationClass
, DomainInformation
);
1632 switch (DomainInformationClass
)
1634 case DomainPasswordInformation
:
1635 case DomainLockoutInformation
:
1636 DesiredAccess
= DOMAIN_WRITE_PASSWORD_PARAMS
;
1639 case DomainLogoffInformation
:
1640 case DomainOemInformation
:
1641 case DomainNameInformation
:
1642 DesiredAccess
= DOMAIN_WRITE_OTHER_PARAMETERS
;
1645 case DomainReplicationInformation
:
1646 case DomainServerRoleInformation
:
1647 case DomainStateInformation
:
1648 DesiredAccess
= DOMAIN_ADMINISTER_SERVER
;
1652 return STATUS_INVALID_INFO_CLASS
;
1655 /* Validate the server handle */
1656 Status
= SampValidateDbObject(DomainHandle
,
1660 if (!NT_SUCCESS(Status
))
1663 switch (DomainInformationClass
)
1665 case DomainPasswordInformation
:
1666 Status
= SampSetDomainPassword(DomainObject
,
1670 case DomainLogoffInformation
:
1671 Status
= SampSetDomainLogoff(DomainObject
,
1675 case DomainOemInformation
:
1676 Status
= SampSetObjectAttribute(DomainObject
,
1679 DomainInformation
->Oem
.OemInformation
.Buffer
,
1680 DomainInformation
->Oem
.OemInformation
.Length
+ sizeof(WCHAR
));
1683 case DomainNameInformation
:
1684 Status
= SampSetObjectAttribute(DomainObject
,
1687 DomainInformation
->Name
.DomainName
.Buffer
,
1688 DomainInformation
->Name
.DomainName
.Length
+ sizeof(WCHAR
));
1691 case DomainReplicationInformation
:
1692 Status
= SampSetObjectAttribute(DomainObject
,
1693 L
"ReplicaSourceNodeName",
1695 DomainInformation
->Replication
.ReplicaSourceNodeName
.Buffer
,
1696 DomainInformation
->Replication
.ReplicaSourceNodeName
.Length
+ sizeof(WCHAR
));
1699 case DomainServerRoleInformation
:
1700 Status
= SampSetDomainServerRole(DomainObject
,
1704 case DomainStateInformation
:
1705 Status
= SampSetDomainState(DomainObject
,
1709 case DomainLockoutInformation
:
1710 Status
= SampSetDomainLockout(DomainObject
,
1715 Status
= STATUS_NOT_IMPLEMENTED
;
1725 SamrCreateGroupInDomain(IN SAMPR_HANDLE DomainHandle
,
1726 IN PRPC_UNICODE_STRING Name
,
1727 IN ACCESS_MASK DesiredAccess
,
1728 OUT SAMPR_HANDLE
*GroupHandle
,
1729 OUT
unsigned long *RelativeId
)
1731 UNICODE_STRING EmptyString
= RTL_CONSTANT_STRING(L
"");
1732 SAM_DOMAIN_FIXED_DATA FixedDomainData
;
1733 SAM_GROUP_FIXED_DATA FixedGroupData
;
1734 PSAM_DB_OBJECT DomainObject
;
1735 PSAM_DB_OBJECT GroupObject
;
1741 TRACE("SamrCreateGroupInDomain(%p %p %lx %p %p)\n",
1742 DomainHandle
, Name
, DesiredAccess
, GroupHandle
, RelativeId
);
1744 /* Map generic access rights */
1745 RtlMapGenericMask(&DesiredAccess
,
1748 /* Validate the domain handle */
1749 Status
= SampValidateDbObject(DomainHandle
,
1751 DOMAIN_CREATE_GROUP
,
1753 if (!NT_SUCCESS(Status
))
1755 TRACE("failed with status 0x%08lx\n", Status
);
1759 /* Check if the group name already exists in the domain */
1760 Status
= SampCheckAccountNameInDomain(DomainObject
,
1762 if (!NT_SUCCESS(Status
))
1764 TRACE("Group name \'%S\' already exists in domain (Status 0x%08lx)\n",
1765 Name
->Buffer
, Status
);
1769 /* Get the fixed domain attributes */
1770 ulSize
= sizeof(SAM_DOMAIN_FIXED_DATA
);
1771 Status
= SampGetObjectAttribute(DomainObject
,
1774 (PVOID
)&FixedDomainData
,
1776 if (!NT_SUCCESS(Status
))
1778 TRACE("failed with status 0x%08lx\n", Status
);
1782 /* Increment the NextRid attribute */
1783 ulRid
= FixedDomainData
.NextRid
;
1784 FixedDomainData
.NextRid
++;
1786 /* Store the fixed domain attributes */
1787 Status
= SampSetObjectAttribute(DomainObject
,
1792 if (!NT_SUCCESS(Status
))
1794 TRACE("failed with status 0x%08lx\n", Status
);
1798 TRACE("RID: %lx\n", ulRid
);
1800 /* Convert the RID into a string (hex) */
1801 swprintf(szRid
, L
"%08lX", ulRid
);
1803 /* Create the group object */
1804 Status
= SampCreateDbObject(DomainObject
,
1811 if (!NT_SUCCESS(Status
))
1813 TRACE("failed with status 0x%08lx\n", Status
);
1817 /* Add the account name of the user object */
1818 Status
= SampSetAccountNameInDomain(DomainObject
,
1822 if (!NT_SUCCESS(Status
))
1824 TRACE("failed with status 0x%08lx\n", Status
);
1828 /* Initialize fixed user data */
1829 memset(&FixedGroupData
, 0, sizeof(SAM_GROUP_FIXED_DATA
));
1830 FixedGroupData
.Version
= 1;
1832 FixedGroupData
.GroupId
= ulRid
;
1834 /* Set fixed user data attribute */
1835 Status
= SampSetObjectAttribute(GroupObject
,
1838 (LPVOID
)&FixedGroupData
,
1839 sizeof(SAM_GROUP_FIXED_DATA
));
1840 if (!NT_SUCCESS(Status
))
1842 TRACE("failed with status 0x%08lx\n", Status
);
1846 /* Set the Name attribute */
1847 Status
= SampSetObjectAttribute(GroupObject
,
1850 (LPVOID
)Name
->Buffer
,
1851 Name
->MaximumLength
);
1852 if (!NT_SUCCESS(Status
))
1854 TRACE("failed with status 0x%08lx\n", Status
);
1858 /* Set the AdminComment attribute */
1859 Status
= SampSetObjectAttribute(GroupObject
,
1863 EmptyString
.MaximumLength
);
1864 if (!NT_SUCCESS(Status
))
1866 TRACE("failed with status 0x%08lx\n", Status
);
1870 if (NT_SUCCESS(Status
))
1872 *GroupHandle
= (SAMPR_HANDLE
)GroupObject
;
1873 *RelativeId
= ulRid
;
1876 TRACE("returns with status 0x%08lx\n", Status
);
1885 SamrEnumerateGroupsInDomain(IN SAMPR_HANDLE DomainHandle
,
1886 IN OUT
unsigned long *EnumerationContext
,
1887 OUT PSAMPR_ENUMERATION_BUFFER
*Buffer
,
1888 IN
unsigned long PreferedMaximumLength
,
1889 OUT
unsigned long *CountReturned
)
1891 PSAMPR_ENUMERATION_BUFFER EnumBuffer
= NULL
;
1892 PSAM_DB_OBJECT DomainObject
;
1893 HANDLE GroupsKeyHandle
= NULL
;
1894 HANDLE NamesKeyHandle
= NULL
;
1895 WCHAR GroupName
[64];
1897 ULONG EnumCount
= 0;
1898 ULONG RequiredLength
= 0;
1903 BOOLEAN MoreEntries
= FALSE
;
1906 TRACE("SamrEnumerateUsersInDomain(%p %p %p %lu %p)\n",
1907 DomainHandle
, EnumerationContext
, Buffer
,
1908 PreferedMaximumLength
, CountReturned
);
1910 /* Validate the domain handle */
1911 Status
= SampValidateDbObject(DomainHandle
,
1913 DOMAIN_LIST_ACCOUNTS
,
1915 if (!NT_SUCCESS(Status
))
1918 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
1922 if (!NT_SUCCESS(Status
))
1925 Status
= SampRegOpenKey(GroupsKeyHandle
,
1929 if (!NT_SUCCESS(Status
))
1934 EnumIndex
= *EnumerationContext
;
1938 NameLength
= 64 * sizeof(WCHAR
);
1939 Status
= SampRegEnumerateValue(NamesKeyHandle
,
1946 if (!NT_SUCCESS(Status
))
1948 if (Status
== STATUS_NO_MORE_ENTRIES
)
1949 Status
= STATUS_SUCCESS
;
1953 TRACE("EnumIndex: %lu\n", EnumIndex
);
1954 TRACE("Group name: %S\n", GroupName
);
1955 TRACE("Name length: %lu\n", NameLength
);
1957 if ((RequiredLength
+ NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
)) > PreferedMaximumLength
)
1963 RequiredLength
+= (NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
));
1969 TRACE("EnumCount: %lu\n", EnumCount
);
1970 TRACE("RequiredLength: %lu\n", RequiredLength
);
1972 if (!NT_SUCCESS(Status
))
1975 EnumBuffer
= midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER
));
1976 if (EnumBuffer
== NULL
)
1978 Status
= STATUS_INSUFFICIENT_RESOURCES
;
1982 EnumBuffer
->EntriesRead
= EnumCount
;
1986 EnumBuffer
->Buffer
= midl_user_allocate(EnumCount
* sizeof(SAMPR_RID_ENUMERATION
));
1987 if (EnumBuffer
->Buffer
== NULL
)
1989 Status
= STATUS_INSUFFICIENT_RESOURCES
;
1995 EnumIndex
= *EnumerationContext
;
1996 for (i
= 0; i
< EnumCount
; i
++, EnumIndex
++)
1998 NameLength
= 64 * sizeof(WCHAR
);
1999 DataLength
= sizeof(ULONG
);
2000 Status
= SampRegEnumerateValue(NamesKeyHandle
,
2007 if (!NT_SUCCESS(Status
))
2009 if (Status
== STATUS_NO_MORE_ENTRIES
)
2010 Status
= STATUS_SUCCESS
;
2014 TRACE("EnumIndex: %lu\n", EnumIndex
);
2015 TRACE("Group name: %S\n", GroupName
);
2016 TRACE("Name length: %lu\n", NameLength
);
2017 TRACE("RID: %lu\n", Rid
);
2019 EnumBuffer
->Buffer
[i
].RelativeId
= Rid
;
2021 EnumBuffer
->Buffer
[i
].Name
.Length
= (USHORT
)NameLength
;
2022 EnumBuffer
->Buffer
[i
].Name
.MaximumLength
= (USHORT
)(DataLength
+ sizeof(UNICODE_NULL
));
2024 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
2026 EnumBuffer
->Buffer
[i
].Name
.Buffer
= midl_user_allocate(EnumBuffer
->Buffer
[i
].Name
.MaximumLength
);
2027 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
== NULL
)
2029 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2033 memcpy(EnumBuffer
->Buffer
[i
].Name
.Buffer
,
2035 EnumBuffer
->Buffer
[i
].Name
.Length
);
2040 if (NT_SUCCESS(Status
))
2042 *EnumerationContext
+= EnumCount
;
2043 *Buffer
= EnumBuffer
;
2044 *CountReturned
= EnumCount
;
2048 *EnumerationContext
= 0;
2052 if (EnumBuffer
!= NULL
)
2054 if (EnumBuffer
->Buffer
!= NULL
)
2056 if (EnumBuffer
->EntriesRead
!= 0)
2058 for (i
= 0; i
< EnumBuffer
->EntriesRead
; i
++)
2060 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
!= NULL
)
2061 midl_user_free(EnumBuffer
->Buffer
[i
].Name
.Buffer
);
2065 midl_user_free(EnumBuffer
->Buffer
);
2068 midl_user_free(EnumBuffer
);
2072 if (NamesKeyHandle
!= NULL
)
2073 SampRegCloseKey(NamesKeyHandle
);
2075 if (GroupsKeyHandle
!= NULL
)
2076 SampRegCloseKey(GroupsKeyHandle
);
2078 if ((Status
== STATUS_SUCCESS
) && (MoreEntries
== TRUE
))
2079 Status
= STATUS_MORE_ENTRIES
;
2088 SamrCreateUserInDomain(IN SAMPR_HANDLE DomainHandle
,
2089 IN PRPC_UNICODE_STRING Name
,
2090 IN ACCESS_MASK DesiredAccess
,
2091 OUT SAMPR_HANDLE
*UserHandle
,
2092 OUT
unsigned long *RelativeId
)
2094 UNICODE_STRING EmptyString
= RTL_CONSTANT_STRING(L
"");
2095 SAM_DOMAIN_FIXED_DATA FixedDomainData
;
2096 SAM_USER_FIXED_DATA FixedUserData
;
2097 PSAM_DB_OBJECT DomainObject
;
2098 PSAM_DB_OBJECT UserObject
;
2099 GROUP_MEMBERSHIP GroupMembership
;
2100 UCHAR LogonHours
[23];
2106 TRACE("SamrCreateUserInDomain(%p %p %lx %p %p)\n",
2107 DomainHandle
, Name
, DesiredAccess
, UserHandle
, RelativeId
);
2110 Name
->Length
== 0 ||
2111 Name
->Buffer
== NULL
||
2112 UserHandle
== NULL
||
2114 return STATUS_INVALID_PARAMETER
;
2116 /* Map generic access rights */
2117 RtlMapGenericMask(&DesiredAccess
,
2120 /* Validate the domain handle */
2121 Status
= SampValidateDbObject(DomainHandle
,
2125 if (!NT_SUCCESS(Status
))
2127 TRACE("failed with status 0x%08lx\n", Status
);
2131 /* Check if the user name already exists in the domain */
2132 Status
= SampCheckAccountNameInDomain(DomainObject
,
2134 if (!NT_SUCCESS(Status
))
2136 TRACE("User name \'%S\' already exists in domain (Status 0x%08lx)\n",
2137 Name
->Buffer
, Status
);
2141 /* Get the fixed domain attributes */
2142 ulSize
= sizeof(SAM_DOMAIN_FIXED_DATA
);
2143 Status
= SampGetObjectAttribute(DomainObject
,
2146 (PVOID
)&FixedDomainData
,
2148 if (!NT_SUCCESS(Status
))
2150 TRACE("failed with status 0x%08lx\n", Status
);
2154 /* Increment the NextRid attribute */
2155 ulRid
= FixedDomainData
.NextRid
;
2156 FixedDomainData
.NextRid
++;
2158 /* Store the fixed domain attributes */
2159 Status
= SampSetObjectAttribute(DomainObject
,
2164 if (!NT_SUCCESS(Status
))
2166 TRACE("failed with status 0x%08lx\n", Status
);
2170 TRACE("RID: %lx\n", ulRid
);
2172 /* Convert the RID into a string (hex) */
2173 swprintf(szRid
, L
"%08lX", ulRid
);
2175 /* Create the user object */
2176 Status
= SampCreateDbObject(DomainObject
,
2183 if (!NT_SUCCESS(Status
))
2185 TRACE("failed with status 0x%08lx\n", Status
);
2189 /* Add the account name for the user object */
2190 Status
= SampSetAccountNameInDomain(DomainObject
,
2194 if (!NT_SUCCESS(Status
))
2196 TRACE("failed with status 0x%08lx\n", Status
);
2200 /* Initialize fixed user data */
2201 memset(&FixedUserData
, 0, sizeof(SAM_USER_FIXED_DATA
));
2202 FixedUserData
.Version
= 1;
2203 FixedUserData
.Reserved
= 0;
2204 FixedUserData
.LastLogon
.QuadPart
= 0;
2205 FixedUserData
.LastLogoff
.QuadPart
= 0;
2206 FixedUserData
.PasswordLastSet
.QuadPart
= 0;
2207 FixedUserData
.AccountExpires
.LowPart
= MAXULONG
;
2208 FixedUserData
.AccountExpires
.HighPart
= MAXLONG
;
2209 FixedUserData
.LastBadPasswordTime
.QuadPart
= 0;
2210 FixedUserData
.UserId
= ulRid
;
2211 FixedUserData
.PrimaryGroupId
= DOMAIN_GROUP_RID_USERS
;
2212 FixedUserData
.UserAccountControl
= USER_ACCOUNT_DISABLED
|
2213 USER_PASSWORD_NOT_REQUIRED
|
2214 USER_NORMAL_ACCOUNT
;
2215 FixedUserData
.CountryCode
= 0;
2216 FixedUserData
.CodePage
= 0;
2217 FixedUserData
.BadPasswordCount
= 0;
2218 FixedUserData
.LogonCount
= 0;
2219 FixedUserData
.AdminCount
= 0;
2220 FixedUserData
.OperatorCount
= 0;
2222 /* Set fixed user data attribute */
2223 Status
= SampSetObjectAttribute(UserObject
,
2226 (LPVOID
)&FixedUserData
,
2227 sizeof(SAM_USER_FIXED_DATA
));
2228 if (!NT_SUCCESS(Status
))
2230 TRACE("failed with status 0x%08lx\n", Status
);
2234 /* Set the Name attribute */
2235 Status
= SampSetObjectAttribute(UserObject
,
2238 (LPVOID
)Name
->Buffer
,
2239 Name
->MaximumLength
);
2240 if (!NT_SUCCESS(Status
))
2242 TRACE("failed with status 0x%08lx\n", Status
);
2246 /* Set the FullName attribute */
2247 Status
= SampSetObjectAttribute(UserObject
,
2251 EmptyString
.MaximumLength
);
2252 if (!NT_SUCCESS(Status
))
2254 TRACE("failed with status 0x%08lx\n", Status
);
2258 /* Set the HomeDirectory attribute */
2259 Status
= SampSetObjectAttribute(UserObject
,
2263 EmptyString
.MaximumLength
);
2264 if (!NT_SUCCESS(Status
))
2266 TRACE("failed with status 0x%08lx\n", Status
);
2270 /* Set the HomeDirectoryDrive attribute */
2271 Status
= SampSetObjectAttribute(UserObject
,
2272 L
"HomeDirectoryDrive",
2275 EmptyString
.MaximumLength
);
2276 if (!NT_SUCCESS(Status
))
2278 TRACE("failed with status 0x%08lx\n", Status
);
2282 /* Set the ScriptPath attribute */
2283 Status
= SampSetObjectAttribute(UserObject
,
2287 EmptyString
.MaximumLength
);
2288 if (!NT_SUCCESS(Status
))
2290 TRACE("failed with status 0x%08lx\n", Status
);
2294 /* Set the ProfilePath attribute */
2295 Status
= SampSetObjectAttribute(UserObject
,
2299 EmptyString
.MaximumLength
);
2300 if (!NT_SUCCESS(Status
))
2302 TRACE("failed with status 0x%08lx\n", Status
);
2306 /* Set the AdminComment attribute */
2307 Status
= SampSetObjectAttribute(UserObject
,
2311 EmptyString
.MaximumLength
);
2312 if (!NT_SUCCESS(Status
))
2314 TRACE("failed with status 0x%08lx\n", Status
);
2318 /* Set the UserComment attribute */
2319 Status
= SampSetObjectAttribute(UserObject
,
2323 EmptyString
.MaximumLength
);
2324 if (!NT_SUCCESS(Status
))
2326 TRACE("failed with status 0x%08lx\n", Status
);
2330 /* Set the WorkStations attribute */
2331 Status
= SampSetObjectAttribute(UserObject
,
2335 EmptyString
.MaximumLength
);
2336 if (!NT_SUCCESS(Status
))
2338 TRACE("failed with status 0x%08lx\n", Status
);
2342 /* Set the Parameters attribute */
2343 Status
= SampSetObjectAttribute(UserObject
,
2347 EmptyString
.MaximumLength
);
2348 if (!NT_SUCCESS(Status
))
2350 TRACE("failed with status 0x%08lx\n", Status
);
2354 /* Set LogonHours attribute*/
2355 *((PUSHORT
)LogonHours
) = 168;
2356 memset(&(LogonHours
[2]), 0xff, 21);
2358 Status
= SampSetObjectAttribute(UserObject
,
2362 sizeof(LogonHours
));
2363 if (!NT_SUCCESS(Status
))
2365 TRACE("failed with status 0x%08lx\n", Status
);
2369 /* Set Groups attribute*/
2370 GroupMembership
.RelativeId
= DOMAIN_GROUP_RID_USERS
;
2371 GroupMembership
.Attributes
= SE_GROUP_MANDATORY
|
2373 SE_GROUP_ENABLED_BY_DEFAULT
;
2375 Status
= SampSetObjectAttribute(UserObject
,
2379 sizeof(GROUP_MEMBERSHIP
));
2380 if (!NT_SUCCESS(Status
))
2382 TRACE("failed with status 0x%08lx\n", Status
);
2386 /* Set LMPwd attribute*/
2387 Status
= SampSetObjectAttribute(UserObject
,
2391 sizeof(ENCRYPTED_LM_OWF_PASSWORD
));
2392 if (!NT_SUCCESS(Status
))
2394 TRACE("failed with status 0x%08lx\n", Status
);
2398 /* Set NTPwd attribute*/
2399 Status
= SampSetObjectAttribute(UserObject
,
2403 sizeof(ENCRYPTED_NT_OWF_PASSWORD
));
2404 if (!NT_SUCCESS(Status
))
2406 TRACE("failed with status 0x%08lx\n", Status
);
2410 /* Set LMPwdHistory attribute*/
2411 Status
= SampSetObjectAttribute(UserObject
,
2416 if (!NT_SUCCESS(Status
))
2418 TRACE("failed with status 0x%08lx\n", Status
);
2422 /* Set NTPwdHistory attribute*/
2423 Status
= SampSetObjectAttribute(UserObject
,
2428 if (!NT_SUCCESS(Status
))
2430 TRACE("failed with status 0x%08lx\n", Status
);
2434 /* FIXME: Set SecDesc attribute*/
2436 if (NT_SUCCESS(Status
))
2438 *UserHandle
= (SAMPR_HANDLE
)UserObject
;
2439 *RelativeId
= ulRid
;
2442 TRACE("returns with status 0x%08lx\n", Status
);
2451 SamrEnumerateUsersInDomain(IN SAMPR_HANDLE DomainHandle
,
2452 IN OUT
unsigned long *EnumerationContext
,
2453 IN
unsigned long UserAccountControl
,
2454 OUT PSAMPR_ENUMERATION_BUFFER
*Buffer
,
2455 IN
unsigned long PreferedMaximumLength
,
2456 OUT
unsigned long *CountReturned
)
2458 PSAMPR_ENUMERATION_BUFFER EnumBuffer
= NULL
;
2459 PSAM_DB_OBJECT DomainObject
;
2460 HANDLE UsersKeyHandle
= NULL
;
2461 HANDLE NamesKeyHandle
= NULL
;
2464 ULONG EnumCount
= 0;
2465 ULONG RequiredLength
= 0;
2470 BOOLEAN MoreEntries
= FALSE
;
2473 TRACE("SamrEnumerateUsersInDomain(%p %p %lx %p %lu %p)\n",
2474 DomainHandle
, EnumerationContext
, UserAccountControl
, Buffer
,
2475 PreferedMaximumLength
, CountReturned
);
2477 /* Validate the domain handle */
2478 Status
= SampValidateDbObject(DomainHandle
,
2480 DOMAIN_LIST_ACCOUNTS
,
2482 if (!NT_SUCCESS(Status
))
2485 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
2489 if (!NT_SUCCESS(Status
))
2492 Status
= SampRegOpenKey(UsersKeyHandle
,
2496 if (!NT_SUCCESS(Status
))
2501 EnumIndex
= *EnumerationContext
;
2505 NameLength
= 64 * sizeof(WCHAR
);
2506 Status
= SampRegEnumerateValue(NamesKeyHandle
,
2513 if (!NT_SUCCESS(Status
))
2515 if (Status
== STATUS_NO_MORE_ENTRIES
)
2516 Status
= STATUS_SUCCESS
;
2520 TRACE("EnumIndex: %lu\n", EnumIndex
);
2521 TRACE("User name: %S\n", UserName
);
2522 TRACE("Name length: %lu\n", NameLength
);
2524 if ((RequiredLength
+ NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
)) > PreferedMaximumLength
)
2530 RequiredLength
+= (NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
));
2536 TRACE("EnumCount: %lu\n", EnumCount
);
2537 TRACE("RequiredLength: %lu\n", RequiredLength
);
2539 if (!NT_SUCCESS(Status
))
2542 EnumBuffer
= midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER
));
2543 if (EnumBuffer
== NULL
)
2545 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2549 EnumBuffer
->EntriesRead
= EnumCount
;
2553 EnumBuffer
->Buffer
= midl_user_allocate(EnumCount
* sizeof(SAMPR_RID_ENUMERATION
));
2554 if (EnumBuffer
->Buffer
== NULL
)
2556 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2562 EnumIndex
= *EnumerationContext
;
2563 for (i
= 0; i
< EnumCount
; i
++, EnumIndex
++)
2565 NameLength
= 64 * sizeof(WCHAR
);
2566 DataLength
= sizeof(ULONG
);
2567 Status
= SampRegEnumerateValue(NamesKeyHandle
,
2574 if (!NT_SUCCESS(Status
))
2576 if (Status
== STATUS_NO_MORE_ENTRIES
)
2577 Status
= STATUS_SUCCESS
;
2581 TRACE("EnumIndex: %lu\n", EnumIndex
);
2582 TRACE("User name: %S\n", UserName
);
2583 TRACE("Name length: %lu\n", NameLength
);
2584 TRACE("RID: %lu\n", Rid
);
2586 EnumBuffer
->Buffer
[i
].RelativeId
= Rid
;
2588 EnumBuffer
->Buffer
[i
].Name
.Length
= (USHORT
)NameLength
;
2589 EnumBuffer
->Buffer
[i
].Name
.MaximumLength
= (USHORT
)(DataLength
+ sizeof(UNICODE_NULL
));
2591 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
2593 EnumBuffer
->Buffer
[i
].Name
.Buffer
= midl_user_allocate(EnumBuffer
->Buffer
[i
].Name
.MaximumLength
);
2594 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
== NULL
)
2596 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2600 memcpy(EnumBuffer
->Buffer
[i
].Name
.Buffer
,
2602 EnumBuffer
->Buffer
[i
].Name
.Length
);
2607 if (NT_SUCCESS(Status
))
2609 *EnumerationContext
+= EnumCount
;
2610 *Buffer
= EnumBuffer
;
2611 *CountReturned
= EnumCount
;
2615 *EnumerationContext
= 0;
2619 if (EnumBuffer
!= NULL
)
2621 if (EnumBuffer
->Buffer
!= NULL
)
2623 if (EnumBuffer
->EntriesRead
!= 0)
2625 for (i
= 0; i
< EnumBuffer
->EntriesRead
; i
++)
2627 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
!= NULL
)
2628 midl_user_free(EnumBuffer
->Buffer
[i
].Name
.Buffer
);
2632 midl_user_free(EnumBuffer
->Buffer
);
2635 midl_user_free(EnumBuffer
);
2639 if (NamesKeyHandle
!= NULL
)
2640 SampRegCloseKey(NamesKeyHandle
);
2642 if (UsersKeyHandle
!= NULL
)
2643 SampRegCloseKey(UsersKeyHandle
);
2645 if ((Status
== STATUS_SUCCESS
) && (MoreEntries
== TRUE
))
2646 Status
= STATUS_MORE_ENTRIES
;
2655 SamrCreateAliasInDomain(IN SAMPR_HANDLE DomainHandle
,
2656 IN PRPC_UNICODE_STRING AccountName
,
2657 IN ACCESS_MASK DesiredAccess
,
2658 OUT SAMPR_HANDLE
*AliasHandle
,
2659 OUT
unsigned long *RelativeId
)
2661 SAM_DOMAIN_FIXED_DATA FixedDomainData
;
2662 PSAM_DB_OBJECT DomainObject
;
2663 PSAM_DB_OBJECT AliasObject
;
2664 UNICODE_STRING EmptyString
= RTL_CONSTANT_STRING(L
"");
2670 TRACE("SamrCreateAliasInDomain(%p %p %lx %p %p)\n",
2671 DomainHandle
, AccountName
, DesiredAccess
, AliasHandle
, RelativeId
);
2673 /* Map generic access rights */
2674 RtlMapGenericMask(&DesiredAccess
,
2677 /* Validate the domain handle */
2678 Status
= SampValidateDbObject(DomainHandle
,
2680 DOMAIN_CREATE_ALIAS
,
2682 if (!NT_SUCCESS(Status
))
2684 TRACE("failed with status 0x%08lx\n", Status
);
2688 /* Check if the alias name already exists in the domain */
2689 Status
= SampCheckAccountNameInDomain(DomainObject
,
2690 AccountName
->Buffer
);
2691 if (!NT_SUCCESS(Status
))
2693 TRACE("Alias name \'%S\' already exists in domain (Status 0x%08lx)\n",
2694 AccountName
->Buffer
, Status
);
2698 /* Get the fixed domain attributes */
2699 ulSize
= sizeof(SAM_DOMAIN_FIXED_DATA
);
2700 Status
= SampGetObjectAttribute(DomainObject
,
2703 (PVOID
)&FixedDomainData
,
2705 if (!NT_SUCCESS(Status
))
2707 TRACE("failed with status 0x%08lx\n", Status
);
2711 /* Increment the NextRid attribute */
2712 ulRid
= FixedDomainData
.NextRid
;
2713 FixedDomainData
.NextRid
++;
2715 /* Store the fixed domain attributes */
2716 Status
= SampSetObjectAttribute(DomainObject
,
2721 if (!NT_SUCCESS(Status
))
2723 TRACE("failed with status 0x%08lx\n", Status
);
2727 TRACE("RID: %lx\n", ulRid
);
2729 /* Convert the RID into a string (hex) */
2730 swprintf(szRid
, L
"%08lX", ulRid
);
2732 /* Create the alias object */
2733 Status
= SampCreateDbObject(DomainObject
,
2740 if (!NT_SUCCESS(Status
))
2742 TRACE("failed with status 0x%08lx\n", Status
);
2746 /* Add the account name for the alias object */
2747 Status
= SampSetAccountNameInDomain(DomainObject
,
2749 AccountName
->Buffer
,
2751 if (!NT_SUCCESS(Status
))
2753 TRACE("failed with status 0x%08lx\n", Status
);
2757 /* Set the Name attribute */
2758 Status
= SampSetObjectAttribute(AliasObject
,
2761 (LPVOID
)AccountName
->Buffer
,
2762 AccountName
->MaximumLength
);
2763 if (!NT_SUCCESS(Status
))
2765 TRACE("failed with status 0x%08lx\n", Status
);
2769 /* Set the Description attribute */
2770 Status
= SampSetObjectAttribute(AliasObject
,
2774 EmptyString
.MaximumLength
);
2775 if (!NT_SUCCESS(Status
))
2777 TRACE("failed with status 0x%08lx\n", Status
);
2781 if (NT_SUCCESS(Status
))
2783 *AliasHandle
= (SAMPR_HANDLE
)AliasObject
;
2784 *RelativeId
= ulRid
;
2787 TRACE("returns with status 0x%08lx\n", Status
);
2796 SamrEnumerateAliasesInDomain(IN SAMPR_HANDLE DomainHandle
,
2797 IN OUT
unsigned long *EnumerationContext
,
2798 OUT PSAMPR_ENUMERATION_BUFFER
*Buffer
,
2799 IN
unsigned long PreferedMaximumLength
,
2800 OUT
unsigned long *CountReturned
)
2802 PSAMPR_ENUMERATION_BUFFER EnumBuffer
= NULL
;
2803 PSAM_DB_OBJECT DomainObject
;
2804 HANDLE AliasesKeyHandle
= NULL
;
2805 HANDLE NamesKeyHandle
= NULL
;
2806 WCHAR AliasName
[64];
2808 ULONG EnumCount
= 0;
2809 ULONG RequiredLength
= 0;
2814 BOOLEAN MoreEntries
= FALSE
;
2817 TRACE("SamrEnumerateAliasesInDomain(%p %p %p %lu %p)\n",
2818 DomainHandle
, EnumerationContext
, Buffer
,
2819 PreferedMaximumLength
, CountReturned
);
2821 /* Validate the domain handle */
2822 Status
= SampValidateDbObject(DomainHandle
,
2824 DOMAIN_LIST_ACCOUNTS
,
2826 if (!NT_SUCCESS(Status
))
2829 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
2833 if (!NT_SUCCESS(Status
))
2836 Status
= SampRegOpenKey(AliasesKeyHandle
,
2840 if (!NT_SUCCESS(Status
))
2845 EnumIndex
= *EnumerationContext
;
2849 NameLength
= 64 * sizeof(WCHAR
);
2850 Status
= SampRegEnumerateValue(NamesKeyHandle
,
2857 if (!NT_SUCCESS(Status
))
2859 if (Status
== STATUS_NO_MORE_ENTRIES
)
2860 Status
= STATUS_SUCCESS
;
2864 TRACE("EnumIndex: %lu\n", EnumIndex
);
2865 TRACE("Alias name: %S\n", AliasName
);
2866 TRACE("Name length: %lu\n", NameLength
);
2868 if ((RequiredLength
+ NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
)) > PreferedMaximumLength
)
2874 RequiredLength
+= (NameLength
+ sizeof(UNICODE_NULL
) + sizeof(SAMPR_RID_ENUMERATION
));
2880 TRACE("EnumCount: %lu\n", EnumCount
);
2881 TRACE("RequiredLength: %lu\n", RequiredLength
);
2883 if (!NT_SUCCESS(Status
))
2886 EnumBuffer
= midl_user_allocate(sizeof(SAMPR_ENUMERATION_BUFFER
));
2887 if (EnumBuffer
== NULL
)
2889 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2893 EnumBuffer
->EntriesRead
= EnumCount
;
2897 EnumBuffer
->Buffer
= midl_user_allocate(EnumCount
* sizeof(SAMPR_RID_ENUMERATION
));
2898 if (EnumBuffer
->Buffer
== NULL
)
2900 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2906 EnumIndex
= *EnumerationContext
;
2907 for (i
= 0; i
< EnumCount
; i
++, EnumIndex
++)
2909 NameLength
= 64 * sizeof(WCHAR
);
2910 DataLength
= sizeof(ULONG
);
2911 Status
= SampRegEnumerateValue(NamesKeyHandle
,
2918 if (!NT_SUCCESS(Status
))
2920 if (Status
== STATUS_NO_MORE_ENTRIES
)
2921 Status
= STATUS_SUCCESS
;
2925 TRACE("EnumIndex: %lu\n", EnumIndex
);
2926 TRACE("Alias name: %S\n", AliasName
);
2927 TRACE("Name length: %lu\n", NameLength
);
2928 TRACE("RID: %lu\n", Rid
);
2930 EnumBuffer
->Buffer
[i
].RelativeId
= Rid
;
2932 EnumBuffer
->Buffer
[i
].Name
.Length
= (USHORT
)NameLength
;
2933 EnumBuffer
->Buffer
[i
].Name
.MaximumLength
= (USHORT
)(DataLength
+ sizeof(UNICODE_NULL
));
2935 /* FIXME: Disabled because of bugs in widl and rpcrt4 */
2937 EnumBuffer
->Buffer
[i
].Name
.Buffer
= midl_user_allocate(EnumBuffer
->Buffer
[i
].Name
.MaximumLength
);
2938 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
== NULL
)
2940 Status
= STATUS_INSUFFICIENT_RESOURCES
;
2944 memcpy(EnumBuffer
->Buffer
[i
].Name
.Buffer
,
2946 EnumBuffer
->Buffer
[i
].Name
.Length
);
2951 if (NT_SUCCESS(Status
))
2953 *EnumerationContext
+= EnumCount
;
2954 *Buffer
= EnumBuffer
;
2955 *CountReturned
= EnumCount
;
2959 *EnumerationContext
= 0;
2963 if (EnumBuffer
!= NULL
)
2965 if (EnumBuffer
->Buffer
!= NULL
)
2967 if (EnumBuffer
->EntriesRead
!= 0)
2969 for (i
= 0; i
< EnumBuffer
->EntriesRead
; i
++)
2971 if (EnumBuffer
->Buffer
[i
].Name
.Buffer
!= NULL
)
2972 midl_user_free(EnumBuffer
->Buffer
[i
].Name
.Buffer
);
2976 midl_user_free(EnumBuffer
->Buffer
);
2979 midl_user_free(EnumBuffer
);
2983 if (NamesKeyHandle
!= NULL
)
2984 SampRegCloseKey(NamesKeyHandle
);
2986 if (AliasesKeyHandle
!= NULL
)
2987 SampRegCloseKey(AliasesKeyHandle
);
2989 if ((Status
== STATUS_SUCCESS
) && (MoreEntries
== TRUE
))
2990 Status
= STATUS_MORE_ENTRIES
;
2999 SamrGetAliasMembership(IN SAMPR_HANDLE DomainHandle
,
3000 IN PSAMPR_PSID_ARRAY SidArray
,
3001 OUT PSAMPR_ULONG_ARRAY Membership
)
3003 PSAM_DB_OBJECT DomainObject
;
3004 HANDLE AliasesKeyHandle
= NULL
;
3005 HANDLE MembersKeyHandle
= NULL
;
3006 HANDLE MemberKeyHandle
= NULL
;
3007 LPWSTR MemberSidString
= NULL
;
3008 PULONG RidArray
= NULL
;
3009 ULONG MaxSidCount
= 0;
3014 WCHAR NameBuffer
[9];
3016 TRACE("SamrGetAliasMembership(%p %p %p)\n",
3017 DomainHandle
, SidArray
, Membership
);
3019 /* Validate the domain handle */
3020 Status
= SampValidateDbObject(DomainHandle
,
3022 DOMAIN_GET_ALIAS_MEMBERSHIP
,
3024 if (!NT_SUCCESS(Status
))
3027 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3031 TRACE("SampRegOpenKey returned %08lX\n", Status
);
3032 if (!NT_SUCCESS(Status
))
3035 Status
= SampRegOpenKey(AliasesKeyHandle
,
3039 TRACE("SampRegOpenKey returned %08lX\n", Status
);
3041 if (Status
== STATUS_OBJECT_NAME_NOT_FOUND
)
3043 Status
= STATUS_SUCCESS
;
3047 if (!NT_SUCCESS(Status
))
3050 for (i
= 0; i
< SidArray
->Count
; i
++)
3052 ConvertSidToStringSid(SidArray
->Sids
[i
].SidPointer
, &MemberSidString
);
3053 TRACE("Open %S\n", MemberSidString
);
3055 Status
= SampRegOpenKey(MembersKeyHandle
,
3059 TRACE("SampRegOpenKey returned %08lX\n", Status
);
3060 if (NT_SUCCESS(Status
))
3062 Status
= SampRegQueryKeyInfo(MemberKeyHandle
,
3065 if (NT_SUCCESS(Status
))
3067 TRACE("Found %lu values\n", ValueCount
);
3068 MaxSidCount
+= ValueCount
;
3071 NtClose(MemberKeyHandle
);
3074 if (Status
== STATUS_OBJECT_NAME_NOT_FOUND
)
3075 Status
= STATUS_SUCCESS
;
3077 LocalFree(MemberSidString
);
3080 if (MaxSidCount
== 0)
3082 Status
= STATUS_SUCCESS
;
3086 TRACE("Maximum sid count: %lu\n", MaxSidCount
);
3087 RidArray
= midl_user_allocate(MaxSidCount
* sizeof(ULONG
));
3088 if (RidArray
== NULL
)
3090 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3094 for (i
= 0; i
< SidArray
->Count
; i
++)
3096 ConvertSidToStringSid(SidArray
->Sids
[i
].SidPointer
, &MemberSidString
);
3097 TRACE("Open %S\n", MemberSidString
);
3099 Status
= SampRegOpenKey(MembersKeyHandle
,
3103 TRACE("SampRegOpenKey returned %08lX\n", Status
);
3104 if (NT_SUCCESS(Status
))
3106 Status
= SampRegQueryKeyInfo(MemberKeyHandle
,
3109 if (NT_SUCCESS(Status
))
3111 TRACE("Found %lu values\n", ValueCount
);
3113 for (j
= 0; j
< ValueCount
; j
++)
3115 DataLength
= 9 * sizeof(WCHAR
);
3116 Status
= SampRegEnumerateValue(MemberKeyHandle
,
3123 if (NT_SUCCESS(Status
))
3125 RidArray
[j
] = wcstoul(NameBuffer
, NULL
, 16);
3130 NtClose(MemberKeyHandle
);
3133 LocalFree(MemberSidString
);
3137 if (NT_SUCCESS(Status
))
3139 Membership
->Count
= MaxSidCount
;
3140 Membership
->Element
= RidArray
;
3144 if (RidArray
!= NULL
)
3145 midl_user_free(RidArray
);
3148 if (MembersKeyHandle
!= NULL
)
3149 NtClose(MembersKeyHandle
);
3151 if (MembersKeyHandle
!= NULL
)
3152 NtClose(MembersKeyHandle
);
3154 if (AliasesKeyHandle
!= NULL
)
3155 NtClose(AliasesKeyHandle
);
3164 SamrLookupNamesInDomain(IN SAMPR_HANDLE DomainHandle
,
3166 IN RPC_UNICODE_STRING Names
[],
3167 OUT PSAMPR_ULONG_ARRAY RelativeIds
,
3168 OUT PSAMPR_ULONG_ARRAY Use
)
3170 PSAM_DB_OBJECT DomainObject
;
3171 HANDLE AccountsKeyHandle
;
3172 HANDLE NamesKeyHandle
;
3173 ULONG MappedCount
= 0;
3179 TRACE("SamrLookupNamesInDomain(%p %lu %p %p %p)\n",
3180 DomainHandle
, Count
, Names
, RelativeIds
, Use
);
3182 /* Validate the domain handle */
3183 Status
= SampValidateDbObject(DomainHandle
,
3187 if (!NT_SUCCESS(Status
))
3189 TRACE("failed with status 0x%08lx\n", Status
);
3193 RelativeIds
->Count
= 0;
3197 return STATUS_SUCCESS
;
3199 /* Allocate the relative IDs array */
3200 RelativeIds
->Element
= midl_user_allocate(Count
* sizeof(ULONG
));
3201 if (RelativeIds
->Element
== NULL
)
3203 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3207 /* Allocate the use array */
3208 Use
->Element
= midl_user_allocate(Count
* sizeof(ULONG
));
3209 if (Use
->Element
== NULL
)
3211 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3215 RelativeIds
->Count
= Count
;
3218 for (i
= 0; i
< Count
; i
++)
3220 TRACE("Name: %S\n", Names
[i
].Buffer
);
3224 /* Lookup aliases */
3225 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3228 &AccountsKeyHandle
);
3229 if (NT_SUCCESS(Status
))
3231 Status
= SampRegOpenKey(AccountsKeyHandle
,
3235 if (NT_SUCCESS(Status
))
3237 DataLength
= sizeof(ULONG
);
3238 Status
= SampRegQueryValue(NamesKeyHandle
,
3244 SampRegCloseKey(NamesKeyHandle
);
3247 SampRegCloseKey(AccountsKeyHandle
);
3250 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3253 /* Return alias account */
3254 if (NT_SUCCESS(Status
) && RelativeId
!= 0)
3256 TRACE("Rid: %lu\n", RelativeId
);
3257 RelativeIds
->Element
[i
] = RelativeId
;
3258 Use
->Element
[i
] = SidTypeAlias
;
3264 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3267 &AccountsKeyHandle
);
3268 if (NT_SUCCESS(Status
))
3270 Status
= SampRegOpenKey(AccountsKeyHandle
,
3274 if (NT_SUCCESS(Status
))
3276 DataLength
= sizeof(ULONG
);
3277 Status
= SampRegQueryValue(NamesKeyHandle
,
3283 SampRegCloseKey(NamesKeyHandle
);
3286 SampRegCloseKey(AccountsKeyHandle
);
3289 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3292 /* Return group account */
3293 if (NT_SUCCESS(Status
) && RelativeId
!= 0)
3295 TRACE("Rid: %lu\n", RelativeId
);
3296 RelativeIds
->Element
[i
] = RelativeId
;
3297 Use
->Element
[i
] = SidTypeGroup
;
3303 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3306 &AccountsKeyHandle
);
3307 if (NT_SUCCESS(Status
))
3309 Status
= SampRegOpenKey(AccountsKeyHandle
,
3313 if (NT_SUCCESS(Status
))
3315 DataLength
= sizeof(ULONG
);
3316 Status
= SampRegQueryValue(NamesKeyHandle
,
3322 SampRegCloseKey(NamesKeyHandle
);
3325 SampRegCloseKey(AccountsKeyHandle
);
3328 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3331 /* Return user account */
3332 if (NT_SUCCESS(Status
) && RelativeId
!= 0)
3334 TRACE("Rid: %lu\n", RelativeId
);
3335 RelativeIds
->Element
[i
] = RelativeId
;
3336 Use
->Element
[i
] = SidTypeUser
;
3341 /* Return unknown account */
3342 RelativeIds
->Element
[i
] = 0;
3343 Use
->Element
[i
] = SidTypeUnknown
;
3347 if (Status
== STATUS_OBJECT_NAME_NOT_FOUND
)
3348 Status
= STATUS_SUCCESS
;
3350 if (NT_SUCCESS(Status
))
3352 if (MappedCount
== 0)
3353 Status
= STATUS_NONE_MAPPED
;
3354 else if (MappedCount
< Count
)
3355 Status
= STATUS_SOME_NOT_MAPPED
;
3359 if (RelativeIds
->Element
!= NULL
)
3361 midl_user_free(RelativeIds
->Element
);
3362 RelativeIds
->Element
= NULL
;
3365 RelativeIds
->Count
= 0;
3367 if (Use
->Element
!= NULL
)
3369 midl_user_free(Use
->Element
);
3370 Use
->Element
= NULL
;
3376 TRACE("Returned Status %lx\n", Status
);
3385 SamrLookupIdsInDomain(IN SAMPR_HANDLE DomainHandle
,
3387 IN ULONG
*RelativeIds
,
3388 OUT PSAMPR_RETURNED_USTRING_ARRAY Names
,
3389 OUT PSAMPR_ULONG_ARRAY Use
)
3391 PSAM_DB_OBJECT DomainObject
;
3393 HANDLE AccountsKeyHandle
;
3394 HANDLE AccountKeyHandle
;
3395 ULONG MappedCount
= 0;
3400 TRACE("SamrLookupIdsInDomain(%p %lu %p %p %p)\n",
3401 DomainHandle
, Count
, RelativeIds
, Names
, Use
);
3403 /* Validate the domain handle */
3404 Status
= SampValidateDbObject(DomainHandle
,
3408 if (!NT_SUCCESS(Status
))
3410 TRACE("failed with status 0x%08lx\n", Status
);
3418 return STATUS_SUCCESS
;
3420 /* Allocate the names array */
3421 Names
->Element
= midl_user_allocate(Count
* sizeof(ULONG
));
3422 if (Names
->Element
== NULL
)
3424 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3428 /* Allocate the use array */
3429 Use
->Element
= midl_user_allocate(Count
* sizeof(ULONG
));
3430 if (Use
->Element
== NULL
)
3432 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3436 Names
->Count
= Count
;
3439 for (i
= 0; i
< Count
; i
++)
3441 TRACE("RID: %lu\n", RelativeIds
[i
]);
3443 swprintf(RidString
, L
"%08lx", RelativeIds
[i
]);
3445 /* Lookup aliases */
3446 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3449 &AccountsKeyHandle
);
3450 if (NT_SUCCESS(Status
))
3452 Status
= SampRegOpenKey(AccountsKeyHandle
,
3456 if (NT_SUCCESS(Status
))
3459 Status
= SampRegQueryValue(AccountKeyHandle
,
3464 if (NT_SUCCESS(Status
))
3466 Names
->Element
[i
].Buffer
= midl_user_allocate(DataLength
);
3467 if (Names
->Element
[i
].Buffer
== NULL
)
3468 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3470 if (NT_SUCCESS(Status
))
3472 Names
->Element
[i
].MaximumLength
= (USHORT
)DataLength
;
3473 Names
->Element
[i
].Length
= (USHORT
)(DataLength
- sizeof(WCHAR
));
3475 Status
= SampRegQueryValue(AccountKeyHandle
,
3478 Names
->Element
[i
].Buffer
,
3483 SampRegCloseKey(AccountKeyHandle
);
3486 SampRegCloseKey(AccountsKeyHandle
);
3489 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3492 /* Return alias account */
3493 if (NT_SUCCESS(Status
) && Names
->Element
[i
].Buffer
!= NULL
)
3495 TRACE("Name: %S\n", Names
->Element
[i
].Buffer
);
3496 Use
->Element
[i
] = SidTypeAlias
;
3502 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3505 &AccountsKeyHandle
);
3506 if (NT_SUCCESS(Status
))
3508 Status
= SampRegOpenKey(AccountsKeyHandle
,
3512 if (NT_SUCCESS(Status
))
3515 Status
= SampRegQueryValue(AccountKeyHandle
,
3520 if (NT_SUCCESS(Status
))
3522 Names
->Element
[i
].Buffer
= midl_user_allocate(DataLength
);
3523 if (Names
->Element
[i
].Buffer
== NULL
)
3524 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3526 if (NT_SUCCESS(Status
))
3528 Names
->Element
[i
].MaximumLength
= (USHORT
)DataLength
;
3529 Names
->Element
[i
].Length
= (USHORT
)(DataLength
- sizeof(WCHAR
));
3531 Status
= SampRegQueryValue(AccountKeyHandle
,
3534 Names
->Element
[i
].Buffer
,
3539 SampRegCloseKey(AccountKeyHandle
);
3542 SampRegCloseKey(AccountsKeyHandle
);
3545 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3548 /* Return group account */
3549 if (NT_SUCCESS(Status
) && Names
->Element
[i
].Buffer
!= NULL
)
3551 TRACE("Name: %S\n", Names
->Element
[i
].Buffer
);
3552 Use
->Element
[i
] = SidTypeGroup
;
3558 Status
= SampRegOpenKey(DomainObject
->KeyHandle
,
3561 &AccountsKeyHandle
);
3562 if (NT_SUCCESS(Status
))
3564 Status
= SampRegOpenKey(AccountsKeyHandle
,
3568 if (NT_SUCCESS(Status
))
3571 Status
= SampRegQueryValue(AccountKeyHandle
,
3576 if (NT_SUCCESS(Status
))
3578 TRACE("DataLength: %lu\n", DataLength
);
3580 Names
->Element
[i
].Buffer
= midl_user_allocate(DataLength
);
3581 if (Names
->Element
[i
].Buffer
== NULL
)
3582 Status
= STATUS_INSUFFICIENT_RESOURCES
;
3584 if (NT_SUCCESS(Status
))
3586 Names
->Element
[i
].MaximumLength
= (USHORT
)DataLength
;
3587 Names
->Element
[i
].Length
= (USHORT
)(DataLength
- sizeof(WCHAR
));
3589 Status
= SampRegQueryValue(AccountKeyHandle
,
3592 Names
->Element
[i
].Buffer
,
3597 SampRegCloseKey(AccountKeyHandle
);
3600 SampRegCloseKey(AccountsKeyHandle
);
3603 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3606 /* Return user account */
3607 if (NT_SUCCESS(Status
) && Names
->Element
[i
].Buffer
!= NULL
)
3609 TRACE("Name: %S\n", Names
->Element
[i
].Buffer
);
3610 Use
->Element
[i
] = SidTypeUser
;
3615 /* Return unknown account */
3616 Use
->Element
[i
] = SidTypeUnknown
;
3620 if (Status
== STATUS_OBJECT_NAME_NOT_FOUND
)
3621 Status
= STATUS_SUCCESS
;
3623 if (NT_SUCCESS(Status
))
3625 if (MappedCount
== 0)
3626 Status
= STATUS_NONE_MAPPED
;
3627 else if (MappedCount
< Count
)
3628 Status
= STATUS_SOME_NOT_MAPPED
;
3632 if (Names
->Element
!= NULL
)
3634 for (i
= 0; i
< Count
; i
++)
3636 if (Names
->Element
[i
].Buffer
!= NULL
)
3637 midl_user_free(Names
->Element
[i
].Buffer
);
3640 midl_user_free(Names
->Element
);
3641 Names
->Element
= NULL
;
3646 if (Use
->Element
!= NULL
)
3648 midl_user_free(Use
->Element
);
3649 Use
->Element
= NULL
;
3662 SamrOpenGroup(IN SAMPR_HANDLE DomainHandle
,
3663 IN ACCESS_MASK DesiredAccess
,
3664 IN
unsigned long GroupId
,
3665 OUT SAMPR_HANDLE
*GroupHandle
)
3667 PSAM_DB_OBJECT DomainObject
;
3668 PSAM_DB_OBJECT GroupObject
;
3672 TRACE("SamrOpenGroup(%p %lx %lx %p)\n",
3673 DomainHandle
, DesiredAccess
, GroupId
, GroupHandle
);
3675 /* Map generic access rights */
3676 RtlMapGenericMask(&DesiredAccess
,
3679 /* Validate the domain handle */
3680 Status
= SampValidateDbObject(DomainHandle
,
3684 if (!NT_SUCCESS(Status
))
3686 TRACE("failed with status 0x%08lx\n", Status
);
3690 /* Convert the RID into a string (hex) */
3691 swprintf(szRid
, L
"%08lX", GroupId
);
3693 /* Create the group object */
3694 Status
= SampOpenDbObject(DomainObject
,
3701 if (!NT_SUCCESS(Status
))
3703 TRACE("failed with status 0x%08lx\n", Status
);
3707 *GroupHandle
= (SAMPR_HANDLE
)GroupObject
;
3709 return STATUS_SUCCESS
;
3714 SampQueryGroupGeneral(PSAM_DB_OBJECT GroupObject
,
3715 PSAMPR_GROUP_INFO_BUFFER
*Buffer
)
3717 PSAMPR_GROUP_INFO_BUFFER InfoBuffer
= NULL
;
3718 SAM_GROUP_FIXED_DATA FixedData
;
3719 ULONG MembersLength
= 0;
3725 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER
));
3726 if (InfoBuffer
== NULL
)
3727 return STATUS_INSUFFICIENT_RESOURCES
;
3729 Status
= SampGetObjectAttributeString(GroupObject
,
3731 &InfoBuffer
->General
.Name
);
3732 if (!NT_SUCCESS(Status
))
3734 TRACE("Status 0x%08lx\n", Status
);
3738 Status
= SampGetObjectAttributeString(GroupObject
,
3740 &InfoBuffer
->General
.AdminComment
);
3741 if (!NT_SUCCESS(Status
))
3743 TRACE("Status 0x%08lx\n", Status
);
3747 Length
= sizeof(SAM_GROUP_FIXED_DATA
);
3748 Status
= SampGetObjectAttribute(GroupObject
,
3753 if (!NT_SUCCESS(Status
))
3756 InfoBuffer
->General
.Attributes
= FixedData
.Attributes
;
3758 Status
= SampGetObjectAttribute(GroupObject
,
3763 if (!NT_SUCCESS(Status
) && Status
!= STATUS_OBJECT_NAME_NOT_FOUND
)
3766 if (Status
== STATUS_OBJECT_NAME_NOT_FOUND
)
3767 InfoBuffer
->General
.MemberCount
= 0;
3769 InfoBuffer
->General
.MemberCount
= MembersLength
/ sizeof(ULONG
);
3771 *Buffer
= InfoBuffer
;
3774 if (!NT_SUCCESS(Status
))
3776 if (InfoBuffer
!= NULL
)
3778 if (InfoBuffer
->General
.Name
.Buffer
!= NULL
)
3779 midl_user_free(InfoBuffer
->General
.Name
.Buffer
);
3781 if (InfoBuffer
->General
.AdminComment
.Buffer
!= NULL
)
3782 midl_user_free(InfoBuffer
->General
.AdminComment
.Buffer
);
3784 midl_user_free(InfoBuffer
);
3793 SampQueryGroupName(PSAM_DB_OBJECT GroupObject
,
3794 PSAMPR_GROUP_INFO_BUFFER
*Buffer
)
3796 PSAMPR_GROUP_INFO_BUFFER InfoBuffer
= NULL
;
3801 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER
));
3802 if (InfoBuffer
== NULL
)
3803 return STATUS_INSUFFICIENT_RESOURCES
;
3805 Status
= SampGetObjectAttributeString(GroupObject
,
3807 &InfoBuffer
->Name
.Name
);
3808 if (!NT_SUCCESS(Status
))
3810 TRACE("Status 0x%08lx\n", Status
);
3814 *Buffer
= InfoBuffer
;
3817 if (!NT_SUCCESS(Status
))
3819 if (InfoBuffer
!= NULL
)
3821 if (InfoBuffer
->Name
.Name
.Buffer
!= NULL
)
3822 midl_user_free(InfoBuffer
->Name
.Name
.Buffer
);
3824 midl_user_free(InfoBuffer
);
3833 SampQueryGroupAttribute(PSAM_DB_OBJECT GroupObject
,
3834 PSAMPR_GROUP_INFO_BUFFER
*Buffer
)
3836 PSAMPR_GROUP_INFO_BUFFER InfoBuffer
= NULL
;
3837 SAM_GROUP_FIXED_DATA FixedData
;
3843 InfoBuffer
= midl_user_allocate(sizeof(SAMPR_GROUP_INFO_BUFFER
));
3844 if (InfoBuffer
== NULL
)
3845 return STATUS_INSUFFICIENT_RESOURCES
;
3847 Length
= sizeof(SAM_GROUP_FIXED_DATA
);
3848 Status
= SampGetObjectAttribute(GroupObject
,