ceph_client_gid(fsc->client));
 }
 
+static ssize_t ceph_vxattrcb_caps(struct ceph_inode_info *ci, char *val,
+                                       size_t size)
+{
+       int issued;
+
+       spin_lock(&ci->i_ceph_lock);
+       issued = __ceph_caps_issued(ci, NULL);
+       spin_unlock(&ci->i_ceph_lock);
+
+       return ceph_fmt_xattr(val, size, "%s/0x%x",
+                             ceph_cap_string(issued), issued);
+}
+
 #define CEPH_XATTR_NAME(_type, _name)  XATTR_CEPH_PREFIX #_type "." #_name
 #define CEPH_XATTR_NAME2(_type, _name, _name2) \
        XATTR_CEPH_PREFIX #_type "." #_name "." #_name2
                .exists_cb = ceph_vxattrcb_snap_btime_exists,
                .flags = VXATTR_FLAG_READONLY,
        },
+       {
+               .name = "ceph.caps",
+               .name_size = sizeof("ceph.caps"),
+               .getxattr_cb = ceph_vxattrcb_caps,
+               .exists_cb = NULL,
+               .flags = VXATTR_FLAG_HIDDEN,
+       },
        { .name = NULL, 0 }     /* Required table terminator */
 };
 
                .exists_cb = ceph_vxattrcb_snap_btime_exists,
                .flags = VXATTR_FLAG_READONLY,
        },
+       {
+               .name = "ceph.caps",
+               .name_size = sizeof("ceph.caps"),
+               .getxattr_cb = ceph_vxattrcb_caps,
+               .exists_cb = NULL,
+               .flags = VXATTR_FLAG_HIDDEN,
+       },
        { .name = NULL, 0 }     /* Required table terminator */
 };