通用netlink中的策略和属性的概念是什么?

Kin*_*ing 5 linux kernel module netlink

我是netlink编程新手.我正在编写一个generic netlink用于创建netlink协议族的程序.我在互联网上搜索了很多文档,我发现了一些"属性和策略",就像定义一个netlink家庭一样.

我对这些事情感到很困惑.

我发现了关于属性的轰鸣声 linux/netlink.h

 <------- NLA_HDRLEN ------> <-- NLA_ALIGN(payload)-->
+---------------------------+- - -+- - - - - - - - - -+- - -+
|        Header             | Pad |    Payload        | Pad | 
|   (struct nlattr)         | ing |                   | ing |
+---------------------------+- - -+- - - - - - - - - -+- - -+
 <-------------------- nlattr->nla_len -------------->
Run Code Online (Sandbox Code Playgroud)

政策是一系列nla_policy结构.我的问题是:

  1. 标题和属性之间有什么关系?请解释"属性".
  2. 什么是政策,需要什么,为什么我们为此使用数组?

我找到了一些关于"它定义属性类型"的政策,
这是什么意思?我的意思是"属性类型的含义是什么?"

这可能是一个无意义的问题,但我完全感到困惑.我一直试图了解这些事情超过三天,请帮助我.

谢谢..

tij*_*jko 5

在创建/使用netlink协议时,netlink属性旨在为协议提供干净的自我文档布局,以允许将来的可扩展性.这意味着如果您想要使用除当前协议中已存在的数据类型之外的其他数据类型,则代码将兼容而不会破坏已存在的操作.

"属性"是依赖于协议的,并且涉及使用所述协议发送的特定消息.

taskstats接口为例:

taskstat属性:

enum {
    TASKSTATS_CMD_ATTR_UNSPEC = 0,
    TASKSTATS_CMD_ATTR_PID,
    TASKSTATS_CMD_ATTR_TGID,
    TASKSTATS_CMD_ATTR_REGISTER_CPUMASK,
    TASKSTATS_CMD_ATTR_DEREGISTER_CPUMASK,
    __TASKSTATS_CMD_ATTR_MAX,
};
Run Code Online (Sandbox Code Playgroud)

在这些属性中,您可以通过在属性之间添加自定义属性UNSPEC并将MAX该属性映射到所需的特定函数或操作来轻松地"扩展"它们.

内核空间taskstat策略:

static const struct nla_policy taskstats_cmd_get_policy[TASKSTATS_CMD_ATTR_MAX+1] = {
    [TASKSTATS_CMD_ATTR_PID]  = { .type = NLA_U32 },
    [TASKSTATS_CMD_ATTR_TGID] = { .type = NLA_U32 },
    [TASKSTATS_CMD_ATTR_REGISTER_CPUMASK] = { .type = NLA_STRING },
    [TASKSTATS_CMD_ATTR_DEREGISTER_CPUMASK] = { .type = NLA_STRING },};
Run Code Online (Sandbox Code Playgroud)

我相信你已经遇到了定义struct nlattr,使用NETLINK_GENERIC协议和taskstats接口加载这个struct的字段的例子:

struct nlattr na;
na.nla_type = CTRL_ATTR_FAMILY_NAME;         // defined in linux/genetlink.h 
na.nla_len = strlen(TASKSTATS_GENL_NAME) + 1 // defined in linux/taskstats.h

// note: you will need to copy/access nlattr data in the same way the NLMSG_DATA
//       macro operates.
Run Code Online (Sandbox Code Playgroud)

现在在内核方面,在解析这些属性时,将调用关联的函数以及有关如何继续的预期操作.

我不确定你发布的图表是否会让你失望,但是,为了给你一个更大的视角,你可以缩小一点:

根据内核源码v3.16 include/net/netlink.h:

/* ========================================================================
 *         Netlink Messages and Attributes Interface (As Seen On TV)
 * ------------------------------------------------------------------------
 *                          Messages Interface
 * ------------------------------------------------------------------------
 *
 * Message Format:
 *    <--- nlmsg_total_size(payload)  --->
 *    <-- nlmsg_msg_size(payload) ->
 *   +----------+- - -+-------------+- - -+-------- - -
 *   | nlmsghdr | Pad |   Payload   | Pad | nlmsghdr
 *   +----------+- - -+-------------+- - -+-------- - -
 *   nlmsg_data(nlh)---^                   ^
 *   nlmsg_next(nlh)-----------------------+
 *
 * Payload Format:
 *    <---------------------- nlmsg_len(nlh) --------------------->
 *    <------ hdrlen ------>       <- nlmsg_attrlen(nlh, hdrlen) ->
 *   +----------------------+- - -+--------------------------------+
 *   |     Family Header    | Pad |           Attributes           |
 *   +----------------------+- - -+--------------------------------+
 *   nlmsg_attrdata(nlh, hdrlen)---^
Run Code Online (Sandbox Code Playgroud)

在这里,您可以看到您发布的标头和有效负载图表只是更大负载的一部分.该段与struct nlmsghdr消息格式中的a一起出现.

现在在策略上,当发送netlink消息时,发送者需要遵守协议格式.消息的接收方将用于struct nla_policy在访问有效负载之前验证属性.

内核使用"系列"或标识符来跟踪要与之通信的适当协议接口,无论是标准协议还是自定义通用Netlink.

当你问"我们可以避免这种情况吗?"时,如果你通过编写自己的自定义通用netlink协议来扩展netlink,那么这些协议可以很容易地调整和维护,而无需通过和更改/修复与之相关的所有操作或让协议崩溃的扁平化.如果没有关联的长度或类型,您建议如何解析具有不同数据类型的嵌套消息?类型和长度允许在正确的对齐上解析消息并允许进行所需的操作.如果没有为有效负载提供标签的属性类型,您将如何解释它,"有效负载是什么"?没有长度你怎么知道有效载荷"有多大"?可能存在多个具有不同长度的有效载荷,而无需区分它们的大小,因此无法分辨出一个开始和另一个结束的位置.

以下是libnl(用于处理netlink套接字的库,强烈建议使用)文档属性的链接.