/* Unix SMB/CIFS implementation. uid/user handling Copyright (C) Andrew Tridgell 1992-1998 Copyright (C) Gerald (Jerry) Carter 2003 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 "includes.h" /***************************************************************** Dissect a user-provided name into domain, name, sid and type. If an explicit domain name was given in the form domain\user, it has to try that. If no explicit domain name was given, we have to do guesswork. *****************************************************************/ BOOL lookup_name(TALLOC_CTX *mem_ctx, const char *full_name, int flags, const char **ret_domain, const char **ret_name, DOM_SID *ret_sid, enum SID_NAME_USE *ret_type) { char *p; const char *tmp; const char *domain = NULL; const char *name = NULL; uint32 rid; DOM_SID sid; enum SID_NAME_USE type; TALLOC_CTX *tmp_ctx = talloc_new(mem_ctx); if (tmp_ctx == NULL) { DEBUG(0, ("talloc_new failed\n")); return False; } p = strchr_m(full_name, '\\'); if (p != NULL) { domain = talloc_strndup(tmp_ctx, full_name, PTR_DIFF(p, full_name)); name = talloc_strdup(tmp_ctx, p+1); } else { domain = talloc_strdup(tmp_ctx, ""); name = talloc_strdup(tmp_ctx, full_name); } if ((domain == NULL) || (name == NULL)) { DEBUG(0, ("talloc failed\n")); return False; } if (strequal(domain, get_global_sam_name())) { /* It's our own domain, lookup the name in passdb */ if (lookup_global_sam_name(name, &rid, &type)) { sid_copy(&sid, get_global_sam_sid()); sid_append_rid(&sid, rid); goto ok; } goto failed; } if (strequal(domain, builtin_domain_name())) { /* Explicit request for a name in BUILTIN */ if (lookup_builtin_name(name, &rid)) { sid_copy(&sid, &global_sid_Builtin); sid_append_rid(&sid, rid); type = SID_NAME_ALIAS; goto ok; } goto failed; } if (domain[0] != '\0') { /* An explicit domain name was given, here our last resort is * winbind. */ if (winbind_lookup_name(domain, name, &sid, &type)) { goto ok; } goto failed; } if (!(flags & LOOKUP_NAME_ISOLATED)) { goto failed; } /* Now the guesswork begins, we haven't been given an explicit * domain. Try the sequence as documented on * http://msdn.microsoft.com/library/en-us/secmgmt/security/lsalookupnames.asp * November 27, 2005 */ /* 1. well-known names */ { if (lookup_wellknown_name(tmp_ctx, name, &sid, &domain)) { type = SID_NAME_WKN_GRP; goto ok; } } /* 2. Builtin domain as such */ if (strequal(name, builtin_domain_name())) { /* Swap domain and name */ tmp = name; name = domain; domain = tmp; sid_copy(&sid, &global_sid_Builtin); type = SID_NAME_DOMAIN; goto ok; } /* 3. Account domain */ if (strequal(name, get_global_sam_name())) { if (!secrets_fetch_domain_sid(name, &sid)) { DEBUG(3, ("Could not fetch my SID\n")); goto failed; } /* Swap domain and name */ tmp = name; name = domain; domain = tmp; type = SID_NAME_DOMAIN; goto ok; } /* 4. Primary domain */ if (!IS_DC && strequal(name, lp_workgroup())) { if (!secrets_fetch_domain_sid(name, &sid)) { DEBUG(3, ("Could not fetch the domain SID\n")); goto failed; } /* Swap domain and name */ tmp = name; name = domain; domain = tmp; type = SID_NAME_DOMAIN; goto ok; } /* 5. Trusted domains as such, to me it looks as if members don't do this, tested an XP workstation in a NT domain -- vl */ if (IS_DC && (secrets_fetch_trusted_domain_password(name, NULL, &sid, NULL))) { /* Swap domain and name */ tmp = name; name = domain; domain = tmp; type = SID_NAME_DOMAIN; goto ok; } /* 6. Builtin aliases */ if (lookup_builtin_name(name, &rid)) { domain = talloc_strdup(tmp_ctx, builtin_domain_name()); sid_copy(&sid, &global_sid_Builtin); sid_append_rid(&sid, rid); type = SID_NAME_ALIAS; goto ok; } /* 7. Local systems' SAM (DCs don't have a local SAM) */ /* 8. Primary SAM (On members, this is the domain) */ /* Both cases are done by looking at our passdb */ if (lookup_global_sam_name(name, &rid, &type)) { domain = talloc_strdup(tmp_ctx, get_global_sam_name()); sid_copy(&sid, get_global_sam_sid()); sid_append_rid(&sid, rid); goto ok; } /* Now our local possibilities are exhausted. */ if (!(flags & LOOKUP_NAME_REMOTE)) { goto failed; } /* If we are not a DC, we have to ask in our primary domain. Let * winbind do that. */ if (!IS_DC && (winbind_lookup_name(lp_workgroup(), name, &sid, &type))) { domain = talloc_strdup(tmp_ctx, lp_workgroup()); goto ok; } /* 9. Trusted domains */ /* If we're a DC we have to ask all trusted DC's. Winbind does not do * that (yet), but give it a chance. */ if (IS_DC && winbind_lookup_name("", name, &sid, &type)) { DOM_SID dom_sid; uint32 tmp_rid; enum SID_NAME_USE domain_type; if (type == SID_NAME_DOMAIN) { /* Swap name and type */ tmp = name; name = domain; domain = tmp; goto ok; } /* Here we have to cope with a little deficiency in the * winbind API: We have to ask it again for the name of the * domain it figured out itself. Maybe fix that later... */ sid_copy(&dom_sid, &sid); sid_split_rid(&dom_sid, &tmp_rid); if (!winbind_lookup_sid(tmp_ctx, &dom_sid, &domain, NULL, &domain_type) || (domain_type != SID_NAME_DOMAIN)) { DEBUG(2, ("winbind could not find the domain's name " "it just looked up for us\n")); goto failed; } goto ok; } /* 10. Don't translate */ failed: talloc_free(tmp_ctx); return False; ok: if ((domain == NULL) || (name == NULL)) { DEBUG(0, ("talloc failed\n")); talloc_free(tmp_ctx); return False; } if (ret_name != NULL) { *ret_name = talloc_steal(mem_ctx, name); } if (ret_domain != NULL) { char *tmp_dom = talloc_strdup(tmp_ctx, domain); strupper_m(tmp_dom); *ret_domain = talloc_steal(mem_ctx, tmp_dom); } if (ret_sid != NULL) { sid_copy(ret_sid, &sid); } if (ret_type != NULL) { *ret_type = type; } talloc_free(tmp_ctx); return True; } /***************************************************************** *THE CANONICAL* convert SID to name function. Tries local lookup first - for local sids, then tries winbind. *****************************************************************/ BOOL lookup_sid(TALLOC_CTX *mem_ctx, const DOM_SID *sid, const char **ret_domain, const char **ret_name, enum SID_NAME_USE *ret_type) { const char *domain = NULL; const char *name = NULL; enum SID_NAME_USE type; TALLOC_CTX *tmp_ctx = talloc_new(mem_ctx); /* Check if this is our own sid. This should perhaps be done by winbind? For the moment handle it here. */ if (tmp_ctx == NULL) { DEBUG(0, ("talloc_new failed\n")); return False; } if (sid_check_is_domain(sid)) { domain = talloc_strdup(tmp_ctx, get_global_sam_name()); name = talloc_strdup(tmp_ctx, ""); type = SID_NAME_DOMAIN; goto ok; } if (sid_check_is_in_our_domain(sid)) { uint32 rid; SMB_ASSERT(sid_peek_rid(sid, &rid)); /* For our own domain passdb is responsible */ if (!lookup_global_sam_rid(tmp_ctx, rid, &name, &type)) { goto failed; } domain = talloc_strdup(tmp_ctx, get_global_sam_name()); goto ok; } if (sid_check_is_builtin(sid)) { domain = talloc_strdup(tmp_ctx, builtin_domain_name()); /* Yes, W2k3 returns "BUILTIN" both as domain and name here */ name = talloc_strdup(tmp_ctx, builtin_domain_name()); type = SID_NAME_DOMAIN; goto ok; } if (sid_check_is_in_builtin(sid)) { uint32 rid; SMB_ASSERT(sid_peek_rid(sid, &rid)); if (!lookup_builtin_rid(tmp_ctx, rid, &name)) { goto failed; } /* There's only aliases in S-1-5-32 */ type = SID_NAME_ALIAS; domain = talloc_strdup(tmp_ctx, builtin_domain_name()); goto ok; } if (winbind_lookup_sid(tmp_ctx, sid, &domain, &name, &type)) { goto ok; } DEBUG(10,("lookup_sid: winbind lookup for SID %s failed - trying " "special SIDs.\n", sid_string_static(sid))); if (lookup_wellknown_sid(tmp_ctx, sid, &domain, &name)) { type = SID_NAME_WKN_GRP; goto ok; } failed: DEBUG(10, ("Failed to lookup sid %s\n", sid_string_static(sid))); talloc_free(tmp_ctx); return False; ok: if ((domain == NULL) || (name == NULL)) { DEBUG(0, ("talloc failed\n")); talloc_free(tmp_ctx); return False; } if (ret_domain != NULL) { *ret_domain = talloc_steal(mem_ctx, domain); } if (ret_name != NULL) { *ret_name = talloc_steal(mem_ctx, name); } if (ret_type != NULL) { *ret_type = type; } talloc_free(tmp_ctx); return True; } /***************************************************************** Id mapping cache. This is to avoid Winbind mappings already seen by smbd to be queried too frequently, keeping winbindd busy, and blocking smbd while winbindd is busy with other stuff. Written by Michael Steffens , modified to use linked lists by jra. *****************************************************************/ #define MAX_UID_SID_CACHE_SIZE 100 #define TURNOVER_UID_SID_CACHE_SIZE 10 #define MAX_GID_SID_CACHE_SIZE 100 #define TURNOVER_GID_SID_CACHE_SIZE 10 static size_t n_uid_sid_cache = 0; static size_t n_gid_sid_cache = 0; static struct uid_sid_cache { struct uid_sid_cache *next, *prev; uid_t uid; DOM_SID sid; enum SID_NAME_USE sidtype; } *uid_sid_cache_head; static struct gid_sid_cache { struct gid_sid_cache *next, *prev; gid_t gid; DOM_SID sid; enum SID_NAME_USE sidtype; } *gid_sid_cache_head; /***************************************************************** Find a SID given a uid. *****************************************************************/ static BOOL fetch_sid_from_uid_cache(DOM_SID *psid, uid_t uid) { struct uid_sid_cache *pc; for (pc = uid_sid_cache_head; pc; pc = pc->next) { if (pc->uid == uid) { *psid = pc->sid; DEBUG(3,("fetch sid from uid cache %u -> %s\n", (unsigned int)uid, sid_string_static(psid))); DLIST_PROMOTE(uid_sid_cache_head, pc); return True; } } return False; } /***************************************************************** Find a uid given a SID. *****************************************************************/ static BOOL fetch_uid_from_cache( uid_t *puid, const DOM_SID *psid ) { struct uid_sid_cache *pc; for (pc = uid_sid_cache_head; pc; pc = pc->next) { if (sid_compare(&pc->sid, psid) == 0) { *puid = pc->uid; DEBUG(3,("fetch uid from cache %u -> %s\n", (unsigned int)*puid, sid_string_static(psid))); DLIST_PROMOTE(uid_sid_cache_head, pc); return True; } } return False; } /***************************************************************** Store uid to SID mapping in cache. *****************************************************************/ static void store_uid_sid_cache(const DOM_SID *psid, uid_t uid) { struct uid_sid_cache *pc; if (n_uid_sid_cache >= MAX_UID_SID_CACHE_SIZE && n_uid_sid_cache > TURNOVER_UID_SID_CACHE_SIZE) { /* Delete the last TURNOVER_UID_SID_CACHE_SIZE entries. */ struct uid_sid_cache *pc_next; size_t i; for (i = 0, pc = uid_sid_cache_head; i < (n_uid_sid_cache - TURNOVER_UID_SID_CACHE_SIZE); i++, pc = pc->next) ; for(; pc; pc = pc_next) { pc_next = pc->next; DLIST_REMOVE(uid_sid_cache_head,pc); SAFE_FREE(pc); n_uid_sid_cache--; } } pc = SMB_MALLOC_P(struct uid_sid_cache); if (!pc) return; pc->uid = uid; sid_copy(&pc->sid, psid); DLIST_ADD(uid_sid_cache_head, pc); n_uid_sid_cache++; } /***************************************************************** Find a SID given a gid. *****************************************************************/ static BOOL fetch_sid_from_gid_cache(DOM_SID *psid, gid_t gid) { struct gid_sid_cache *pc; for (pc = gid_sid_cache_head; pc; pc = pc->next) { if (pc->gid == gid) { *psid = pc->sid; DEBUG(3,("fetch sid from gid cache %u -> %s\n", (unsigned int)gid, sid_string_static(psid))); DLIST_PROMOTE(gid_sid_cache_head, pc); return True; } } return False; } /***************************************************************** Find a gid given a SID. *****************************************************************/ static BOOL fetch_gid_from_cache(gid_t *pgid, const DOM_SID *psid) { struct gid_sid_cache *pc; for (pc = gid_sid_cache_head; pc; pc = pc->next) { if (sid_compare(&pc->sid, psid) == 0) { *pgid = pc->gid; DEBUG(3,("fetch gid from cache %u -> %s\n", (unsigned int)*pgid, sid_string_static(psid))); DLIST_PROMOTE(gid_sid_cache_head, pc); return True; } } return False; } /***************************************************************** Store gid to SID mapping in cache. *****************************************************************/ static void store_gid_sid_cache(const DOM_SID *psid, gid_t gid) { struct gid_sid_cache *pc; if (n_gid_sid_cache >= MAX_GID_SID_CACHE_SIZE && n_gid_sid_cache > TURNOVER_GID_SID_CACHE_SIZE) { /* Delete the last TURNOVER_GID_SID_CACHE_SIZE entries. */ struct gid_sid_cache *pc_next; size_t i; for (i = 0, pc = gid_sid_cache_head; i < (n_gid_sid_cache - TURNOVER_GID_SID_CACHE_SIZE); i++, pc = pc->next) ; for(; pc; pc = pc_next) { pc_next = pc->next; DLIST_REMOVE(gid_sid_cache_head,pc); SAFE_FREE(pc); n_gid_sid_cache--; } } pc = SMB_MALLOC_P(struct gid_sid_cache); if (!pc) return; pc->gid = gid; sid_copy(&pc->sid, psid); DLIST_ADD(gid_sid_cache_head, pc); n_gid_sid_cache++; } /***************************************************************** *THE CANONICAL* convert uid_t to SID function. *****************************************************************/ NTSTATUS uid_to_sid(DOM_SID *psid, uid_t uid) { uid_t low, high; ZERO_STRUCTP(psid); if (fetch_sid_from_uid_cache(psid, uid)) return ( psid ? NT_STATUS_OK : NT_STATUS_UNSUCCESSFUL ); /* DC's never use winbindd to resolve users outside the defined idmap range */ if ( lp_server_role()==ROLE_DOMAIN_MEMBER || (lp_idmap_uid(&low, &high) && uid >= low && uid <= high) ) { if (winbind_uid_to_sid(psid, uid)) { DEBUG(10,("uid_to_sid: winbindd %u -> %s\n", (unsigned int)uid, sid_string_static(psid))); if (psid) store_uid_sid_cache(psid, uid); return ( psid ? NT_STATUS_OK : NT_STATUS_UNSUCCESSFUL ); } } if (!local_uid_to_sid(psid, uid)) { DEBUG(10,("uid_to_sid: local %u failed to map to sid\n", (unsigned int)uid )); return NT_STATUS_UNSUCCESSFUL; } DEBUG(10,("uid_to_sid: local %u -> %s\n", (unsigned int)uid, sid_string_static(psid))); store_uid_sid_cache(psid, uid); return NT_STATUS_OK; } /***************************************************************** *THE CANONICAL* convert gid_t to SID function. *****************************************************************/ NTSTATUS gid_to_sid(DOM_SID *psid, gid_t gid) { gid_t low, high; ZERO_STRUCTP(psid); if (fetch_sid_from_gid_cache(psid, gid)) return ( psid ? NT_STATUS_OK : NT_STATUS_UNSUCCESSFUL ); /* DC's never use winbindd to resolve groups outside the defined idmap range */ if ( lp_server_role()==ROLE_DOMAIN_MEMBER || (lp_idmap_gid(&low, &high) && gid >= low && gid <= high) ) { if (winbind_gid_to_sid(psid, gid)) { DEBUG(10,("gid_to_sid: winbindd %u -> %s\n", (unsigned int)gid, sid_string_static(psid))); if (psid) store_gid_sid_cache(psid, gid); return ( psid ? NT_STATUS_OK : NT_STATUS_UNSUCCESSFUL ); } } if (!local_gid_to_sid(psid, gid)) { DEBUG(10,("gid_to_sid: local %u failed to map to sid\n", (unsigned int)gid )); return NT_STATUS_UNSUCCESSFUL; } DEBUG(10,("gid_to_sid: local %u -> %s\n", (unsigned int)gid, sid_string_static(psid))); store_gid_sid_cache(psid, gid); return NT_STATUS_OK; } /***************************************************************** *THE CANONICAL* convert SID to uid function. *****************************************************************/ NTSTATUS sid_to_uid(const DOM_SID *psid, uid_t *puid) { enum SID_NAME_USE name_type; if (fetch_uid_from_cache(puid, psid)) return NT_STATUS_OK; /* if this is our SID then go straight to a local lookup */ if ( sid_compare_domain(get_global_sam_sid(), psid) == 0 ) { DEBUG(10,("sid_to_uid: my domain (%s) - trying local.\n", sid_string_static(psid) )); if ( local_sid_to_uid(puid, psid, &name_type) ) goto success; DEBUG(10,("sid_to_uid: local lookup failed\n")); return NT_STATUS_UNSUCCESSFUL; } /* If it is not our local domain, only hope is winbindd */ if ( !winbind_lookup_sid(NULL, psid, NULL, NULL, &name_type) ) { DEBUG(10,("sid_to_uid: winbind lookup for non-local sid %s failed\n", sid_string_static(psid) )); return NT_STATUS_UNSUCCESSFUL; } /* If winbindd does know the SID, ensure this is a user */ if (name_type != SID_NAME_USER) { DEBUG(10,("sid_to_uid: winbind lookup succeeded but SID is not a user (%u)\n", (unsigned int)name_type )); return NT_STATUS_INVALID_PARAMETER; } /* get the uid. Has to work or else we are dead in the water */ if ( !winbind_sid_to_uid(puid, psid) ) { DEBUG(10,("sid_to_uid: winbind failed to allocate a new uid for sid %s\n", sid_string_static(psid))); return NT_STATUS_UNSUCCESSFUL; } success: DEBUG(10,("sid_to_uid: %s -> %u\n", sid_string_static(psid), (unsigned int)*puid )); store_uid_sid_cache(psid, *puid); return NT_STATUS_OK; } /***************************************************************** *THE CANONICAL* convert SID to gid function. Group mapping is used for gids that maps to Wellknown SIDs *****************************************************************/ NTSTATUS sid_to_gid(const DOM_SID *psid, gid_t *pgid) { enum SID_NAME_USE name_type; if (fetch_gid_from_cache(pgid, psid)) return NT_STATUS_OK; /* * First we must look up the name and decide if this is a group sid. * Group mapping can deal with foreign SIDs */ if ( local_sid_to_gid(pgid, psid, &name_type) ) goto success; if (!winbind_lookup_sid(NULL, psid, NULL, NULL, &name_type)) { DEBUG(10,("sid_to_gid: no one knows the SID %s (tried local, then " "winbind)\n", sid_string_static(psid))); return NT_STATUS_UNSUCCESSFUL; } /* winbindd knows it; Ensure this is a group sid */ if ((name_type != SID_NAME_DOM_GRP) && (name_type != SID_NAME_ALIAS) && (name_type != SID_NAME_WKN_GRP)) { DEBUG(10,("sid_to_gid: winbind lookup succeeded but SID is not a known group (%u)\n", (unsigned int)name_type )); /* winbindd is running and knows about this SID. Just the wrong type. Don't fallback to a local lookup here */ return NT_STATUS_INVALID_PARAMETER; } /* winbindd knows it and it is a type of group; sid_to_gid must succeed or we are dead in the water */ if ( !winbind_sid_to_gid(pgid, psid) ) { DEBUG(10,("sid_to_gid: winbind failed to allocate a new gid for sid %s\n", sid_string_static(psid))); return NT_STATUS_UNSUCCESSFUL; } success: DEBUG(10,("sid_to_gid: %s -> %u\n", sid_string_static(psid), (unsigned int)*pgid )); store_gid_sid_cache(psid, *pgid); return NT_STATUS_OK; }