op-msg.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. /*
  2. * Linux WiMAX
  3. * Generic messaging interface between userspace and driver/device
  4. *
  5. *
  6. * Copyright (C) 2007-2008 Intel Corporation <linux-wimax@intel.com>
  7. * Inaky Perez-Gonzalez <inaky.perez-gonzalez@intel.com>
  8. *
  9. * This program is free software; you can redistribute it and/or
  10. * modify it under the terms of the GNU General Public License version
  11. * 2 as published by the Free Software Foundation.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  21. * 02110-1301, USA.
  22. *
  23. *
  24. * This implements a direct communication channel between user space and
  25. * the driver/device, by which free form messages can be sent back and
  26. * forth.
  27. *
  28. * This is intended for device-specific features, vendor quirks, etc.
  29. *
  30. * See include/net/wimax.h
  31. *
  32. * GENERIC NETLINK ENCODING AND CAPACITY
  33. *
  34. * A destination "pipe name" is added to each message; it is up to the
  35. * drivers to assign or use those names (if using them at all).
  36. *
  37. * Messages are encoded as a binary netlink attribute using nla_put()
  38. * using type NLA_UNSPEC (as some versions of libnl still in
  39. * deployment don't yet understand NLA_BINARY).
  40. *
  41. * The maximum capacity of this transport is PAGESIZE per message (so
  42. * the actual payload will be bit smaller depending on the
  43. * netlink/generic netlink attributes and headers).
  44. *
  45. * RECEPTION OF MESSAGES
  46. *
  47. * When a message is received from user space, it is passed verbatim
  48. * to the driver calling wimax_dev->op_msg_from_user(). The return
  49. * value from this function is passed back to user space as an ack
  50. * over the generic netlink protocol.
  51. *
  52. * The stack doesn't do any processing or interpretation of these
  53. * messages.
  54. *
  55. * SENDING MESSAGES
  56. *
  57. * Messages can be sent with wimax_msg().
  58. *
  59. * If the message delivery needs to happen on a different context to
  60. * that of its creation, wimax_msg_alloc() can be used to get a
  61. * pointer to the message that can be delivered later on with
  62. * wimax_msg_send().
  63. *
  64. * ROADMAP
  65. *
  66. * wimax_gnl_doit_msg_from_user() Process a message from user space
  67. * wimax_dev_get_by_genl_info()
  68. * wimax_dev->op_msg_from_user() Delivery of message to the driver
  69. *
  70. * wimax_msg() Send a message to user space
  71. * wimax_msg_alloc()
  72. * wimax_msg_send()
  73. */
  74. #include <linux/device.h>
  75. #include <linux/slab.h>
  76. #include <net/genetlink.h>
  77. #include <linux/netdevice.h>
  78. #include <linux/wimax.h>
  79. #include <linux/security.h>
  80. #include <linux/export.h>
  81. #include "wimax-internal.h"
  82. #define D_SUBMODULE op_msg
  83. #include "debug-levels.h"
  84. /**
  85. * wimax_msg_alloc - Create a new skb for sending a message to userspace
  86. *
  87. * @wimax_dev: WiMAX device descriptor
  88. * @pipe_name: "named pipe" the message will be sent to
  89. * @msg: pointer to the message data to send
  90. * @size: size of the message to send (in bytes), including the header.
  91. * @gfp_flags: flags for memory allocation.
  92. *
  93. * Returns: %0 if ok, negative errno code on error
  94. *
  95. * Description:
  96. *
  97. * Allocates an skb that will contain the message to send to user
  98. * space over the messaging pipe and initializes it, copying the
  99. * payload.
  100. *
  101. * Once this call is done, you can deliver it with
  102. * wimax_msg_send().
  103. *
  104. * IMPORTANT:
  105. *
  106. * Don't use skb_push()/skb_pull()/skb_reserve() on the skb, as
  107. * wimax_msg_send() depends on skb->data being placed at the
  108. * beginning of the user message.
  109. *
  110. * Unlike other WiMAX stack calls, this call can be used way early,
  111. * even before wimax_dev_add() is called, as long as the
  112. * wimax_dev->net_dev pointer is set to point to a proper
  113. * net_dev. This is so that drivers can use it early in case they need
  114. * to send stuff around or communicate with user space.
  115. */
  116. struct sk_buff *wimax_msg_alloc(struct wimax_dev *wimax_dev,
  117. const char *pipe_name,
  118. const void *msg, size_t size,
  119. gfp_t gfp_flags)
  120. {
  121. int result;
  122. struct device *dev = wimax_dev_to_dev(wimax_dev);
  123. size_t msg_size;
  124. void *genl_msg;
  125. struct sk_buff *skb;
  126. msg_size = nla_total_size(size)
  127. + nla_total_size(sizeof(u32))
  128. + (pipe_name ? nla_total_size(strlen(pipe_name)) : 0);
  129. result = -ENOMEM;
  130. skb = genlmsg_new(msg_size, gfp_flags);
  131. if (skb == NULL)
  132. goto error_new;
  133. genl_msg = genlmsg_put(skb, 0, 0, &wimax_gnl_family,
  134. 0, WIMAX_GNL_OP_MSG_TO_USER);
  135. if (genl_msg == NULL) {
  136. dev_err(dev, "no memory to create generic netlink message\n");
  137. goto error_genlmsg_put;
  138. }
  139. result = nla_put_u32(skb, WIMAX_GNL_MSG_IFIDX,
  140. wimax_dev->net_dev->ifindex);
  141. if (result < 0) {
  142. dev_err(dev, "no memory to add ifindex attribute\n");
  143. goto error_nla_put;
  144. }
  145. if (pipe_name) {
  146. result = nla_put_string(skb, WIMAX_GNL_MSG_PIPE_NAME,
  147. pipe_name);
  148. if (result < 0) {
  149. dev_err(dev, "no memory to add pipe_name attribute\n");
  150. goto error_nla_put;
  151. }
  152. }
  153. result = nla_put(skb, WIMAX_GNL_MSG_DATA, size, msg);
  154. if (result < 0) {
  155. dev_err(dev, "no memory to add payload (msg %p size %zu) in "
  156. "attribute: %d\n", msg, size, result);
  157. goto error_nla_put;
  158. }
  159. genlmsg_end(skb, genl_msg);
  160. return skb;
  161. error_nla_put:
  162. error_genlmsg_put:
  163. error_new:
  164. nlmsg_free(skb);
  165. return ERR_PTR(result);
  166. }
  167. EXPORT_SYMBOL_GPL(wimax_msg_alloc);
  168. /**
  169. * wimax_msg_data_len - Return a pointer and size of a message's payload
  170. *
  171. * @msg: Pointer to a message created with wimax_msg_alloc()
  172. * @size: Pointer to where to store the message's size
  173. *
  174. * Returns the pointer to the message data.
  175. */
  176. const void *wimax_msg_data_len(struct sk_buff *msg, size_t *size)
  177. {
  178. struct nlmsghdr *nlh = (void *) msg->head;
  179. struct nlattr *nla;
  180. nla = nlmsg_find_attr(nlh, sizeof(struct genlmsghdr),
  181. WIMAX_GNL_MSG_DATA);
  182. if (nla == NULL) {
  183. printk(KERN_ERR "Cannot find attribute WIMAX_GNL_MSG_DATA\n");
  184. return NULL;
  185. }
  186. *size = nla_len(nla);
  187. return nla_data(nla);
  188. }
  189. EXPORT_SYMBOL_GPL(wimax_msg_data_len);
  190. /**
  191. * wimax_msg_data - Return a pointer to a message's payload
  192. *
  193. * @msg: Pointer to a message created with wimax_msg_alloc()
  194. */
  195. const void *wimax_msg_data(struct sk_buff *msg)
  196. {
  197. struct nlmsghdr *nlh = (void *) msg->head;
  198. struct nlattr *nla;
  199. nla = nlmsg_find_attr(nlh, sizeof(struct genlmsghdr),
  200. WIMAX_GNL_MSG_DATA);
  201. if (nla == NULL) {
  202. printk(KERN_ERR "Cannot find attribute WIMAX_GNL_MSG_DATA\n");
  203. return NULL;
  204. }
  205. return nla_data(nla);
  206. }
  207. EXPORT_SYMBOL_GPL(wimax_msg_data);
  208. /**
  209. * wimax_msg_len - Return a message's payload length
  210. *
  211. * @msg: Pointer to a message created with wimax_msg_alloc()
  212. */
  213. ssize_t wimax_msg_len(struct sk_buff *msg)
  214. {
  215. struct nlmsghdr *nlh = (void *) msg->head;
  216. struct nlattr *nla;
  217. nla = nlmsg_find_attr(nlh, sizeof(struct genlmsghdr),
  218. WIMAX_GNL_MSG_DATA);
  219. if (nla == NULL) {
  220. printk(KERN_ERR "Cannot find attribute WIMAX_GNL_MSG_DATA\n");
  221. return -EINVAL;
  222. }
  223. return nla_len(nla);
  224. }
  225. EXPORT_SYMBOL_GPL(wimax_msg_len);
  226. /**
  227. * wimax_msg_send - Send a pre-allocated message to user space
  228. *
  229. * @wimax_dev: WiMAX device descriptor
  230. *
  231. * @skb: &struct sk_buff returned by wimax_msg_alloc(). Note the
  232. * ownership of @skb is transferred to this function.
  233. *
  234. * Returns: 0 if ok, < 0 errno code on error
  235. *
  236. * Description:
  237. *
  238. * Sends a free-form message that was preallocated with
  239. * wimax_msg_alloc() and filled up.
  240. *
  241. * Assumes that once you pass an skb to this function for sending, it
  242. * owns it and will release it when done (on success).
  243. *
  244. * IMPORTANT:
  245. *
  246. * Don't use skb_push()/skb_pull()/skb_reserve() on the skb, as
  247. * wimax_msg_send() depends on skb->data being placed at the
  248. * beginning of the user message.
  249. *
  250. * Unlike other WiMAX stack calls, this call can be used way early,
  251. * even before wimax_dev_add() is called, as long as the
  252. * wimax_dev->net_dev pointer is set to point to a proper
  253. * net_dev. This is so that drivers can use it early in case they need
  254. * to send stuff around or communicate with user space.
  255. */
  256. int wimax_msg_send(struct wimax_dev *wimax_dev, struct sk_buff *skb)
  257. {
  258. struct device *dev = wimax_dev_to_dev(wimax_dev);
  259. void *msg = skb->data;
  260. size_t size = skb->len;
  261. might_sleep();
  262. d_printf(1, dev, "CTX: wimax msg, %zu bytes\n", size);
  263. d_dump(2, dev, msg, size);
  264. genlmsg_multicast(skb, 0, wimax_gnl_mcg.id, GFP_KERNEL);
  265. d_printf(1, dev, "CTX: genl multicast done\n");
  266. return 0;
  267. }
  268. EXPORT_SYMBOL_GPL(wimax_msg_send);
  269. /**
  270. * wimax_msg - Send a message to user space
  271. *
  272. * @wimax_dev: WiMAX device descriptor (properly referenced)
  273. * @pipe_name: "named pipe" the message will be sent to
  274. * @buf: pointer to the message to send.
  275. * @size: size of the buffer pointed to by @buf (in bytes).
  276. * @gfp_flags: flags for memory allocation.
  277. *
  278. * Returns: %0 if ok, negative errno code on error.
  279. *
  280. * Description:
  281. *
  282. * Sends a free-form message to user space on the device @wimax_dev.
  283. *
  284. * NOTES:
  285. *
  286. * Once the @skb is given to this function, who will own it and will
  287. * release it when done (unless it returns error).
  288. */
  289. int wimax_msg(struct wimax_dev *wimax_dev, const char *pipe_name,
  290. const void *buf, size_t size, gfp_t gfp_flags)
  291. {
  292. int result = -ENOMEM;
  293. struct sk_buff *skb;
  294. skb = wimax_msg_alloc(wimax_dev, pipe_name, buf, size, gfp_flags);
  295. if (IS_ERR(skb))
  296. result = PTR_ERR(skb);
  297. else
  298. result = wimax_msg_send(wimax_dev, skb);
  299. return result;
  300. }
  301. EXPORT_SYMBOL_GPL(wimax_msg);
  302. static const struct nla_policy wimax_gnl_msg_policy[WIMAX_GNL_ATTR_MAX + 1] = {
  303. [WIMAX_GNL_MSG_IFIDX] = {
  304. .type = NLA_U32,
  305. },
  306. [WIMAX_GNL_MSG_DATA] = {
  307. .type = NLA_UNSPEC, /* libnl doesn't grok BINARY yet */
  308. },
  309. };
  310. /*
  311. * Relays a message from user space to the driver
  312. *
  313. * The skb is passed to the driver-specific function with the netlink
  314. * and generic netlink headers already stripped.
  315. *
  316. * This call will block while handling/relaying the message.
  317. */
  318. static
  319. int wimax_gnl_doit_msg_from_user(struct sk_buff *skb, struct genl_info *info)
  320. {
  321. int result, ifindex;
  322. struct wimax_dev *wimax_dev;
  323. struct device *dev;
  324. struct nlmsghdr *nlh = info->nlhdr;
  325. char *pipe_name;
  326. void *msg_buf;
  327. size_t msg_len;
  328. might_sleep();
  329. d_fnstart(3, NULL, "(skb %p info %p)\n", skb, info);
  330. result = -ENODEV;
  331. if (info->attrs[WIMAX_GNL_MSG_IFIDX] == NULL) {
  332. printk(KERN_ERR "WIMAX_GNL_MSG_FROM_USER: can't find IFIDX "
  333. "attribute\n");
  334. goto error_no_wimax_dev;
  335. }
  336. ifindex = nla_get_u32(info->attrs[WIMAX_GNL_MSG_IFIDX]);
  337. wimax_dev = wimax_dev_get_by_genl_info(info, ifindex);
  338. if (wimax_dev == NULL)
  339. goto error_no_wimax_dev;
  340. dev = wimax_dev_to_dev(wimax_dev);
  341. /* Unpack arguments */
  342. result = -EINVAL;
  343. if (info->attrs[WIMAX_GNL_MSG_DATA] == NULL) {
  344. dev_err(dev, "WIMAX_GNL_MSG_FROM_USER: can't find MSG_DATA "
  345. "attribute\n");
  346. goto error_no_data;
  347. }
  348. msg_buf = nla_data(info->attrs[WIMAX_GNL_MSG_DATA]);
  349. msg_len = nla_len(info->attrs[WIMAX_GNL_MSG_DATA]);
  350. if (info->attrs[WIMAX_GNL_MSG_PIPE_NAME] == NULL)
  351. pipe_name = NULL;
  352. else {
  353. struct nlattr *attr = info->attrs[WIMAX_GNL_MSG_PIPE_NAME];
  354. size_t attr_len = nla_len(attr);
  355. /* libnl-1.1 does not yet support NLA_NUL_STRING */
  356. result = -ENOMEM;
  357. pipe_name = kstrndup(nla_data(attr), attr_len + 1, GFP_KERNEL);
  358. if (pipe_name == NULL)
  359. goto error_alloc;
  360. pipe_name[attr_len] = 0;
  361. }
  362. mutex_lock(&wimax_dev->mutex);
  363. result = wimax_dev_is_ready(wimax_dev);
  364. if (result == -ENOMEDIUM)
  365. result = 0;
  366. if (result < 0)
  367. goto error_not_ready;
  368. result = -ENOSYS;
  369. if (wimax_dev->op_msg_from_user == NULL)
  370. goto error_noop;
  371. d_printf(1, dev,
  372. "CRX: nlmsghdr len %u type %u flags 0x%04x seq 0x%x pid %u\n",
  373. nlh->nlmsg_len, nlh->nlmsg_type, nlh->nlmsg_flags,
  374. nlh->nlmsg_seq, nlh->nlmsg_pid);
  375. d_printf(1, dev, "CRX: wimax message %zu bytes\n", msg_len);
  376. d_dump(2, dev, msg_buf, msg_len);
  377. result = wimax_dev->op_msg_from_user(wimax_dev, pipe_name,
  378. msg_buf, msg_len, info);
  379. error_noop:
  380. error_not_ready:
  381. mutex_unlock(&wimax_dev->mutex);
  382. error_alloc:
  383. kfree(pipe_name);
  384. error_no_data:
  385. dev_put(wimax_dev->net_dev);
  386. error_no_wimax_dev:
  387. d_fnend(3, NULL, "(skb %p info %p) = %d\n", skb, info, result);
  388. return result;
  389. }
  390. /*
  391. * Generic Netlink glue
  392. */
  393. struct genl_ops wimax_gnl_msg_from_user = {
  394. .cmd = WIMAX_GNL_OP_MSG_FROM_USER,
  395. .flags = GENL_ADMIN_PERM,
  396. .policy = wimax_gnl_msg_policy,
  397. .doit = wimax_gnl_doit_msg_from_user,
  398. .dumpit = NULL,
  399. };