mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-26 07:40:48 +00:00 
			
		
		
		
	 134e63756d
			
		
	
	
		134e63756d
		
	
	
	
	
		
			
			This makes generic netlink network namespace aware. No generic netlink families except for the controller family are made namespace aware, they need to be checked one by one and then set the family->netnsok member to true. A new function genlmsg_multicast_netns() is introduced to allow sending a multicast message in a given namespace, for example when it applies to an object that lives in that namespace, a new function genlmsg_multicast_allns() to send a message to all network namespaces (for objects that do not have an associated netns). The function genlmsg_multicast() is changed to multicast the message in just init_net, which is currently correct for all generic netlink families since they only work in init_net right now. Some will later want to work in all net namespaces because they do not care about the netns at all -- those will have to be converted to use one of the new functions genlmsg_multicast_allns() or genlmsg_multicast_netns() whenever they are made netns aware in some way. After this patch families can easily decide whether or not they should be available in all net namespaces. Many genl families us it for objects not related to networking and should therefore be available in all namespaces, but that will have to be done on a per family basis. Note that this doesn't touch on the checkpoint/restart problem where network namespaces could be used, genl families and multicast groups are numbered globally and I see no easy way of changing that, especially since it must be possible to multicast to all network namespaces for those families that do not care about netns. Signed-off-by: Johannes Berg <johannes@sipsolutions.net> Signed-off-by: David S. Miller <davem@davemloft.net>
		
			
				
	
	
		
			109 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			109 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * net/tipc/netlink.c: TIPC configuration handling
 | |
|  *
 | |
|  * Copyright (c) 2005-2006, Ericsson AB
 | |
|  * Copyright (c) 2005-2007, Wind River Systems
 | |
|  * All rights reserved.
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without
 | |
|  * modification, are permitted provided that the following conditions are met:
 | |
|  *
 | |
|  * 1. Redistributions of source code must retain the above copyright
 | |
|  *    notice, this list of conditions and the following disclaimer.
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright
 | |
|  *    notice, this list of conditions and the following disclaimer in the
 | |
|  *    documentation and/or other materials provided with the distribution.
 | |
|  * 3. Neither the names of the copyright holders nor the names of its
 | |
|  *    contributors may be used to endorse or promote products derived from
 | |
|  *    this software without specific prior written permission.
 | |
|  *
 | |
|  * Alternatively, this software may be distributed under the terms of the
 | |
|  * GNU General Public License ("GPL") version 2 as published by the Free
 | |
|  * Software Foundation.
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | |
|  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | |
|  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 | |
|  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 | |
|  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 | |
|  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | |
|  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 | |
|  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 | |
|  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 | |
|  * POSSIBILITY OF SUCH DAMAGE.
 | |
|  */
 | |
| 
 | |
| #include "core.h"
 | |
| #include "config.h"
 | |
| #include <net/genetlink.h>
 | |
| 
 | |
| static int handle_cmd(struct sk_buff *skb, struct genl_info *info)
 | |
| {
 | |
| 	struct sk_buff *rep_buf;
 | |
| 	struct nlmsghdr *rep_nlh;
 | |
| 	struct nlmsghdr *req_nlh = info->nlhdr;
 | |
| 	struct tipc_genlmsghdr *req_userhdr = info->userhdr;
 | |
| 	int hdr_space = NLMSG_SPACE(GENL_HDRLEN + TIPC_GENL_HDRLEN);
 | |
| 	u16 cmd;
 | |
| 
 | |
| 	if ((req_userhdr->cmd & 0xC000) && (!capable(CAP_NET_ADMIN)))
 | |
| 		cmd = TIPC_CMD_NOT_NET_ADMIN;
 | |
| 	else
 | |
| 		cmd = req_userhdr->cmd;
 | |
| 
 | |
| 	rep_buf = tipc_cfg_do_cmd(req_userhdr->dest, cmd,
 | |
| 			NLMSG_DATA(req_nlh) + GENL_HDRLEN + TIPC_GENL_HDRLEN,
 | |
| 			NLMSG_PAYLOAD(req_nlh, GENL_HDRLEN + TIPC_GENL_HDRLEN),
 | |
| 			hdr_space);
 | |
| 
 | |
| 	if (rep_buf) {
 | |
| 		skb_push(rep_buf, hdr_space);
 | |
| 		rep_nlh = nlmsg_hdr(rep_buf);
 | |
| 		memcpy(rep_nlh, req_nlh, hdr_space);
 | |
| 		rep_nlh->nlmsg_len = rep_buf->len;
 | |
| 		genlmsg_unicast(&init_net, rep_buf, NETLINK_CB(skb).pid);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct genl_family tipc_genl_family = {
 | |
| 	.id		= GENL_ID_GENERATE,
 | |
| 	.name		= TIPC_GENL_NAME,
 | |
| 	.version	= TIPC_GENL_VERSION,
 | |
| 	.hdrsize	= TIPC_GENL_HDRLEN,
 | |
| 	.maxattr	= 0,
 | |
| };
 | |
| 
 | |
| static struct genl_ops tipc_genl_ops = {
 | |
| 	.cmd		= TIPC_GENL_CMD,
 | |
| 	.doit		= handle_cmd,
 | |
| };
 | |
| 
 | |
| static int tipc_genl_family_registered;
 | |
| 
 | |
| int tipc_netlink_start(void)
 | |
| {
 | |
| 	int res;
 | |
| 
 | |
| 	res = genl_register_family_with_ops(&tipc_genl_family,
 | |
| 		&tipc_genl_ops, 1);
 | |
| 	if (res) {
 | |
| 		err("Failed to register netlink interface\n");
 | |
| 		return res;
 | |
| 	}
 | |
| 
 | |
| 	tipc_genl_family_registered = 1;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void tipc_netlink_stop(void)
 | |
| {
 | |
| 	if (!tipc_genl_family_registered)
 | |
| 		return;
 | |
| 
 | |
| 	genl_unregister_family(&tipc_genl_family);
 | |
| 	tipc_genl_family_registered = 0;
 | |
| }
 |