/* Unix SMB/CIFS implementation. In-Child server implementation of the routines defined in wbint.idl Copyright (C) Volker Lendecke 2009 Copyright (C) Guenther Deschner 2009 This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include "includes.h" #include "winbindd/winbindd.h" #include "winbindd/winbindd_proto.h" #include "rpc_client/cli_pipe.h" #include "ntdomain.h" #include "librpc/gen_ndr/srv_winbind.h" #include "../librpc/gen_ndr/ndr_netlogon_c.h" #include "idmap.h" #include "../libcli/security/security.h" #include "../libcli/auth/netlogon_creds_cli.h" void _wbint_Ping(struct pipes_struct *p, struct wbint_Ping *r) { *r->out.out_data = r->in.in_data; } static bool reset_cm_connection_on_error(struct winbindd_domain *domain, NTSTATUS status) { if (NT_STATUS_EQUAL(status, NT_STATUS_IO_TIMEOUT)) { invalidate_cm_connection(&domain->conn); /* We invalidated the connection. */ return true; } return false; } NTSTATUS _wbint_LookupSid(struct pipes_struct *p, struct wbint_LookupSid *r) { struct winbindd_domain *domain = wb_child_domain(); char *dom_name; char *name; enum lsa_SidType type; NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->sid_to_name(domain, p->mem_ctx, r->in.sid, &dom_name, &name, &type); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { return status; } *r->out.domain = dom_name; *r->out.name = name; *r->out.type = type; return NT_STATUS_OK; } NTSTATUS _wbint_LookupSids(struct pipes_struct *p, struct wbint_LookupSids *r) { struct winbindd_domain *domain = wb_child_domain(); struct lsa_RefDomainList *domains = r->out.domains; NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } /* * This breaks the winbindd_domain->methods abstraction: This * is only called for remote domains, and both winbindd_msrpc * and winbindd_ad call into lsa_lookupsids anyway. Caching is * done at the wbint RPC layer. */ status = rpc_lookup_sids(p->mem_ctx, domain, r->in.sids, &domains, &r->out.names); if (domains != NULL) { r->out.domains = domains; } reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_LookupName(struct pipes_struct *p, struct wbint_LookupName *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->name_to_sid( domain, p->mem_ctx, r->in.domain, r->in.name, r->in.flags, r->out.sid, r->out.type); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_Sids2UnixIDs(struct pipes_struct *p, struct wbint_Sids2UnixIDs *r) { uint32_t i, j; struct id_map *ids = NULL; struct id_map **id_ptrs = NULL; struct dom_sid *sids = NULL; uint32_t *id_idx = NULL; NTSTATUS status = NT_STATUS_NO_MEMORY; for (i=0; iin.domains->count; i++) { struct lsa_DomainInfo *d = &r->in.domains->domains[i]; struct idmap_domain *dom; uint32_t num_ids; dom = idmap_find_domain_with_sid(d->name.string, d->sid); if (dom == NULL) { DEBUG(10, ("idmap domain %s:%s not found\n", d->name.string, sid_string_dbg(d->sid))); continue; } num_ids = 0; for (j=0; jin.ids->num_ids; j++) { if (r->in.ids->ids[j].domain_index == i) { num_ids += 1; } } ids = talloc_realloc(talloc_tos(), ids, struct id_map, num_ids); if (ids == NULL) { goto nomem; } id_ptrs = talloc_realloc(talloc_tos(), id_ptrs, struct id_map *, num_ids+1); if (id_ptrs == NULL) { goto nomem; } id_idx = talloc_realloc(talloc_tos(), id_idx, uint32_t, num_ids); if (id_idx == NULL) { goto nomem; } sids = talloc_realloc(talloc_tos(), sids, struct dom_sid, num_ids); if (sids == NULL) { goto nomem; } num_ids = 0; /* * Convert the input data into a list of * id_map structs suitable for handing in * to the idmap sids_to_unixids method. */ for (j=0; jin.ids->num_ids; j++) { struct wbint_TransID *id = &r->in.ids->ids[j]; if (id->domain_index != i) { continue; } id_idx[num_ids] = j; id_ptrs[num_ids] = &ids[num_ids]; ids[num_ids].sid = &sids[num_ids]; sid_compose(ids[num_ids].sid, d->sid, id->rid); ids[num_ids].xid.type = id->type; ids[num_ids].status = ID_UNKNOWN; num_ids += 1; } id_ptrs[num_ids] = NULL; status = dom->methods->sids_to_unixids(dom, id_ptrs); DEBUG(10, ("sids_to_unixids returned %s\n", nt_errstr(status))); /* * Extract the results for handing them back to the caller. */ for (j=0; jin.ids->ids[id_idx[j]]; if (ids[j].status != ID_MAPPED) { id->xid.id = UINT32_MAX; id->xid.type = ID_TYPE_NOT_SPECIFIED; continue; } id->xid = ids[j].xid; } } status = NT_STATUS_OK; nomem: TALLOC_FREE(ids); TALLOC_FREE(id_ptrs); TALLOC_FREE(id_idx); TALLOC_FREE(sids); return status; } NTSTATUS _wbint_Uid2Sid(struct pipes_struct *p, struct wbint_Uid2Sid *r) { return idmap_uid_to_sid(r->in.dom_name ? r->in.dom_name : "", r->out.sid, r->in.uid); } NTSTATUS _wbint_Gid2Sid(struct pipes_struct *p, struct wbint_Gid2Sid *r) { return idmap_gid_to_sid(r->in.dom_name ? r->in.dom_name : "", r->out.sid, r->in.gid); } NTSTATUS _wbint_AllocateUid(struct pipes_struct *p, struct wbint_AllocateUid *r) { struct unixid xid; NTSTATUS status; status = idmap_allocate_uid(&xid); if (!NT_STATUS_IS_OK(status)) { return status; } *r->out.uid = xid.id; return NT_STATUS_OK; } NTSTATUS _wbint_AllocateGid(struct pipes_struct *p, struct wbint_AllocateGid *r) { struct unixid xid; NTSTATUS status; status = idmap_allocate_gid(&xid); if (!NT_STATUS_IS_OK(status)) { return status; } *r->out.gid = xid.id; return NT_STATUS_OK; } NTSTATUS _wbint_QueryUser(struct pipes_struct *p, struct wbint_QueryUser *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->query_user(domain, p->mem_ctx, r->in.sid, r->out.info); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_LookupUserAliases(struct pipes_struct *p, struct wbint_LookupUserAliases *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->lookup_useraliases( domain, p->mem_ctx, r->in.sids->num_sids, r->in.sids->sids, &r->out.rids->num_rids, &r->out.rids->rids); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_LookupUserGroups(struct pipes_struct *p, struct wbint_LookupUserGroups *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->lookup_usergroups( domain, p->mem_ctx, r->in.sid, &r->out.sids->num_sids, &r->out.sids->sids); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_QuerySequenceNumber(struct pipes_struct *p, struct wbint_QuerySequenceNumber *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->sequence_number(domain, r->out.sequence); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_LookupGroupMembers(struct pipes_struct *p, struct wbint_LookupGroupMembers *r) { struct winbindd_domain *domain = wb_child_domain(); uint32_t i, num_names; struct dom_sid *sid_mem; char **names; uint32_t *name_types; NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->lookup_groupmem( domain, p->mem_ctx, r->in.sid, r->in.type, &num_names, &sid_mem, &names, &name_types); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { return status; } r->out.members->num_principals = num_names; r->out.members->principals = talloc_array( r->out.members, struct wbint_Principal, num_names); if (r->out.members->principals == NULL) { return NT_STATUS_NO_MEMORY; } for (i=0; iout.members->principals[i]; sid_copy(&m->sid, &sid_mem[i]); m->name = talloc_move(r->out.members->principals, &names[i]); m->type = (enum lsa_SidType)name_types[i]; } return NT_STATUS_OK; } NTSTATUS _wbint_QueryUserList(struct pipes_struct *p, struct wbint_QueryUserList *r) { struct winbindd_domain *domain = wb_child_domain(); NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->query_user_list( domain, p->mem_ctx, &r->out.users->num_userinfos, &r->out.users->userinfos); reset_cm_connection_on_error(domain, status); return status; } NTSTATUS _wbint_QueryGroupList(struct pipes_struct *p, struct wbint_QueryGroupList *r) { struct winbindd_domain *domain = wb_child_domain(); uint32_t i, num_groups; struct wb_acct_info *groups; struct wbint_Principal *result; NTSTATUS status; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->enum_dom_groups(domain, talloc_tos(), &num_groups, &groups); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { return status; } result = talloc_array(r->out.groups, struct wbint_Principal, num_groups); if (result == NULL) { return NT_STATUS_NO_MEMORY; } for (i=0; isid, groups[i].rid); result[i].type = SID_NAME_DOM_GRP; result[i].name = talloc_strdup(result, groups[i].acct_name); if (result[i].name == NULL) { TALLOC_FREE(result); TALLOC_FREE(groups); return NT_STATUS_NO_MEMORY; } } r->out.groups->num_principals = num_groups; r->out.groups->principals = result; TALLOC_FREE(groups); return NT_STATUS_OK; } NTSTATUS _wbint_DsGetDcName(struct pipes_struct *p, struct wbint_DsGetDcName *r) { struct winbindd_domain *domain = wb_child_domain(); struct rpc_pipe_client *netlogon_pipe; struct netr_DsRGetDCNameInfo *dc_info; NTSTATUS status; WERROR werr; unsigned int orig_timeout; struct dcerpc_binding_handle *b; if (domain == NULL) { return dsgetdcname(p->mem_ctx, winbind_messaging_context(), r->in.domain_name, r->in.domain_guid, r->in.site_name ? r->in.site_name : "", r->in.flags, r->out.dc_info); } status = cm_connect_netlogon(domain, &netlogon_pipe); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { DEBUG(10, ("Can't contact the NETLOGON pipe\n")); return status; } b = netlogon_pipe->binding_handle; /* This call can take a long time - allow the server to time out. 35 seconds should do it. */ orig_timeout = rpccli_set_timeout(netlogon_pipe, 35000); if (domain->active_directory) { status = dcerpc_netr_DsRGetDCName(b, p->mem_ctx, domain->dcname, r->in.domain_name, NULL, r->in.domain_guid, r->in.flags, r->out.dc_info, &werr); if (NT_STATUS_IS_OK(status) && W_ERROR_IS_OK(werr)) { goto done; } if (reset_cm_connection_on_error(domain, status)) { /* Re-initialize. */ status = cm_connect_netlogon(domain, &netlogon_pipe); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { DEBUG(10, ("Can't contact the NETLOGON pipe\n")); return status; } b = netlogon_pipe->binding_handle; /* This call can take a long time - allow the server to time out. 35 seconds should do it. */ orig_timeout = rpccli_set_timeout(netlogon_pipe, 35000); } } /* * Fallback to less capable methods */ dc_info = talloc_zero(r->out.dc_info, struct netr_DsRGetDCNameInfo); if (dc_info == NULL) { status = NT_STATUS_NO_MEMORY; goto done; } if (r->in.flags & DS_PDC_REQUIRED) { status = dcerpc_netr_GetDcName(b, p->mem_ctx, domain->dcname, r->in.domain_name, &dc_info->dc_unc, &werr); } else { status = dcerpc_netr_GetAnyDCName(b, p->mem_ctx, domain->dcname, r->in.domain_name, &dc_info->dc_unc, &werr); } reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { DEBUG(10, ("dcerpc_netr_Get[Any]DCName failed: %s\n", nt_errstr(status))); goto done; } if (!W_ERROR_IS_OK(werr)) { DEBUG(10, ("dcerpc_netr_Get[Any]DCName failed: %s\n", win_errstr(werr))); status = werror_to_ntstatus(werr); goto done; } *r->out.dc_info = dc_info; status = NT_STATUS_OK; done: /* And restore our original timeout. */ rpccli_set_timeout(netlogon_pipe, orig_timeout); return status; } NTSTATUS _wbint_LookupRids(struct pipes_struct *p, struct wbint_LookupRids *r) { struct winbindd_domain *domain = wb_child_domain(); char *domain_name; char **names; enum lsa_SidType *types; struct wbint_Principal *result; NTSTATUS status; int i; if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = domain->methods->rids_to_names( domain, talloc_tos(), r->in.domain_sid, r->in.rids->rids, r->in.rids->num_rids, &domain_name, &names, &types); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { return status; } *r->out.domain_name = talloc_move(r->out.domain_name, &domain_name); result = talloc_array(p->mem_ctx, struct wbint_Principal, r->in.rids->num_rids); if (result == NULL) { return NT_STATUS_NO_MEMORY; } for (i=0; iin.rids->num_rids; i++) { sid_compose(&result[i].sid, r->in.domain_sid, r->in.rids->rids[i]); result[i].type = types[i]; result[i].name = talloc_move(result, &names[i]); } TALLOC_FREE(types); TALLOC_FREE(names); r->out.names->num_principals = r->in.rids->num_rids; r->out.names->principals = result; return NT_STATUS_OK; } NTSTATUS _wbint_CheckMachineAccount(struct pipes_struct *p, struct wbint_CheckMachineAccount *r) { struct winbindd_domain *domain; int num_retries = 0; NTSTATUS status; domain = wb_child_domain(); if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } again: invalidate_cm_connection(&domain->conn); domain->conn.netlogon_force_reauth = true; { struct rpc_pipe_client *netlogon_pipe; status = cm_connect_netlogon(domain, &netlogon_pipe); } /* There is a race condition between fetching the trust account password and the periodic machine password change. So it's possible that the trust account password has been changed on us. We are returned NT_STATUS_ACCESS_DENIED if this happens. */ #define MAX_RETRIES 3 if ((num_retries < MAX_RETRIES) && NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) { num_retries++; goto again; } if (!NT_STATUS_IS_OK(status)) { DEBUG(3, ("could not open handle to NETLOGON pipe\n")); goto done; } /* Pass back result code - zero for success, other values for specific failures. */ DEBUG(3,("domain %s secret is %s\n", domain->name, NT_STATUS_IS_OK(status) ? "good" : "bad")); done: DEBUG(NT_STATUS_IS_OK(status) ? 5 : 2, ("Checking the trust account password for domain %s returned %s\n", domain->name, nt_errstr(status))); return status; } NTSTATUS _wbint_ChangeMachineAccount(struct pipes_struct *p, struct wbint_ChangeMachineAccount *r) { struct messaging_context *msg_ctx = winbind_messaging_context(); struct winbindd_domain *domain; NTSTATUS status; struct rpc_pipe_client *netlogon_pipe; domain = wb_child_domain(); if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = cm_connect_netlogon(domain, &netlogon_pipe); if (!NT_STATUS_IS_OK(status)) { DEBUG(3, ("could not open handle to NETLOGON pipe\n")); goto done; } status = trust_pw_change(domain->conn.netlogon_creds, msg_ctx, netlogon_pipe->binding_handle, domain->name, true); /* force */ /* Pass back result code - zero for success, other values for specific failures. */ DEBUG(3,("domain %s secret %s\n", domain->name, NT_STATUS_IS_OK(status) ? "changed" : "unchanged")); done: DEBUG(NT_STATUS_IS_OK(status) ? 5 : 2, ("Changing the trust account password for domain %s returned %s\n", domain->name, nt_errstr(status))); return status; } NTSTATUS _wbint_PingDc(struct pipes_struct *p, struct wbint_PingDc *r) { NTSTATUS status; struct winbindd_domain *domain; struct rpc_pipe_client *netlogon_pipe; union netr_CONTROL_QUERY_INFORMATION info; WERROR werr; fstring logon_server; struct dcerpc_binding_handle *b; domain = wb_child_domain(); if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = cm_connect_netlogon(domain, &netlogon_pipe); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { DEBUG(3, ("could not open handle to NETLOGON pipe: %s\n", nt_errstr(status))); return status; } b = netlogon_pipe->binding_handle; fstr_sprintf(logon_server, "\\\\%s", domain->dcname); *r->out.dcname = talloc_strdup(p->mem_ctx, domain->dcname); if (*r->out.dcname == NULL) { DEBUG(2, ("Could not allocate memory\n")); return NT_STATUS_NO_MEMORY; } /* * This provokes a WERR_NOT_SUPPORTED error message. This is * documented in the wspp docs. I could not get a successful * call to work, but the main point here is testing that the * netlogon pipe works. */ status = dcerpc_netr_LogonControl(b, p->mem_ctx, logon_server, NETLOGON_CONTROL_QUERY, 2, &info, &werr); reset_cm_connection_on_error(domain, status); if (!NT_STATUS_IS_OK(status)) { DEBUG(2, ("dcerpc_netr_LogonControl failed: %s\n", nt_errstr(status))); return status; } if (!W_ERROR_EQUAL(werr, WERR_NOT_SUPPORTED)) { DEBUG(2, ("dcerpc_netr_LogonControl returned %s, expected " "WERR_NOT_SUPPORTED\n", win_errstr(werr))); return werror_to_ntstatus(werr); } DEBUG(5, ("winbindd_dual_ping_dc succeeded\n")); return NT_STATUS_OK; } NTSTATUS _winbind_DsrUpdateReadOnlyServerDnsRecords(struct pipes_struct *p, struct winbind_DsrUpdateReadOnlyServerDnsRecords *r) { struct winbindd_domain *domain; NTSTATUS status; struct rpc_pipe_client *netlogon_pipe; domain = wb_child_domain(); if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } status = cm_connect_netlogon(domain, &netlogon_pipe); if (!NT_STATUS_IS_OK(status)) { DEBUG(3, ("could not open handle to NETLOGON pipe\n")); goto done; } status = netlogon_creds_cli_DsrUpdateReadOnlyServerDnsRecords(domain->conn.netlogon_creds, netlogon_pipe->binding_handle, r->in.site_name, r->in.dns_ttl, r->in.dns_names); /* Pass back result code - zero for success, other values for specific failures. */ DEBUG(3,("DNS records for domain %s %s\n", domain->name, NT_STATUS_IS_OK(status) ? "changed" : "unchanged")); done: DEBUG(NT_STATUS_IS_OK(status) ? 5 : 2, ("Update of DNS records via RW DC %s returned %s\n", domain->name, nt_errstr(status))); return status; } NTSTATUS _winbind_SamLogon(struct pipes_struct *p, struct winbind_SamLogon *r) { struct winbindd_domain *domain; NTSTATUS status; DATA_BLOB lm_response, nt_response; domain = wb_child_domain(); if (domain == NULL) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } /* TODO: Handle interactive logons here */ if (r->in.validation_level != 3 || r->in.logon.network == NULL || (r->in.logon_level != NetlogonNetworkInformation && r->in.logon_level != NetlogonNetworkTransitiveInformation)) { return NT_STATUS_REQUEST_NOT_ACCEPTED; } lm_response = data_blob_talloc(p->mem_ctx, r->in.logon.network->lm.data, r->in.logon.network->lm.length); nt_response = data_blob_talloc(p->mem_ctx, r->in.logon.network->nt.data, r->in.logon.network->nt.length); status = winbind_dual_SamLogon(domain, p->mem_ctx, r->in.logon.network->identity_info.parameter_control, r->in.logon.network->identity_info.account_name.string, r->in.logon.network->identity_info.domain_name.string, r->in.logon.network->identity_info.workstation.string, r->in.logon.network->challenge, lm_response, nt_response, &r->out.validation.sam3); return status; }