ftp://ftp.kernel.org/pub/linux/kernel/v2.6/linux-2.6.6.tar.bz2
[linux-2.6.git] / include / asm-m68knommu / checksum.h
1 #ifndef _M68K_CHECKSUM_H
2 #define _M68K_CHECKSUM_H
3
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 /*
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
26 unsigned int csum_partial_copy(const char *src, char *dst, int len, int sum);
27
28
29 /*
30  * the same as csum_partial_copy, but copies from user space.
31  *
32  * here even more important to align src and dst on a 32-bit (or even
33  * better 64-bit) boundary
34  */
35
36 extern unsigned int csum_partial_copy_from_user(const char *src, char *dst,
37                                                 int len, int sum, int *csum_err);
38
39 #define csum_partial_copy_nocheck(src, dst, len, sum)   \
40         csum_partial_copy((src), (dst), (len), (sum))
41
42 unsigned short ip_fast_csum(unsigned char *iph, unsigned int ihl);
43
44 /*
45  *      Fold a partial checksum
46  */
47
48 static inline unsigned int csum_fold(unsigned int sum)
49 {
50 #ifdef CONFIG_COLDFIRE
51         sum = (sum & 0xffff) + (sum >> 16);
52         sum = (sum & 0xffff) + (sum >> 16);
53 #else
54         unsigned int tmp = sum;
55         __asm__("swap %1\n\t"
56                 "addw %1, %0\n\t"
57                 "clrw %1\n\t"
58                 "addxw %1, %0"
59                 : "=&d" (sum), "=&d" (tmp)
60                 : "0" (sum), "1" (sum));
61 #endif
62         return ~sum;
63 }
64
65
66 /*
67  * computes the checksum of the TCP/UDP pseudo-header
68  * returns a 16-bit checksum, already complemented
69  */
70
71 static inline unsigned int
72 csum_tcpudp_nofold(unsigned long saddr, unsigned long daddr, unsigned short len,
73                   unsigned short proto, unsigned int sum)
74 {
75         __asm__ ("addl  %1,%0\n\t"
76                  "addxl %4,%0\n\t"
77                  "addxl %5,%0\n\t"
78                  "clrl %1\n\t"
79                  "addxl %1,%0"
80                  : "=&d" (sum), "=&d" (saddr)
81                  : "0" (daddr), "1" (saddr), "d" (len + proto),
82                    "d"(sum));
83         return sum;
84 }
85
86 static inline unsigned short int
87 csum_tcpudp_magic(unsigned long saddr, unsigned long daddr, unsigned short len,
88                   unsigned short proto, unsigned int sum)
89 {
90         return csum_fold(csum_tcpudp_nofold(saddr,daddr,len,proto,sum));
91 }
92
93 /*
94  * this routine is used for miscellaneous IP-like checksums, mainly
95  * in icmp.c
96  */
97
98 extern unsigned short ip_compute_csum(const unsigned char * buff, int len);
99
100 #define _HAVE_ARCH_IPV6_CSUM
101 static __inline__ unsigned short int
102 csum_ipv6_magic(struct in6_addr *saddr, struct in6_addr *daddr,
103                 __u32 len, unsigned short proto, unsigned int sum) 
104 {
105         register unsigned long tmp;
106         __asm__("addl %2@,%0\n\t"
107                 "movel %2@(4),%1\n\t"
108                 "addxl %1,%0\n\t"
109                 "movel %2@(8),%1\n\t"
110                 "addxl %1,%0\n\t"
111                 "movel %2@(12),%1\n\t"
112                 "addxl %1,%0\n\t"
113                 "movel %3@,%1\n\t"
114                 "addxl %1,%0\n\t"
115                 "movel %3@(4),%1\n\t"
116                 "addxl %1,%0\n\t"
117                 "movel %3@(8),%1\n\t"
118                 "addxl %1,%0\n\t"
119                 "movel %3@(12),%1\n\t"
120                 "addxl %1,%0\n\t"
121                 "addxl %4,%0\n\t"
122                 "clrl %1\n\t"
123                 "addxl %1,%0"
124                 : "=&d" (sum), "=&d" (tmp)
125                 : "a" (saddr), "a" (daddr), "d" (len + proto),
126                   "0" (sum));
127
128         return csum_fold(sum);
129 }
130
131 #endif /* _M68K_CHECKSUM_H */