/* Unix SMB/CIFS implementation. Security Descriptor (SD) helper functions Copyright (C) Andrew Tridgell 2000 Copyright (C) Tim Potter 2000 Copyright (C) Jeremy Allison 2000 Copyright (C) Jelmer Vernooij 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 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 "libsmb/libsmb.h" #include "util_sd.h" #include "librpc/gen_ndr/ndr_lsa.h" #include "../libcli/security/security.h" #include "rpc_client/cli_pipe.h" #include "rpc_client/cli_lsarpc.h" #include "lib/util/string_wrappers.h" /* These values discovered by inspection */ struct perm_value { const char *perm; uint32_t mask; }; static const struct perm_value special_values[] = { { "R", SEC_RIGHTS_FILE_READ }, { "W", SEC_RIGHTS_FILE_WRITE }, { "X", SEC_RIGHTS_FILE_EXECUTE }, { "D", SEC_STD_DELETE }, { "P", SEC_STD_WRITE_DAC }, { "O", SEC_STD_WRITE_OWNER }, { NULL, 0 }, }; static const struct perm_value standard_values[] = { { "READ", SEC_RIGHTS_DIR_READ|SEC_DIR_TRAVERSE }, { "CHANGE", SEC_RIGHTS_DIR_READ|SEC_STD_DELETE|\ SEC_DIR_DELETE_CHILD|\ SEC_RIGHTS_DIR_WRITE|SEC_DIR_TRAVERSE }, { "FULL", SEC_RIGHTS_DIR_ALL }, { NULL, 0 }, }; static const struct { uint16_t mask; const char *str; const char *desc; } sec_desc_ctrl_bits[] = { {SEC_DESC_OWNER_DEFAULTED, "OD", "Owner Defaulted"}, {SEC_DESC_GROUP_DEFAULTED, "GD", "Group Defaulted"}, {SEC_DESC_DACL_PRESENT, "DP", "DACL Present"}, {SEC_DESC_DACL_DEFAULTED, "DD", "DACL Defaulted"}, {SEC_DESC_SACL_PRESENT, "SP", "SACL Present"}, {SEC_DESC_SACL_DEFAULTED, "SD", "SACL Defaulted"}, {SEC_DESC_DACL_TRUSTED, "DT", "DACL Trusted"}, {SEC_DESC_SERVER_SECURITY, "SS", "Server Security"}, {SEC_DESC_DACL_AUTO_INHERIT_REQ, "DR", "DACL Inheritance Required"}, {SEC_DESC_SACL_AUTO_INHERIT_REQ, "SR", "SACL Inheritance Required"}, {SEC_DESC_DACL_AUTO_INHERITED, "DI", "DACL Auto Inherited"}, {SEC_DESC_SACL_AUTO_INHERITED, "SI", "SACL Auto Inherited"}, {SEC_DESC_DACL_PROTECTED, "PD", "DACL Protected"}, {SEC_DESC_SACL_PROTECTED, "PS", "SACL Protected"}, {SEC_DESC_RM_CONTROL_VALID, "RM", "RM Control Valid"}, {SEC_DESC_SELF_RELATIVE , "SR", "Self Relative"}, }; /* Open cli connection and policy handle */ static NTSTATUS cli_lsa_lookup_sid(struct cli_state *cli, const struct dom_sid *sid, TALLOC_CTX *mem_ctx, enum lsa_SidType *type, char **domain, char **name) { struct smbXcli_tcon *orig_tcon = NULL; struct rpc_pipe_client *p = NULL; struct policy_handle handle; NTSTATUS status; TALLOC_CTX *frame = talloc_stackframe(); enum lsa_SidType *types; char **domains; char **names; if (cli_state_has_tcon(cli)) { orig_tcon = cli_state_save_tcon(cli); if (orig_tcon == NULL) { status = NT_STATUS_NO_MEMORY; goto tcon_fail; } } status = cli_tree_connect(cli, "IPC$", "?????", NULL); if (!NT_STATUS_IS_OK(status)) { goto tcon_fail; } status = cli_rpc_pipe_open_noauth(cli, &ndr_table_lsarpc, &p); if (!NT_STATUS_IS_OK(status)) { goto fail; } status = rpccli_lsa_open_policy(p, talloc_tos(), True, GENERIC_EXECUTE_ACCESS, &handle); if (!NT_STATUS_IS_OK(status)) { goto fail; } status = rpccli_lsa_lookup_sids(p, talloc_tos(), &handle, 1, sid, &domains, &names, &types); if (!NT_STATUS_IS_OK(status)) { goto fail; } *type = types[0]; *domain = talloc_move(mem_ctx, &domains[0]); *name = talloc_move(mem_ctx, &names[0]); status = NT_STATUS_OK; fail: TALLOC_FREE(p); cli_tdis(cli); tcon_fail: cli_state_restore_tcon(cli, orig_tcon); TALLOC_FREE(frame); return status; } /* convert a SID to a string, either numeric or username/group */ void SidToString(struct cli_state *cli, fstring str, const struct dom_sid *sid, bool numeric) { char *domain = NULL; char *name = NULL; enum lsa_SidType type; NTSTATUS status; sid_to_fstring(str, sid); if (numeric || cli == NULL) { return; } status = cli_lsa_lookup_sid(cli, sid, talloc_tos(), &type, &domain, &name); if (!NT_STATUS_IS_OK(status)) { return; } if (*domain) { slprintf(str, sizeof(fstring) - 1, "%s%s%s", domain, lp_winbind_separator(), name); } else { fstrcpy(str, name); } } static NTSTATUS cli_lsa_lookup_name(struct cli_state *cli, const char *name, enum lsa_SidType *type, struct dom_sid *sid) { struct smbXcli_tcon *orig_tcon = NULL; struct rpc_pipe_client *p = NULL; struct policy_handle handle; NTSTATUS status; TALLOC_CTX *frame = talloc_stackframe(); struct dom_sid *sids; enum lsa_SidType *types; if (cli_state_has_tcon(cli)) { orig_tcon = cli_state_save_tcon(cli); if (orig_tcon == NULL) { status = NT_STATUS_NO_MEMORY; goto tcon_fail; } } status = cli_tree_connect(cli, "IPC$", "?????", NULL); if (!NT_STATUS_IS_OK(status)) { goto tcon_fail; } status = cli_rpc_pipe_open_noauth(cli, &ndr_table_lsarpc, &p); if (!NT_STATUS_IS_OK(status)) { goto fail; } status = rpccli_lsa_open_policy(p, talloc_tos(), True, GENERIC_EXECUTE_ACCESS, &handle); if (!NT_STATUS_IS_OK(status)) { goto fail; } status = rpccli_lsa_lookup_names(p, talloc_tos(), &handle, 1, &name, NULL, 1, &sids, &types); if (!NT_STATUS_IS_OK(status)) { goto fail; } *type = types[0]; *sid = sids[0]; status = NT_STATUS_OK; fail: TALLOC_FREE(p); cli_tdis(cli); tcon_fail: cli_state_restore_tcon(cli, orig_tcon); TALLOC_FREE(frame); return status; } /* convert a string to a SID, either numeric or username/group */ bool StringToSid(struct cli_state *cli, struct dom_sid *sid, const char *str) { enum lsa_SidType type; if (string_to_sid(sid, str)) { return true; } if (cli == NULL) { return false; } return NT_STATUS_IS_OK(cli_lsa_lookup_name(cli, str, &type, sid)); } static void print_ace_flags(FILE *f, uint8_t flags) { char *str = talloc_strdup(NULL, ""); if (!str) { goto out; } if (flags & SEC_ACE_FLAG_OBJECT_INHERIT) { str = talloc_asprintf(str, "%s%s", str, "OI|"); if (!str) { goto out; } } if (flags & SEC_ACE_FLAG_CONTAINER_INHERIT) { str = talloc_asprintf(str, "%s%s", str, "CI|"); if (!str) { goto out; } } if (flags & SEC_ACE_FLAG_NO_PROPAGATE_INHERIT) { str = talloc_asprintf(str, "%s%s", str, "NP|"); if (!str) { goto out; } } if (flags & SEC_ACE_FLAG_INHERIT_ONLY) { str = talloc_asprintf(str, "%s%s", str, "IO|"); if (!str) { goto out; } } if (flags & SEC_ACE_FLAG_INHERITED_ACE) { str = talloc_asprintf(str, "%s%s", str, "I|"); if (!str) { goto out; } } /* Ignore define SEC_ACE_FLAG_SUCCESSFUL_ACCESS ( 0x40 ) and SEC_ACE_FLAG_FAILED_ACCESS ( 0x80 ) as they're audit ace flags. */ if (str[strlen(str)-1] == '|') { str[strlen(str)-1] = '\0'; fprintf(f, "/%s/", str); } else { fprintf(f, "/0x%x/", flags); } TALLOC_FREE(str); return; out: fprintf(f, "/0x%x/", flags); } /* print an ACE on a FILE, using either numeric or ascii representation */ void print_ace(struct cli_state *cli, FILE *f, struct security_ace *ace, bool numeric) { const struct perm_value *v; fstring sidstr; int do_print = 0; uint32_t got_mask; SidToString(cli, sidstr, &ace->trustee, numeric); fprintf(f, "%s:", sidstr); if (numeric) { fprintf(f, "%d/0x%x/0x%08x", ace->type, ace->flags, ace->access_mask); return; } /* Ace type */ if (ace->type == SEC_ACE_TYPE_ACCESS_ALLOWED) { fprintf(f, "ALLOWED"); } else if (ace->type == SEC_ACE_TYPE_ACCESS_DENIED) { fprintf(f, "DENIED"); } else { fprintf(f, "%d", ace->type); } print_ace_flags(f, ace->flags); /* Standard permissions */ for (v = standard_values; v->perm; v++) { if (ace->access_mask == v->mask) { fprintf(f, "%s", v->perm); return; } } /* Special permissions. Print out a hex value if we have leftover bits in the mask. */ got_mask = ace->access_mask; again: for (v = special_values; v->perm; v++) { if ((ace->access_mask & v->mask) == v->mask) { if (do_print) { fprintf(f, "%s", v->perm); } got_mask &= ~v->mask; } } if (!do_print) { if (got_mask != 0) { fprintf(f, "0x%08x", ace->access_mask); } else { do_print = 1; goto again; } } } static bool parse_ace_flags(const char *str, unsigned int *pflags) { const char *p = str; *pflags = 0; while (*p) { if (strnequal(p, "OI", 2)) { *pflags |= SEC_ACE_FLAG_OBJECT_INHERIT; p += 2; } else if (strnequal(p, "CI", 2)) { *pflags |= SEC_ACE_FLAG_CONTAINER_INHERIT; p += 2; } else if (strnequal(p, "NP", 2)) { *pflags |= SEC_ACE_FLAG_NO_PROPAGATE_INHERIT; p += 2; } else if (strnequal(p, "IO", 2)) { *pflags |= SEC_ACE_FLAG_INHERIT_ONLY; p += 2; } else if (*p == 'I') { *pflags |= SEC_ACE_FLAG_INHERITED_ACE; p += 1; } else if (*p) { return false; } switch (*p) { case '|': p++; FALL_THROUGH; case '\0': continue; default: return false; } } return true; } /* parse an ACE in the same format as print_ace() */ bool parse_ace(struct cli_state *cli, struct security_ace *ace, const char *orig_str) { char *p; const char *cp; char *tok; unsigned int atype = 0; unsigned int aflags = 0; unsigned int amask = 0; struct dom_sid sid; uint32_t mask; const struct perm_value *v; char *str = SMB_STRDUP(orig_str); TALLOC_CTX *frame = talloc_stackframe(); if (!str) { TALLOC_FREE(frame); return False; } ZERO_STRUCTP(ace); p = strchr_m(str,':'); if (!p) { printf("ACE '%s': missing ':'.\n", orig_str); SAFE_FREE(str); TALLOC_FREE(frame); return False; } *p = '\0'; p++; if (!StringToSid(cli, &sid, str)) { printf("ACE '%s': failed to convert '%s' to SID\n", orig_str, str); SAFE_FREE(str); TALLOC_FREE(frame); return False; } cp = p; if (!next_token_talloc(frame, &cp, &tok, "/")) { printf("ACE '%s': failed to find '/' character.\n", orig_str); SAFE_FREE(str); TALLOC_FREE(frame); return False; } if (strncmp(tok, "ALLOWED", strlen("ALLOWED")) == 0) { atype = SEC_ACE_TYPE_ACCESS_ALLOWED; } else if (strncmp(tok, "DENIED", strlen("DENIED")) == 0) { atype = SEC_ACE_TYPE_ACCESS_DENIED; } else if (strnequal(tok, "0x", 2)) { int result; result = sscanf(tok, "%x", &atype); if (result == 0 || (atype != SEC_ACE_TYPE_ACCESS_ALLOWED && atype != SEC_ACE_TYPE_ACCESS_DENIED)) { printf("ACE '%s': bad hex value for type at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return false; } } else if(tok[0] >= '0' && tok[0] <= '9') { int result; result = sscanf(tok, "%u", &atype); if (result == 0 || (atype != SEC_ACE_TYPE_ACCESS_ALLOWED && atype != SEC_ACE_TYPE_ACCESS_DENIED)) { printf("ACE '%s': bad integer value for type at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return false; } } else { printf("ACE '%s': missing 'ALLOWED' or 'DENIED' entry at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } if (!next_token_talloc(frame, &cp, &tok, "/")) { printf("ACE '%s': bad flags entry at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } if (tok[0] < '0' || tok[0] > '9') { if (!parse_ace_flags(tok, &aflags)) { printf("ACE '%s': bad named flags entry at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } } else if (strnequal(tok, "0x", 2)) { if (!sscanf(tok, "%x", &aflags)) { printf("ACE '%s': bad hex flags entry at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } } else { if (!sscanf(tok, "%u", &aflags)) { printf("ACE '%s': bad integer flags entry at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } } if (!next_token_talloc(frame, &cp, &tok, "/")) { printf("ACE '%s': missing / at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } if (strncmp(tok, "0x", 2) == 0) { if (sscanf(tok, "%x", &amask) != 1) { printf("ACE '%s': bad hex number at '%s'\n", orig_str, tok); SAFE_FREE(str); TALLOC_FREE(frame); return False; } goto done; } for (v = standard_values; v->perm; v++) { if (strcmp(tok, v->perm) == 0) { amask = v->mask; goto done; } } p = tok; while(*p) { bool found = False; for (v = special_values; v->perm; v++) { if (v->perm[0] == *p) { amask |= v->mask; found = True; } } if (!found) { printf("ACE '%s': bad permission value at '%s'\n", orig_str, p); SAFE_FREE(str); TALLOC_FREE(frame); return False; } p++; } if (*p) { TALLOC_FREE(frame); SAFE_FREE(str); return False; } done: mask = amask; init_sec_ace(ace, &sid, atype, mask, aflags); TALLOC_FREE(frame); SAFE_FREE(str); return True; } static void print_acl_ctrl(FILE *file, uint16_t ctrl, bool numeric) { int i; const char* separator = ""; fprintf(file, "CONTROL:"); if (numeric) { fprintf(file, "0x%x\n", ctrl); return; } for (i = ARRAY_SIZE(sec_desc_ctrl_bits) - 1; i >= 0; i--) { if (ctrl & sec_desc_ctrl_bits[i].mask) { fprintf(file, "%s%s", separator, sec_desc_ctrl_bits[i].str); separator = "|"; } } fputc('\n', file); } /* print a ascii version of a security descriptor on a FILE handle */ void sec_desc_print(struct cli_state *cli, FILE *f, struct security_descriptor *sd, bool numeric) { fstring sidstr; uint32_t i; fprintf(f, "REVISION:%d\n", sd->revision); print_acl_ctrl(f, sd->type, numeric); /* Print owner and group sid */ if (sd->owner_sid) { SidToString(cli, sidstr, sd->owner_sid, numeric); } else { fstrcpy(sidstr, ""); } fprintf(f, "OWNER:%s\n", sidstr); if (sd->group_sid) { SidToString(cli, sidstr, sd->group_sid, numeric); } else { fstrcpy(sidstr, ""); } fprintf(f, "GROUP:%s\n", sidstr); /* Print aces */ for (i = 0; sd->dacl && i < sd->dacl->num_aces; i++) { struct security_ace *ace = &sd->dacl->aces[i]; fprintf(f, "ACL:"); print_ace(cli, f, ace, numeric); fprintf(f, "\n"); } }