checksum.h 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. #ifndef _BFIN_CHECKSUM_H
  2. #define _BFIN_CHECKSUM_H
  3. /*
  4. * MODIFIED FOR BFIN April 30, 2001 akbar.hussain@lineo.com
  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. __wsum csum_partial(const void *buff, int len, __wsum sum);
  18. /*
  19. * the same as csum_partial, but copies from src while it
  20. * checksums
  21. *
  22. * here even more important to align src and dst on a 32-bit (or even
  23. * better 64-bit) boundary
  24. */
  25. __wsum csum_partial_copy(const void *src, void *dst,
  26. int len, __wsum sum);
  27. /*
  28. * the same as csum_partial_copy, but copies from user space.
  29. *
  30. * here even more important to align src and dst on a 32-bit (or even
  31. * better 64-bit) boundary
  32. */
  33. extern __wsum csum_partial_copy_from_user(const void __user *src, void *dst,
  34. int len, __wsum sum, int *csum_err);
  35. #define csum_partial_copy_nocheck(src, dst, len, sum) \
  36. csum_partial_copy((src), (dst), (len), (sum))
  37. __sum16 ip_fast_csum(unsigned char *iph, unsigned int ihl);
  38. /*
  39. * Fold a partial checksum
  40. */
  41. static inline __sum16 csum_fold(__wsum sum)
  42. {
  43. while (sum >> 16)
  44. sum = (sum & 0xffff) + (sum >> 16);
  45. return ((~(sum << 16)) >> 16);
  46. }
  47. /*
  48. * computes the checksum of the TCP/UDP pseudo-header
  49. * returns a 16-bit checksum, already complemented
  50. */
  51. static inline __wsum
  52. csum_tcpudp_nofold(__be32 saddr, __be32 daddr, unsigned short len,
  53. unsigned short proto, __wsum sum)
  54. {
  55. unsigned int carry;
  56. __asm__ ("%0 = %0 + %2;\n\t"
  57. "CC = AC0;\n\t"
  58. "%1 = CC;\n\t"
  59. "%0 = %0 + %1;\n\t"
  60. "%0 = %0 + %3;\n\t"
  61. "CC = AC0;\n\t"
  62. "%1 = CC;\n\t"
  63. "%0 = %0 + %1;\n\t"
  64. "%0 = %0 + %4;\n\t"
  65. "CC = AC0;\n\t"
  66. "%1 = CC;\n\t"
  67. "%0 = %0 + %1;\n\t"
  68. : "=d" (sum), "=&d" (carry)
  69. : "d" (daddr), "d" (saddr), "d" ((len + proto) << 8), "0"(sum)
  70. : "CC");
  71. return (sum);
  72. }
  73. static inline __sum16
  74. csum_tcpudp_magic(__be32 saddr, __be32 daddr, unsigned short len,
  75. unsigned short proto, __wsum sum)
  76. {
  77. return csum_fold(csum_tcpudp_nofold(saddr, daddr, len, proto, sum));
  78. }
  79. /*
  80. * this routine is used for miscellaneous IP-like checksums, mainly
  81. * in icmp.c
  82. */
  83. extern __sum16 ip_compute_csum(const void *buff, int len);
  84. #endif /* _BFIN_CHECKSUM_H */