2 * Netlink interface for IEEE 802.15.4 stack
4 * Copyright 2007, 2008 Siemens AG
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2
8 * as published by the Free Software Foundation.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
16 * Sergey Lapin <slapin@ossfans.org>
17 * Dmitry Eremin-Solenikov <dbaryshkov@gmail.com>
18 * Maxim Osipov <maxim.osipov@siemens.com>
21 #include <linux/kernel.h>
22 #include <linux/gfp.h>
23 #include <net/genetlink.h>
24 #include <linux/nl802154.h>
26 #include "ieee802154.h"
28 static unsigned int ieee802154_seq_num
;
29 static DEFINE_SPINLOCK(ieee802154_seq_lock
);
31 /* Requests to userspace */
32 struct sk_buff
*ieee802154_nl_create(int flags
, u8 req
)
35 struct sk_buff
*msg
= nlmsg_new(NLMSG_DEFAULT_SIZE
, GFP_ATOMIC
);
41 spin_lock_irqsave(&ieee802154_seq_lock
, f
);
42 hdr
= genlmsg_put(msg
, 0, ieee802154_seq_num
++,
43 &nl802154_family
, flags
, req
);
44 spin_unlock_irqrestore(&ieee802154_seq_lock
, f
);
53 int ieee802154_nl_mcast(struct sk_buff
*msg
, unsigned int group
)
55 struct nlmsghdr
*nlh
= nlmsg_hdr(msg
);
56 void *hdr
= genlmsg_data(nlmsg_data(nlh
));
58 genlmsg_end(msg
, hdr
);
60 return genlmsg_multicast(&nl802154_family
, msg
, 0, group
, GFP_ATOMIC
);
63 struct sk_buff
*ieee802154_nl_new_reply(struct genl_info
*info
,
67 struct sk_buff
*msg
= nlmsg_new(NLMSG_DEFAULT_SIZE
, GFP_ATOMIC
);
72 hdr
= genlmsg_put_reply(msg
, info
,
73 &nl802154_family
, flags
, req
);
82 int ieee802154_nl_reply(struct sk_buff
*msg
, struct genl_info
*info
)
84 struct nlmsghdr
*nlh
= nlmsg_hdr(msg
);
85 void *hdr
= genlmsg_data(nlmsg_data(nlh
));
87 genlmsg_end(msg
, hdr
);
89 return genlmsg_reply(msg
, info
);
92 static const struct genl_ops ieee802154_ops
[] = {
94 IEEE802154_DUMP(IEEE802154_LIST_PHY
, ieee802154_list_phy
,
96 IEEE802154_OP(IEEE802154_ADD_IFACE
, ieee802154_add_iface
),
97 IEEE802154_OP(IEEE802154_DEL_IFACE
, ieee802154_del_iface
),
99 IEEE802154_OP(IEEE802154_ASSOCIATE_REQ
, ieee802154_associate_req
),
100 IEEE802154_OP(IEEE802154_ASSOCIATE_RESP
, ieee802154_associate_resp
),
101 IEEE802154_OP(IEEE802154_DISASSOCIATE_REQ
, ieee802154_disassociate_req
),
102 IEEE802154_OP(IEEE802154_SCAN_REQ
, ieee802154_scan_req
),
103 IEEE802154_OP(IEEE802154_START_REQ
, ieee802154_start_req
),
104 IEEE802154_DUMP(IEEE802154_LIST_IFACE
, ieee802154_list_iface
,
105 ieee802154_dump_iface
),
106 IEEE802154_OP(IEEE802154_SET_MACPARAMS
, ieee802154_set_macparams
),
107 IEEE802154_OP(IEEE802154_LLSEC_GETPARAMS
, ieee802154_llsec_getparams
),
108 IEEE802154_OP(IEEE802154_LLSEC_SETPARAMS
, ieee802154_llsec_setparams
),
109 IEEE802154_DUMP(IEEE802154_LLSEC_LIST_KEY
, NULL
,
110 ieee802154_llsec_dump_keys
),
111 IEEE802154_OP(IEEE802154_LLSEC_ADD_KEY
, ieee802154_llsec_add_key
),
112 IEEE802154_OP(IEEE802154_LLSEC_DEL_KEY
, ieee802154_llsec_del_key
),
113 IEEE802154_DUMP(IEEE802154_LLSEC_LIST_DEV
, NULL
,
114 ieee802154_llsec_dump_devs
),
115 IEEE802154_OP(IEEE802154_LLSEC_ADD_DEV
, ieee802154_llsec_add_dev
),
116 IEEE802154_OP(IEEE802154_LLSEC_DEL_DEV
, ieee802154_llsec_del_dev
),
117 IEEE802154_DUMP(IEEE802154_LLSEC_LIST_DEVKEY
, NULL
,
118 ieee802154_llsec_dump_devkeys
),
119 IEEE802154_OP(IEEE802154_LLSEC_ADD_DEVKEY
, ieee802154_llsec_add_devkey
),
120 IEEE802154_OP(IEEE802154_LLSEC_DEL_DEVKEY
, ieee802154_llsec_del_devkey
),
121 IEEE802154_DUMP(IEEE802154_LLSEC_LIST_SECLEVEL
, NULL
,
122 ieee802154_llsec_dump_seclevels
),
123 IEEE802154_OP(IEEE802154_LLSEC_ADD_SECLEVEL
,
124 ieee802154_llsec_add_seclevel
),
125 IEEE802154_OP(IEEE802154_LLSEC_DEL_SECLEVEL
,
126 ieee802154_llsec_del_seclevel
),
129 static const struct genl_multicast_group ieee802154_mcgrps
[] = {
130 [IEEE802154_COORD_MCGRP
] = { .name
= IEEE802154_MCAST_COORD_NAME
, },
131 [IEEE802154_BEACON_MCGRP
] = { .name
= IEEE802154_MCAST_BEACON_NAME
, },
134 struct genl_family nl802154_family __ro_after_init
= {
136 .name
= IEEE802154_NL_NAME
,
138 .maxattr
= IEEE802154_ATTR_MAX
,
139 .module
= THIS_MODULE
,
140 .ops
= ieee802154_ops
,
141 .n_ops
= ARRAY_SIZE(ieee802154_ops
),
142 .mcgrps
= ieee802154_mcgrps
,
143 .n_mcgrps
= ARRAY_SIZE(ieee802154_mcgrps
),
146 int __init
ieee802154_nl_init(void)
148 return genl_register_family(&nl802154_family
);
151 void ieee802154_nl_exit(void)
153 genl_unregister_family(&nl802154_family
);