/* Unix SMB/Netbios implementation. Version 2.0 Winbind daemon for ntdom nss module Copyright (C) Tim Potter 2000 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 2 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, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "winbindd.h" #include "sids.h" /* Add a trusted domain to our list of domains */ static struct winbindd_domain *add_trusted_domain(char *domain_name, DOM_SID *domain_sid) { struct winbindd_domain *domain, *tmp; for (tmp = domain_list; tmp != NULL; tmp = tmp->next) { if (strcmp(domain_name, tmp->name) == 0) { DEBUG(3, ("domain %s already in domain list\n", domain_name)); return tmp; } } DEBUG(1, ("adding domain %s\n", domain_name)); /* Create new domain entry */ if ((domain = (struct winbindd_domain *)malloc(sizeof(*domain))) == NULL) return NULL; /* Fill in fields */ ZERO_STRUCTP(domain); fstrcpy(domain->name, domain_name); sid_copy(&domain->sid, domain_sid); /* Link to domain list */ DLIST_ADD(domain_list, domain); return domain; } /* Look up global info for the winbind daemon */ BOOL get_domain_info(void) { uint32 enum_ctx = 0; uint32 num_doms = 0; char **domains = NULL; DOM_SID *sids = NULL, domain_sid; NTSTATUS result; CLI_POLICY_HND *hnd; int i; fstring level5_dom; DEBUG(1, ("getting trusted domain list\n")); /* Add our workgroup - keep handle to look up trusted domains */ if (!(hnd = cm_get_lsa_handle(lp_workgroup()))) return False; result = cli_lsa_query_info_policy(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, 0x05, level5_dom, &domain_sid); if (!NT_STATUS_IS_OK(result)) return False; add_trusted_domain(lp_workgroup(), &domain_sid); /* Enumerate list of trusted domains */ if (!(hnd = cm_get_lsa_handle(lp_workgroup()))) return False; result = cli_lsa_enum_trust_dom(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, &enum_ctx, &num_doms, &domains, &sids); if (!NT_STATUS_IS_OK(result)) return False; /* Add each domain to the trusted domain list */ for(i = 0; i < num_doms; i++) add_trusted_domain(domains[i], &sids[i]); return True; } #if 0 /* Open sam and sam domain handles */ static BOOL open_sam_handles(struct winbindd_domain *domain) { /* Get domain info (sid and controller name) */ if (!domain->got_domain_info) { domain->got_domain_info = get_domain_info(domain); if (!domain->got_domain_info) return False; } /* Shut down existing sam handles */ if (domain->sam_dom_handle_open) { wb_samr_close(&domain->sam_dom_handle); domain->sam_dom_handle_open = False; } if (domain->sam_handle_open) { wb_samr_close(&domain->sam_handle); domain->sam_handle_open = False; } /* Open sam handle */ domain->sam_handle_open = wb_samr_connect(domain->controller, SEC_RIGHTS_MAXIMUM_ALLOWED, &domain->sam_handle); if (!domain->sam_handle_open) return False; /* Open sam domain handle */ domain->sam_dom_handle_open = wb_samr_open_domain(&domain->sam_handle, SEC_RIGHTS_MAXIMUM_ALLOWED, &domain->sid, &domain->sam_dom_handle); if (!domain->sam_dom_handle_open) return False; return True; } static BOOL rpc_hnd_ok(CLI_POLICY_HND *hnd) { return hnd->cli->fd != -1; } /* Return true if the SAM domain handles are open and responding. */ BOOL domain_handles_open(struct winbindd_domain *domain) { time_t t; BOOL result; /* Check we haven't checked too recently */ t = time(NULL); if ((t - domain->last_check) < WINBINDD_ESTABLISH_LOOP) { return domain->sam_handle_open && domain->sam_dom_handle_open; } DEBUG(3, ("checking domain handles for domain %s\n", domain->name)); domain->last_check = t; /* Open sam handles if they are marked as closed */ if (!domain->sam_handle_open || !domain->sam_dom_handle_open) { reopen: DEBUG(3, ("opening sam handles\n")); return open_sam_handles(domain); } /* Check sam handles are ok - the domain controller may have failed and we need to move to a BDC. */ if (!rpc_hnd_ok(&domain->sam_handle) || !rpc_hnd_ok(&domain->sam_dom_handle)) { /* We want to close the current connection but attempt to open a new set, possibly to a new dc. If this doesn't work then return False as we have no dc to talk to. */ DEBUG(3, ("sam handles not responding\n")); winbindd_kill_connections(domain); goto reopen; } result = domain->sam_handle_open && domain->sam_dom_handle_open; return result; } /* Shut down connections to all domain controllers */ static void winbindd_kill_connections(struct winbindd_domain *domain) { /* Kill all connections */ if (!domain) { struct winbindd_domain *tmp; for (tmp = domain_list; tmp; tmp = tmp->next) { winbindd_kill_connections(domain); } return; } /* Log a level 0 message - this is probably a domain controller failure */ if (!domain->controller[0]) return; DEBUG(0, ("killing connections to domain %s with controller %s\n", domain->name, domain->controller)); /* Close LSA connections if we are killing connections to the dc that has them open. */ if (strequal(server_state.controller, domain->controller)) { server_state.pwdb_initialised = False; server_state.lsa_handle_open = False; wb_lsa_close(&server_state.lsa_handle); } /* Close domain sam handles but don't free them as this severely traumatises the getent state. The connections will be reopened later. */ if (domain->sam_dom_handle_open) { wb_samr_close(&domain->sam_dom_handle); domain->sam_dom_handle_open = False; } if (domain->sam_handle_open) { wb_samr_close(&domain->sam_handle); domain->sam_handle_open = False; } /* Re-lookup domain info which includes domain controller name */ domain->got_domain_info = False; } /* Kill connections to all servers */ void winbindd_kill_all_connections(void) { struct winbindd_domain *domain; /* Iterate over domain list */ domain = domain_list; while (domain) { struct winbindd_domain *next; /* Kill conections */ winbindd_kill_connections(domain); /* Remove domain from list */ next = domain->next; DLIST_REMOVE(domain_list, domain); SAFE_FREE(domain); domain = next; } } /* Attempt to connect to all domain controllers we know about */ void establish_connections(BOOL force_reestablish) { static time_t lastt; time_t t; /* Check we haven't checked too recently */ t = time(NULL); if ((t - lastt < WINBINDD_ESTABLISH_LOOP) && !force_reestablish) { return; } lastt = t; DEBUG(3, ("establishing connections\n")); /* Maybe the connection died - if so then close up and restart */ if (server_state.pwdb_initialised && server_state.lsa_handle_open && !rpc_hnd_ok(&server_state.lsa_handle)) { winbindd_kill_connections(NULL); } if (!server_state.pwdb_initialised) { /* Lookup domain controller name */ if (!get_any_dc_name(lp_workgroup(), server_state.controller)) { DEBUG(3, ("could not find any domain controllers " "for domain %s\n", lp_workgroup())); return; } /* Initialise password database and sids */ /* server_state.pwdb_initialised = pwdb_initialise(False); */ server_state.pwdb_initialised = True; if (!server_state.pwdb_initialised) { DEBUG(3, ("could not initialise pwdb\n")); return; } } /* Open lsa handle if it isn't already open */ if (!server_state.lsa_handle_open) { server_state.lsa_handle_open = wb_lsa_open_policy(server_state.controller, False, SEC_RIGHTS_MAXIMUM_ALLOWED, &server_state.lsa_handle); if (!server_state.lsa_handle_open) { DEBUG(0, ("error opening lsa handle on dc %s\n", server_state.controller)); return; } /* Now we can talk to the server we can get some info */ get_trusted_domains(); } } #endif /* Connect to a domain controller using get_any_dc_name() to discover the domain name and sid */ BOOL lookup_domain_sid(char *domain_name, struct winbindd_domain *domain) { fstring level5_dom; uint32 enum_ctx = 0; uint32 num_doms = 0; char **domains = NULL; DOM_SID *sids = NULL; CLI_POLICY_HND *hnd; NTSTATUS result; DEBUG(1, ("looking up sid for domain %s\n", domain_name)); if (!(hnd = cm_get_lsa_handle(domain_name))) return False; /* Do a level 5 query info policy if we are looking up the SID for our own domain. */ if (strequal(domain_name, lp_workgroup())) { result = cli_lsa_query_info_policy(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, 0x05, level5_dom, &domain->sid); return NT_STATUS_IS_OK(result); } /* Use lsaenumdomains to get sid for this domain */ result = cli_lsa_enum_trust_dom(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, &enum_ctx, &num_doms, &domains, &sids); /* Look for domain name */ if (NT_STATUS_IS_OK(result) && domains && sids) { int found = False; int i; for(i = 0; i < num_doms; i++) { if (strequal(domain_name, domains[i])) { sid_copy(&domain->sid, &sids[i]); found = True; break; } } return found; } return NT_STATUS_IS_OK(result); } #if 0 /* Lookup domain controller and sid for a domain */ BOOL get_domain_info(struct winbindd_domain *domain) { fstring sid_str; DEBUG(1, ("Getting domain info for domain %s\n", domain->name)); /* Lookup domain sid */ if (!lookup_domain_sid(domain->name, domain)) { DEBUG(0, ("could not find sid for domain %s\n", domain->name)); return False; } /* Lookup OK */ domain->got_domain_info = 1; sid_to_string(sid_str, &domain->sid); DEBUG(1, ("found sid %s for domain %s\n", sid_str, domain->name)); return True; } #endif /* Lookup a sid in a domain from a name */ BOOL winbindd_lookup_sid_by_name(char *name, DOM_SID *sid, enum SID_NAME_USE *type) { int num_sids = 0, num_names = 1; DOM_SID *sids = NULL; uint32 *types = NULL; CLI_POLICY_HND *hnd; NTSTATUS result; /* Don't bother with machine accounts */ if (name[strlen(name) - 1] == '$') return False; /* Lookup name */ if (!(hnd = cm_get_lsa_handle(lp_workgroup()))) return False; result = cli_lsa_lookup_names(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, num_names, (char **)&name, &sids, &types, &num_sids); /* Return rid and type if lookup successful */ if (NT_STATUS_IS_OK(result)) { /* Return sid */ if ((sid != NULL) && (sids != NULL)) sid_copy(sid, &sids[0]); /* Return name type */ if ((type != NULL) && (types != NULL)) *type = types[0]; return True; } return False; } /* Lookup a name in a domain from a sid */ BOOL winbindd_lookup_name_by_sid(DOM_SID *sid, fstring name, enum SID_NAME_USE *type) { int num_sids = 1, num_names = 0; uint32 *types = NULL; char **names; CLI_POLICY_HND *hnd; NTSTATUS result; /* Lookup name */ if (!(hnd = cm_get_lsa_handle(lp_workgroup()))) return False; result = cli_lsa_lookup_sids(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, num_sids, sid, &names, &types, &num_names); /* Return name and type if successful */ if (NT_STATUS_IS_OK(result)) { /* Return name */ if ((names != NULL) && (name != NULL)) fstrcpy(name, names[0]); /* Return name type */ if ((type != NULL) && (types != NULL)) *type = types[0]; return True; } return False; } /* Lookup user information from a rid */ BOOL winbindd_lookup_userinfo(struct winbindd_domain *domain, uint32 user_rid, SAM_USERINFO_CTR **user_info) { CLI_POLICY_HND *hnd; uint16 info_level = 0x15; NTSTATUS result; if (!(hnd = cm_get_sam_user_handle(domain->name, &domain->sid, user_rid))) return False; result = cli_samr_query_userinfo(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, info_level, user_info); return NT_STATUS_IS_OK(result); } /* Lookup groups a user is a member of. I wish Unix had a call like this! */ BOOL winbindd_lookup_usergroups(struct winbindd_domain *domain, uint32 user_rid, uint32 *num_groups, DOM_GID **user_groups) { CLI_POLICY_HND *hnd; NTSTATUS result; if (!(hnd = cm_get_sam_user_handle(domain->name, &domain->sid, user_rid))) return False; result = cli_samr_query_usergroups(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, num_groups, user_groups); return NT_STATUS_IS_OK(result); } /* Lookup group membership given a rid. */ BOOL winbindd_lookup_groupmem(struct winbindd_domain *domain, uint32 group_rid, uint32 *num_names, uint32 **rid_mem, char ***names, uint32 **name_types) { CLI_POLICY_HND *group_hnd, *dom_hnd; NTSTATUS result; uint32 i, total_names = 0; if (!(group_hnd = cm_get_sam_group_handle(domain->name, &domain->sid, group_rid))) return False; /* Get group membership. This is a list of rids. */ result = cli_samr_query_groupmem(group_hnd->cli, group_hnd->cli->mem_ctx, &group_hnd->pol, num_names, rid_mem, name_types); if (!NT_STATUS_IS_OK(result)) return NT_STATUS_IS_OK(result); /* Convert list of rids into list of names. Do this in bunches of ~1000 to avoid crashing NT4. It looks like there is a buffer overflow or something like that lurking around somewhere. */ if (!(dom_hnd = cm_get_sam_dom_handle(domain->name, &domain->sid))) return False; #define MAX_LOOKUP_RIDS 900 *names = talloc(dom_hnd->cli->mem_ctx, *num_names * sizeof(char *)); *name_types = talloc(dom_hnd->cli->mem_ctx, *num_names * sizeof(uint32)); for (i = 0; i < *num_names; i += MAX_LOOKUP_RIDS) { int num_lookup_rids = MIN(*num_names - i, MAX_LOOKUP_RIDS); uint32 tmp_num_names = 0; char **tmp_names = NULL; uint32 *tmp_types = NULL; /* Lookup a chunk of rids */ result = cli_samr_lookup_rids(dom_hnd->cli, dom_hnd->cli->mem_ctx, &dom_hnd->pol, 1000, /* flags */ num_lookup_rids, &(*rid_mem)[i], &tmp_num_names, &tmp_names, &tmp_types); if (!NT_STATUS_IS_OK(result)) return False; /* Copy result into array. The talloc system will take care of freeing the temporary arrays later on. */ memcpy(&(*names)[i], tmp_names, sizeof(char *) * tmp_num_names); memcpy(&(*name_types)[i], tmp_types, sizeof(uint32) * tmp_num_names); total_names += tmp_num_names; } *num_names = total_names; return NT_STATUS_IS_OK(result); } /* Globals for domain list stuff */ struct winbindd_domain *domain_list = NULL; /* Given a domain name, return the struct winbindd domain info for it if it is actually working. */ struct winbindd_domain *find_domain_from_name(char *domain_name) { struct winbindd_domain *tmp; /* Search through list */ for (tmp = domain_list; tmp != NULL; tmp = tmp->next) { if (strcmp(domain_name, tmp->name) == 0) return tmp; } /* Not found */ return NULL; } /* Given a domain name, return the struct winbindd domain info for it */ struct winbindd_domain *find_domain_from_sid(DOM_SID *sid) { struct winbindd_domain *tmp; /* Search through list */ for (tmp = domain_list; tmp != NULL; tmp = tmp->next) { if (sid_equal(sid, &tmp->sid)) { if (!tmp->got_domain_info) return NULL; return tmp; } } /* Not found */ return NULL; } /* Free state information held for {set,get,end}{pw,gr}ent() functions */ void free_getent_state(struct getent_state *state) { struct getent_state *temp; /* Iterate over state list */ temp = state; while(temp != NULL) { struct getent_state *next; /* Free sam entries then list entry */ SAFE_FREE(state->sam_entries); DLIST_REMOVE(state, state); next = temp->next; SAFE_FREE(temp); temp = next; } } /* Parse list of arguments to winbind uid or winbind gid parameters */ static BOOL parse_id_list(char *paramstr, BOOL is_user) { uid_t id_low, id_high = 0; /* Give a nicer error message if no parameters specified */ if (strequal(paramstr, "")) { DEBUG(0, ("winbind %s parameter missing\n", is_user ? "uid" : "gid")); return False; } /* Parse entry */ if (sscanf(paramstr, "%u-%u", &id_low, &id_high) != 2) { DEBUG(0, ("winbind %s parameter invalid\n", is_user ? "uid" : "gid")); return False; } /* Store id info */ if (is_user) { server_state.uid_low = id_low; server_state.uid_high = id_high; } else { server_state.gid_low = id_low; server_state.gid_high = id_high; } return True; } /* Initialise trusted domain info */ BOOL winbindd_param_init(void) { /* Parse winbind uid and winbind_gid parameters */ if (!(parse_id_list(lp_winbind_uid(), True) && parse_id_list(lp_winbind_gid(), False))) { return False; } /* Check for reversed uid and gid ranges */ if (server_state.uid_low > server_state.uid_high) { DEBUG(0, ("uid range invalid\n")); return False; } if (server_state.gid_low > server_state.gid_high) { DEBUG(0, ("gid range invalid\n")); return False; } return True; } /* Query display info for a domain. This returns enough information plus a bit extra to give an overview of domain users for the User Manager application. */ NTSTATUS winbindd_query_dispinfo(struct winbindd_domain *domain, uint32 *start_ndx, uint16 info_level, uint32 *num_entries, SAM_DISPINFO_CTR *ctr) { CLI_POLICY_HND *hnd; NTSTATUS result = NT_STATUS_UNSUCCESSFUL; if (!(hnd = cm_get_sam_dom_handle(domain->name, &domain->sid))) return result; result = cli_samr_query_dispinfo(hnd->cli, hnd->cli->mem_ctx, &hnd->pol, start_ndx, info_level, num_entries, 0xffff, ctr); if (!NT_STATUS_IS_OK(result)) return result; return NT_STATUS_OK; } /* Check if a domain is present in a comma-separated list of domains */ BOOL check_domain_env(char *domain_env, char *domain) { fstring name; char *tmp = domain_env; while(next_token(&tmp, name, ",", sizeof(fstring))) { if (strequal(name, domain)) { return True; } } return False; } /* Parse a string of the form DOMAIN/user into a domain and a user */ void parse_domain_user(char *domuser, fstring domain, fstring user) { char *p; char *sep = lp_winbind_separator(); if (!sep) sep = "\\"; p = strchr(domuser,*sep); if (!p) p = strchr(domuser,'\\'); if (!p) { fstrcpy(domain,""); fstrcpy(user, domuser); return; } fstrcpy(user, p+1); fstrcpy(domain, domuser); domain[PTR_DIFF(p, domuser)] = 0; strupper(domain); }