Smack: UDS revision
This patch addresses a number of long standing issues with the way Smack treats UNIX domain sockets. All access control was being done based on the label of the file system object. This is inconsistant with the internet domain, in which access is done based on the IPIN and IPOUT attributes of the socket. As a result of the inode label policy it was not possible to use a UDS socket for label cognizant services, including dbus and the X11 server. Support for SCM_PEERSEC on UDS sockets is also provided. Signed-off-by: Casey Schaufler <casey@schaufler-ca.com> Signed-off-by: James Morris <jmorris@namei.org>
This commit is contained in:
parent
7e70cb4978
commit
b4e0d5f079
|
@ -1667,10 +1667,13 @@ static int smack_inode_setsecurity(struct inode *inode, const char *name,
|
||||||
ssp->smk_in = sp;
|
ssp->smk_in = sp;
|
||||||
else if (strcmp(name, XATTR_SMACK_IPOUT) == 0) {
|
else if (strcmp(name, XATTR_SMACK_IPOUT) == 0) {
|
||||||
ssp->smk_out = sp;
|
ssp->smk_out = sp;
|
||||||
rc = smack_netlabel(sock->sk, SMACK_CIPSO_SOCKET);
|
if (sock->sk->sk_family != PF_UNIX) {
|
||||||
if (rc != 0)
|
rc = smack_netlabel(sock->sk, SMACK_CIPSO_SOCKET);
|
||||||
printk(KERN_WARNING "Smack: \"%s\" netlbl error %d.\n",
|
if (rc != 0)
|
||||||
__func__, -rc);
|
printk(KERN_WARNING
|
||||||
|
"Smack: \"%s\" netlbl error %d.\n",
|
||||||
|
__func__, -rc);
|
||||||
|
}
|
||||||
} else
|
} else
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
@ -2267,9 +2270,10 @@ static void smack_d_instantiate(struct dentry *opt_dentry, struct inode *inode)
|
||||||
break;
|
break;
|
||||||
case SOCKFS_MAGIC:
|
case SOCKFS_MAGIC:
|
||||||
/*
|
/*
|
||||||
* Casey says sockets get the smack of the task.
|
* Socket access is controlled by the socket
|
||||||
|
* structures associated with the task involved.
|
||||||
*/
|
*/
|
||||||
final = csp;
|
final = smack_known_star.smk_known;
|
||||||
break;
|
break;
|
||||||
case PROC_SUPER_MAGIC:
|
case PROC_SUPER_MAGIC:
|
||||||
/*
|
/*
|
||||||
|
@ -2296,7 +2300,16 @@ static void smack_d_instantiate(struct dentry *opt_dentry, struct inode *inode)
|
||||||
/*
|
/*
|
||||||
* This isn't an understood special case.
|
* This isn't an understood special case.
|
||||||
* Get the value from the xattr.
|
* Get the value from the xattr.
|
||||||
*
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* UNIX domain sockets use lower level socket data.
|
||||||
|
*/
|
||||||
|
if (S_ISSOCK(inode->i_mode)) {
|
||||||
|
final = smack_known_star.smk_known;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/*
|
||||||
* No xattr support means, alas, no SMACK label.
|
* No xattr support means, alas, no SMACK label.
|
||||||
* Use the aforeapplied default.
|
* Use the aforeapplied default.
|
||||||
* It would be curious if the label of the task
|
* It would be curious if the label of the task
|
||||||
|
@ -2418,14 +2431,18 @@ static int smack_setprocattr(struct task_struct *p, char *name,
|
||||||
static int smack_unix_stream_connect(struct socket *sock,
|
static int smack_unix_stream_connect(struct socket *sock,
|
||||||
struct socket *other, struct sock *newsk)
|
struct socket *other, struct sock *newsk)
|
||||||
{
|
{
|
||||||
struct inode *sp = SOCK_INODE(sock);
|
struct socket_smack *ssp = sock->sk->sk_security;
|
||||||
struct inode *op = SOCK_INODE(other);
|
struct socket_smack *osp = other->sk->sk_security;
|
||||||
struct smk_audit_info ad;
|
struct smk_audit_info ad;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
smk_ad_init(&ad, __func__, LSM_AUDIT_DATA_NET);
|
smk_ad_init(&ad, __func__, LSM_AUDIT_DATA_NET);
|
||||||
smk_ad_setfield_u_net_sk(&ad, other->sk);
|
smk_ad_setfield_u_net_sk(&ad, other->sk);
|
||||||
return smk_access(smk_of_inode(sp), smk_of_inode(op),
|
|
||||||
MAY_READWRITE, &ad);
|
if (!capable(CAP_MAC_OVERRIDE))
|
||||||
|
rc = smk_access(ssp->smk_out, osp->smk_in, MAY_WRITE, &ad);
|
||||||
|
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2438,13 +2455,18 @@ static int smack_unix_stream_connect(struct socket *sock,
|
||||||
*/
|
*/
|
||||||
static int smack_unix_may_send(struct socket *sock, struct socket *other)
|
static int smack_unix_may_send(struct socket *sock, struct socket *other)
|
||||||
{
|
{
|
||||||
struct inode *sp = SOCK_INODE(sock);
|
struct socket_smack *ssp = sock->sk->sk_security;
|
||||||
struct inode *op = SOCK_INODE(other);
|
struct socket_smack *osp = other->sk->sk_security;
|
||||||
struct smk_audit_info ad;
|
struct smk_audit_info ad;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
smk_ad_init(&ad, __func__, LSM_AUDIT_DATA_NET);
|
smk_ad_init(&ad, __func__, LSM_AUDIT_DATA_NET);
|
||||||
smk_ad_setfield_u_net_sk(&ad, other->sk);
|
smk_ad_setfield_u_net_sk(&ad, other->sk);
|
||||||
return smk_access(smk_of_inode(sp), smk_of_inode(op), MAY_WRITE, &ad);
|
|
||||||
|
if (!capable(CAP_MAC_OVERRIDE))
|
||||||
|
rc = smk_access(ssp->smk_out, osp->smk_in, MAY_WRITE, &ad);
|
||||||
|
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2629,7 +2651,7 @@ static int smack_socket_getpeersec_stream(struct socket *sock,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* smack_socket_getpeersec_dgram - pull in packet label
|
* smack_socket_getpeersec_dgram - pull in packet label
|
||||||
* @sock: the socket
|
* @sock: the peer socket
|
||||||
* @skb: packet data
|
* @skb: packet data
|
||||||
* @secid: pointer to where to put the secid of the packet
|
* @secid: pointer to where to put the secid of the packet
|
||||||
*
|
*
|
||||||
|
@ -2640,41 +2662,39 @@ static int smack_socket_getpeersec_dgram(struct socket *sock,
|
||||||
|
|
||||||
{
|
{
|
||||||
struct netlbl_lsm_secattr secattr;
|
struct netlbl_lsm_secattr secattr;
|
||||||
struct sock *sk;
|
struct socket_smack *sp;
|
||||||
char smack[SMK_LABELLEN];
|
char smack[SMK_LABELLEN];
|
||||||
int family = PF_INET;
|
int family = PF_UNSPEC;
|
||||||
u32 s;
|
u32 s = 0; /* 0 is the invalid secid */
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
/*
|
if (skb != NULL) {
|
||||||
* Only works for families with packets.
|
if (skb->protocol == htons(ETH_P_IP))
|
||||||
*/
|
family = PF_INET;
|
||||||
if (sock != NULL) {
|
else if (skb->protocol == htons(ETH_P_IPV6))
|
||||||
sk = sock->sk;
|
family = PF_INET6;
|
||||||
if (sk->sk_family != PF_INET && sk->sk_family != PF_INET6)
|
|
||||||
return 0;
|
|
||||||
family = sk->sk_family;
|
|
||||||
}
|
}
|
||||||
/*
|
if (family == PF_UNSPEC && sock != NULL)
|
||||||
* Translate what netlabel gave us.
|
family = sock->sk->sk_family;
|
||||||
*/
|
|
||||||
netlbl_secattr_init(&secattr);
|
|
||||||
rc = netlbl_skbuff_getattr(skb, family, &secattr);
|
|
||||||
if (rc == 0)
|
|
||||||
smack_from_secattr(&secattr, smack);
|
|
||||||
netlbl_secattr_destroy(&secattr);
|
|
||||||
|
|
||||||
/*
|
if (family == PF_UNIX) {
|
||||||
* Give up if we couldn't get anything
|
sp = sock->sk->sk_security;
|
||||||
*/
|
s = smack_to_secid(sp->smk_out);
|
||||||
if (rc != 0)
|
} else if (family == PF_INET || family == PF_INET6) {
|
||||||
return rc;
|
/*
|
||||||
|
* Translate what netlabel gave us.
|
||||||
s = smack_to_secid(smack);
|
*/
|
||||||
|
netlbl_secattr_init(&secattr);
|
||||||
|
rc = netlbl_skbuff_getattr(skb, family, &secattr);
|
||||||
|
if (rc == 0) {
|
||||||
|
smack_from_secattr(&secattr, smack);
|
||||||
|
s = smack_to_secid(smack);
|
||||||
|
}
|
||||||
|
netlbl_secattr_destroy(&secattr);
|
||||||
|
}
|
||||||
|
*secid = s;
|
||||||
if (s == 0)
|
if (s == 0)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
*secid = s;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue