Commit a31f2d17 authored by Pablo Neira Ayuso's avatar Pablo Neira Ayuso Committed by David S. Miller

netlink: add netlink_kernel_cfg parameter to netlink_kernel_create

This patch adds the following structure:

struct netlink_kernel_cfg {
        unsigned int    groups;
        void            (*input)(struct sk_buff *skb);
        struct mutex    *cb_mutex;
};

That can be passed to netlink_kernel_create to set optional configurations
for netlink kernel sockets.

I've populated this structure by looking for NULL and zero parameters at the
existing code. The remaining parameters that always need to be set are still
left in the original interface.

That includes optional parameters for the netlink socket creation. This allows
easy extensibility of this interface in the future.

This patch also adapts all callers to use this new interface.
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent dd7f36ba
...@@ -496,9 +496,12 @@ static void crypto_netlink_rcv(struct sk_buff *skb) ...@@ -496,9 +496,12 @@ static void crypto_netlink_rcv(struct sk_buff *skb)
static int __init crypto_user_init(void) static int __init crypto_user_init(void)
{ {
struct netlink_kernel_cfg cfg = {
.input = crypto_netlink_rcv,
};
crypto_nlsk = netlink_kernel_create(&init_net, NETLINK_CRYPTO, crypto_nlsk = netlink_kernel_create(&init_net, NETLINK_CRYPTO,
0, crypto_netlink_rcv, THIS_MODULE, &cfg);
NULL, THIS_MODULE);
if (!crypto_nlsk) if (!crypto_nlsk)
return -ENOMEM; return -ENOMEM;
......
...@@ -251,15 +251,20 @@ static const struct file_operations cn_file_ops = { ...@@ -251,15 +251,20 @@ static const struct file_operations cn_file_ops = {
.release = single_release .release = single_release
}; };
static struct cn_dev cdev = {
.input = cn_rx_skb,
};
static int __devinit cn_init(void) static int __devinit cn_init(void)
{ {
struct cn_dev *dev = &cdev; struct cn_dev *dev = &cdev;
struct netlink_kernel_cfg cfg = {
dev->input = cn_rx_skb; .groups = CN_NETLINK_USERS + 0xf,
.input = dev->input,
};
dev->nls = netlink_kernel_create(&init_net, NETLINK_CONNECTOR, dev->nls = netlink_kernel_create(&init_net, NETLINK_CONNECTOR,
CN_NETLINK_USERS + 0xf, THIS_MODULE, &cfg);
dev->input, NULL, THIS_MODULE);
if (!dev->nls) if (!dev->nls)
return -EIO; return -EIO;
......
...@@ -173,8 +173,11 @@ static void ibnl_rcv(struct sk_buff *skb) ...@@ -173,8 +173,11 @@ static void ibnl_rcv(struct sk_buff *skb)
int __init ibnl_init(void) int __init ibnl_init(void)
{ {
nls = netlink_kernel_create(&init_net, NETLINK_RDMA, 0, ibnl_rcv, struct netlink_kernel_cfg cfg = {
NULL, THIS_MODULE); .input = ibnl_rcv,
};
nls = netlink_kernel_create(&init_net, NETLINK_RDMA, THIS_MODULE, &cfg);
if (!nls) { if (!nls) {
pr_warn("Failed to create netlink socket\n"); pr_warn("Failed to create netlink socket\n");
return -ENOMEM; return -ENOMEM;
......
...@@ -486,6 +486,10 @@ void ...@@ -486,6 +486,10 @@ void
scsi_netlink_init(void) scsi_netlink_init(void)
{ {
int error; int error;
struct netlink_kernel_cfg cfg = {
.input = scsi_nl_rcv_msg,
.groups = SCSI_NL_GRP_CNT,
};
INIT_LIST_HEAD(&scsi_nl_drivers); INIT_LIST_HEAD(&scsi_nl_drivers);
...@@ -497,8 +501,7 @@ scsi_netlink_init(void) ...@@ -497,8 +501,7 @@ scsi_netlink_init(void)
} }
scsi_nl_sock = netlink_kernel_create(&init_net, NETLINK_SCSITRANSPORT, scsi_nl_sock = netlink_kernel_create(&init_net, NETLINK_SCSITRANSPORT,
SCSI_NL_GRP_CNT, scsi_nl_rcv_msg, NULL, THIS_MODULE, &cfg);
THIS_MODULE);
if (!scsi_nl_sock) { if (!scsi_nl_sock) {
printk(KERN_ERR "%s: register of receive handler failed\n", printk(KERN_ERR "%s: register of receive handler failed\n",
__func__); __func__);
......
...@@ -2936,7 +2936,10 @@ EXPORT_SYMBOL_GPL(iscsi_unregister_transport); ...@@ -2936,7 +2936,10 @@ EXPORT_SYMBOL_GPL(iscsi_unregister_transport);
static __init int iscsi_transport_init(void) static __init int iscsi_transport_init(void)
{ {
int err; int err;
struct netlink_kernel_cfg cfg = {
.groups = 1,
.input = iscsi_if_rx,
};
printk(KERN_INFO "Loading iSCSI transport class v%s.\n", printk(KERN_INFO "Loading iSCSI transport class v%s.\n",
ISCSI_TRANSPORT_VERSION); ISCSI_TRANSPORT_VERSION);
...@@ -2966,8 +2969,8 @@ static __init int iscsi_transport_init(void) ...@@ -2966,8 +2969,8 @@ static __init int iscsi_transport_init(void)
if (err) if (err)
goto unregister_conn_class; goto unregister_conn_class;
nls = netlink_kernel_create(&init_net, NETLINK_ISCSI, 1, iscsi_if_rx, nls = netlink_kernel_create(&init_net, NETLINK_ISCSI,
NULL, THIS_MODULE); THIS_MODULE, &cfg);
if (!nls) { if (!nls) {
err = -ENOBUFS; err = -ENOBUFS;
goto unregister_session_class; goto unregister_session_class;
......
...@@ -88,13 +88,15 @@ struct sock *netlink_init(int unit, void (*cb)(struct net_device *dev, u16 type, ...@@ -88,13 +88,15 @@ struct sock *netlink_init(int unit, void (*cb)(struct net_device *dev, u16 type,
void *msg, int len)) void *msg, int len))
{ {
struct sock *sock; struct sock *sock;
struct netlink_kernel_cfg cfg = {
.input = netlink_rcv,
};
#if !defined(DEFINE_MUTEX) #if !defined(DEFINE_MUTEX)
init_MUTEX(&netlink_mutex); init_MUTEX(&netlink_mutex);
#endif #endif
sock = netlink_kernel_create(&init_net, unit, 0, netlink_rcv, NULL, sock = netlink_kernel_create(&init_net, unit, THIS_MODULE, &cfg);
THIS_MODULE);
if (sock) if (sock)
rcv_cb = cb; rcv_cb = cb;
......
...@@ -174,11 +174,16 @@ struct netlink_skb_parms { ...@@ -174,11 +174,16 @@ struct netlink_skb_parms {
extern void netlink_table_grab(void); extern void netlink_table_grab(void);
extern void netlink_table_ungrab(void); extern void netlink_table_ungrab(void);
extern struct sock *netlink_kernel_create(struct net *net, /* optional Netlink kernel configuration parameters */
int unit,unsigned int groups, struct netlink_kernel_cfg {
void (*input)(struct sk_buff *skb), unsigned int groups;
struct mutex *cb_mutex, void (*input)(struct sk_buff *skb);
struct module *module); struct mutex *cb_mutex;
};
extern struct sock *netlink_kernel_create(struct net *net, int unit,
struct module *module,
struct netlink_kernel_cfg *cfg);
extern void netlink_kernel_release(struct sock *sk); extern void netlink_kernel_release(struct sock *sk);
extern int __netlink_change_ngroups(struct sock *sk, unsigned int groups); extern int __netlink_change_ngroups(struct sock *sk, unsigned int groups);
extern int netlink_change_ngroups(struct sock *sk, unsigned int groups); extern int netlink_change_ngroups(struct sock *sk, unsigned int groups);
......
...@@ -962,14 +962,17 @@ static void audit_receive(struct sk_buff *skb) ...@@ -962,14 +962,17 @@ static void audit_receive(struct sk_buff *skb)
static int __init audit_init(void) static int __init audit_init(void)
{ {
int i; int i;
struct netlink_kernel_cfg cfg = {
.input = audit_receive,
};
if (audit_initialized == AUDIT_DISABLED) if (audit_initialized == AUDIT_DISABLED)
return 0; return 0;
printk(KERN_INFO "audit: initializing netlink socket (%s)\n", printk(KERN_INFO "audit: initializing netlink socket (%s)\n",
audit_default ? "enabled" : "disabled"); audit_default ? "enabled" : "disabled");
audit_sock = netlink_kernel_create(&init_net, NETLINK_AUDIT, 0, audit_sock = netlink_kernel_create(&init_net, NETLINK_AUDIT,
audit_receive, NULL, THIS_MODULE); THIS_MODULE, &cfg);
if (!audit_sock) if (!audit_sock)
audit_panic("cannot initialize netlink socket"); audit_panic("cannot initialize netlink socket");
else else
......
...@@ -373,13 +373,16 @@ EXPORT_SYMBOL_GPL(add_uevent_var); ...@@ -373,13 +373,16 @@ EXPORT_SYMBOL_GPL(add_uevent_var);
static int uevent_net_init(struct net *net) static int uevent_net_init(struct net *net)
{ {
struct uevent_sock *ue_sk; struct uevent_sock *ue_sk;
struct netlink_kernel_cfg cfg = {
.groups = 1,
};
ue_sk = kzalloc(sizeof(*ue_sk), GFP_KERNEL); ue_sk = kzalloc(sizeof(*ue_sk), GFP_KERNEL);
if (!ue_sk) if (!ue_sk)
return -ENOMEM; return -ENOMEM;
ue_sk->sk = netlink_kernel_create(net, NETLINK_KOBJECT_UEVENT, ue_sk->sk = netlink_kernel_create(net, NETLINK_KOBJECT_UEVENT,
1, NULL, NULL, THIS_MODULE); THIS_MODULE, &cfg);
if (!ue_sk->sk) { if (!ue_sk->sk) {
printk(KERN_ERR printk(KERN_ERR
"kobject_uevent: unable to create netlink socket!\n"); "kobject_uevent: unable to create netlink socket!\n");
......
...@@ -282,6 +282,9 @@ static int __init ebt_ulog_init(void) ...@@ -282,6 +282,9 @@ static int __init ebt_ulog_init(void)
{ {
int ret; int ret;
int i; int i;
struct netlink_kernel_cfg cfg = {
.groups = EBT_ULOG_MAXNLGROUPS,
};
if (nlbufsiz >= 128*1024) { if (nlbufsiz >= 128*1024) {
pr_warning("Netlink buffer has to be <= 128kB," pr_warning("Netlink buffer has to be <= 128kB,"
...@@ -296,8 +299,7 @@ static int __init ebt_ulog_init(void) ...@@ -296,8 +299,7 @@ static int __init ebt_ulog_init(void)
} }
ebtulognl = netlink_kernel_create(&init_net, NETLINK_NFLOG, ebtulognl = netlink_kernel_create(&init_net, NETLINK_NFLOG,
EBT_ULOG_MAXNLGROUPS, NULL, NULL, THIS_MODULE, &cfg);
THIS_MODULE);
if (!ebtulognl) if (!ebtulognl)
ret = -ENOMEM; ret = -ENOMEM;
else if ((ret = xt_register_target(&ebt_ulog_tg_reg)) != 0) else if ((ret = xt_register_target(&ebt_ulog_tg_reg)) != 0)
......
...@@ -2353,8 +2353,13 @@ static struct notifier_block rtnetlink_dev_notifier = { ...@@ -2353,8 +2353,13 @@ static struct notifier_block rtnetlink_dev_notifier = {
static int __net_init rtnetlink_net_init(struct net *net) static int __net_init rtnetlink_net_init(struct net *net)
{ {
struct sock *sk; struct sock *sk;
sk = netlink_kernel_create(net, NETLINK_ROUTE, RTNLGRP_MAX, struct netlink_kernel_cfg cfg = {
rtnetlink_rcv, &rtnl_mutex, THIS_MODULE); .groups = RTNLGRP_MAX,
.input = rtnetlink_rcv,
.cb_mutex = &rtnl_mutex,
};
sk = netlink_kernel_create(net, NETLINK_ROUTE, THIS_MODULE, &cfg);
if (!sk) if (!sk)
return -ENOMEM; return -ENOMEM;
net->rtnl = sk; net->rtnl = sk;
......
...@@ -171,8 +171,12 @@ EXPORT_SYMBOL_GPL(sock_diag_nlsk); ...@@ -171,8 +171,12 @@ EXPORT_SYMBOL_GPL(sock_diag_nlsk);
static int __init sock_diag_init(void) static int __init sock_diag_init(void)
{ {
sock_diag_nlsk = netlink_kernel_create(&init_net, NETLINK_SOCK_DIAG, 0, struct netlink_kernel_cfg cfg = {
sock_diag_rcv, NULL, THIS_MODULE); .input = sock_diag_rcv,
};
sock_diag_nlsk = netlink_kernel_create(&init_net, NETLINK_SOCK_DIAG,
THIS_MODULE, &cfg);
return sock_diag_nlsk == NULL ? -ENOMEM : 0; return sock_diag_nlsk == NULL ? -ENOMEM : 0;
} }
......
...@@ -125,11 +125,13 @@ static struct nf_hook_ops dnrmg_ops __read_mostly = { ...@@ -125,11 +125,13 @@ static struct nf_hook_ops dnrmg_ops __read_mostly = {
static int __init dn_rtmsg_init(void) static int __init dn_rtmsg_init(void)
{ {
int rv = 0; int rv = 0;
struct netlink_kernel_cfg cfg = {
.groups = DNRNG_NLGRP_MAX,
.input = dnrmg_receive_user_skb,
};
dnrmg = netlink_kernel_create(&init_net, dnrmg = netlink_kernel_create(&init_net,
NETLINK_DNRTMSG, DNRNG_NLGRP_MAX, NETLINK_DNRTMSG, THIS_MODULE, &cfg);
dnrmg_receive_user_skb,
NULL, THIS_MODULE);
if (dnrmg == NULL) { if (dnrmg == NULL) {
printk(KERN_ERR "dn_rtmsg: Cannot create netlink socket"); printk(KERN_ERR "dn_rtmsg: Cannot create netlink socket");
return -ENOMEM; return -ENOMEM;
......
...@@ -976,8 +976,11 @@ static void nl_fib_input(struct sk_buff *skb) ...@@ -976,8 +976,11 @@ static void nl_fib_input(struct sk_buff *skb)
static int __net_init nl_fib_lookup_init(struct net *net) static int __net_init nl_fib_lookup_init(struct net *net)
{ {
struct sock *sk; struct sock *sk;
sk = netlink_kernel_create(net, NETLINK_FIB_LOOKUP, 0, struct netlink_kernel_cfg cfg = {
nl_fib_input, NULL, THIS_MODULE); .input = nl_fib_input,
};
sk = netlink_kernel_create(net, NETLINK_FIB_LOOKUP, THIS_MODULE, &cfg);
if (sk == NULL) if (sk == NULL)
return -EAFNOSUPPORT; return -EAFNOSUPPORT;
net->ipv4.fibnl = sk; net->ipv4.fibnl = sk;
......
...@@ -381,6 +381,9 @@ static struct nf_logger ipt_ulog_logger __read_mostly = { ...@@ -381,6 +381,9 @@ static struct nf_logger ipt_ulog_logger __read_mostly = {
static int __init ulog_tg_init(void) static int __init ulog_tg_init(void)
{ {
int ret, i; int ret, i;
struct netlink_kernel_cfg cfg = {
.groups = ULOG_MAXNLGROUPS,
};
pr_debug("init module\n"); pr_debug("init module\n");
...@@ -393,9 +396,8 @@ static int __init ulog_tg_init(void) ...@@ -393,9 +396,8 @@ static int __init ulog_tg_init(void)
for (i = 0; i < ULOG_MAXNLGROUPS; i++) for (i = 0; i < ULOG_MAXNLGROUPS; i++)
setup_timer(&ulog_buffers[i].timer, ulog_timer, i); setup_timer(&ulog_buffers[i].timer, ulog_timer, i);
nflognl = netlink_kernel_create(&init_net, nflognl = netlink_kernel_create(&init_net, NETLINK_NFLOG,
NETLINK_NFLOG, ULOG_MAXNLGROUPS, NULL, THIS_MODULE, &cfg);
NULL, THIS_MODULE);
if (!nflognl) if (!nflognl)
return -ENOMEM; return -ENOMEM;
......
...@@ -203,9 +203,12 @@ static void nfnetlink_rcv(struct sk_buff *skb) ...@@ -203,9 +203,12 @@ static void nfnetlink_rcv(struct sk_buff *skb)
static int __net_init nfnetlink_net_init(struct net *net) static int __net_init nfnetlink_net_init(struct net *net)
{ {
struct sock *nfnl; struct sock *nfnl;
struct netlink_kernel_cfg cfg = {
.groups = NFNLGRP_MAX,
.input = nfnetlink_rcv,
};
nfnl = netlink_kernel_create(net, NETLINK_NETFILTER, NFNLGRP_MAX, nfnl = netlink_kernel_create(net, NETLINK_NETFILTER, THIS_MODULE, &cfg);
nfnetlink_rcv, NULL, THIS_MODULE);
if (!nfnl) if (!nfnl)
return -ENOMEM; return -ENOMEM;
net->nfnl_stash = nfnl; net->nfnl_stash = nfnl;
......
...@@ -1503,14 +1503,16 @@ static void netlink_data_ready(struct sock *sk, int len) ...@@ -1503,14 +1503,16 @@ static void netlink_data_ready(struct sock *sk, int len)
*/ */
struct sock * struct sock *
netlink_kernel_create(struct net *net, int unit, unsigned int groups, netlink_kernel_create(struct net *net, int unit,
void (*input)(struct sk_buff *skb), struct module *module,
struct mutex *cb_mutex, struct module *module) struct netlink_kernel_cfg *cfg)
{ {
struct socket *sock; struct socket *sock;
struct sock *sk; struct sock *sk;
struct netlink_sock *nlk; struct netlink_sock *nlk;
struct listeners *listeners = NULL; struct listeners *listeners = NULL;
struct mutex *cb_mutex = cfg ? cfg->cb_mutex : NULL;
unsigned int groups;
BUG_ON(!nl_table); BUG_ON(!nl_table);
...@@ -1532,16 +1534,18 @@ netlink_kernel_create(struct net *net, int unit, unsigned int groups, ...@@ -1532,16 +1534,18 @@ netlink_kernel_create(struct net *net, int unit, unsigned int groups,
sk = sock->sk; sk = sock->sk;
sk_change_net(sk, net); sk_change_net(sk, net);
if (groups < 32) if (!cfg || cfg->groups < 32)
groups = 32; groups = 32;
else
groups = cfg->groups;
listeners = kzalloc(sizeof(*listeners) + NLGRPSZ(groups), GFP_KERNEL); listeners = kzalloc(sizeof(*listeners) + NLGRPSZ(groups), GFP_KERNEL);
if (!listeners) if (!listeners)
goto out_sock_release; goto out_sock_release;
sk->sk_data_ready = netlink_data_ready; sk->sk_data_ready = netlink_data_ready;
if (input) if (cfg && cfg->input)
nlk_sk(sk)->netlink_rcv = input; nlk_sk(sk)->netlink_rcv = cfg->input;
if (netlink_insert(sk, net, 0)) if (netlink_insert(sk, net, 0))
goto out_sock_release; goto out_sock_release;
......
...@@ -915,10 +915,14 @@ static struct genl_multicast_group notify_grp = { ...@@ -915,10 +915,14 @@ static struct genl_multicast_group notify_grp = {
static int __net_init genl_pernet_init(struct net *net) static int __net_init genl_pernet_init(struct net *net)
{ {
struct netlink_kernel_cfg cfg = {
.input = genl_rcv,
.cb_mutex = &genl_mutex,
};
/* we'll bump the group number right afterwards */ /* we'll bump the group number right afterwards */
net->genl_sock = netlink_kernel_create(net, NETLINK_GENERIC, 0, net->genl_sock = netlink_kernel_create(net, NETLINK_GENERIC,
genl_rcv, &genl_mutex, THIS_MODULE, &cfg);
THIS_MODULE);
if (!net->genl_sock && net_eq(net, &init_net)) if (!net->genl_sock && net_eq(net, &init_net))
panic("GENL: Cannot initialize generic netlink\n"); panic("GENL: Cannot initialize generic netlink\n");
......
...@@ -2959,9 +2959,12 @@ static struct xfrm_mgr netlink_mgr = { ...@@ -2959,9 +2959,12 @@ static struct xfrm_mgr netlink_mgr = {
static int __net_init xfrm_user_net_init(struct net *net) static int __net_init xfrm_user_net_init(struct net *net)
{ {
struct sock *nlsk; struct sock *nlsk;
struct netlink_kernel_cfg cfg = {
.groups = XFRMNLGRP_MAX,
.input = xfrm_netlink_rcv,
};
nlsk = netlink_kernel_create(net, NETLINK_XFRM, XFRMNLGRP_MAX, nlsk = netlink_kernel_create(net, NETLINK_XFRM, THIS_MODULE, &cfg);
xfrm_netlink_rcv, NULL, THIS_MODULE);
if (nlsk == NULL) if (nlsk == NULL)
return -ENOMEM; return -ENOMEM;
net->xfrm.nlsk_stash = nlsk; /* Don't set to NULL */ net->xfrm.nlsk_stash = nlsk; /* Don't set to NULL */
......
...@@ -111,8 +111,12 @@ void selnl_notify_policyload(u32 seqno) ...@@ -111,8 +111,12 @@ void selnl_notify_policyload(u32 seqno)
static int __init selnl_init(void) static int __init selnl_init(void)
{ {
struct netlink_kernel_cfg cfg = {
.groups = SELNLGRP_MAX,
};
selnl = netlink_kernel_create(&init_net, NETLINK_SELINUX, selnl = netlink_kernel_create(&init_net, NETLINK_SELINUX,
SELNLGRP_MAX, NULL, NULL, THIS_MODULE); THIS_MODULE, &cfg);
if (selnl == NULL) if (selnl == NULL)
panic("SELinux: Cannot create netlink socket."); panic("SELinux: Cannot create netlink socket.");
netlink_set_nonroot(NETLINK_SELINUX, NL_NONROOT_RECV); netlink_set_nonroot(NETLINK_SELINUX, NL_NONROOT_RECV);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment