decode.h 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. #ifndef __CEPH_DECODE_H
  2. #define __CEPH_DECODE_H
  3. #include <linux/bug.h>
  4. #include <linux/time.h>
  5. #include <asm/unaligned.h>
  6. #include "types.h"
  7. /*
  8. * in all cases,
  9. * void **p pointer to position pointer
  10. * void *end pointer to end of buffer (last byte + 1)
  11. */
  12. static inline u64 ceph_decode_64(void **p)
  13. {
  14. u64 v = get_unaligned_le64(*p);
  15. *p += sizeof(u64);
  16. return v;
  17. }
  18. static inline u32 ceph_decode_32(void **p)
  19. {
  20. u32 v = get_unaligned_le32(*p);
  21. *p += sizeof(u32);
  22. return v;
  23. }
  24. static inline u16 ceph_decode_16(void **p)
  25. {
  26. u16 v = get_unaligned_le16(*p);
  27. *p += sizeof(u16);
  28. return v;
  29. }
  30. static inline u8 ceph_decode_8(void **p)
  31. {
  32. u8 v = *(u8 *)*p;
  33. (*p)++;
  34. return v;
  35. }
  36. static inline void ceph_decode_copy(void **p, void *pv, size_t n)
  37. {
  38. memcpy(pv, *p, n);
  39. *p += n;
  40. }
  41. /*
  42. * bounds check input.
  43. */
  44. #define ceph_decode_need(p, end, n, bad) \
  45. do { \
  46. if (unlikely(*(p) + (n) > (end))) \
  47. goto bad; \
  48. } while (0)
  49. #define ceph_decode_64_safe(p, end, v, bad) \
  50. do { \
  51. ceph_decode_need(p, end, sizeof(u64), bad); \
  52. v = ceph_decode_64(p); \
  53. } while (0)
  54. #define ceph_decode_32_safe(p, end, v, bad) \
  55. do { \
  56. ceph_decode_need(p, end, sizeof(u32), bad); \
  57. v = ceph_decode_32(p); \
  58. } while (0)
  59. #define ceph_decode_16_safe(p, end, v, bad) \
  60. do { \
  61. ceph_decode_need(p, end, sizeof(u16), bad); \
  62. v = ceph_decode_16(p); \
  63. } while (0)
  64. #define ceph_decode_8_safe(p, end, v, bad) \
  65. do { \
  66. ceph_decode_need(p, end, sizeof(u8), bad); \
  67. v = ceph_decode_8(p); \
  68. } while (0)
  69. #define ceph_decode_copy_safe(p, end, pv, n, bad) \
  70. do { \
  71. ceph_decode_need(p, end, n, bad); \
  72. ceph_decode_copy(p, pv, n); \
  73. } while (0)
  74. /*
  75. * struct ceph_timespec <-> struct timespec
  76. */
  77. static inline void ceph_decode_timespec(struct timespec *ts,
  78. const struct ceph_timespec *tv)
  79. {
  80. ts->tv_sec = le32_to_cpu(tv->tv_sec);
  81. ts->tv_nsec = le32_to_cpu(tv->tv_nsec);
  82. }
  83. static inline void ceph_encode_timespec(struct ceph_timespec *tv,
  84. const struct timespec *ts)
  85. {
  86. tv->tv_sec = cpu_to_le32(ts->tv_sec);
  87. tv->tv_nsec = cpu_to_le32(ts->tv_nsec);
  88. }
  89. /*
  90. * sockaddr_storage <-> ceph_sockaddr
  91. */
  92. static inline void ceph_encode_addr(struct ceph_entity_addr *a)
  93. {
  94. __be16 ss_family = htons(a->in_addr.ss_family);
  95. a->in_addr.ss_family = *(__u16 *)&ss_family;
  96. }
  97. static inline void ceph_decode_addr(struct ceph_entity_addr *a)
  98. {
  99. __be16 ss_family = *(__be16 *)&a->in_addr.ss_family;
  100. a->in_addr.ss_family = ntohs(ss_family);
  101. WARN_ON(a->in_addr.ss_family == 512);
  102. }
  103. /*
  104. * encoders
  105. */
  106. static inline void ceph_encode_64(void **p, u64 v)
  107. {
  108. put_unaligned_le64(v, (__le64 *)*p);
  109. *p += sizeof(u64);
  110. }
  111. static inline void ceph_encode_32(void **p, u32 v)
  112. {
  113. put_unaligned_le32(v, (__le32 *)*p);
  114. *p += sizeof(u32);
  115. }
  116. static inline void ceph_encode_16(void **p, u16 v)
  117. {
  118. put_unaligned_le16(v, (__le16 *)*p);
  119. *p += sizeof(u16);
  120. }
  121. static inline void ceph_encode_8(void **p, u8 v)
  122. {
  123. *(u8 *)*p = v;
  124. (*p)++;
  125. }
  126. static inline void ceph_encode_copy(void **p, const void *s, int len)
  127. {
  128. memcpy(*p, s, len);
  129. *p += len;
  130. }
  131. /*
  132. * filepath, string encoders
  133. */
  134. static inline void ceph_encode_filepath(void **p, void *end,
  135. u64 ino, const char *path)
  136. {
  137. u32 len = path ? strlen(path) : 0;
  138. BUG_ON(*p + sizeof(ino) + sizeof(len) + len > end);
  139. ceph_encode_8(p, 1);
  140. ceph_encode_64(p, ino);
  141. ceph_encode_32(p, len);
  142. if (len)
  143. memcpy(*p, path, len);
  144. *p += len;
  145. }
  146. static inline void ceph_encode_string(void **p, void *end,
  147. const char *s, u32 len)
  148. {
  149. BUG_ON(*p + sizeof(len) + len > end);
  150. ceph_encode_32(p, len);
  151. if (len)
  152. memcpy(*p, s, len);
  153. *p += len;
  154. }
  155. #define ceph_encode_need(p, end, n, bad) \
  156. do { \
  157. if (unlikely(*(p) + (n) > (end))) \
  158. goto bad; \
  159. } while (0)
  160. #define ceph_encode_64_safe(p, end, v, bad) \
  161. do { \
  162. ceph_encode_need(p, end, sizeof(u64), bad); \
  163. ceph_encode_64(p, v); \
  164. } while (0)
  165. #define ceph_encode_32_safe(p, end, v, bad) \
  166. do { \
  167. ceph_encode_need(p, end, sizeof(u32), bad); \
  168. ceph_encode_32(p, v); \
  169. } while (0)
  170. #define ceph_encode_16_safe(p, end, v, bad) \
  171. do { \
  172. ceph_encode_need(p, end, sizeof(u16), bad); \
  173. ceph_encode_16(p, v); \
  174. } while (0)
  175. #define ceph_encode_copy_safe(p, end, pv, n, bad) \
  176. do { \
  177. ceph_encode_need(p, end, n, bad); \
  178. ceph_encode_copy(p, pv, n); \
  179. } while (0)
  180. #define ceph_encode_string_safe(p, end, s, n, bad) \
  181. do { \
  182. ceph_encode_need(p, end, n, bad); \
  183. ceph_encode_string(p, end, s, n); \
  184. } while (0)
  185. #endif