header_ops.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2014 Fraunhofer ITWM
  4. *
  5. * Written by:
  6. * Phoebe Buckheister <phoebe.buckheister@itwm.fraunhofer.de>
  7. */
  8. #include <linux/ieee802154.h>
  9. #include <net/mac802154.h>
  10. #include <net/ieee802154_netdev.h>
  11. static int
  12. ieee802154_hdr_push_addr(u8 *buf, const struct ieee802154_addr *addr,
  13. bool omit_pan)
  14. {
  15. int pos = 0;
  16. if (addr->mode == IEEE802154_ADDR_NONE)
  17. return 0;
  18. if (!omit_pan) {
  19. memcpy(buf + pos, &addr->pan_id, 2);
  20. pos += 2;
  21. }
  22. switch (addr->mode) {
  23. case IEEE802154_ADDR_SHORT:
  24. memcpy(buf + pos, &addr->short_addr, 2);
  25. pos += 2;
  26. break;
  27. case IEEE802154_ADDR_LONG:
  28. memcpy(buf + pos, &addr->extended_addr, IEEE802154_ADDR_LEN);
  29. pos += IEEE802154_ADDR_LEN;
  30. break;
  31. default:
  32. return -EINVAL;
  33. }
  34. return pos;
  35. }
  36. static int
  37. ieee802154_hdr_push_sechdr(u8 *buf, const struct ieee802154_sechdr *hdr)
  38. {
  39. int pos = 5;
  40. memcpy(buf, hdr, 1);
  41. memcpy(buf + 1, &hdr->frame_counter, 4);
  42. switch (hdr->key_id_mode) {
  43. case IEEE802154_SCF_KEY_IMPLICIT:
  44. return pos;
  45. case IEEE802154_SCF_KEY_INDEX:
  46. break;
  47. case IEEE802154_SCF_KEY_SHORT_INDEX:
  48. memcpy(buf + pos, &hdr->short_src, 4);
  49. pos += 4;
  50. break;
  51. case IEEE802154_SCF_KEY_HW_INDEX:
  52. memcpy(buf + pos, &hdr->extended_src, IEEE802154_ADDR_LEN);
  53. pos += IEEE802154_ADDR_LEN;
  54. break;
  55. }
  56. buf[pos++] = hdr->key_id;
  57. return pos;
  58. }
  59. int
  60. ieee802154_hdr_push(struct sk_buff *skb, struct ieee802154_hdr *hdr)
  61. {
  62. u8 buf[IEEE802154_MAX_HEADER_LEN];
  63. int pos = 2;
  64. int rc;
  65. struct ieee802154_hdr_fc *fc = &hdr->fc;
  66. buf[pos++] = hdr->seq;
  67. fc->dest_addr_mode = hdr->dest.mode;
  68. rc = ieee802154_hdr_push_addr(buf + pos, &hdr->dest, false);
  69. if (rc < 0)
  70. return -EINVAL;
  71. pos += rc;
  72. fc->source_addr_mode = hdr->source.mode;
  73. if (hdr->source.pan_id == hdr->dest.pan_id &&
  74. hdr->dest.mode != IEEE802154_ADDR_NONE)
  75. fc->intra_pan = true;
  76. rc = ieee802154_hdr_push_addr(buf + pos, &hdr->source, fc->intra_pan);
  77. if (rc < 0)
  78. return -EINVAL;
  79. pos += rc;
  80. if (fc->security_enabled) {
  81. fc->version = 1;
  82. rc = ieee802154_hdr_push_sechdr(buf + pos, &hdr->sec);
  83. if (rc < 0)
  84. return -EINVAL;
  85. pos += rc;
  86. }
  87. memcpy(buf, fc, 2);
  88. memcpy(skb_push(skb, pos), buf, pos);
  89. return pos;
  90. }
  91. EXPORT_SYMBOL_GPL(ieee802154_hdr_push);
  92. int ieee802154_mac_cmd_push(struct sk_buff *skb, void *f,
  93. const void *pl, unsigned int pl_len)
  94. {
  95. struct ieee802154_mac_cmd_frame *frame = f;
  96. struct ieee802154_mac_cmd_pl *mac_pl = &frame->mac_pl;
  97. struct ieee802154_hdr *mhr = &frame->mhr;
  98. int ret;
  99. skb_reserve(skb, sizeof(*mhr));
  100. ret = ieee802154_hdr_push(skb, mhr);
  101. if (ret < 0)
  102. return ret;
  103. skb_reset_mac_header(skb);
  104. skb->mac_len = ret;
  105. skb_put_data(skb, mac_pl, sizeof(*mac_pl));
  106. skb_put_data(skb, pl, pl_len);
  107. return 0;
  108. }
  109. EXPORT_SYMBOL_GPL(ieee802154_mac_cmd_push);
  110. int ieee802154_beacon_push(struct sk_buff *skb,
  111. struct ieee802154_beacon_frame *beacon)
  112. {
  113. struct ieee802154_beacon_hdr *mac_pl = &beacon->mac_pl;
  114. struct ieee802154_hdr *mhr = &beacon->mhr;
  115. int ret;
  116. skb_reserve(skb, sizeof(*mhr));
  117. ret = ieee802154_hdr_push(skb, mhr);
  118. if (ret < 0)
  119. return ret;
  120. skb_reset_mac_header(skb);
  121. skb->mac_len = ret;
  122. skb_put_data(skb, mac_pl, sizeof(*mac_pl));
  123. if (mac_pl->pend_short_addr_count || mac_pl->pend_ext_addr_count)
  124. return -EOPNOTSUPP;
  125. return 0;
  126. }
  127. EXPORT_SYMBOL_GPL(ieee802154_beacon_push);
  128. static int
  129. ieee802154_hdr_get_addr(const u8 *buf, int mode, bool omit_pan,
  130. struct ieee802154_addr *addr)
  131. {
  132. int pos = 0;
  133. addr->mode = mode;
  134. if (mode == IEEE802154_ADDR_NONE)
  135. return 0;
  136. if (!omit_pan) {
  137. memcpy(&addr->pan_id, buf + pos, 2);
  138. pos += 2;
  139. }
  140. if (mode == IEEE802154_ADDR_SHORT) {
  141. memcpy(&addr->short_addr, buf + pos, 2);
  142. return pos + 2;
  143. } else {
  144. memcpy(&addr->extended_addr, buf + pos, IEEE802154_ADDR_LEN);
  145. return pos + IEEE802154_ADDR_LEN;
  146. }
  147. }
  148. static int ieee802154_hdr_addr_len(int mode, bool omit_pan)
  149. {
  150. int pan_len = omit_pan ? 0 : 2;
  151. switch (mode) {
  152. case IEEE802154_ADDR_NONE: return 0;
  153. case IEEE802154_ADDR_SHORT: return 2 + pan_len;
  154. case IEEE802154_ADDR_LONG: return IEEE802154_ADDR_LEN + pan_len;
  155. default: return -EINVAL;
  156. }
  157. }
  158. static int
  159. ieee802154_hdr_get_sechdr(const u8 *buf, struct ieee802154_sechdr *hdr)
  160. {
  161. int pos = 5;
  162. memcpy(hdr, buf, 1);
  163. memcpy(&hdr->frame_counter, buf + 1, 4);
  164. switch (hdr->key_id_mode) {
  165. case IEEE802154_SCF_KEY_IMPLICIT:
  166. return pos;
  167. case IEEE802154_SCF_KEY_INDEX:
  168. break;
  169. case IEEE802154_SCF_KEY_SHORT_INDEX:
  170. memcpy(&hdr->short_src, buf + pos, 4);
  171. pos += 4;
  172. break;
  173. case IEEE802154_SCF_KEY_HW_INDEX:
  174. memcpy(&hdr->extended_src, buf + pos, IEEE802154_ADDR_LEN);
  175. pos += IEEE802154_ADDR_LEN;
  176. break;
  177. }
  178. hdr->key_id = buf[pos++];
  179. return pos;
  180. }
  181. static int ieee802154_sechdr_lengths[4] = {
  182. [IEEE802154_SCF_KEY_IMPLICIT] = 5,
  183. [IEEE802154_SCF_KEY_INDEX] = 6,
  184. [IEEE802154_SCF_KEY_SHORT_INDEX] = 10,
  185. [IEEE802154_SCF_KEY_HW_INDEX] = 14,
  186. };
  187. static int ieee802154_hdr_sechdr_len(u8 sc)
  188. {
  189. return ieee802154_sechdr_lengths[IEEE802154_SCF_KEY_ID_MODE(sc)];
  190. }
  191. static int ieee802154_hdr_minlen(const struct ieee802154_hdr *hdr)
  192. {
  193. int dlen, slen;
  194. dlen = ieee802154_hdr_addr_len(hdr->fc.dest_addr_mode, false);
  195. slen = ieee802154_hdr_addr_len(hdr->fc.source_addr_mode,
  196. hdr->fc.intra_pan);
  197. if (slen < 0 || dlen < 0)
  198. return -EINVAL;
  199. return 3 + dlen + slen + hdr->fc.security_enabled;
  200. }
  201. static int
  202. ieee802154_hdr_get_addrs(const u8 *buf, struct ieee802154_hdr *hdr)
  203. {
  204. int pos = 0;
  205. pos += ieee802154_hdr_get_addr(buf + pos, hdr->fc.dest_addr_mode,
  206. false, &hdr->dest);
  207. pos += ieee802154_hdr_get_addr(buf + pos, hdr->fc.source_addr_mode,
  208. hdr->fc.intra_pan, &hdr->source);
  209. if (hdr->fc.intra_pan)
  210. hdr->source.pan_id = hdr->dest.pan_id;
  211. return pos;
  212. }
  213. int
  214. ieee802154_hdr_pull(struct sk_buff *skb, struct ieee802154_hdr *hdr)
  215. {
  216. int pos = 3, rc;
  217. if (!pskb_may_pull(skb, 3))
  218. return -EINVAL;
  219. memcpy(hdr, skb->data, 3);
  220. rc = ieee802154_hdr_minlen(hdr);
  221. if (rc < 0 || !pskb_may_pull(skb, rc))
  222. return -EINVAL;
  223. pos += ieee802154_hdr_get_addrs(skb->data + pos, hdr);
  224. if (hdr->fc.security_enabled) {
  225. int want = pos + ieee802154_hdr_sechdr_len(skb->data[pos]);
  226. if (!pskb_may_pull(skb, want))
  227. return -EINVAL;
  228. pos += ieee802154_hdr_get_sechdr(skb->data + pos, &hdr->sec);
  229. }
  230. skb_pull(skb, pos);
  231. return pos;
  232. }
  233. EXPORT_SYMBOL_GPL(ieee802154_hdr_pull);
  234. int ieee802154_mac_cmd_pl_pull(struct sk_buff *skb,
  235. struct ieee802154_mac_cmd_pl *mac_pl)
  236. {
  237. if (!pskb_may_pull(skb, sizeof(*mac_pl)))
  238. return -EINVAL;
  239. memcpy(mac_pl, skb->data, sizeof(*mac_pl));
  240. skb_pull(skb, sizeof(*mac_pl));
  241. return 0;
  242. }
  243. EXPORT_SYMBOL_GPL(ieee802154_mac_cmd_pl_pull);
  244. int
  245. ieee802154_hdr_peek_addrs(const struct sk_buff *skb, struct ieee802154_hdr *hdr)
  246. {
  247. const u8 *buf = skb_mac_header(skb);
  248. int pos = 3, rc;
  249. if (buf + 3 > skb_tail_pointer(skb))
  250. return -EINVAL;
  251. memcpy(hdr, buf, 3);
  252. rc = ieee802154_hdr_minlen(hdr);
  253. if (rc < 0 || buf + rc > skb_tail_pointer(skb))
  254. return -EINVAL;
  255. pos += ieee802154_hdr_get_addrs(buf + pos, hdr);
  256. return pos;
  257. }
  258. EXPORT_SYMBOL_GPL(ieee802154_hdr_peek_addrs);
  259. int
  260. ieee802154_hdr_peek(const struct sk_buff *skb, struct ieee802154_hdr *hdr)
  261. {
  262. const u8 *buf = skb_mac_header(skb);
  263. int pos;
  264. pos = ieee802154_hdr_peek_addrs(skb, hdr);
  265. if (pos < 0)
  266. return -EINVAL;
  267. if (hdr->fc.security_enabled) {
  268. u8 key_id_mode = IEEE802154_SCF_KEY_ID_MODE(*(buf + pos));
  269. int want = pos + ieee802154_sechdr_lengths[key_id_mode];
  270. if (buf + want > skb_tail_pointer(skb))
  271. return -EINVAL;
  272. pos += ieee802154_hdr_get_sechdr(buf + pos, &hdr->sec);
  273. }
  274. return pos;
  275. }
  276. EXPORT_SYMBOL_GPL(ieee802154_hdr_peek);
  277. int ieee802154_max_payload(const struct ieee802154_hdr *hdr)
  278. {
  279. int hlen = ieee802154_hdr_minlen(hdr);
  280. if (hdr->fc.security_enabled) {
  281. hlen += ieee802154_sechdr_lengths[hdr->sec.key_id_mode] - 1;
  282. hlen += ieee802154_sechdr_authtag_len(&hdr->sec);
  283. }
  284. return IEEE802154_MTU - hlen - IEEE802154_MFR_SIZE;
  285. }
  286. EXPORT_SYMBOL_GPL(ieee802154_max_payload);