1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37#include "core.h"
38#include "socket.h"
39#include "name_table.h"
40#include "bearer.h"
41#include "link.h"
42#include "node.h"
43#include "net.h"
44#include <net/genetlink.h>
45
46static const struct nla_policy tipc_nl_policy[TIPC_NLA_MAX + 1] = {
47 [TIPC_NLA_UNSPEC] = { .type = NLA_UNSPEC, },
48 [TIPC_NLA_BEARER] = { .type = NLA_NESTED, },
49 [TIPC_NLA_SOCK] = { .type = NLA_NESTED, },
50 [TIPC_NLA_PUBL] = { .type = NLA_NESTED, },
51 [TIPC_NLA_LINK] = { .type = NLA_NESTED, },
52 [TIPC_NLA_MEDIA] = { .type = NLA_NESTED, },
53 [TIPC_NLA_NODE] = { .type = NLA_NESTED, },
54 [TIPC_NLA_NET] = { .type = NLA_NESTED, },
55 [TIPC_NLA_NAME_TABLE] = { .type = NLA_NESTED, }
56};
57
58
59
60
61struct genl_family tipc_genl_family = {
62 .id = GENL_ID_GENERATE,
63 .name = TIPC_GENL_V2_NAME,
64 .version = TIPC_GENL_V2_VERSION,
65 .hdrsize = 0,
66 .maxattr = TIPC_NLA_MAX,
67 .netnsok = true,
68};
69
70static const struct genl_ops tipc_genl_v2_ops[] = {
71 {
72 .cmd = TIPC_NL_BEARER_DISABLE,
73 .doit = tipc_nl_bearer_disable,
74 .policy = tipc_nl_policy,
75 },
76 {
77 .cmd = TIPC_NL_BEARER_ENABLE,
78 .doit = tipc_nl_bearer_enable,
79 .policy = tipc_nl_policy,
80 },
81 {
82 .cmd = TIPC_NL_BEARER_GET,
83 .doit = tipc_nl_bearer_get,
84 .dumpit = tipc_nl_bearer_dump,
85 .policy = tipc_nl_policy,
86 },
87 {
88 .cmd = TIPC_NL_BEARER_SET,
89 .doit = tipc_nl_bearer_set,
90 .policy = tipc_nl_policy,
91 },
92 {
93 .cmd = TIPC_NL_SOCK_GET,
94 .dumpit = tipc_nl_sk_dump,
95 .policy = tipc_nl_policy,
96 },
97 {
98 .cmd = TIPC_NL_PUBL_GET,
99 .dumpit = tipc_nl_publ_dump,
100 .policy = tipc_nl_policy,
101 },
102 {
103 .cmd = TIPC_NL_LINK_GET,
104 .doit = tipc_nl_link_get,
105 .dumpit = tipc_nl_link_dump,
106 .policy = tipc_nl_policy,
107 },
108 {
109 .cmd = TIPC_NL_LINK_SET,
110 .doit = tipc_nl_link_set,
111 .policy = tipc_nl_policy,
112 },
113 {
114 .cmd = TIPC_NL_LINK_RESET_STATS,
115 .doit = tipc_nl_link_reset_stats,
116 .policy = tipc_nl_policy,
117 },
118 {
119 .cmd = TIPC_NL_MEDIA_GET,
120 .doit = tipc_nl_media_get,
121 .dumpit = tipc_nl_media_dump,
122 .policy = tipc_nl_policy,
123 },
124 {
125 .cmd = TIPC_NL_MEDIA_SET,
126 .doit = tipc_nl_media_set,
127 .policy = tipc_nl_policy,
128 },
129 {
130 .cmd = TIPC_NL_NODE_GET,
131 .dumpit = tipc_nl_node_dump,
132 .policy = tipc_nl_policy,
133 },
134 {
135 .cmd = TIPC_NL_NET_GET,
136 .dumpit = tipc_nl_net_dump,
137 .policy = tipc_nl_policy,
138 },
139 {
140 .cmd = TIPC_NL_NET_SET,
141 .doit = tipc_nl_net_set,
142 .policy = tipc_nl_policy,
143 },
144 {
145 .cmd = TIPC_NL_NAME_TABLE_GET,
146 .dumpit = tipc_nl_name_table_dump,
147 .policy = tipc_nl_policy,
148 }
149};
150
151int tipc_nlmsg_parse(const struct nlmsghdr *nlh, struct nlattr ***attr)
152{
153 u32 maxattr = tipc_genl_family.maxattr;
154
155 *attr = tipc_genl_family.attrbuf;
156 if (!*attr)
157 return -EOPNOTSUPP;
158
159 return nlmsg_parse(nlh, GENL_HDRLEN, *attr, maxattr, tipc_nl_policy);
160}
161
162int tipc_netlink_start(void)
163{
164 int res;
165
166 res = genl_register_family_with_ops(&tipc_genl_family,
167 tipc_genl_v2_ops);
168 if (res) {
169 pr_err("Failed to register netlink interface\n");
170 return res;
171 }
172 return 0;
173}
174
175void tipc_netlink_stop(void)
176{
177 genl_unregister_family(&tipc_genl_family);
178}
179