mirror of
https://github.com/Fishwaldo/Star64_linux.git
synced 2025-04-04 13:34:39 +00:00
netlink: have netlink per-protocol bind function return an error code.
Have the netlink per-protocol optional bind function return an int error code rather than void to signal a failure. This will enable netlink protocols to perform extra checks including capabilities and permissions verifications when updating memberships in multicast groups. In netlink_bind() and netlink_setsockopt() the call to the per-protocol bind function was moved above the multicast group update to prevent any access to the multicast socket groups before checking with the per-protocol bind function. This will enable the per-protocol bind function to be used to check permissions which could be denied before making them available, and to avoid the messy job of undoing the addition should the per-protocol bind function fail. The netfilter subsystem seems to be the only one currently using the per-protocol bind function. Signed-off-by: Richard Guy Briggs <rgb@redhat.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
bfe4bc71c6
commit
4f52090052
4 changed files with 56 additions and 24 deletions
|
@ -45,7 +45,8 @@ struct netlink_kernel_cfg {
|
||||||
unsigned int flags;
|
unsigned int flags;
|
||||||
void (*input)(struct sk_buff *skb);
|
void (*input)(struct sk_buff *skb);
|
||||||
struct mutex *cb_mutex;
|
struct mutex *cb_mutex;
|
||||||
void (*bind)(int group);
|
int (*bind)(int group);
|
||||||
|
void (*unbind)(int group);
|
||||||
bool (*compare)(struct net *net, struct sock *sk);
|
bool (*compare)(struct net *net, struct sock *sk);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -400,7 +400,7 @@ static void nfnetlink_rcv(struct sk_buff *skb)
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_MODULES
|
#ifdef CONFIG_MODULES
|
||||||
static void nfnetlink_bind(int group)
|
static int nfnetlink_bind(int group)
|
||||||
{
|
{
|
||||||
const struct nfnetlink_subsystem *ss;
|
const struct nfnetlink_subsystem *ss;
|
||||||
int type = nfnl_group2type[group];
|
int type = nfnl_group2type[group];
|
||||||
|
@ -410,6 +410,7 @@ static void nfnetlink_bind(int group)
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
if (!ss)
|
if (!ss)
|
||||||
request_module("nfnetlink-subsys-%d", type);
|
request_module("nfnetlink-subsys-%d", type);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -1206,7 +1206,8 @@ static int netlink_create(struct net *net, struct socket *sock, int protocol,
|
||||||
struct module *module = NULL;
|
struct module *module = NULL;
|
||||||
struct mutex *cb_mutex;
|
struct mutex *cb_mutex;
|
||||||
struct netlink_sock *nlk;
|
struct netlink_sock *nlk;
|
||||||
void (*bind)(int group);
|
int (*bind)(int group);
|
||||||
|
void (*unbind)(int group);
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
sock->state = SS_UNCONNECTED;
|
sock->state = SS_UNCONNECTED;
|
||||||
|
@ -1232,6 +1233,7 @@ static int netlink_create(struct net *net, struct socket *sock, int protocol,
|
||||||
err = -EPROTONOSUPPORT;
|
err = -EPROTONOSUPPORT;
|
||||||
cb_mutex = nl_table[protocol].cb_mutex;
|
cb_mutex = nl_table[protocol].cb_mutex;
|
||||||
bind = nl_table[protocol].bind;
|
bind = nl_table[protocol].bind;
|
||||||
|
unbind = nl_table[protocol].unbind;
|
||||||
netlink_unlock_table();
|
netlink_unlock_table();
|
||||||
|
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
|
@ -1248,6 +1250,7 @@ static int netlink_create(struct net *net, struct socket *sock, int protocol,
|
||||||
nlk = nlk_sk(sock->sk);
|
nlk = nlk_sk(sock->sk);
|
||||||
nlk->module = module;
|
nlk->module = module;
|
||||||
nlk->netlink_bind = bind;
|
nlk->netlink_bind = bind;
|
||||||
|
nlk->netlink_unbind = unbind;
|
||||||
out:
|
out:
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
|
@ -1301,6 +1304,7 @@ static int netlink_release(struct socket *sock)
|
||||||
kfree_rcu(old, rcu);
|
kfree_rcu(old, rcu);
|
||||||
nl_table[sk->sk_protocol].module = NULL;
|
nl_table[sk->sk_protocol].module = NULL;
|
||||||
nl_table[sk->sk_protocol].bind = NULL;
|
nl_table[sk->sk_protocol].bind = NULL;
|
||||||
|
nl_table[sk->sk_protocol].unbind = NULL;
|
||||||
nl_table[sk->sk_protocol].flags = 0;
|
nl_table[sk->sk_protocol].flags = 0;
|
||||||
nl_table[sk->sk_protocol].registered = 0;
|
nl_table[sk->sk_protocol].registered = 0;
|
||||||
}
|
}
|
||||||
|
@ -1411,6 +1415,19 @@ static int netlink_realloc_groups(struct sock *sk)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void netlink_unbind(int group, long unsigned int groups,
|
||||||
|
struct netlink_sock *nlk)
|
||||||
|
{
|
||||||
|
int undo;
|
||||||
|
|
||||||
|
if (!nlk->netlink_unbind)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (undo = 0; undo < group; undo++)
|
||||||
|
if (test_bit(group, &groups))
|
||||||
|
nlk->netlink_unbind(undo);
|
||||||
|
}
|
||||||
|
|
||||||
static int netlink_bind(struct socket *sock, struct sockaddr *addr,
|
static int netlink_bind(struct socket *sock, struct sockaddr *addr,
|
||||||
int addr_len)
|
int addr_len)
|
||||||
{
|
{
|
||||||
|
@ -1419,6 +1436,7 @@ static int netlink_bind(struct socket *sock, struct sockaddr *addr,
|
||||||
struct netlink_sock *nlk = nlk_sk(sk);
|
struct netlink_sock *nlk = nlk_sk(sk);
|
||||||
struct sockaddr_nl *nladdr = (struct sockaddr_nl *)addr;
|
struct sockaddr_nl *nladdr = (struct sockaddr_nl *)addr;
|
||||||
int err;
|
int err;
|
||||||
|
long unsigned int groups = nladdr->nl_groups;
|
||||||
|
|
||||||
if (addr_len < sizeof(struct sockaddr_nl))
|
if (addr_len < sizeof(struct sockaddr_nl))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -1427,7 +1445,7 @@ static int netlink_bind(struct socket *sock, struct sockaddr *addr,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
/* Only superuser is allowed to listen multicasts */
|
/* Only superuser is allowed to listen multicasts */
|
||||||
if (nladdr->nl_groups) {
|
if (groups) {
|
||||||
if (!netlink_capable(sock, NL_CFG_F_NONROOT_RECV))
|
if (!netlink_capable(sock, NL_CFG_F_NONROOT_RECV))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
err = netlink_realloc_groups(sk);
|
err = netlink_realloc_groups(sk);
|
||||||
|
@ -1435,37 +1453,45 @@ static int netlink_bind(struct socket *sock, struct sockaddr *addr,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nlk->portid) {
|
if (nlk->portid)
|
||||||
if (nladdr->nl_pid != nlk->portid)
|
if (nladdr->nl_pid != nlk->portid)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
} else {
|
|
||||||
|
if (nlk->netlink_bind && groups) {
|
||||||
|
int group;
|
||||||
|
|
||||||
|
for (group = 0; group < nlk->ngroups; group++) {
|
||||||
|
if (!test_bit(group, &groups))
|
||||||
|
continue;
|
||||||
|
err = nlk->netlink_bind(group);
|
||||||
|
if (!err)
|
||||||
|
continue;
|
||||||
|
netlink_unbind(group, groups, nlk);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!nlk->portid) {
|
||||||
err = nladdr->nl_pid ?
|
err = nladdr->nl_pid ?
|
||||||
netlink_insert(sk, net, nladdr->nl_pid) :
|
netlink_insert(sk, net, nladdr->nl_pid) :
|
||||||
netlink_autobind(sock);
|
netlink_autobind(sock);
|
||||||
if (err)
|
if (err) {
|
||||||
|
netlink_unbind(nlk->ngroups - 1, groups, nlk);
|
||||||
return err;
|
return err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!nladdr->nl_groups && (nlk->groups == NULL || !(u32)nlk->groups[0]))
|
if (!groups && (nlk->groups == NULL || !(u32)nlk->groups[0]))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
netlink_table_grab();
|
netlink_table_grab();
|
||||||
netlink_update_subscriptions(sk, nlk->subscriptions +
|
netlink_update_subscriptions(sk, nlk->subscriptions +
|
||||||
hweight32(nladdr->nl_groups) -
|
hweight32(groups) -
|
||||||
hweight32(nlk->groups[0]));
|
hweight32(nlk->groups[0]));
|
||||||
nlk->groups[0] = (nlk->groups[0] & ~0xffffffffUL) | nladdr->nl_groups;
|
nlk->groups[0] = (nlk->groups[0] & ~0xffffffffUL) | groups;
|
||||||
netlink_update_listeners(sk);
|
netlink_update_listeners(sk);
|
||||||
netlink_table_ungrab();
|
netlink_table_ungrab();
|
||||||
|
|
||||||
if (nlk->netlink_bind && nlk->groups[0]) {
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < nlk->ngroups; i++) {
|
|
||||||
if (test_bit(i, nlk->groups))
|
|
||||||
nlk->netlink_bind(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2103,14 +2129,16 @@ static int netlink_setsockopt(struct socket *sock, int level, int optname,
|
||||||
return err;
|
return err;
|
||||||
if (!val || val - 1 >= nlk->ngroups)
|
if (!val || val - 1 >= nlk->ngroups)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
if (nlk->netlink_bind) {
|
||||||
|
err = nlk->netlink_bind(val);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
}
|
||||||
netlink_table_grab();
|
netlink_table_grab();
|
||||||
netlink_update_socket_mc(nlk, val,
|
netlink_update_socket_mc(nlk, val,
|
||||||
optname == NETLINK_ADD_MEMBERSHIP);
|
optname == NETLINK_ADD_MEMBERSHIP);
|
||||||
netlink_table_ungrab();
|
netlink_table_ungrab();
|
||||||
|
|
||||||
if (nlk->netlink_bind)
|
|
||||||
nlk->netlink_bind(val);
|
|
||||||
|
|
||||||
err = 0;
|
err = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,7 +38,8 @@ struct netlink_sock {
|
||||||
struct mutex *cb_mutex;
|
struct mutex *cb_mutex;
|
||||||
struct mutex cb_def_mutex;
|
struct mutex cb_def_mutex;
|
||||||
void (*netlink_rcv)(struct sk_buff *skb);
|
void (*netlink_rcv)(struct sk_buff *skb);
|
||||||
void (*netlink_bind)(int group);
|
int (*netlink_bind)(int group);
|
||||||
|
void (*netlink_unbind)(int group);
|
||||||
struct module *module;
|
struct module *module;
|
||||||
#ifdef CONFIG_NETLINK_MMAP
|
#ifdef CONFIG_NETLINK_MMAP
|
||||||
struct mutex pg_vec_lock;
|
struct mutex pg_vec_lock;
|
||||||
|
@ -74,7 +75,8 @@ struct netlink_table {
|
||||||
unsigned int groups;
|
unsigned int groups;
|
||||||
struct mutex *cb_mutex;
|
struct mutex *cb_mutex;
|
||||||
struct module *module;
|
struct module *module;
|
||||||
void (*bind)(int group);
|
int (*bind)(int group);
|
||||||
|
void (*unbind)(int group);
|
||||||
bool (*compare)(struct net *net, struct sock *sock);
|
bool (*compare)(struct net *net, struct sock *sock);
|
||||||
int registered;
|
int registered;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue