6lowpan.h 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. /*
  2. * Copyright 2011, Siemens AG
  3. * written by Alexander Smirnov <alex.bluesman.smirnov@gmail.com>
  4. */
  5. /*
  6. * Based on patches from Jon Smirl <jonsmirl@gmail.com>
  7. * Copyright (c) 2011 Jon Smirl <jonsmirl@gmail.com>
  8. *
  9. * This program is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License version 2
  11. * 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 along
  19. * with this program; if not, write to the Free Software Foundation, Inc.,
  20. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. */
  22. /* Jon's code is based on 6lowpan implementation for Contiki which is:
  23. * Copyright (c) 2008, Swedish Institute of Computer Science.
  24. * All rights reserved.
  25. *
  26. * Redistribution and use in source and binary forms, with or without
  27. * modification, are permitted provided that the following conditions
  28. * are met:
  29. * 1. Redistributions of source code must retain the above copyright
  30. * notice, this list of conditions and the following disclaimer.
  31. * 2. Redistributions in binary form must reproduce the above copyright
  32. * notice, this list of conditions and the following disclaimer in the
  33. * documentation and/or other materials provided with the distribution.
  34. * 3. Neither the name of the Institute nor the names of its contributors
  35. * may be used to endorse or promote products derived from this software
  36. * without specific prior written permission.
  37. *
  38. * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
  39. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  40. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  41. * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
  42. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  43. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  44. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  45. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  46. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  47. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  48. * SUCH DAMAGE.
  49. */
  50. #ifndef __6LOWPAN_H__
  51. #define __6LOWPAN_H__
  52. #include <net/ipv6.h>
  53. #include <net/net_namespace.h>
  54. #define EUI64_ADDR_LEN 8
  55. #define LOWPAN_NHC_MAX_ID_LEN 1
  56. /* Maximum next header compression length which we currently support inclusive
  57. * possible inline data.
  58. */
  59. #define LOWPAN_NHC_MAX_HDR_LEN (sizeof(struct udphdr))
  60. /* Max IPHC Header len without IPv6 hdr specific inline data.
  61. * Useful for getting the "extra" bytes we need at worst case compression.
  62. *
  63. * LOWPAN_IPHC + CID + LOWPAN_NHC_MAX_ID_LEN
  64. */
  65. #define LOWPAN_IPHC_MAX_HEADER_LEN (2 + 1 + LOWPAN_NHC_MAX_ID_LEN)
  66. /* Maximum worst case IPHC header buffer size */
  67. #define LOWPAN_IPHC_MAX_HC_BUF_LEN (sizeof(struct ipv6hdr) + \
  68. LOWPAN_IPHC_MAX_HEADER_LEN + \
  69. LOWPAN_NHC_MAX_HDR_LEN)
  70. #define LOWPAN_DISPATCH_IPV6 0x41 /* 01000001 = 65 */
  71. #define LOWPAN_DISPATCH_IPHC 0x60 /* 011xxxxx = ... */
  72. #define LOWPAN_DISPATCH_IPHC_MASK 0xe0
  73. static inline bool lowpan_is_ipv6(u8 dispatch)
  74. {
  75. return dispatch == LOWPAN_DISPATCH_IPV6;
  76. }
  77. static inline bool lowpan_is_iphc(u8 dispatch)
  78. {
  79. return (dispatch & LOWPAN_DISPATCH_IPHC_MASK) == LOWPAN_DISPATCH_IPHC;
  80. }
  81. #define LOWPAN_PRIV_SIZE(llpriv_size) \
  82. (sizeof(struct lowpan_priv) + llpriv_size)
  83. enum lowpan_lltypes {
  84. LOWPAN_LLTYPE_BTLE,
  85. LOWPAN_LLTYPE_IEEE802154,
  86. };
  87. struct lowpan_priv {
  88. enum lowpan_lltypes lltype;
  89. /* must be last */
  90. u8 priv[0] __aligned(sizeof(void *));
  91. };
  92. static inline
  93. struct lowpan_priv *lowpan_priv(const struct net_device *dev)
  94. {
  95. return netdev_priv(dev);
  96. }
  97. struct lowpan_802154_cb {
  98. u16 d_tag;
  99. unsigned int d_size;
  100. u8 d_offset;
  101. };
  102. static inline
  103. struct lowpan_802154_cb *lowpan_802154_cb(const struct sk_buff *skb)
  104. {
  105. BUILD_BUG_ON(sizeof(struct lowpan_802154_cb) > sizeof(skb->cb));
  106. return (struct lowpan_802154_cb *)skb->cb;
  107. }
  108. #ifdef DEBUG
  109. /* print data in line */
  110. static inline void raw_dump_inline(const char *caller, char *msg,
  111. const unsigned char *buf, int len)
  112. {
  113. if (msg)
  114. pr_debug("%s():%s: ", caller, msg);
  115. print_hex_dump_debug("", DUMP_PREFIX_NONE, 16, 1, buf, len, false);
  116. }
  117. /* print data in a table format:
  118. *
  119. * addr: xx xx xx xx xx xx
  120. * addr: xx xx xx xx xx xx
  121. * ...
  122. */
  123. static inline void raw_dump_table(const char *caller, char *msg,
  124. const unsigned char *buf, int len)
  125. {
  126. if (msg)
  127. pr_debug("%s():%s:\n", caller, msg);
  128. print_hex_dump_debug("\t", DUMP_PREFIX_OFFSET, 16, 1, buf, len, false);
  129. }
  130. #else
  131. static inline void raw_dump_table(const char *caller, char *msg,
  132. const unsigned char *buf, int len) { }
  133. static inline void raw_dump_inline(const char *caller, char *msg,
  134. const unsigned char *buf, int len) { }
  135. #endif
  136. /**
  137. * lowpan_fetch_skb - getting inline data from 6LoWPAN header
  138. *
  139. * This function will pull data from sk buffer and put it into data to
  140. * remove the 6LoWPAN inline data. This function returns true if the
  141. * sk buffer is too small to pull the amount of data which is specified
  142. * by len.
  143. *
  144. * @skb: the buffer where the inline data should be pulled from.
  145. * @data: destination buffer for the inline data.
  146. * @len: amount of data which should be pulled in bytes.
  147. */
  148. static inline bool lowpan_fetch_skb(struct sk_buff *skb, void *data,
  149. unsigned int len)
  150. {
  151. if (unlikely(!pskb_may_pull(skb, len)))
  152. return true;
  153. skb_copy_from_linear_data(skb, data, len);
  154. skb_pull(skb, len);
  155. return false;
  156. }
  157. static inline void lowpan_push_hc_data(u8 **hc_ptr, const void *data,
  158. const size_t len)
  159. {
  160. memcpy(*hc_ptr, data, len);
  161. *hc_ptr += len;
  162. }
  163. void lowpan_netdev_setup(struct net_device *dev, enum lowpan_lltypes lltype);
  164. /**
  165. * lowpan_header_decompress - replace 6LoWPAN header with IPv6 header
  166. *
  167. * This function replaces the IPHC 6LoWPAN header which should be pointed at
  168. * skb->data and skb_network_header, with the IPv6 header.
  169. * It would be nice that the caller have the necessary headroom of IPv6 header
  170. * and greatest Transport layer header, this would reduce the overhead for
  171. * reallocate headroom.
  172. *
  173. * @skb: the buffer which should be manipulate.
  174. * @dev: the lowpan net device pointer.
  175. * @daddr: destination lladdr of mac header which is used for compression
  176. * methods.
  177. * @saddr: source lladdr of mac header which is used for compression
  178. * methods.
  179. */
  180. int lowpan_header_decompress(struct sk_buff *skb, const struct net_device *dev,
  181. const void *daddr, const void *saddr);
  182. /**
  183. * lowpan_header_compress - replace IPv6 header with 6LoWPAN header
  184. *
  185. * This function replaces the IPv6 header which should be pointed at
  186. * skb->data and skb_network_header, with the IPHC 6LoWPAN header.
  187. * The caller need to be sure that the sk buffer is not shared and at have
  188. * at least a headroom which is smaller or equal LOWPAN_IPHC_MAX_HEADER_LEN,
  189. * which is the IPHC "more bytes than IPv6 header" at worst case.
  190. *
  191. * @skb: the buffer which should be manipulate.
  192. * @dev: the lowpan net device pointer.
  193. * @daddr: destination lladdr of mac header which is used for compression
  194. * methods.
  195. * @saddr: source lladdr of mac header which is used for compression
  196. * methods.
  197. */
  198. int lowpan_header_compress(struct sk_buff *skb, const struct net_device *dev,
  199. const void *daddr, const void *saddr);
  200. #endif /* __6LOWPAN_H__ */