af_unix: Allow connecting to sockets in other network namespaces.
Remove the restriction that only allows connecting to a unix domain socket identified by unix path that is in the same network namespace. Crossing network namespaces is always tricky and we did not support this at first, because of a strict policy of don't mix the namespaces. Later after Pavel proposed this we did not support this because no one had performed the audit to make certain using unix domain sockets across namespaces is safe. What fundamentally makes connecting to af_unix sockets in other namespaces is safe is that you have to have the proper permissions on the unix domain socket inode that lives in the filesystem. If you want strict isolation you just don't create inodes where unfriendlys can get at them, or with permissions that allow unfriendlys to open them. All nicely handled for us by the mount namespace and other standard file system facilities. I looked through unix domain sockets and they are a very controlled environment so none of the work that goes on in dev_forward_skb to make crossing namespaces safe appears needed, we are not loosing controll of the skb and so do not need to set up the skb to look like it is comming in fresh from the outside world. Further the fields in struct unix_skb_parms should not have any problems crossing network namespaces. Now that we handle SCM_CREDENTIALS in a way that gives useable values across namespaces. There does not appear to be any operational problems with encouraging the use of unix domain sockets across containers either. Signed-off-by: Eric W. Biederman <ebiederm@xmission.com> Acked-by: Daniel Lezcano <daniel.lezcano@free.fr> Acked-by: Pavel Emelyanov <xemul@openvz.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
7361c36c52
commit
6616f7888c
1 changed files with 2 additions and 5 deletions
|
@ -282,7 +282,7 @@ static inline struct sock *unix_find_socket_byname(struct net *net,
|
|||
return s;
|
||||
}
|
||||
|
||||
static struct sock *unix_find_socket_byinode(struct net *net, struct inode *i)
|
||||
static struct sock *unix_find_socket_byinode(struct inode *i)
|
||||
{
|
||||
struct sock *s;
|
||||
struct hlist_node *node;
|
||||
|
@ -292,9 +292,6 @@ static struct sock *unix_find_socket_byinode(struct net *net, struct inode *i)
|
|||
&unix_socket_table[i->i_ino & (UNIX_HASH_SIZE - 1)]) {
|
||||
struct dentry *dentry = unix_sk(s)->dentry;
|
||||
|
||||
if (!net_eq(sock_net(s), net))
|
||||
continue;
|
||||
|
||||
if (dentry && dentry->d_inode == i) {
|
||||
sock_hold(s);
|
||||
goto found;
|
||||
|
@ -758,7 +755,7 @@ static struct sock *unix_find_other(struct net *net,
|
|||
err = -ECONNREFUSED;
|
||||
if (!S_ISSOCK(inode->i_mode))
|
||||
goto put_fail;
|
||||
u = unix_find_socket_byinode(net, inode);
|
||||
u = unix_find_socket_byinode(inode);
|
||||
if (!u)
|
||||
goto put_fail;
|
||||
|
||||
|
|
Loading…
Reference in a new issue