checksum.h
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:3k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /* TODO: csum_tcpudp_magic could be speeded up, and csum_fold as well */
  2. #ifndef _CRIS_CHECKSUM_H
  3. #define _CRIS_CHECKSUM_H
  4. /*
  5.  * computes the checksum of a memory block at buff, length len,
  6.  * and adds in "sum" (32-bit)
  7.  *
  8.  * returns a 32-bit number suitable for feeding into itself
  9.  * or csum_tcpudp_magic
  10.  *
  11.  * this function must be called with even lengths, except
  12.  * for the last fragment, which may be odd
  13.  *
  14.  * it's best to have buff aligned on a 32-bit boundary
  15.  */
  16. unsigned int csum_partial(const unsigned char * buff, int len, unsigned int sum);
  17. /*
  18.  * the same as csum_partial, but copies from src while it
  19.  * checksums
  20.  *
  21.  * here even more important to align src and dst on a 32-bit (or even
  22.  * better 64-bit) boundary
  23.  */
  24. unsigned int csum_partial_copy_nocheck(const char *src, char *dst,
  25.        int len, unsigned int sum);
  26. /*
  27.  * Fold a partial checksum into a word
  28.  */
  29. static inline unsigned int csum_fold(unsigned int sum)
  30. {
  31. /* the while loop is unnecessary really, it's always enough with two
  32.    iterations */
  33. while(sum >> 16)
  34. sum = (sum & 0xffff) + (sum >> 16); /* add in end-around carry */
  35. return ~sum;
  36. }
  37. /* Checksum some values used in TCP/UDP headers.
  38.  *
  39.  * The gain by doing this in asm is that C will not generate carry-additions
  40.  * for the 32-bit components of the checksum, so otherwise we would have had
  41.  * to split all of those into 16-bit components, then add.
  42.  */
  43. static inline unsigned int
  44. csum_tcpudp_nofold(unsigned long saddr, unsigned long daddr, unsigned short len,
  45.    unsigned short proto, unsigned int sum)
  46. {
  47. int res;
  48. __asm__ ("add.d %2, %0nt"
  49.  "axnt"
  50.  "add.d %3, %0nt"
  51.  "axnt"
  52.  "add.d %4, %0nt"
  53.  "axnt"
  54.  "addq 0, %0n"
  55. : "=r" (res)
  56. : "0" (sum), "r" (daddr), "r" (saddr), "r" ((ntohs(len) << 16) + (proto << 8)));
  57. return res;
  58. }
  59. extern unsigned int csum_partial_copy_from_user(const char *src, char *dst,
  60. int len, unsigned int sum, 
  61. int *errptr);
  62. /*
  63.  * This is a version of ip_compute_csum() optimized for IP headers,
  64.  * which always checksum on 4 octet boundaries.
  65.  *
  66.  */
  67. static inline unsigned short ip_fast_csum(unsigned char * iph,
  68.   unsigned int ihl)
  69. {
  70. return csum_fold(csum_partial(iph, ihl * 4, 0));
  71. }
  72.  
  73. /*
  74.  * computes the checksum of the TCP/UDP pseudo-header
  75.  * returns a 16-bit checksum, already complemented
  76.  */
  77. static inline unsigned short int csum_tcpudp_magic(unsigned long saddr,
  78.    unsigned long daddr,
  79.    unsigned short len,
  80.    unsigned short proto,
  81.    unsigned int sum)
  82. {
  83. return csum_fold(csum_tcpudp_nofold(saddr,daddr,len,proto,sum));
  84. }
  85. /*
  86.  * this routine is used for miscellaneous IP-like checksums, mainly
  87.  * in icmp.c
  88.  */
  89. static inline unsigned short ip_compute_csum(unsigned char * buff, int len) {
  90. return csum_fold (csum_partial(buff, len, 0));
  91. }
  92. #endif