checksum.h 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. #ifndef _ASM_SCORE_CHECKSUM_H
  2. #define _ASM_SCORE_CHECKSUM_H
  3. #include <linux/in6.h>
  4. #include <asm/uaccess.h>
  5. /*
  6. * computes the checksum of a memory block at buff, length len,
  7. * and adds in "sum" (32-bit)
  8. *
  9. * returns a 32-bit number suitable for feeding into itself
  10. * or csum_tcpudp_magic
  11. *
  12. * this function must be called with even lengths, except
  13. * for the last fragment, which may be odd
  14. *
  15. * it's best to have buff aligned on a 32-bit boundary
  16. */
  17. unsigned int csum_partial(const void *buff, int len, __wsum sum);
  18. unsigned int csum_partial_copy_from_user(const char *src, char *dst, int len,
  19. unsigned int sum, int *csum_err);
  20. unsigned int csum_partial_copy(const char *src, char *dst,
  21. int len, unsigned int sum);
  22. /*
  23. * this is a new version of the above that records errors it finds in *errp,
  24. * but continues and zeros the rest of the buffer.
  25. */
  26. /*
  27. * Copy and checksum to user
  28. */
  29. #define HAVE_CSUM_COPY_USER
  30. static inline
  31. __wsum csum_and_copy_to_user(const void *src, void __user *dst, int len,
  32. __wsum sum, int *err_ptr)
  33. {
  34. sum = csum_partial(src, len, sum);
  35. if (copy_to_user(dst, src, len)) {
  36. *err_ptr = -EFAULT;
  37. return (__force __wsum) -1; /* invalid checksum */
  38. }
  39. return sum;
  40. }
  41. #define csum_partial_copy_nocheck csum_partial_copy
  42. /*
  43. * Fold a partial checksum without adding pseudo headers
  44. */
  45. static inline __sum16 csum_fold(__wsum sum)
  46. {
  47. /* the while loop is unnecessary really, it's always enough with two
  48. iterations */
  49. __asm__ __volatile__(
  50. ".set volatile\n\t"
  51. ".set\tr1\n\t"
  52. "slli\tr1,%0, 16\n\t"
  53. "add\t%0,%0, r1\n\t"
  54. "cmp.c\tr1, %0\n\t"
  55. "srli\t%0, %0, 16\n\t"
  56. "bleu\t1f\n\t"
  57. "addi\t%0, 0x1\n\t"
  58. "1:ldi\tr30, 0xffff\n\t"
  59. "xor\t%0, %0, r30\n\t"
  60. "slli\t%0, %0, 16\n\t"
  61. "srli\t%0, %0, 16\n\t"
  62. ".set\tnor1\n\t"
  63. ".set optimize\n\t"
  64. : "=r" (sum)
  65. : "0" (sum));
  66. return sum;
  67. }
  68. /*
  69. * This is a version of ip_compute_csum() optimized for IP headers,
  70. * which always checksum on 4 octet boundaries.
  71. *
  72. * By Jorge Cwik <jorge@laser.satlink.net>, adapted for linux by
  73. * Arnt Gulbrandsen.
  74. */
  75. static inline __sum16 ip_fast_csum(const void *iph, unsigned int ihl)
  76. {
  77. unsigned int sum;
  78. unsigned long dummy;
  79. __asm__ __volatile__(
  80. ".set volatile\n\t"
  81. ".set\tnor1\n\t"
  82. "lw\t%0, [%1]\n\t"
  83. "subri\t%2, %2, 4\n\t"
  84. "slli\t%2, %2, 2\n\t"
  85. "lw\t%3, [%1, 4]\n\t"
  86. "add\t%2, %2, %1\n\t"
  87. "add\t%0, %0, %3\n\t"
  88. "cmp.c\t%3, %0\n\t"
  89. "lw\t%3, [%1, 8]\n\t"
  90. "bleu\t1f\n\t"
  91. "addi\t%0, 0x1\n\t"
  92. "1:\n\t"
  93. "add\t%0, %0, %3\n\t"
  94. "cmp.c\t%3, %0\n\t"
  95. "lw\t%3, [%1, 12]\n\t"
  96. "bleu\t1f\n\t"
  97. "addi\t%0, 0x1\n\t"
  98. "1:add\t%0, %0, %3\n\t"
  99. "cmp.c\t%3, %0\n\t"
  100. "bleu\t1f\n\t"
  101. "addi\t%0, 0x1\n"
  102. "1:\tlw\t%3, [%1, 16]\n\t"
  103. "addi\t%1, 4\n\t"
  104. "add\t%0, %0, %3\n\t"
  105. "cmp.c\t%3, %0\n\t"
  106. "bleu\t2f\n\t"
  107. "addi\t%0, 0x1\n"
  108. "2:cmp.c\t%2, %1\n\t"
  109. "bne\t1b\n\t"
  110. ".set\tr1\n\t"
  111. ".set optimize\n\t"
  112. : "=&r" (sum), "=&r" (iph), "=&r" (ihl), "=&r" (dummy)
  113. : "1" (iph), "2" (ihl));
  114. return csum_fold(sum);
  115. }
  116. static inline __wsum
  117. csum_tcpudp_nofold(__be32 saddr, __be32 daddr, unsigned short len,
  118. unsigned short proto, __wsum sum)
  119. {
  120. unsigned long tmp = (ntohs(len) << 16) + proto * 256;
  121. __asm__ __volatile__(
  122. ".set volatile\n\t"
  123. "add\t%0, %0, %2\n\t"
  124. "cmp.c\t%2, %0\n\t"
  125. "bleu\t1f\n\t"
  126. "addi\t%0, 0x1\n\t"
  127. "1:\n\t"
  128. "add\t%0, %0, %3\n\t"
  129. "cmp.c\t%3, %0\n\t"
  130. "bleu\t1f\n\t"
  131. "addi\t%0, 0x1\n\t"
  132. "1:\n\t"
  133. "add\t%0, %0, %4\n\t"
  134. "cmp.c\t%4, %0\n\t"
  135. "bleu\t1f\n\t"
  136. "addi\t%0, 0x1\n\t"
  137. "1:\n\t"
  138. ".set optimize\n\t"
  139. : "=r" (sum)
  140. : "0" (daddr), "r"(saddr),
  141. "r" (tmp),
  142. "r" (sum));
  143. return sum;
  144. }
  145. /*
  146. * computes the checksum of the TCP/UDP pseudo-header
  147. * returns a 16-bit checksum, already complemented
  148. */
  149. static inline __sum16
  150. csum_tcpudp_magic(__be32 saddr, __be32 daddr, unsigned short len,
  151. unsigned short proto, __wsum sum)
  152. {
  153. return csum_fold(csum_tcpudp_nofold(saddr, daddr, len, proto, sum));
  154. }
  155. /*
  156. * this routine is used for miscellaneous IP-like checksums, mainly
  157. * in icmp.c
  158. */
  159. static inline unsigned short ip_compute_csum(const void *buff, int len)
  160. {
  161. return csum_fold(csum_partial(buff, len, 0));
  162. }
  163. #define _HAVE_ARCH_IPV6_CSUM
  164. static inline __sum16 csum_ipv6_magic(const struct in6_addr *saddr,
  165. const struct in6_addr *daddr,
  166. __u32 len, unsigned short proto,
  167. __wsum sum)
  168. {
  169. __asm__ __volatile__(
  170. ".set\tnoreorder\t\t\t# csum_ipv6_magic\n\t"
  171. ".set\tnoat\n\t"
  172. "addu\t%0, %5\t\t\t# proto (long in network byte order)\n\t"
  173. "sltu\t$1, %0, %5\n\t"
  174. "addu\t%0, $1\n\t"
  175. "addu\t%0, %6\t\t\t# csum\n\t"
  176. "sltu\t$1, %0, %6\n\t"
  177. "lw\t%1, 0(%2)\t\t\t# four words source address\n\t"
  178. "addu\t%0, $1\n\t"
  179. "addu\t%0, %1\n\t"
  180. "sltu\t$1, %0, %1\n\t"
  181. "lw\t%1, 4(%2)\n\t"
  182. "addu\t%0, $1\n\t"
  183. "addu\t%0, %1\n\t"
  184. "sltu\t$1, %0, %1\n\t"
  185. "lw\t%1, 8(%2)\n\t"
  186. "addu\t%0, $1\n\t"
  187. "addu\t%0, %1\n\t"
  188. "sltu\t$1, %0, %1\n\t"
  189. "lw\t%1, 12(%2)\n\t"
  190. "addu\t%0, $1\n\t"
  191. "addu\t%0, %1\n\t"
  192. "sltu\t$1, %0, %1\n\t"
  193. "lw\t%1, 0(%3)\n\t"
  194. "addu\t%0, $1\n\t"
  195. "addu\t%0, %1\n\t"
  196. "sltu\t$1, %0, %1\n\t"
  197. "lw\t%1, 4(%3)\n\t"
  198. "addu\t%0, $1\n\t"
  199. "addu\t%0, %1\n\t"
  200. "sltu\t$1, %0, %1\n\t"
  201. "lw\t%1, 8(%3)\n\t"
  202. "addu\t%0, $1\n\t"
  203. "addu\t%0, %1\n\t"
  204. "sltu\t$1, %0, %1\n\t"
  205. "lw\t%1, 12(%3)\n\t"
  206. "addu\t%0, $1\n\t"
  207. "addu\t%0, %1\n\t"
  208. "sltu\t$1, %0, %1\n\t"
  209. "addu\t%0, $1\t\t\t# Add final carry\n\t"
  210. ".set\tnoat\n\t"
  211. ".set\tnoreorder"
  212. : "=r" (sum), "=r" (proto)
  213. : "r" (saddr), "r" (daddr),
  214. "0" (htonl(len)), "1" (htonl(proto)), "r" (sum));
  215. return csum_fold(sum);
  216. }
  217. #endif /* _ASM_SCORE_CHECKSUM_H */