e02d066755fce908dcc8ebd0fd59e74ca34c8483
[dragonfly.git] / usr.sbin / IPXrouted / af.c
1 /*
2  * Copyright (c) 1985, 1993
3  *      The Regents of the University of California.  All rights reserved.
4  *
5  * Copyright (c) 1995 John Hay.  All rights reserved.
6  *
7  * This file includes significant work done at Cornell University by
8  * Bill Nesheim.  That work included by permission.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  * 3. All advertising materials mentioning features or use of this software
19  *    must display the following acknowledgement:
20  *      This product includes software developed by the University of
21  *      California, Berkeley and its contributors.
22  * 4. Neither the name of the University nor the names of its contributors
23  *    may be used to endorse or promote products derived from this software
24  *    without specific prior written permission.
25  *
26  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
27  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
28  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
29  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
30  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
31  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
32  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
33  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
34  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
35  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36  * SUCH DAMAGE.
37  *
38  * $FreeBSD: src/usr.sbin/IPXrouted/af.c,v 1.6 1999/08/28 01:15:01 peter Exp $
39  * $DragonFly: src/usr.sbin/IPXrouted/af.c,v 1.4 2004/12/18 22:48:02 swildner Exp $
40  *
41  * @(#)af.c     8.1 (Berkeley) 6/5/93
42  */
43
44 #include "defs.h"
45
46 /*
47  * Address family support routines
48  */
49 af_hash_t       null_hash;
50 af_netmatch_t   null_netmatch;
51 af_output_t     null_output;
52 af_portmatch_t  null_portmatch;
53 af_portcheck_t  null_portcheck;
54 af_checkhost_t  null_checkhost;
55 af_ishost_t     null_ishost;
56 af_canon_t      null_canon;
57
58 void    ipxnet_hash(struct sockaddr_ipx *, struct afhash *);
59 int     ipxnet_netmatch(struct sockaddr_ipx *, struct sockaddr_ipx *);
60 void    ipxnet_output(int, int, struct sockaddr_ipx *, int);
61 int     ipxnet_portmatch(struct sockaddr_ipx *);
62 int     ipxnet_checkhost(struct sockaddr_ipx *);
63 int     ipxnet_ishost(struct sockaddr_ipx *);
64 void    ipxnet_canon(struct sockaddr_ipx *);
65
66 #define NIL \
67         { null_hash,            null_netmatch,          null_output, \
68           null_portmatch,       null_portcheck,         null_checkhost, \
69           null_ishost,          null_canon }
70 #define IPXNET \
71         { (af_hash_t *)ipxnet_hash, \
72           (af_netmatch_t *)ipxnet_netmatch, \
73           (af_output_t *)ipxnet_output, \
74           (af_portmatch_t *)ipxnet_portmatch, \
75           (af_portcheck_t *)ipxnet_portmatch, \
76           (af_checkhost_t *)ipxnet_checkhost, \
77           (af_ishost_t *)ipxnet_ishost, \
78           (af_canon_t *)ipxnet_canon }
79
80 struct afswitch afswitch[AF_MAX] =
81         { NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL,
82           NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL, NIL,
83           NIL, NIL, NIL, IPXNET, NIL, NIL };
84
85 struct sockaddr_ipx ipxnet_default = { sizeof(struct sockaddr_ipx), AF_IPX };
86
87 union ipx_net ipx_anynet;
88 union ipx_net ipx_zeronet;
89
90 void
91 ipxnet_hash(struct sockaddr_ipx *sipx, struct afhash *hp)
92 {
93         long hash;
94 #if 0
95         u_short *s = sipx->sipx_addr.x_host.s_host;
96 #endif
97         u_char *c;
98
99         c = sipx->sipx_addr.x_net.c_net;
100
101 #define IMVAL   33
102         hash = 0;
103         hash = hash * IMVAL + *c++;
104         hash = hash * IMVAL + *c++;
105         hash = hash * IMVAL + *c++;
106         hash = hash * IMVAL + *c++;
107 #undef IMVAL
108
109         hp->afh_nethash = hash;
110         hp->afh_nethash ^= (hash >> 8);
111         hp->afh_nethash ^= (hash >> 16);
112         hp->afh_nethash ^= (hash >> 24);
113
114 #if 0
115         hash = 0;
116         hash = *s++; hash <<= 8; hash += *s++; hash <<= 8; hash += *s;
117         hp->afh_hosthash = hash;
118 #endif
119 }
120
121 int
122 ipxnet_netmatch(struct sockaddr_ipx *sxn1, struct sockaddr_ipx *sxn2)
123 {
124         return (ipx_neteq(sxn1->sipx_addr, sxn2->sipx_addr));
125 }
126
127 /*
128  * Verify the message is from the right port.
129  */
130 int
131 ipxnet_portmatch(struct sockaddr_ipx *sipx)
132 {
133         
134         return (ntohs(sipx->sipx_addr.x_port) == IPXPORT_RIP );
135 }
136
137
138 /*
139  * ipx output routine.
140  */
141 #ifdef DEBUG
142 int do_output = 0;
143 #endif
144 void
145 ipxnet_output(int s, int flags, struct sockaddr_ipx *sipx, int size)
146 {
147         struct sockaddr_ipx dst;
148
149         dst = *sipx;
150         sipx = &dst;
151         if (sipx->sipx_addr.x_port == 0)
152                 sipx->sipx_addr.x_port = htons(IPXPORT_RIP);
153 #ifdef DEBUG
154         if(do_output || ntohs(msg->rip_cmd) == RIPCMD_REQUEST)
155 #endif  
156         /*
157          * Kludge to allow us to get routes out to machines that
158          * don't know their addresses yet; send to that address on
159          * ALL connected nets
160          */
161          if (ipx_neteqnn(sipx->sipx_addr.x_net, ipx_zeronet)) {
162                 extern struct interface *ifnet;
163                 struct interface *ifp;
164                 
165                 for (ifp = ifnet; ifp; ifp = ifp->int_next) {
166                         sipx->sipx_addr.x_net = 
167                                 satoipx_addr(ifp->int_addr).x_net;
168                         sendto(s, msg, size, flags,
169                             (struct sockaddr *)sipx, sizeof (*sipx));
170                 }
171                 return;
172         }
173         
174         sendto(s, msg, size, flags,
175             (struct sockaddr *)sipx, sizeof (*sipx));
176 }
177
178 /*
179  * Return 1 if we want this route.
180  * We use this to disallow route net G entries for one for multiple
181  * point to point links.
182  */
183 int
184 ipxnet_checkhost(struct sockaddr_ipx *sipx)
185 {
186         struct interface *ifp = if_ifwithnet((struct sockaddr *)sipx);
187         /*
188          * We want this route if there is no more than one 
189          * point to point interface with this network.
190          */
191         if (ifp == 0 || (ifp->int_flags & IFF_POINTOPOINT)==0) return (1);
192         return (ifp->int_sq.n == ifp->int_sq.p);
193 }
194
195 /*
196  * Return 1 if the address is
197  * for a host, 0 for a network.
198  */
199 int
200 ipxnet_ishost(struct sockaddr_ipx *sipx)
201 {
202         u_short *s = sipx->sipx_addr.x_host.s_host;
203
204         if ((s[0]==0x0000) && (s[1]==0x0000) && (s[2]==0x0000))
205                 return (0);
206         if ((s[0]==0xffff) && (s[1]==0xffff) && (s[2]==0xffff))
207                 return (0);
208
209         return (1);
210 }
211
212 void
213 ipxnet_canon(struct sockaddr_ipx *sipx)
214 {
215
216         sipx->sipx_addr.x_port = 0;
217 }
218
219 void
220 null_hash(struct sockaddr *addr, struct afhash *hp)
221 {
222
223         hp->afh_nethash = hp->afh_hosthash = 0;
224 }
225
226 int
227 null_netmatch(struct sockaddr *a1, struct sockaddr *a2)
228 {
229
230         return (0);
231 }
232
233 void
234 null_output(int s, int f, struct sockaddr *a1, int n)
235 {
236
237         ;
238 }
239
240 int
241 null_portmatch(struct sockaddr *a1)
242 {
243
244         return (0);
245 }
246
247 int
248 null_portcheck(struct sockaddr *a1)
249 {
250
251         return (0);
252 }
253
254 int
255 null_ishost(struct sockaddr *a1)
256 {
257
258         return (0);
259 }
260
261 int
262 null_checkhost(struct sockaddr *a1)
263 {
264
265         return (0);
266 }
267
268 void
269 null_canon(struct sockaddr *a1)
270 {
271
272         ;
273 }
274