- count是一个标准的使用计数器,在使用特定的net实例前后,需要分别调用辅助函数get_net和put_net. 在count降低到0时,将释放该命名空间,并将其从系统中删除.
- 所有可用的命名空间都保存在一个双链表上,表头是net_namespace_list 。list用作链表元素. copy_net_ns函数向该链表添加一个新的命名空间 。在用create_new_namespace创建一组新的命名空间时,会自动调用该函数 。
- 由于每个命名空间都包含不同的网络设备,这必然会反映到procfs的内容上(参见10.1节) 。各命名空间的处理需要三个数据项:/proc/net由proc_net表示,而/proc/net/stats由proc_net_stats表示,proc_net_root指向当前命名空间的procfs实例的根结点,即/proc
- 每个命名空间都可以有一个不同的环回设备,而loopback_dev指向履行该职责的(虚拟)网络设备.
- 网络设备由struct net_device表示 。与特定命名空间关联的所有设备都保存在一个双链表上,表头为dev_base_head 。各个设备还通过另外两个双链表维护:一个将设备名用作散列键(dev_name_head),另一个将接口索引用作散列键(dev_index_head) 。
- 请注意,术语“设备”和“接口”有细微的差别 。设备表示提供物理传输能力的硬件设备,而接口可以是纯虚拟的实体,可能在真正的设备上实现 。例如,一个网卡可以提供两个接口 。对我们来说,两个术语的区别不那么重要,在下文中将交替使用这两个术语 。网络子系统的许多组件仍然需要做很多工作才能正确处理命名空间,要使网络子系统能够完全感知命名空间,还有相当长的路要走 。例如,内核版本2.6.25(在撰写本章时,仍处于开发中)将开始一些最初的准备工作,以便使特定的协议能够感知到命名空间:
struct net{ /*......*/struct net_device*loopback_dev;/* The loopback */struct netns_corecore;struct netns_mibmib;struct netns_packetpacket;struct netns_unixunx;struct netns_ipv4ipv4;#if IS_ENABLED(CONFIG_IPV6)struct netns_ipv6ipv6;#endif /*......*/}
- ipv4用于存储协议参数(此前是全局的), 为此引入了特定于协议的结构. 这个方 法是逐步进行的:首先设置好基本框架,后续的各个步骤,将全局属性迁移到各命名空间的表示,这些结构最初都是空的 。在未来的内核版本中,还将引入更多此类代码
- linux系统包括默认的命名空间 : "init_net"和用户自定义的net
- 我们通常说的namespace 一般是默认的命名空间 : "init_net", 也就是所有的"网络通信协议"+"网络设备"都是属于默认的命名空间.
- 大多数计算机通常都只需要一个网络命名空间. 即只有默认命名空间init_net(该变量实际上是全局的,并未包含在另一个命名空间中,其定义如下:
//http://lxr.free-electrons.com/source/net/core/net_namespace.c?v=4.7#L35struct net init_net = {.dev_base_head = LIST_HEAD_INIT(init_net.dev_base_head),};EXPORT_SYMBOL(init_net);
- init_net会被链接到net_namespace_list这个双向链表上, 定义如下所示:
- net_namespace_list就包含了所有的网络命令空间, 其以init_net为表头
LIST_HEAD(net_namespace_list);EXPORT_SYMBOL_GPL(net_namespace_list);
2.2 初始化 & 清理元组 pernet_operations(创建命名空间)初始化 & 清理元组pernet_operations- 每个网络命名空间由几个部分组成, 例如, 在procfs中的表示. 每当创建一个新的网络命名空间时, 必须初始化这些部分. 在删除命名空间时, 也同样需要一些清理工作. 内核采用下列结构来跟踪所有必需的初始化/清理元组.
//http://lxr.free-electrons.com/source/include/net/net_namespace.h?v=4.7#L288struct pernet_operations {struct list_head list;int (*init)(struct net *net);void (*exit)(struct net *net);void (*exit_batch)(struct list_head *net_exit_list);int *id;size_t size;};//http://lxr.free-electrons.com/source/include/net/net_namespace.h?v=4.7#L316int register_pernet_subsys(struct pernet_operations *);void unregister_pernet_subsys(struct pernet_operations *);int register_pernet_device(struct pernet_operations *);void unregister_pernet_device(struct pernet_operations *);
推荐阅读
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- Linux运维三剑客 grep、awk、sed实用笔记
- CVE-2021-4034 关于 Linux Polkit 权限提升漏洞的修复方法
- 2022 年保护 Linux 服务器的 10 种流行开源工具
- 全宇宙最全的Linux运维bash脚本常见用法总结
- 从linux内核看io_uring的实现
- 一文了解在 Python 中如何使用列表推导
- Linux系统中 tar.xz文件解压
- C语言进行Linux网络编程
- linux系统下安装mysql 8.0.26配置文件my.cnf详细注解
- Linux运维常用工具