+ file-capabilities-change-xattr-format-v2.patch added to -mm tree

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



The patch titled
     file capabilities: change xattr format (v2)
has been added to the -mm tree.  Its filename is
     file-capabilities-change-xattr-format-v2.patch

*** Remember to use Documentation/SubmitChecklist when testing your code ***

See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find
out what to do about this

------------------------------------------------------
Subject: file capabilities: change xattr format (v2)
From: Serge E. Hallyn <serue@xxxxxxxxxx>

This patch implements three suggestions by Andrew Morgan.  The first is to
change the on disk format.  The __le32 version, is renamed 'magic_etc', and
contains a revision number (starting at 1), and a bit representing Fe.  The
Fe capset is no longer stored, so the xattr is now 3 ints instead of 4.

The Fe was previously a full capset which was masked with the calculated
new_permitted to get the process' new effective set.  It is now a single
bit in the xattr.  Soon, the bprm->cap_effective will also be changed to a
boolean.  When that boolean is false, the P'e will be the empty set.  When
the boolean is true, then P'e will be set equal to P'p (new_permitted). 
The rationale for this is that either the application does not know about
capabilities, and needs to start with all permitted caps in its effective
set, or it does know about capabilities, and can start with an empty
effective set and enable the caps it wants when it wants.

Finally, future format compatibility is reduced.  If a security.capability
xattr is found with too new a version, don't run the binary.

(I can send an ltp patch to test this to anyone who asks.)

Signed-off-by: Serge E. Hallyn <serue@xxxxxxxxxx>
Cc: Andrew Morgan <morgan@xxxxxxxxxx>
Cc: Casey Schaufler <casey@xxxxxxxxxxxxxxxx>
Cc: Chris Wright <chrisw@xxxxxxxxxxxx>
Cc: James Morris <jmorris@xxxxxxxxx>
Cc: KaiGai Kohei <kaigai@xxxxxxxxxxxx>
Cc: Serge E. Hallyn <serue@xxxxxxxxxx>
Cc: Stephen Smalley <sds@xxxxxxxxxxxxx>
Signed-off-by: Andrew Morton <akpm@xxxxxxxxxxxxxxxxxxxx>
---

 include/linux/capability.h |   35 +++++-------------
 security/commoncap.c       |   65 ++++++++++++-----------------------
 2 files changed, 34 insertions(+), 66 deletions(-)

diff -puN include/linux/capability.h~file-capabilities-change-xattr-format-v2 include/linux/capability.h
--- a/include/linux/capability.h~file-capabilities-change-xattr-format-v2
+++ a/include/linux/capability.h
@@ -47,28 +47,12 @@ typedef struct __user_cap_data_struct {
 #define XATTR_CAPS_SUFFIX "capability"
 #define XATTR_NAME_CAPS XATTR_SECURITY_PREFIX XATTR_CAPS_SUFFIX
 
-/* size of caps that we work with */
-#define XATTR_CAPS_SZ (4*sizeof(__le32))
+#define XATTR_CAPS_SZ (3*sizeof(__le32))
+#define VFS_CAP_REVISION_MASK	0xFF000000
+#define VFS_CAP_REVISION	0x01000000
 
-/*
- * data[] is organized as:
- *   effective[0]
- *   permitted[0]
- *   inheritable[0]
- *   effective[1]
- *   ...
- * this way we can just read as much of the on-disk capability as
- * we know should exist and know we'll get the data we'll need.
- */
-struct vfs_cap_data_disk {
-	__le32 version;
-	__le32 data[];  /* eff[0], perm[0], inh[0], eff[1], ... */
-};
-
-struct vfs_cap_data_disk_v1 {
-	__le32 version;
-	__le32 data[3];  /* eff[0], perm[0], inh[0] */
-};
+#define VFS_CAP_FLAGS_MASK	~VFS_CAP_REVISION_MASK
+#define VFS_CAP_FLAGS_EFFECTIVE	0x000001
 
 #ifdef __KERNEL__
 
@@ -76,10 +60,11 @@ struct vfs_cap_data_disk_v1 {
 #include <asm/current.h>
 
 struct vfs_cap_data {
-	__u32 version;
-	__u32 effective;
-	__u32 permitted;
-	__u32 inheritable;
+	__u32 magic_etc;  /* Little endian */
+	struct {
+		__u32 permitted;    /* Little endian */
+		__u32 inheritable;  /* Little endian */
+	} data[1];
 };
 
 /* #define STRICT_CAP_T_TYPECHECKS */
diff -puN security/commoncap.c~file-capabilities-change-xattr-format-v2 security/commoncap.c
--- a/security/commoncap.c~file-capabilities-change-xattr-format-v2
+++ a/security/commoncap.c
@@ -118,27 +118,26 @@ static inline void bprm_clear_caps(struc
 
 #ifdef CONFIG_SECURITY_FILE_CAPABILITIES
 
-static inline int cap_from_disk(struct vfs_cap_data_disk *dcap,
-					struct linux_binprm *bprm, int size)
+static inline int cap_from_disk(__le32 *caps, struct linux_binprm *bprm,
+				int size)
 {
-	int version;
+	__u32 magic_etc;
 
-	version = le32_to_cpu(dcap->version);
-	if (version != _LINUX_CAPABILITY_VERSION)
+	if (size != XATTR_CAPS_SZ)
 		return -EINVAL;
 
-	size /= sizeof(u32);
-	if ((size-1)%3) {
-		printk(KERN_WARNING "%s: size is an invalid size %d for %s\n",
-						__FUNCTION__, size, bprm->filename);
+	magic_etc = le32_to_cpu(caps[0]);
+
+	switch ((magic_etc & VFS_CAP_REVISION_MASK)) {
+	case VFS_CAP_REVISION:
+		bprm->cap_effective = (magic_etc & VFS_CAP_FLAGS_EFFECTIVE)
+					? CAP_FULL_SET : CAP_EMPTY_SET;
+		bprm->cap_permitted = to_cap_t( le32_to_cpu(caps[1]) );
+		bprm->cap_inheritable = to_cap_t( le32_to_cpu(caps[2]) );
+		return 0;
+	default:
 		return -EINVAL;
 	}
-
-	bprm->cap_effective = le32_to_cpu(dcap->data[0]);
-	bprm->cap_permitted = le32_to_cpu(dcap->data[1]);
-	bprm->cap_inheritable = le32_to_cpu(dcap->data[2]);
-
-	return 0;
 }
 
 /* Locate any VFS capabilities: */
@@ -146,11 +145,9 @@ static int get_file_caps(struct linux_bi
 {
 	struct dentry *dentry;
 	int rc = 0;
-	struct vfs_cap_data_disk_v1 v1caps;
-	struct vfs_cap_data_disk *dcaps;
+	__le32 v1caps[XATTR_CAPS_SZ];
 	struct inode *inode;
 
-	dcaps = (struct vfs_cap_data_disk *)&v1caps;
 	if (bprm->file->f_vfsmnt->mnt_flags & MNT_NOSUID) {
 		bprm_clear_caps(bprm);
 		return 0;
@@ -161,40 +158,23 @@ static int get_file_caps(struct linux_bi
 	if (!inode->i_op || !inode->i_op->getxattr)
 		goto out;
 
-	rc = inode->i_op->getxattr(dentry, XATTR_NAME_CAPS, dcaps,
+	rc = inode->i_op->getxattr(dentry, XATTR_NAME_CAPS, &v1caps,
 							XATTR_CAPS_SZ);
 	if (rc == -ENODATA || rc == -EOPNOTSUPP) {
+		/* no data, that's ok */
 		rc = 0;
 		goto out;
 	}
-	if (rc == -ERANGE) {
-		int size;
-		size = inode->i_op->getxattr(dentry, XATTR_NAME_CAPS, NULL, 0);
-		if (size <= 0) {  /* shouldn't ever happen */
-			rc = -EINVAL;
-			goto out;
-		}
-		dcaps = kmalloc(size, GFP_KERNEL);
-		if (!dcaps) {
-			rc = -ENOMEM;
-			goto out;
-		}
-		rc = inode->i_op->getxattr(dentry, XATTR_NAME_CAPS, dcaps,
-							size);
-	}
 	if (rc < 0)
 		goto out;
-	if (rc < sizeof(struct vfs_cap_data_disk_v1)) {
-		rc = -EINVAL;
-		goto out;
-	}
 
-	rc = cap_from_disk(dcaps, bprm, rc);
+	rc = cap_from_disk(v1caps, bprm, rc);
+	if (rc)
+		printk(KERN_NOTICE "%s: cap_from_disk returned %d for %s\n",
+			__FUNCTION__, rc, bprm->filename);
 
 out:
 	dput(dentry);
-	if ((void *)dcaps != (void *)&v1caps)
-		kfree(dcaps);
 	if (rc)
 		bprm_clear_caps(bprm);
 
@@ -214,6 +194,9 @@ int cap_bprm_set_security (struct linux_
 	int ret;
 
 	ret = get_file_caps(bprm);
+	if (ret)
+		printk(KERN_NOTICE "%s: get_file_caps returned %d for %s\n",
+			__FUNCTION__, ret, bprm->filename);
 
 	/*  To support inheritance of root-permissions and suid-root
 	 *  executables under compatibility mode, we raise all three
_

Patches currently in -mm which might be from serue@xxxxxxxxxx are

implement-file-posix-capabilities.patch
implement-file-posix-capabilities-fix.patch
file-capabilities-introduce-cap_setfcap.patch
file-capabilities-get_file_caps-cleanups.patch
file-caps-update-selinux-xattr-hooks.patch
file-capabilities-clear-caps-cleanup.patch
file-capabilities-clear-caps-cleanup-fix.patch
file-capabilities-change-xattr-format-v2.patch
file-capabilities-change-fe-to-a-bool.patch
remove-config_uts_ns-and-config_ipc_ns.patch
user-namespace-add-the-framework.patch
user-namespace-add-unshare.patch
mm-fix-create_new_namespaces-return-value.patch
cpuset-zero-malloc-revert-the-old-cpuset-fix.patch
containersv10-basic-container-framework.patch
containersv10-basic-container-framework-fix.patch
containersv10-example-cpu-accounting-subsystem.patch
containersv10-example-cpu-accounting-subsystem-fix.patch
containersv10-add-tasks-file-interface.patch
containersv10-add-tasks-file-interface-fix.patch
containersv10-add-fork-exit-hooks.patch
containersv10-add-fork-exit-hooks-fix.patch
containersv10-add-container_clone-interface.patch
containersv10-add-container_clone-interface-fix.patch
containersv10-add-procfs-interface.patch
containersv10-add-procfs-interface-fix.patch
containersv10-make-cpusets-a-client-of-containers.patch
containersv10-share-css_group-arrays-between-tasks-with-same-container-memberships.patch
containersv10-share-css_group-arrays-between-tasks-with-same-container-memberships-fix.patch
containersv10-share-css_group-arrays-between-tasks-with-same-container-memberships-cpuset-zero-malloc-fix-for-new-containers.patch
containersv10-simple-debug-info-subsystem.patch
containersv10-simple-debug-info-subsystem-fix.patch
containersv10-simple-debug-info-subsystem-fix-2.patch
containersv10-support-for-automatic-userspace-release-agents.patch
containers-implement-subsys-post_clone.patch
containers-implement-namespace-tracking-subsystem-v3.patch
pid-namespaces-round-up-the-api.patch

-
To unsubscribe from this list: send the line "unsubscribe mm-commits" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Kernel Newbies FAQ]     [Kernel Archive]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [Bugtraq]     [Photo]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]

  Powered by Linux