/* Unix SMB/CIFS implementation. POSIX NTVFS backend - ACL support Copyright (C) Andrew Tridgell 2004 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" #include "auth/auth.h" #include "system/filesys.h" #include "vfs_posix.h" #include "librpc/gen_ndr/ndr_xattr.h" /* map a single access_mask from generic to specific bits for files/dirs */ static uint32_t pvfs_translate_mask(uint32_t access_mask) { if (access_mask & SEC_MASK_GENERIC) { if (access_mask & SEC_GENERIC_READ) access_mask |= SEC_RIGHTS_FILE_READ; if (access_mask & SEC_GENERIC_WRITE) access_mask |= SEC_RIGHTS_FILE_WRITE; if (access_mask & SEC_GENERIC_EXECUTE) access_mask |= SEC_RIGHTS_FILE_EXECUTE; if (access_mask & SEC_GENERIC_ALL) access_mask |= SEC_RIGHTS_FILE_ALL; access_mask &= ~SEC_MASK_GENERIC; } return access_mask; } /* map any generic access bits in the given acl this relies on the fact that the mappings for files and directories are the same */ static void pvfs_translate_generic_bits(struct security_acl *acl) { unsigned i; for (i=0;inum_aces;i++) { struct security_ace *ace = &acl->aces[i]; ace->access_mask = pvfs_translate_mask(ace->access_mask); } } /* setup a default ACL for a file */ static NTSTATUS pvfs_default_acl(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, int fd, struct xattr_NTACL *acl) { struct security_descriptor *sd; NTSTATUS status; struct security_ace ace; mode_t mode; sd = security_descriptor_initialise(req); if (sd == NULL) { return NT_STATUS_NO_MEMORY; } status = sidmap_uid_to_sid(pvfs->sidmap, sd, name->st.st_uid, &sd->owner_sid); if (!NT_STATUS_IS_OK(status)) { return status; } status = sidmap_gid_to_sid(pvfs->sidmap, sd, name->st.st_gid, &sd->group_sid); if (!NT_STATUS_IS_OK(status)) { return status; } sd->type |= SEC_DESC_DACL_PRESENT; mode = name->st.st_mode; /* we provide up to 4 ACEs - Owner - Group - Everyone - Administrator */ /* setup owner ACE */ ace.type = SEC_ACE_TYPE_ACCESS_ALLOWED; ace.flags = 0; ace.trustee = *sd->owner_sid; ace.access_mask = 0; if (mode & S_IRUSR) { if (mode & S_IWUSR) { ace.access_mask |= SEC_RIGHTS_FILE_ALL; } else { ace.access_mask |= SEC_RIGHTS_FILE_READ | SEC_FILE_EXECUTE; } } if (mode & S_IWUSR) { ace.access_mask |= SEC_RIGHTS_FILE_WRITE | SEC_STD_DELETE; } if (ace.access_mask) { security_descriptor_dacl_add(sd, &ace); } /* setup group ACE */ ace.trustee = *sd->group_sid; ace.access_mask = 0; if (mode & S_IRGRP) { ace.access_mask |= SEC_RIGHTS_FILE_READ | SEC_FILE_EXECUTE; } if (mode & S_IWGRP) { /* note that delete is not granted - this matches posix behaviour */ ace.access_mask |= SEC_RIGHTS_FILE_WRITE; } if (ace.access_mask) { security_descriptor_dacl_add(sd, &ace); } /* setup other ACE */ ace.trustee = *dom_sid_parse_talloc(req, SID_WORLD); ace.access_mask = 0; if (mode & S_IROTH) { ace.access_mask |= SEC_RIGHTS_FILE_READ | SEC_FILE_EXECUTE; } if (mode & S_IWOTH) { ace.access_mask |= SEC_RIGHTS_FILE_WRITE; } if (ace.access_mask) { security_descriptor_dacl_add(sd, &ace); } /* setup system ACE */ ace.trustee = *dom_sid_parse_talloc(req, SID_NT_SYSTEM); ace.access_mask = SEC_RIGHTS_FILE_ALL; security_descriptor_dacl_add(sd, &ace); acl->version = 1; acl->info.sd = sd; return NT_STATUS_OK; } /* omit any security_descriptor elements not specified in the given secinfo flags */ static void normalise_sd_flags(struct security_descriptor *sd, uint32_t secinfo_flags) { if (!(secinfo_flags & SECINFO_OWNER)) { sd->owner_sid = NULL; } if (!(secinfo_flags & SECINFO_GROUP)) { sd->group_sid = NULL; } if (!(secinfo_flags & SECINFO_DACL)) { sd->dacl = NULL; } if (!(secinfo_flags & SECINFO_SACL)) { sd->sacl = NULL; } } /* answer a setfileinfo for an ACL */ NTSTATUS pvfs_acl_set(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, int fd, uint32_t access_mask, union smb_setfileinfo *info) { struct xattr_NTACL *acl; uint32_t secinfo_flags = info->set_secdesc.in.secinfo_flags; struct security_descriptor *new_sd, *sd; NTSTATUS status; uid_t uid = -1; gid_t gid = -1; acl = talloc_p(req, struct xattr_NTACL); if (acl == NULL) { return NT_STATUS_NO_MEMORY; } status = pvfs_acl_load(pvfs, name, fd, acl); if (NT_STATUS_EQUAL(status, NT_STATUS_NOT_FOUND)) { status = pvfs_default_acl(pvfs, req, name, fd, acl); } if (!NT_STATUS_IS_OK(status)) { return status; } switch (acl->version) { case 1: sd = acl->info.sd; break; default: return NT_STATUS_INVALID_ACL; } new_sd = info->set_secdesc.in.sd; uid = name->st.st_uid; gid = name->st.st_gid; /* only set the elements that have been specified */ if ((secinfo_flags & SECINFO_OWNER) && !dom_sid_equal(sd->owner_sid, new_sd->owner_sid)) { if (!(access_mask & SEC_STD_WRITE_OWNER)) { return NT_STATUS_ACCESS_DENIED; } sd->owner_sid = new_sd->owner_sid; status = sidmap_sid_to_unixuid(pvfs->sidmap, sd->owner_sid, &uid); if (!NT_STATUS_IS_OK(status)) { return status; } } if ((secinfo_flags & SECINFO_GROUP) && !dom_sid_equal(sd->group_sid, new_sd->group_sid)) { sd->group_sid = new_sd->group_sid; status = sidmap_sid_to_unixgid(pvfs->sidmap, sd->owner_sid, &gid); if (!NT_STATUS_IS_OK(status)) { return status; } } if (secinfo_flags & SECINFO_DACL) { sd->dacl = new_sd->dacl; pvfs_translate_generic_bits(sd->dacl); } if (secinfo_flags & SECINFO_SACL) { sd->sacl = new_sd->sacl; if (!(access_mask & SEC_FLAG_SYSTEM_SECURITY)) { return NT_STATUS_ACCESS_DENIED; } pvfs_translate_generic_bits(sd->sacl); } if (uid != -1 || gid != -1) { int ret; if (fd == -1) { ret = chown(name->full_name, uid, gid); } else { ret = fchown(fd, uid, gid); } if (ret == -1) { return pvfs_map_errno(pvfs, errno); } } status = pvfs_acl_save(pvfs, name, fd, acl); return status; } /* answer a fileinfo query for the ACL */ NTSTATUS pvfs_acl_query(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, int fd, union smb_fileinfo *info) { struct xattr_NTACL *acl; NTSTATUS status; struct security_descriptor *sd; acl = talloc_p(req, struct xattr_NTACL); if (acl == NULL) { return NT_STATUS_NO_MEMORY; } status = pvfs_acl_load(pvfs, name, fd, acl); if (NT_STATUS_EQUAL(status, NT_STATUS_NOT_FOUND)) { status = pvfs_default_acl(pvfs, req, name, fd, acl); } if (!NT_STATUS_IS_OK(status)) { return status; } switch (acl->version) { case 1: sd = acl->info.sd; break; default: return NT_STATUS_INVALID_ACL; } normalise_sd_flags(sd, info->query_secdesc.in.secinfo_flags); info->query_secdesc.out.sd = sd; return NT_STATUS_OK; } /* default access check function based on unix permissions doing this saves on building a full security descriptor for the common case of access check on files with no specific NT ACL */ NTSTATUS pvfs_access_check_unix(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, uint32_t *access_mask) { uid_t uid = geteuid(); uint32_t max_bits = SEC_RIGHTS_FILE_READ | SEC_FILE_ALL; /* owner and root get extra permissions */ if (uid == 0 || uid == name->st.st_uid) { max_bits |= SEC_STD_ALL; } if (*access_mask == SEC_FLAG_MAXIMUM_ALLOWED) { *access_mask = max_bits; return NT_STATUS_OK; } if (*access_mask & ~max_bits) { return NT_STATUS_ACCESS_DENIED; } *access_mask |= SEC_FILE_READ_ATTRIBUTE; return NT_STATUS_OK; } /* check the security descriptor on a file, if any *access_mask is modified with the access actually granted */ NTSTATUS pvfs_access_check(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, uint32_t *access_mask) { struct security_token *token = req->session->session_info->security_token; struct xattr_NTACL *acl; NTSTATUS status; struct security_descriptor *sd; acl = talloc_p(req, struct xattr_NTACL); if (acl == NULL) { return NT_STATUS_NO_MEMORY; } status = pvfs_acl_load(pvfs, name, -1, acl); if (NT_STATUS_EQUAL(status, NT_STATUS_NOT_FOUND)) { talloc_free(acl); return pvfs_access_check_unix(pvfs, req, name, access_mask); } if (!NT_STATUS_IS_OK(status)) { return status; } switch (acl->version) { case 1: sd = acl->info.sd; break; default: return NT_STATUS_INVALID_ACL; } /* expand the generic access bits to file specific bits */ *access_mask = pvfs_translate_mask(*access_mask); /* check the acl against the required access mask */ status = sec_access_check(sd, token, *access_mask, access_mask); /* this bit is always granted, even if not asked for */ *access_mask |= SEC_FILE_READ_ATTRIBUTE; talloc_free(acl); return status; } /* a simplified interface to access check, designed for calls that do not take or return an access check mask */ NTSTATUS pvfs_access_check_simple(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, uint32_t access_needed) { if (access_needed == 0) { return NT_STATUS_OK; } return pvfs_access_check(pvfs, req, name, &access_needed); } /* access check for creating a new file/directory */ NTSTATUS pvfs_access_check_create(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name) { struct pvfs_filename *parent; NTSTATUS status; status = pvfs_resolve_parent(pvfs, req, name, &parent); if (!NT_STATUS_IS_OK(status)) { return status; } return pvfs_access_check_simple(pvfs, req, parent, SEC_DIR_ADD_FILE); } /* determine if an ACE is inheritable */ static BOOL pvfs_inheritable_ace(struct pvfs_state *pvfs, const struct security_ace *ace, BOOL container) { if (!container) { return (ace->flags & SEC_ACE_FLAG_OBJECT_INHERIT) != 0; } if (ace->flags & SEC_ACE_FLAG_CONTAINER_INHERIT) { return True; } if ((ace->flags & SEC_ACE_FLAG_OBJECT_INHERIT) && !(ace->flags & SEC_ACE_FLAG_NO_PROPAGATE_INHERIT)) { return True; } return False; } /* this is the core of ACL inheritance. It copies any inheritable aces from the parent SD to the child SD. Note that the algorithm depends on whether the child is a container or not */ static NTSTATUS pvfs_acl_inherit_aces(struct pvfs_state *pvfs, struct security_descriptor *parent_sd, struct security_descriptor *sd, BOOL container) { int i; for (i=0;idacl->num_aces;i++) { struct security_ace ace = parent_sd->dacl->aces[i]; NTSTATUS status; if (!pvfs_inheritable_ace(pvfs, &ace, container)) { continue; } /* see the RAW-ACLS inheritance test for details on these rules */ if (!container) { ace.flags = 0; } else { ace.flags &= ~SEC_ACE_FLAG_INHERIT_ONLY; if (!(ace.flags & SEC_ACE_FLAG_CONTAINER_INHERIT)) { ace.flags |= SEC_ACE_FLAG_INHERIT_ONLY; } if (ace.flags & SEC_ACE_FLAG_NO_PROPAGATE_INHERIT) { ace.flags = 0; } } status = security_descriptor_dacl_add(sd, &ace); if (!NT_STATUS_IS_OK(status)) { return status; } } return NT_STATUS_OK; } /* setup an ACL on a new file/directory based on the inherited ACL from the parent. If there is no inherited ACL then we don't set anything, as the default ACL applies anyway */ NTSTATUS pvfs_acl_inherit(struct pvfs_state *pvfs, struct smbsrv_request *req, struct pvfs_filename *name, int fd) { struct xattr_NTACL *acl; NTSTATUS status; struct pvfs_filename *parent; struct security_descriptor *parent_sd, *sd; BOOL container; /* form the parents path */ status = pvfs_resolve_parent(pvfs, req, name, &parent); if (!NT_STATUS_IS_OK(status)) { return status; } acl = talloc_p(req, struct xattr_NTACL); if (acl == NULL) { return NT_STATUS_NO_MEMORY; } status = pvfs_acl_load(pvfs, parent, -1, acl); if (NT_STATUS_EQUAL(status, NT_STATUS_NOT_FOUND)) { return NT_STATUS_OK; } if (!NT_STATUS_IS_OK(status)) { return status; } switch (acl->version) { case 1: parent_sd = acl->info.sd; break; default: return NT_STATUS_INVALID_ACL; } if (parent_sd == NULL || parent_sd->dacl == NULL || parent_sd->dacl->num_aces == 0) { /* go with the default ACL */ return NT_STATUS_OK; } /* create the new sd */ sd = security_descriptor_initialise(req); if (sd == NULL) { return NT_STATUS_NO_MEMORY; } status = sidmap_uid_to_sid(pvfs->sidmap, sd, name->st.st_uid, &sd->owner_sid); if (!NT_STATUS_IS_OK(status)) { return status; } status = sidmap_gid_to_sid(pvfs->sidmap, sd, name->st.st_gid, &sd->group_sid); if (!NT_STATUS_IS_OK(status)) { return status; } sd->type |= SEC_DESC_DACL_PRESENT; container = (name->dos.attrib & FILE_ATTRIBUTE_DIRECTORY) ? True:False; /* fill in the aces from the parent */ status = pvfs_acl_inherit_aces(pvfs, parent_sd, sd, container); if (!NT_STATUS_IS_OK(status)) { return status; } /* if there is nothing to inherit then we fallback to the default acl */ if (sd->dacl == NULL || sd->dacl->num_aces == 0) { return NT_STATUS_OK; } acl->info.sd = sd; status = pvfs_acl_save(pvfs, name, fd, acl); return status; }