2 Unix SMB/CIFS implementation.
4 Winbind rpc backend functions
6 Copyright (C) Tim Potter 2000-2001,2003
7 Copyright (C) Andrew Tridgell 2001
8 Copyright (C) Volker Lendecke 2005
9 Copyright (C) Guenther Deschner 2008 (pidl conversion)
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; either version 3 of the License, or
14 (at your option) any later version.
16 This program is distributed in the hope that it will be useful,
17 but WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 GNU General Public License for more details.
21 You should have received a copy of the GNU General Public License
22 along with this program. If not, see <http://www.gnu.org/licenses/>.
27 #include "winbindd_rpc.h"
29 #include "../librpc/gen_ndr/ndr_samr_c.h"
30 #include "rpc_client/cli_pipe.h"
31 #include "rpc_client/cli_samr.h"
32 #include "rpc_client/cli_lsarpc.h"
33 #include "../libcli/security/security.h"
34 #include "libsmb/samlogon_cache.h"
37 #define DBGC_CLASS DBGC_WINBIND
39 static NTSTATUS
winbindd_lookup_names(TALLOC_CTX
*mem_ctx
,
40 struct winbindd_domain
*domain
,
43 const char ***domains
,
44 struct dom_sid
**sids
,
45 enum lsa_SidType
**types
);
47 /* Query display info for a domain. This returns enough information plus a
48 bit extra to give an overview of domain users for the User Manager
50 static NTSTATUS
msrpc_query_user_list(struct winbindd_domain
*domain
,
54 struct rpc_pipe_client
*samr_pipe
= NULL
;
55 struct policy_handle dom_pol
;
56 uint32_t *rids
= NULL
;
60 DEBUG(3, ("msrpc_query_user_list\n"));
62 tmp_ctx
= talloc_stackframe();
63 if (tmp_ctx
== NULL
) {
64 return NT_STATUS_NO_MEMORY
;
67 if ( !winbindd_can_contact_domain( domain
) ) {
68 DEBUG(10,("query_user_list: No incoming trust for domain %s\n",
70 status
= NT_STATUS_OK
;
74 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
75 if (!NT_STATUS_IS_OK(status
)) {
79 status
= rpc_query_user_list(tmp_ctx
,
84 if (!NT_STATUS_IS_OK(status
)) {
89 *prids
= talloc_move(mem_ctx
, &rids
);
98 /* list all domain groups */
99 static NTSTATUS
msrpc_enum_dom_groups(struct winbindd_domain
*domain
,
102 struct wb_acct_info
**pinfo
)
104 struct rpc_pipe_client
*samr_pipe
;
105 struct policy_handle dom_pol
;
106 struct wb_acct_info
*info
= NULL
;
107 uint32_t num_info
= 0;
111 DEBUG(3,("msrpc_enum_dom_groups\n"));
117 tmp_ctx
= talloc_stackframe();
118 if (tmp_ctx
== NULL
) {
119 return NT_STATUS_NO_MEMORY
;
122 if ( !winbindd_can_contact_domain( domain
) ) {
123 DEBUG(10,("enum_domain_groups: No incoming trust for domain %s\n",
125 status
= NT_STATUS_OK
;
129 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
130 if (!NT_STATUS_IS_OK(status
)) {
134 status
= rpc_enum_dom_groups(tmp_ctx
,
139 if (!NT_STATUS_IS_OK(status
)) {
144 *pnum_info
= num_info
;
148 *pinfo
= talloc_move(mem_ctx
, &info
);
152 TALLOC_FREE(tmp_ctx
);
156 /* List all domain groups */
158 static NTSTATUS
msrpc_enum_local_groups(struct winbindd_domain
*domain
,
161 struct wb_acct_info
**pinfo
)
163 struct rpc_pipe_client
*samr_pipe
;
164 struct policy_handle dom_pol
;
165 struct wb_acct_info
*info
= NULL
;
166 uint32_t num_info
= 0;
170 DEBUG(3,("msrpc_enum_local_groups\n"));
176 tmp_ctx
= talloc_stackframe();
177 if (tmp_ctx
== NULL
) {
178 return NT_STATUS_NO_MEMORY
;
181 if ( !winbindd_can_contact_domain( domain
) ) {
182 DEBUG(10,("enum_local_groups: No incoming trust for domain %s\n",
184 status
= NT_STATUS_OK
;
188 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
189 if (!NT_STATUS_IS_OK(status
)) {
193 status
= rpc_enum_local_groups(mem_ctx
,
198 if (!NT_STATUS_IS_OK(status
)) {
203 *pnum_info
= num_info
;
207 *pinfo
= talloc_move(mem_ctx
, &info
);
211 TALLOC_FREE(tmp_ctx
);
215 /* convert a single name to a sid in a domain */
216 static NTSTATUS
msrpc_name_to_sid(struct winbindd_domain
*domain
,
218 const char *domain_name
,
221 const char **pdom_name
,
223 enum lsa_SidType
*type
)
226 struct dom_sid
*sids
= NULL
;
227 enum lsa_SidType
*types
= NULL
;
228 char *full_name
= NULL
;
229 const char *names
[1] = { NULL
, };
230 const char **domains
= NULL
;
231 NTSTATUS name_map_status
= NT_STATUS_UNSUCCESSFUL
;
232 char *mapped_name
= NULL
;
234 if (name
== NULL
|| *name
=='\0') {
235 full_name
= talloc_asprintf(mem_ctx
, "%s", domain_name
);
236 } else if (domain_name
== NULL
|| *domain_name
== '\0') {
237 full_name
= talloc_asprintf(mem_ctx
, "%s", name
);
239 full_name
= talloc_asprintf(mem_ctx
, "%s\\%s", domain_name
, name
);
242 DEBUG(0, ("talloc_asprintf failed!\n"));
243 return NT_STATUS_NO_MEMORY
;
246 DEBUG(3, ("msrpc_name_to_sid: name=%s\n", full_name
));
248 name_map_status
= normalize_name_unmap(mem_ctx
, full_name
,
251 /* Reset the full_name pointer if we mapped anything */
253 if (NT_STATUS_IS_OK(name_map_status
) ||
254 NT_STATUS_EQUAL(name_map_status
, NT_STATUS_FILE_RENAMED
))
256 full_name
= mapped_name
;
259 DEBUG(3,("name_to_sid [rpc] %s for domain %s\n",
260 full_name
?full_name
:"", domain_name
));
262 names
[0] = full_name
;
264 result
= winbindd_lookup_names(mem_ctx
, domain
, 1,
267 if (!NT_STATUS_IS_OK(result
))
270 /* Return rid and type if lookup successful */
272 if (pdom_name
!= NULL
) {
273 const char *dom_name
= NULL
;
275 if (domains
[0] != NULL
) {
276 dom_name
= talloc_strdup(mem_ctx
, domains
[0]);
277 if (dom_name
== NULL
) {
278 return NT_STATUS_NO_MEMORY
;
281 *pdom_name
= dom_name
;
284 sid_copy(sid
, &sids
[0]);
291 convert a domain SID to a user or group name
293 static NTSTATUS
msrpc_sid_to_name(struct winbindd_domain
*domain
,
295 const struct dom_sid
*sid
,
298 enum lsa_SidType
*type
)
302 enum lsa_SidType
*types
= NULL
;
304 NTSTATUS name_map_status
= NT_STATUS_UNSUCCESSFUL
;
305 char *mapped_name
= NULL
;
306 struct dom_sid_buf buf
;
308 DEBUG(3, ("msrpc_sid_to_name: %s for domain %s\n",
309 dom_sid_str_buf(sid
, &buf
),
312 result
= winbindd_lookup_sids(mem_ctx
,
319 if (!NT_STATUS_IS_OK(result
)) {
320 DEBUG(2,("msrpc_sid_to_name: failed to lookup sids: %s\n",
326 *type
= (enum lsa_SidType
)types
[0];
327 *domain_name
= domains
[0];
330 DEBUG(5,("Mapped sid to [%s]\\[%s]\n", domains
[0], *name
));
332 name_map_status
= normalize_name_map(mem_ctx
, domain
->name
, *name
,
334 if (NT_STATUS_IS_OK(name_map_status
) ||
335 NT_STATUS_EQUAL(name_map_status
, NT_STATUS_FILE_RENAMED
))
338 DEBUG(5,("returning mapped name -- %s\n", *name
));
344 static NTSTATUS
msrpc_rids_to_names(struct winbindd_domain
*domain
,
346 const struct dom_sid
*sid
,
351 enum lsa_SidType
**types
)
355 struct dom_sid
*sids
;
359 DEBUG(3, ("msrpc_rids_to_names: domain %s\n", domain
->name
));
362 sids
= talloc_array(mem_ctx
, struct dom_sid
, num_rids
);
364 return NT_STATUS_NO_MEMORY
;
370 for (i
=0; i
<num_rids
; i
++) {
371 if (!sid_compose(&sids
[i
], sid
, rids
[i
])) {
372 return NT_STATUS_INTERNAL_ERROR
;
376 result
= winbindd_lookup_sids(mem_ctx
,
384 if (!NT_STATUS_IS_OK(result
) &&
385 !NT_STATUS_EQUAL(result
, STATUS_SOME_UNMAPPED
)) {
390 for (i
=0; i
<num_rids
; i
++) {
391 NTSTATUS name_map_status
= NT_STATUS_UNSUCCESSFUL
;
392 char *mapped_name
= NULL
;
394 if ((*types
)[i
] != SID_NAME_UNKNOWN
) {
395 name_map_status
= normalize_name_map(mem_ctx
,
399 if (NT_STATUS_IS_OK(name_map_status
) ||
400 NT_STATUS_EQUAL(name_map_status
, NT_STATUS_FILE_RENAMED
))
402 ret_names
[i
] = mapped_name
;
405 *domain_name
= domains
[i
];
412 /* Lookup groups a user is a member of. I wish Unix had a call like this! */
413 static NTSTATUS
msrpc_lookup_usergroups(struct winbindd_domain
*domain
,
415 const struct dom_sid
*user_sid
,
416 uint32_t *pnum_groups
,
417 struct dom_sid
**puser_grpsids
)
419 struct rpc_pipe_client
*samr_pipe
;
420 struct policy_handle dom_pol
;
421 struct dom_sid
*user_grpsids
= NULL
;
422 struct dom_sid_buf buf
;
423 uint32_t num_groups
= 0;
427 DEBUG(3,("msrpc_lookup_usergroups sid=%s\n",
428 dom_sid_str_buf(user_sid
, &buf
)));
432 tmp_ctx
= talloc_stackframe();
433 if (tmp_ctx
== NULL
) {
434 return NT_STATUS_NO_MEMORY
;
437 /* Check if we have a cached user_info_3 */
438 status
= lookup_usergroups_cached(tmp_ctx
,
442 if (NT_STATUS_IS_OK(status
)) {
446 if ( !winbindd_can_contact_domain( domain
) ) {
447 DEBUG(10,("lookup_usergroups: No incoming trust for domain %s\n",
450 /* Tell the cache manager not to remember this one */
451 status
= NT_STATUS_SYNCHRONIZATION_REQUIRED
;
455 /* no cache; hit the wire */
456 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
457 if (!NT_STATUS_IS_OK(status
)) {
461 status
= rpc_lookup_usergroups(tmp_ctx
,
468 if (!NT_STATUS_IS_OK(status
)) {
473 *pnum_groups
= num_groups
;
476 *puser_grpsids
= talloc_move(mem_ctx
, &user_grpsids
);
480 TALLOC_FREE(tmp_ctx
);
485 #define MAX_SAM_ENTRIES_W2K 0x400 /* 1024 */
487 static NTSTATUS
msrpc_lookup_useraliases(struct winbindd_domain
*domain
,
489 uint32_t num_sids
, const struct dom_sid
*sids
,
490 uint32_t *pnum_aliases
,
491 uint32_t **palias_rids
)
493 struct rpc_pipe_client
*samr_pipe
;
494 struct policy_handle dom_pol
;
495 uint32_t num_aliases
= 0;
496 uint32_t *alias_rids
= NULL
;
500 DEBUG(3,("msrpc_lookup_useraliases\n"));
506 tmp_ctx
= talloc_stackframe();
507 if (tmp_ctx
== NULL
) {
508 return NT_STATUS_NO_MEMORY
;
511 if (!winbindd_can_contact_domain(domain
)) {
512 DEBUG(10,("msrpc_lookup_useraliases: No incoming trust for domain %s\n",
514 /* Tell the cache manager not to remember this one */
515 status
= NT_STATUS_SYNCHRONIZATION_REQUIRED
;
519 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
520 if (!NT_STATUS_IS_OK(status
)) {
524 status
= rpc_lookup_useraliases(tmp_ctx
,
531 if (!NT_STATUS_IS_OK(status
)) {
536 *pnum_aliases
= num_aliases
;
540 *palias_rids
= talloc_move(mem_ctx
, &alias_rids
);
544 TALLOC_FREE(tmp_ctx
);
548 /* lookup alias membership */
549 static NTSTATUS
msrpc_lookup_aliasmem(struct winbindd_domain
*domain
,
551 const struct dom_sid
*alias_sid
,
552 enum lsa_SidType type
,
554 struct dom_sid
**sid_mem
)
556 struct rpc_pipe_client
*samr_pipe
= NULL
;
557 struct policy_handle dom_pol
;
558 struct dom_sid
*alias_members
= NULL
;
559 struct dom_sid_buf buf
;
560 uint32_t num_groups
= 0;
561 TALLOC_CTX
*tmp_ctx
= talloc_stackframe();
564 D_INFO("Lookup alias members in domain=%s for sid=%s.\n",
566 dom_sid_str_buf(alias_sid
, &buf
));
570 if (!winbindd_can_contact_domain(domain
)) {
571 D_DEBUG("No incoming trust for domain %s\n", domain
->name
);
572 status
= NT_STATUS_OK
;
576 status
= cm_connect_sam(domain
, tmp_ctx
, false, &samr_pipe
, &dom_pol
);
577 if (!NT_STATUS_IS_OK(status
)) {
581 status
= rpc_lookup_aliasmem(tmp_ctx
,
589 if (!NT_STATUS_IS_OK(status
)) {
593 *pnum_sids
= num_groups
;
595 *sid_mem
= talloc_move(mem_ctx
, &alias_members
);
599 talloc_free(tmp_ctx
);
603 /* Lookup group membership given a rid. */
604 static NTSTATUS
msrpc_lookup_groupmem(struct winbindd_domain
*domain
,
606 const struct dom_sid
*group_sid
,
607 enum lsa_SidType type
,
609 struct dom_sid
**sid_mem
,
611 uint32_t **name_types
)
613 NTSTATUS status
, result
;
614 uint32_t i
, total_names
= 0;
615 struct policy_handle dom_pol
, group_pol
;
616 uint32_t des_access
= SEC_FLAG_MAXIMUM_ALLOWED
;
617 uint32_t *rid_mem
= NULL
;
620 struct rpc_pipe_client
*cli
;
621 unsigned int orig_timeout
;
622 struct samr_RidAttrArray
*rids
= NULL
;
623 struct dcerpc_binding_handle
*b
;
624 struct dom_sid_buf buf
;
626 DEBUG(3,("msrpc_lookup_groupmem: %s sid=%s\n", domain
->name
,
627 dom_sid_str_buf(group_sid
, &buf
)));
629 if ( !winbindd_can_contact_domain( domain
) ) {
630 DEBUG(10,("lookup_groupmem: No incoming trust for domain %s\n",
635 if (!sid_peek_check_rid(&domain
->sid
, group_sid
, &group_rid
))
636 return NT_STATUS_UNSUCCESSFUL
;
640 result
= cm_connect_sam(domain
, mem_ctx
, false, &cli
, &dom_pol
);
641 if (!NT_STATUS_IS_OK(result
))
644 b
= cli
->binding_handle
;
646 status
= dcerpc_samr_OpenGroup(b
, mem_ctx
,
652 if (any_nt_status_not_ok(status
, result
, &status
)) {
656 /* Step #1: Get a list of user rids that are the members of the
659 /* This call can take a long time - allow the server to time out.
660 35 seconds should do it. */
662 orig_timeout
= rpccli_set_timeout(cli
, 35000);
664 status
= dcerpc_samr_QueryGroupMember(b
, mem_ctx
,
669 /* And restore our original timeout. */
670 rpccli_set_timeout(cli
, orig_timeout
);
674 dcerpc_samr_Close(b
, mem_ctx
, &group_pol
, &_result
);
677 if (any_nt_status_not_ok(status
, result
, &status
)) {
681 if (!rids
|| !rids
->count
) {
688 *num_names
= rids
->count
;
689 rid_mem
= rids
->rids
;
691 /* Step #2: Convert list of rids into list of usernames. Do this
692 in bunches of ~1000 to avoid crashing NT4. It looks like there
693 is a buffer overflow or something like that lurking around
696 #define MAX_LOOKUP_RIDS 900
698 *names
= talloc_zero_array(mem_ctx
, char *, *num_names
);
699 *name_types
= talloc_zero_array(mem_ctx
, uint32_t, *num_names
);
700 *sid_mem
= talloc_zero_array(mem_ctx
, struct dom_sid
, *num_names
);
702 for (j
=0;j
<(*num_names
);j
++)
703 sid_compose(&(*sid_mem
)[j
], &domain
->sid
, rid_mem
[j
]);
705 if (*num_names
>0 && (!*names
|| !*name_types
))
706 return NT_STATUS_NO_MEMORY
;
708 for (i
= 0; i
< *num_names
; i
+= MAX_LOOKUP_RIDS
) {
709 int num_lookup_rids
= MIN(*num_names
- i
, MAX_LOOKUP_RIDS
);
710 struct lsa_Strings tmp_names
;
711 struct samr_Ids tmp_types
;
713 /* Lookup a chunk of rids */
715 status
= dcerpc_samr_LookupRids(b
, mem_ctx
,
722 if (!NT_STATUS_IS_OK(status
)) {
726 /* see if we have a real error (and yes the
727 STATUS_SOME_UNMAPPED is the one returned from 2k) */
729 if (!NT_STATUS_IS_OK(result
) &&
730 !NT_STATUS_EQUAL(result
, STATUS_SOME_UNMAPPED
))
733 /* Copy result into array. The talloc system will take
734 care of freeing the temporary arrays later on. */
736 if (tmp_names
.count
!= num_lookup_rids
) {
737 return NT_STATUS_INVALID_NETWORK_RESPONSE
;
739 if (tmp_types
.count
!= num_lookup_rids
) {
740 return NT_STATUS_INVALID_NETWORK_RESPONSE
;
743 for (r
=0; r
<tmp_names
.count
; r
++) {
744 if (tmp_types
.ids
[r
] == SID_NAME_UNKNOWN
) {
747 if (total_names
>= *num_names
) {
750 (*names
)[total_names
] = fill_domain_username_talloc(
751 mem_ctx
, domain
->name
,
752 tmp_names
.names
[r
].string
, true);
753 (*name_types
)[total_names
] = tmp_types
.ids
[r
];
758 *num_names
= total_names
;
763 /* get a list of trusted domains */
764 static NTSTATUS
msrpc_trusted_domains(struct winbindd_domain
*domain
,
766 struct netr_DomainTrustList
*ptrust_list
)
768 struct rpc_pipe_client
*lsa_pipe
;
769 struct policy_handle lsa_policy
;
770 struct netr_DomainTrust
*trusts
= NULL
;
771 uint32_t num_trusts
= 0;
775 DEBUG(3,("msrpc_trusted_domains\n"));
778 ZERO_STRUCTP(ptrust_list
);
781 tmp_ctx
= talloc_stackframe();
782 if (tmp_ctx
== NULL
) {
783 return NT_STATUS_NO_MEMORY
;
786 status
= cm_connect_lsa(domain
, tmp_ctx
, &lsa_pipe
, &lsa_policy
);
787 if (!NT_STATUS_IS_OK(status
)) {
791 status
= rpc_trusted_domains(tmp_ctx
,
796 if (!NT_STATUS_IS_OK(status
)) {
801 ptrust_list
->count
= num_trusts
;
802 ptrust_list
->array
= talloc_move(mem_ctx
, &trusts
);
806 TALLOC_FREE(tmp_ctx
);
810 /* find the lockout policy for a domain */
811 static NTSTATUS
msrpc_lockout_policy(struct winbindd_domain
*domain
,
813 struct samr_DomInfo12
*lockout_policy
)
815 NTSTATUS status
, result
;
816 struct rpc_pipe_client
*cli
;
817 struct policy_handle dom_pol
;
818 union samr_DomainInfo
*info
= NULL
;
819 struct dcerpc_binding_handle
*b
;
821 DEBUG(3, ("msrpc_lockout_policy: fetch lockout policy for %s\n", domain
->name
));
823 if ( !winbindd_can_contact_domain( domain
) ) {
824 DEBUG(10,("msrpc_lockout_policy: No incoming trust for domain %s\n",
826 return NT_STATUS_NOT_SUPPORTED
;
829 status
= cm_connect_sam(domain
, mem_ctx
, false, &cli
, &dom_pol
);
830 if (!NT_STATUS_IS_OK(status
)) {
834 b
= cli
->binding_handle
;
836 status
= dcerpc_samr_QueryDomainInfo(b
, mem_ctx
,
838 DomainLockoutInformation
,
841 if (any_nt_status_not_ok(status
, result
, &status
)) {
845 *lockout_policy
= info
->info12
;
847 DEBUG(10,("msrpc_lockout_policy: lockout_threshold %d\n",
848 info
->info12
.lockout_threshold
));
855 /* find the password policy for a domain */
856 static NTSTATUS
msrpc_password_policy(struct winbindd_domain
*domain
,
858 struct samr_DomInfo1
*password_policy
)
860 NTSTATUS status
, result
;
861 struct rpc_pipe_client
*cli
;
862 struct policy_handle dom_pol
;
863 union samr_DomainInfo
*info
= NULL
;
864 struct dcerpc_binding_handle
*b
;
866 DEBUG(3, ("msrpc_password_policy: fetch password policy for %s\n",
869 if ( !winbindd_can_contact_domain( domain
) ) {
870 DEBUG(10,("msrpc_password_policy: No incoming trust for domain %s\n",
872 return NT_STATUS_NOT_SUPPORTED
;
875 status
= cm_connect_sam(domain
, mem_ctx
, false, &cli
, &dom_pol
);
876 if (!NT_STATUS_IS_OK(status
)) {
880 b
= cli
->binding_handle
;
882 status
= dcerpc_samr_QueryDomainInfo(b
, mem_ctx
,
884 DomainPasswordInformation
,
887 if (!NT_STATUS_IS_OK(status
)) {
890 if (!NT_STATUS_IS_OK(result
)) {
894 *password_policy
= info
->info1
;
896 DEBUG(10,("msrpc_password_policy: min_length_password %d\n",
897 info
->info1
.min_password_length
));
904 static enum lsa_LookupNamesLevel
winbindd_lookup_level(
905 struct winbindd_domain
*domain
)
907 enum lsa_LookupNamesLevel level
= LSA_LOOKUP_NAMES_DOMAINS_ONLY
;
909 if (domain
->internal
) {
910 level
= LSA_LOOKUP_NAMES_ALL
;
911 } else if (domain
->secure_channel_type
== SEC_CHAN_DNS_DOMAIN
) {
912 if (domain
->domain_flags
& NETR_TRUST_FLAG_IN_FOREST
) {
916 * Depending on what we want to resolve. We need to use:
917 * 1. LsapLookupXForestReferral(5)/LSA_LOOKUP_NAMES_FOREST_TRUSTS_ONLY
918 * if we want to pass the request into the direction of the forest
919 * root domain. The forest root domain uses
920 * LsapLookupXForestResolve(6)/LSA_LOOKUP_NAMES_UPLEVEL_TRUSTS_ONLY2
921 * when passing the request to trusted forests.
922 * 2. LsapLookupGC(4)/LSA_LOOKUP_NAMES_UPLEVEL_TRUSTS_ONLY
923 * if we're not a GC and want to resolve a name within our own forest.
925 * As we don't support more than one domain in our own forest
926 * and always try to be a GC for now, we just set
927 * LSA_LOOKUP_NAMES_FOREST_TRUSTS_ONLY.
929 level
= LSA_LOOKUP_NAMES_FOREST_TRUSTS_ONLY
;
930 } else if (domain
->domain_trust_attribs
& LSA_TRUST_ATTRIBUTE_FOREST_TRANSITIVE
) {
932 * This is LsapLookupXForestResolve(6)/LSA_LOOKUP_NAMES_UPLEVEL_TRUSTS_ONLY2
934 level
= LSA_LOOKUP_NAMES_UPLEVEL_TRUSTS_ONLY2
;
937 * This is LsapLookupTDL(3)/LSA_LOOKUP_NAMES_PRIMARY_DOMAIN_ONLY
939 level
= LSA_LOOKUP_NAMES_PRIMARY_DOMAIN_ONLY
;
941 } else if (domain
->secure_channel_type
== SEC_CHAN_DOMAIN
) {
943 * This is LsapLookupTDL(3)/LSA_LOOKUP_NAMES_PRIMARY_DOMAIN_ONLY
945 level
= LSA_LOOKUP_NAMES_PRIMARY_DOMAIN_ONLY
;
946 } else if (domain
->rodc
) {
947 level
= LSA_LOOKUP_NAMES_RODC_REFERRAL_TO_FULL_DC
;
950 * This is LsapLookupPDC(2)/LSA_LOOKUP_NAMES_DOMAINS_ONLY
952 level
= LSA_LOOKUP_NAMES_DOMAINS_ONLY
;
958 NTSTATUS
winbindd_lookup_sids(TALLOC_CTX
*mem_ctx
,
959 struct winbindd_domain
*domain
,
961 const struct dom_sid
*sids
,
964 enum lsa_SidType
**types
)
968 struct rpc_pipe_client
*cli
= NULL
;
969 struct dcerpc_binding_handle
*b
= NULL
;
970 struct policy_handle lsa_policy
;
971 unsigned int orig_timeout
;
972 bool use_lookupsids3
= false;
973 bool retried
= false;
974 enum lsa_LookupNamesLevel level
= LSA_LOOKUP_NAMES_ALL
;
977 status
= cm_connect_lsat(domain
, mem_ctx
, &cli
, &lsa_policy
);
978 if (!NT_STATUS_IS_OK(status
)) {
982 b
= cli
->binding_handle
;
984 if (cli
->transport
->transport
== NCACN_IP_TCP
) {
985 use_lookupsids3
= true;
988 level
= winbindd_lookup_level(domain
);
991 * This call can take a long time
992 * allow the server to time out.
993 * 35 seconds should do it.
995 orig_timeout
= dcerpc_binding_handle_set_timeout(b
, 35000);
997 status
= dcerpc_lsa_lookup_sids_generic(b
,
1009 /* And restore our original timeout. */
1010 dcerpc_binding_handle_set_timeout(b
, orig_timeout
);
1012 if (reset_cm_connection_on_error(domain
, b
, status
)) {
1014 * This can happen if the schannel key is not
1015 * valid anymore, we need to invalidate the
1016 * all connections to the dc and reestablish
1017 * a netlogon connection first.
1019 domain
->can_do_ncacn_ip_tcp
= domain
->active_directory
;
1024 status
= NT_STATUS_ACCESS_DENIED
;
1027 if (any_nt_status_not_ok(status
, result
, &status
)) {
1031 return NT_STATUS_OK
;
1034 static NTSTATUS
winbindd_lookup_names(TALLOC_CTX
*mem_ctx
,
1035 struct winbindd_domain
*domain
,
1038 const char ***domains
,
1039 struct dom_sid
**sids
,
1040 enum lsa_SidType
**types
)
1044 struct rpc_pipe_client
*cli
= NULL
;
1045 struct dcerpc_binding_handle
*b
= NULL
;
1046 struct policy_handle lsa_policy
;
1047 unsigned int orig_timeout
= 0;
1048 bool use_lookupnames4
= false;
1049 bool retried
= false;
1050 enum lsa_LookupNamesLevel level
= LSA_LOOKUP_NAMES_ALL
;
1053 if (domains
== NULL
) {
1059 status
= cm_connect_lsat(domain
, mem_ctx
, &cli
, &lsa_policy
);
1060 if (!NT_STATUS_IS_OK(status
)) {
1064 b
= cli
->binding_handle
;
1066 if (cli
->transport
->transport
== NCACN_IP_TCP
) {
1067 use_lookupnames4
= true;
1070 level
= winbindd_lookup_level(domain
);
1073 * This call can take a long time
1074 * allow the server to time out.
1075 * 35 seconds should do it.
1077 orig_timeout
= dcerpc_binding_handle_set_timeout(b
, 35000);
1079 status
= dcerpc_lsa_lookup_names_generic(b
,
1083 (const char **) names
,
1091 /* And restore our original timeout. */
1092 dcerpc_binding_handle_set_timeout(b
, orig_timeout
);
1094 if (reset_cm_connection_on_error(domain
, b
, status
)) {
1096 * This can happen if the schannel key is not
1097 * valid anymore, we need to invalidate the
1098 * all connections to the dc and reestablish
1099 * a netlogon connection first.
1105 status
= NT_STATUS_ACCESS_DENIED
;
1108 if (!NT_STATUS_IS_OK(status
)) {
1112 if (NT_STATUS_EQUAL(result
, NT_STATUS_NONE_MAPPED
)) {
1113 if (num_names
> 0) {
1116 *sids
= talloc_zero_array(mem_ctx
, struct dom_sid
, num_names
);
1117 if (*sids
== NULL
) {
1118 return NT_STATUS_NO_MEMORY
;
1121 *types
= talloc_zero_array(mem_ctx
, enum lsa_SidType
, num_names
);
1122 if (*types
== NULL
) {
1123 return NT_STATUS_NO_MEMORY
;
1126 for (i
= 0; i
< num_names
; i
++) {
1127 (*types
)[i
] = SID_NAME_UNKNOWN
;
1130 if (domains
!= NULL
) {
1131 *domains
= talloc_zero_array(mem_ctx
, const char *, num_names
);
1132 if (*domains
== NULL
) {
1133 return NT_STATUS_NO_MEMORY
;
1138 result
= NT_STATUS_OK
;
1139 } else if (NT_STATUS_EQUAL(result
, NT_STATUS_SOME_NOT_MAPPED
)) {
1140 if (talloc_array_length(*sids
) != num_names
) {
1141 return NT_STATUS_INVALID_NETWORK_RESPONSE
;
1143 if (talloc_array_length(*types
) != num_names
) {
1144 return NT_STATUS_INVALID_NETWORK_RESPONSE
;
1146 if (domains
!= NULL
) {
1147 if (talloc_array_length(*domains
) != num_names
) {
1148 return NT_STATUS_INVALID_NETWORK_RESPONSE
;
1151 result
= NT_STATUS_OK
;
1154 if (any_nt_status_not_ok(status
, result
, &status
)) {
1158 return NT_STATUS_OK
;
1161 /* the rpc backend methods are exposed via this structure */
1162 struct winbindd_methods msrpc_methods
= {
1164 msrpc_query_user_list
,
1165 msrpc_enum_dom_groups
,
1166 msrpc_enum_local_groups
,
1169 msrpc_rids_to_names
,
1170 msrpc_lookup_usergroups
,
1171 msrpc_lookup_useraliases
,
1172 msrpc_lookup_groupmem
,
1173 msrpc_lookup_aliasmem
,
1174 msrpc_lockout_policy
,
1175 msrpc_password_policy
,
1176 msrpc_trusted_domains
,