OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rtems-20020807/] [cpukit/] [libnetworking/] [netinet/] [in_cksum.c] - Blame information for rev 1774

Go to most recent revision | Details | Compare with Previous | View Log

Line No. Rev Author Line
1 1026 ivang
/*
2
 * Copyright (c) 1988, 1992, 1993
3
 *      The Regents of the University of California.  All rights reserved.
4
 *
5
 * Redistribution and use in source and binary forms, with or without
6
 * modification, are permitted provided that the following conditions
7
 * are met:
8
 * 1. Redistributions of source code must retain the above copyright
9
 *    notice, this list of conditions and the following disclaimer.
10
 * 2. Redistributions in binary form must reproduce the above copyright
11
 *    notice, this list of conditions and the following disclaimer in the
12
 *    documentation and/or other materials provided with the distribution.
13
 * 3. All advertising materials mentioning features or use of this software
14
 *    must display the following acknowledgement:
15
 *      This product includes software developed by the University of
16
 *      California, Berkeley and its contributors.
17
 * 4. Neither the name of the University nor the names of its contributors
18
 *    may be used to endorse or promote products derived from this software
19
 *    without specific prior written permission.
20
 *
21
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31
 * SUCH DAMAGE.
32
 *
33
 *      @(#)in_cksum.c  8.1 (Berkeley) 6/10/93
34
 * in_cksum.c,v 1.6 2002/07/30 14:33:24 joel Exp
35
 */
36
 
37
#include <sys/param.h>
38
#include <sys/mbuf.h>
39
 
40
/*
41
 *  Try to use a CPU specific version, then punt to the portable C one.
42
 */
43
 
44
 
45
#if (defined(__GNUC__) && defined(__arm__))
46
 
47
#include "in_cksum_arm.c"
48
 
49
#elif (defined(__GNUC__) && defined(__i386__))
50
 
51
#include "in_cksum_i386.c"
52
 
53
#elif (defined(__GNUC__) && (defined(__mc68000__) || defined(__m68k__)))
54
 
55
#include "in_cksum_m68k.c"
56
#elif (defined(__GNUC__) && defined(__PPC__))
57
 
58
#include "in_cksum_powerpc.c"
59
 
60
#else
61
 
62
/*
63
 * Checksum routine for Internet Protocol family headers (Portable Version).
64
 *
65
 * This routine is very heavily used in the network
66
 * code and should be modified for each CPU to be as fast as possible.
67
 */
68
 
69
#define ADDCARRY(x)  (x > 65535 ? x -= 65535 : x)
70
#define REDUCE \
71
  {l_util.l = sum; sum = l_util.s[0] + l_util.s[1];  ADDCARRY(sum);}
72
 
73
int
74
in_cksum(m, len)
75
        register struct mbuf *m;
76
        register int len;
77
{
78
        register u_short *w;
79
        register int sum = 0;
80
        register int mlen = 0;
81
        int byte_swapped = 0;
82
 
83
        union {
84
                char    c[2];
85
                u_short s;
86
        } s_util;
87
        union {
88
                u_short s[2];
89
                long    l;
90
        } l_util;
91
 
92
        for (;m && len; m = m->m_next) {
93
                if (m->m_len == 0)
94
                        continue;
95
                w = mtod(m, u_short *);
96
                if (mlen == -1) {
97
                        /*
98
                         * The first byte of this mbuf is the continuation
99
                         * of a word spanning between this mbuf and the
100
                         * last mbuf.
101
                         *
102
                         * s_util.c[0] is already saved when scanning previous
103
                         * mbuf.
104
                         */
105
                        s_util.c[1] = *(char *)w;
106
                        sum += s_util.s;
107
                        w = (u_short *)((char *)w + 1);
108
                        mlen = m->m_len - 1;
109
                        len--;
110
                } else
111
                        mlen = m->m_len;
112
                if (len < mlen)
113
                        mlen = len;
114
                len -= mlen;
115
                /*
116
                 * Force to even boundary.
117
                 */
118
                if ((1 & (int) w) && (mlen > 0)) {
119
                        REDUCE;
120
                        sum <<= 8;
121
                        s_util.c[0] = *(u_char *)w;
122
                        w = (u_short *)((char *)w + 1);
123
                        mlen--;
124
                        byte_swapped = 1;
125
                }
126
                /*
127
                 * Unroll the loop to make overhead from
128
                 * branches &c small.
129
                 */
130
                while ((mlen -= 32) >= 0) {
131
                        sum += w[0]; sum += w[1]; sum += w[2]; sum += w[3];
132
                        sum += w[4]; sum += w[5]; sum += w[6]; sum += w[7];
133
                        sum += w[8]; sum += w[9]; sum += w[10]; sum += w[11];
134
                        sum += w[12]; sum += w[13]; sum += w[14]; sum += w[15];
135
                        w += 16;
136
                }
137
                mlen += 32;
138
                while ((mlen -= 8) >= 0) {
139
                        sum += w[0]; sum += w[1]; sum += w[2]; sum += w[3];
140
                        w += 4;
141
                }
142
                mlen += 8;
143
                if (mlen == 0 && byte_swapped == 0)
144
                        continue;
145
                REDUCE;
146
                while ((mlen -= 2) >= 0) {
147
                        sum += *w++;
148
                }
149
                if (byte_swapped) {
150
                        REDUCE;
151
                        sum <<= 8;
152
                        byte_swapped = 0;
153
                        if (mlen == -1) {
154
                                s_util.c[1] = *(char *)w;
155
                                sum += s_util.s;
156
                                mlen = 0;
157
                        } else
158
                                mlen = -1;
159
                } else if (mlen == -1)
160
                        s_util.c[0] = *(char *)w;
161
        }
162
        if (len)
163
                puts("cksum: out of data");
164
        if (mlen == -1) {
165
                /* The last mbuf has odd # of bytes. Follow the
166
                   standard (the odd byte may be shifted left by 8 bits
167
                   or not as determined by endian-ness of the machine) */
168
                s_util.c[1] = 0;
169
                sum += s_util.s;
170
        }
171
        REDUCE;
172
        return (~sum & 0xffff);
173
}
174
#endif

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.