checksum.h 2.6 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. unsigned int csum_partial(const unsigned char *buff, int len, unsigned int 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. unsigned int csum_partial_copy(const unsigned char *src, unsigned char *dst,
  26. int len, int 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 unsigned int csum_partial_copy_from_user(const unsigned char *src,
  34. unsigned char *dst, int len,
  35. int sum, int *csum_err);
  36. #define csum_partial_copy_nocheck(src, dst, len, sum) \
  37. csum_partial_copy((src), (dst), (len), (sum))
  38. unsigned short ip_fast_csum(unsigned char *iph, unsigned int ihl);
  39. /*
  40. * Fold a partial checksum
  41. */
  42. static inline unsigned int csum_fold(unsigned int sum)
  43. {
  44. while (sum >> 16)
  45. sum = (sum & 0xffff) + (sum >> 16);
  46. return ((~(sum << 16)) >> 16);
  47. }
  48. /*
  49. * computes the checksum of the TCP/UDP pseudo-header
  50. * returns a 16-bit checksum, already complemented
  51. */
  52. static inline unsigned int
  53. csum_tcpudp_nofold(unsigned long saddr, unsigned long daddr, unsigned short len,
  54. unsigned short proto, unsigned int sum)
  55. {
  56. __asm__ ("%0 = %0 + %1;\n\t"
  57. "CC = AC0;\n\t"
  58. "if !CC jump 4;\n\t"
  59. "%0 = %0 + %4;\n\t"
  60. "%0 = %0 + %2;\n\t"
  61. "CC = AC0;\n\t"
  62. "if !CC jump 4;\n\t"
  63. "%0 = %0 + %4;\n\t"
  64. "%0 = %0 + %3;\n\t"
  65. "CC = AC0;\n\t"
  66. "if !CC jump 4;\n\t"
  67. "%0 = %0 + %4;\n\t"
  68. "NOP;\n\t"
  69. : "=d" (sum)
  70. : "d" (daddr), "d" (saddr), "d" ((ntohs(len)<<16)+proto*256), "d" (1), "0"(sum));
  71. return (sum);
  72. }
  73. static inline unsigned short int
  74. csum_tcpudp_magic(unsigned long saddr, unsigned long daddr, unsigned short len,
  75. unsigned short proto, unsigned int 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 unsigned short ip_compute_csum(const unsigned char *buff, int len);
  84. #endif /* _BFIN_CHECKSUM_H */