xref: /inferno-os/os/ip/ethermedium.c (revision 74a4d8c26dd3c1e9febcb717cfd6cb6512991a7a)
1*74a4d8c2SCharles.Forsyth #include "u.h"
2*74a4d8c2SCharles.Forsyth #include "../port/lib.h"
3*74a4d8c2SCharles.Forsyth #include "mem.h"
4*74a4d8c2SCharles.Forsyth #include "dat.h"
5*74a4d8c2SCharles.Forsyth #include "fns.h"
6*74a4d8c2SCharles.Forsyth #include "../port/error.h"
7*74a4d8c2SCharles.Forsyth 
8*74a4d8c2SCharles.Forsyth #include "ip.h"
9*74a4d8c2SCharles.Forsyth #include "ipv6.h"
10*74a4d8c2SCharles.Forsyth #include "kernel.h"
11*74a4d8c2SCharles.Forsyth 
12*74a4d8c2SCharles.Forsyth typedef struct Etherhdr Etherhdr;
13*74a4d8c2SCharles.Forsyth struct Etherhdr
14*74a4d8c2SCharles.Forsyth {
15*74a4d8c2SCharles.Forsyth 	uchar	d[6];
16*74a4d8c2SCharles.Forsyth 	uchar	s[6];
17*74a4d8c2SCharles.Forsyth 	uchar	t[2];
18*74a4d8c2SCharles.Forsyth };
19*74a4d8c2SCharles.Forsyth 
20*74a4d8c2SCharles.Forsyth static uchar ipbroadcast[IPaddrlen] = {
21*74a4d8c2SCharles.Forsyth 	0xff,0xff,0xff,0xff,
22*74a4d8c2SCharles.Forsyth 	0xff,0xff,0xff,0xff,
23*74a4d8c2SCharles.Forsyth 	0xff,0xff,0xff,0xff,
24*74a4d8c2SCharles.Forsyth 	0xff,0xff,0xff,0xff,
25*74a4d8c2SCharles.Forsyth };
26*74a4d8c2SCharles.Forsyth 
27*74a4d8c2SCharles.Forsyth static uchar etherbroadcast[] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
28*74a4d8c2SCharles.Forsyth 
29*74a4d8c2SCharles.Forsyth static void	etherread4(void *a);
30*74a4d8c2SCharles.Forsyth static void	etherread6(void *a);
31*74a4d8c2SCharles.Forsyth static void	etherbind(Ipifc *ifc, int argc, char **argv);
32*74a4d8c2SCharles.Forsyth static void	etherunbind(Ipifc *ifc);
33*74a4d8c2SCharles.Forsyth static void	etherbwrite(Ipifc *ifc, Block *bp, int version, uchar *ip);
34*74a4d8c2SCharles.Forsyth static void	etheraddmulti(Ipifc *ifc, uchar *a, uchar *ia);
35*74a4d8c2SCharles.Forsyth static void	etherremmulti(Ipifc *ifc, uchar *a, uchar *ia);
36*74a4d8c2SCharles.Forsyth static Block*	multicastarp(Fs *f, Arpent *a, Medium*, uchar *mac);
37*74a4d8c2SCharles.Forsyth static void	sendarp(Ipifc *ifc, Arpent *a);
38*74a4d8c2SCharles.Forsyth static void	sendgarp(Ipifc *ifc, uchar*);
39*74a4d8c2SCharles.Forsyth static int	multicastea(uchar *ea, uchar *ip);
40*74a4d8c2SCharles.Forsyth static void	recvarpproc(void*);
41*74a4d8c2SCharles.Forsyth static void	resolveaddr6(Ipifc *ifc, Arpent *a);
42*74a4d8c2SCharles.Forsyth static void	etherpref2addr(uchar *pref, uchar *ea);
43*74a4d8c2SCharles.Forsyth 
44*74a4d8c2SCharles.Forsyth Medium ethermedium =
45*74a4d8c2SCharles.Forsyth {
46*74a4d8c2SCharles.Forsyth .name=		"ether",
47*74a4d8c2SCharles.Forsyth .hsize=		14,
48*74a4d8c2SCharles.Forsyth .mintu=		60,
49*74a4d8c2SCharles.Forsyth .maxtu=		1514,
50*74a4d8c2SCharles.Forsyth .maclen=	6,
51*74a4d8c2SCharles.Forsyth .bind=		etherbind,
52*74a4d8c2SCharles.Forsyth .unbind=	etherunbind,
53*74a4d8c2SCharles.Forsyth .bwrite=	etherbwrite,
54*74a4d8c2SCharles.Forsyth .addmulti=	etheraddmulti,
55*74a4d8c2SCharles.Forsyth .remmulti=	etherremmulti,
56*74a4d8c2SCharles.Forsyth .ares=		arpenter,
57*74a4d8c2SCharles.Forsyth .areg=		sendgarp,
58*74a4d8c2SCharles.Forsyth .pref2addr=	etherpref2addr,
59*74a4d8c2SCharles.Forsyth };
60*74a4d8c2SCharles.Forsyth 
61*74a4d8c2SCharles.Forsyth Medium gbemedium =
62*74a4d8c2SCharles.Forsyth {
63*74a4d8c2SCharles.Forsyth .name=		"gbe",
64*74a4d8c2SCharles.Forsyth .hsize=		14,
65*74a4d8c2SCharles.Forsyth .mintu=		60,
66*74a4d8c2SCharles.Forsyth .maxtu=		9014,
67*74a4d8c2SCharles.Forsyth .maclen=	6,
68*74a4d8c2SCharles.Forsyth .bind=		etherbind,
69*74a4d8c2SCharles.Forsyth .unbind=	etherunbind,
70*74a4d8c2SCharles.Forsyth .bwrite=	etherbwrite,
71*74a4d8c2SCharles.Forsyth .addmulti=	etheraddmulti,
72*74a4d8c2SCharles.Forsyth .remmulti=	etherremmulti,
73*74a4d8c2SCharles.Forsyth .ares=		arpenter,
74*74a4d8c2SCharles.Forsyth .areg=		sendgarp,
75*74a4d8c2SCharles.Forsyth .pref2addr=	etherpref2addr,
76*74a4d8c2SCharles.Forsyth };
77*74a4d8c2SCharles.Forsyth 
78*74a4d8c2SCharles.Forsyth typedef struct	Etherrock Etherrock;
79*74a4d8c2SCharles.Forsyth struct Etherrock
80*74a4d8c2SCharles.Forsyth {
81*74a4d8c2SCharles.Forsyth 	Fs	*f;		/* file system we belong to */
82*74a4d8c2SCharles.Forsyth 	Proc	*arpp;		/* arp process */
83*74a4d8c2SCharles.Forsyth 	Proc	*read4p;	/* reading process (v4)*/
84*74a4d8c2SCharles.Forsyth 	Proc	*read6p;	/* reading process (v6)*/
85*74a4d8c2SCharles.Forsyth 	Chan	*mchan4;	/* Data channel for v4 */
86*74a4d8c2SCharles.Forsyth 	Chan	*achan;		/* Arp channel */
87*74a4d8c2SCharles.Forsyth 	Chan	*cchan4;	/* Control channel for v4 */
88*74a4d8c2SCharles.Forsyth 	Chan	*mchan6;	/* Data channel for v6 */
89*74a4d8c2SCharles.Forsyth 	Chan	*cchan6;	/* Control channel for v6 */
90*74a4d8c2SCharles.Forsyth };
91*74a4d8c2SCharles.Forsyth 
92*74a4d8c2SCharles.Forsyth /*
93*74a4d8c2SCharles.Forsyth  *  ethernet arp request
94*74a4d8c2SCharles.Forsyth  */
95*74a4d8c2SCharles.Forsyth enum
96*74a4d8c2SCharles.Forsyth {
97*74a4d8c2SCharles.Forsyth 	ETARP		= 0x0806,
98*74a4d8c2SCharles.Forsyth 	ETIP4		= 0x0800,
99*74a4d8c2SCharles.Forsyth 	ETIP6		= 0x86DD,
100*74a4d8c2SCharles.Forsyth 	ARPREQUEST	= 1,
101*74a4d8c2SCharles.Forsyth 	ARPREPLY	= 2,
102*74a4d8c2SCharles.Forsyth };
103*74a4d8c2SCharles.Forsyth 
104*74a4d8c2SCharles.Forsyth typedef struct Etherarp Etherarp;
105*74a4d8c2SCharles.Forsyth struct Etherarp
106*74a4d8c2SCharles.Forsyth {
107*74a4d8c2SCharles.Forsyth 	uchar	d[6];
108*74a4d8c2SCharles.Forsyth 	uchar	s[6];
109*74a4d8c2SCharles.Forsyth 	uchar	type[2];
110*74a4d8c2SCharles.Forsyth 	uchar	hrd[2];
111*74a4d8c2SCharles.Forsyth 	uchar	pro[2];
112*74a4d8c2SCharles.Forsyth 	uchar	hln;
113*74a4d8c2SCharles.Forsyth 	uchar	pln;
114*74a4d8c2SCharles.Forsyth 	uchar	op[2];
115*74a4d8c2SCharles.Forsyth 	uchar	sha[6];
116*74a4d8c2SCharles.Forsyth 	uchar	spa[4];
117*74a4d8c2SCharles.Forsyth 	uchar	tha[6];
118*74a4d8c2SCharles.Forsyth 	uchar	tpa[4];
119*74a4d8c2SCharles.Forsyth };
120*74a4d8c2SCharles.Forsyth 
121*74a4d8c2SCharles.Forsyth static char *nbmsg = "nonblocking";
122*74a4d8c2SCharles.Forsyth 
123*74a4d8c2SCharles.Forsyth /*
124*74a4d8c2SCharles.Forsyth  *  called to bind an IP ifc to an ethernet device
125*74a4d8c2SCharles.Forsyth  *  called with ifc wlock'd
126*74a4d8c2SCharles.Forsyth  */
127*74a4d8c2SCharles.Forsyth static void
etherbind(Ipifc * ifc,int argc,char ** argv)128*74a4d8c2SCharles.Forsyth etherbind(Ipifc *ifc, int argc, char **argv)
129*74a4d8c2SCharles.Forsyth {
130*74a4d8c2SCharles.Forsyth 	Chan *mchan4, *cchan4, *achan, *mchan6, *cchan6;
131*74a4d8c2SCharles.Forsyth 	char addr[Maxpath];	//char addr[2*KNAMELEN];
132*74a4d8c2SCharles.Forsyth 	char dir[Maxpath];	//char dir[2*KNAMELEN];
133*74a4d8c2SCharles.Forsyth 	char *buf;
134*74a4d8c2SCharles.Forsyth 	int fd, cfd, n;
135*74a4d8c2SCharles.Forsyth 	char *ptr;
136*74a4d8c2SCharles.Forsyth 	Etherrock *er;
137*74a4d8c2SCharles.Forsyth 
138*74a4d8c2SCharles.Forsyth 	if(argc < 2)
139*74a4d8c2SCharles.Forsyth 		error(Ebadarg);
140*74a4d8c2SCharles.Forsyth 
141*74a4d8c2SCharles.Forsyth 	mchan4 = cchan4 = achan = mchan6 = cchan6 = nil;
142*74a4d8c2SCharles.Forsyth 	buf = nil;
143*74a4d8c2SCharles.Forsyth 	if(waserror()){
144*74a4d8c2SCharles.Forsyth 		if(mchan4 != nil)
145*74a4d8c2SCharles.Forsyth 			cclose(mchan4);
146*74a4d8c2SCharles.Forsyth 		if(cchan4 != nil)
147*74a4d8c2SCharles.Forsyth 			cclose(cchan4);
148*74a4d8c2SCharles.Forsyth 		if(achan != nil)
149*74a4d8c2SCharles.Forsyth 			cclose(achan);
150*74a4d8c2SCharles.Forsyth 		if(mchan6 != nil)
151*74a4d8c2SCharles.Forsyth 			cclose(mchan6);
152*74a4d8c2SCharles.Forsyth 		if(cchan6 != nil)
153*74a4d8c2SCharles.Forsyth 			cclose(cchan6);
154*74a4d8c2SCharles.Forsyth 		if(buf != nil)
155*74a4d8c2SCharles.Forsyth 			free(buf);
156*74a4d8c2SCharles.Forsyth 		nexterror();
157*74a4d8c2SCharles.Forsyth 	}
158*74a4d8c2SCharles.Forsyth 
159*74a4d8c2SCharles.Forsyth 	/*
160*74a4d8c2SCharles.Forsyth 	 *  open ip converstation
161*74a4d8c2SCharles.Forsyth 	 *
162*74a4d8c2SCharles.Forsyth 	 *  the dial will fail if the type is already open on
163*74a4d8c2SCharles.Forsyth 	 *  this device.
164*74a4d8c2SCharles.Forsyth 	 */
165*74a4d8c2SCharles.Forsyth 	snprint(addr, sizeof(addr), "%s!0x800", argv[2]);
166*74a4d8c2SCharles.Forsyth 	fd = kdial(addr, nil, dir, &cfd);
167*74a4d8c2SCharles.Forsyth 	if(fd < 0)
168*74a4d8c2SCharles.Forsyth 		errorf("dial 0x800 failed: %s", up->env->errstr);
169*74a4d8c2SCharles.Forsyth 	mchan4 = commonfdtochan(fd, ORDWR, 0, 1);
170*74a4d8c2SCharles.Forsyth 	cchan4 = commonfdtochan(cfd, ORDWR, 0, 1);
171*74a4d8c2SCharles.Forsyth 	kclose(fd);
172*74a4d8c2SCharles.Forsyth 	kclose(cfd);
173*74a4d8c2SCharles.Forsyth 
174*74a4d8c2SCharles.Forsyth 	/*
175*74a4d8c2SCharles.Forsyth 	 *  make it non-blocking
176*74a4d8c2SCharles.Forsyth 	 */
177*74a4d8c2SCharles.Forsyth 	devtab[cchan4->type]->write(cchan4, nbmsg, strlen(nbmsg), 0);
178*74a4d8c2SCharles.Forsyth 
179*74a4d8c2SCharles.Forsyth 	/*
180*74a4d8c2SCharles.Forsyth 	 *  get mac address and speed
181*74a4d8c2SCharles.Forsyth 	 */
182*74a4d8c2SCharles.Forsyth 	snprint(addr, sizeof(addr), "%s/stats", dir);
183*74a4d8c2SCharles.Forsyth 	fd = kopen(addr, OREAD);
184*74a4d8c2SCharles.Forsyth 	if(fd < 0)
185*74a4d8c2SCharles.Forsyth 		errorf("can't open ether stats: %s", up->env->errstr);
186*74a4d8c2SCharles.Forsyth 
187*74a4d8c2SCharles.Forsyth 	buf = smalloc(512);
188*74a4d8c2SCharles.Forsyth 	n = kread(fd, buf, 511);
189*74a4d8c2SCharles.Forsyth 	kclose(fd);
190*74a4d8c2SCharles.Forsyth 	if(n <= 0)
191*74a4d8c2SCharles.Forsyth 		error(Eio);
192*74a4d8c2SCharles.Forsyth 	buf[n] = 0;
193*74a4d8c2SCharles.Forsyth 
194*74a4d8c2SCharles.Forsyth 	ptr = strstr(buf, "addr: ");
195*74a4d8c2SCharles.Forsyth 	if(!ptr)
196*74a4d8c2SCharles.Forsyth 		error(Eio);
197*74a4d8c2SCharles.Forsyth 	ptr += 6;
198*74a4d8c2SCharles.Forsyth 	parsemac(ifc->mac, ptr, 6);
199*74a4d8c2SCharles.Forsyth 
200*74a4d8c2SCharles.Forsyth 	ptr = strstr(buf, "mbps: ");
201*74a4d8c2SCharles.Forsyth 	if(ptr){
202*74a4d8c2SCharles.Forsyth 		ptr += 6;
203*74a4d8c2SCharles.Forsyth 		ifc->mbps = atoi(ptr);
204*74a4d8c2SCharles.Forsyth 	} else
205*74a4d8c2SCharles.Forsyth 		ifc->mbps = 100;
206*74a4d8c2SCharles.Forsyth 
207*74a4d8c2SCharles.Forsyth 	/*
208*74a4d8c2SCharles.Forsyth  	 *  open arp conversation
209*74a4d8c2SCharles.Forsyth 	 */
210*74a4d8c2SCharles.Forsyth 	snprint(addr, sizeof(addr), "%s!0x806", argv[2]);
211*74a4d8c2SCharles.Forsyth 	fd = kdial(addr, nil, nil, nil);
212*74a4d8c2SCharles.Forsyth 	if(fd < 0)
213*74a4d8c2SCharles.Forsyth 		errorf("dial 0x806 failed: %s", up->env->errstr);
214*74a4d8c2SCharles.Forsyth 	achan = commonfdtochan(fd, ORDWR, 0, 1);
215*74a4d8c2SCharles.Forsyth 	kclose(fd);
216*74a4d8c2SCharles.Forsyth 
217*74a4d8c2SCharles.Forsyth 	/*
218*74a4d8c2SCharles.Forsyth 	 *  open ip conversation
219*74a4d8c2SCharles.Forsyth 	 *
220*74a4d8c2SCharles.Forsyth 	 *  the dial will fail if the type is already open on
221*74a4d8c2SCharles.Forsyth 	 *  this device.
222*74a4d8c2SCharles.Forsyth 	 */
223*74a4d8c2SCharles.Forsyth 	snprint(addr, sizeof(addr), "%s!0x86DD", argv[2]);
224*74a4d8c2SCharles.Forsyth 	fd = kdial(addr, nil, dir, &cfd);
225*74a4d8c2SCharles.Forsyth 	if(fd < 0)
226*74a4d8c2SCharles.Forsyth 		errorf("dial 0x86DD failed: %s", up->env->errstr);
227*74a4d8c2SCharles.Forsyth 	mchan6 = commonfdtochan(fd, ORDWR, 0, 1);
228*74a4d8c2SCharles.Forsyth 	cchan6 = commonfdtochan(cfd, ORDWR, 0, 1);
229*74a4d8c2SCharles.Forsyth 	kclose(fd);
230*74a4d8c2SCharles.Forsyth 	kclose(cfd);
231*74a4d8c2SCharles.Forsyth 
232*74a4d8c2SCharles.Forsyth 	/*
233*74a4d8c2SCharles.Forsyth 	 *  make it non-blocking
234*74a4d8c2SCharles.Forsyth 	 */
235*74a4d8c2SCharles.Forsyth 	devtab[cchan6->type]->write(cchan6, nbmsg, strlen(nbmsg), 0);
236*74a4d8c2SCharles.Forsyth 
237*74a4d8c2SCharles.Forsyth 	er = smalloc(sizeof(*er));
238*74a4d8c2SCharles.Forsyth 	er->mchan4 = mchan4;
239*74a4d8c2SCharles.Forsyth 	er->cchan4 = cchan4;
240*74a4d8c2SCharles.Forsyth 	er->achan = achan;
241*74a4d8c2SCharles.Forsyth 	er->mchan6 = mchan6;
242*74a4d8c2SCharles.Forsyth 	er->cchan6 = cchan6;
243*74a4d8c2SCharles.Forsyth 	er->f = ifc->conv->p->f;
244*74a4d8c2SCharles.Forsyth 	ifc->arg = er;
245*74a4d8c2SCharles.Forsyth 
246*74a4d8c2SCharles.Forsyth 	free(buf);
247*74a4d8c2SCharles.Forsyth 	poperror();
248*74a4d8c2SCharles.Forsyth 
249*74a4d8c2SCharles.Forsyth 	kproc("etherread4", etherread4, ifc, 0);
250*74a4d8c2SCharles.Forsyth 	kproc("recvarpproc", recvarpproc, ifc, 0);
251*74a4d8c2SCharles.Forsyth 	kproc("etherread6", etherread6, ifc, 0);
252*74a4d8c2SCharles.Forsyth }
253*74a4d8c2SCharles.Forsyth 
254*74a4d8c2SCharles.Forsyth /*
255*74a4d8c2SCharles.Forsyth  *  called with ifc wlock'd
256*74a4d8c2SCharles.Forsyth  */
257*74a4d8c2SCharles.Forsyth static void
etherunbind(Ipifc * ifc)258*74a4d8c2SCharles.Forsyth etherunbind(Ipifc *ifc)
259*74a4d8c2SCharles.Forsyth {
260*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
261*74a4d8c2SCharles.Forsyth 
262*74a4d8c2SCharles.Forsyth 	if(er->read4p)
263*74a4d8c2SCharles.Forsyth 		postnote(er->read4p, 1, "unbind", 0);
264*74a4d8c2SCharles.Forsyth 	if(er->read6p)
265*74a4d8c2SCharles.Forsyth 		postnote(er->read6p, 1, "unbind", 0);
266*74a4d8c2SCharles.Forsyth 	if(er->arpp)
267*74a4d8c2SCharles.Forsyth 		postnote(er->arpp, 1, "unbind", 0);
268*74a4d8c2SCharles.Forsyth 
269*74a4d8c2SCharles.Forsyth 	/* wait for readers to die */
270*74a4d8c2SCharles.Forsyth 	while(er->arpp != 0 || er->read4p != 0 || er->read6p != 0)
271*74a4d8c2SCharles.Forsyth 		tsleep(&up->sleep, return0, 0, 300);
272*74a4d8c2SCharles.Forsyth 
273*74a4d8c2SCharles.Forsyth 	if(er->mchan4 != nil)
274*74a4d8c2SCharles.Forsyth 		cclose(er->mchan4);
275*74a4d8c2SCharles.Forsyth 	if(er->achan != nil)
276*74a4d8c2SCharles.Forsyth 		cclose(er->achan);
277*74a4d8c2SCharles.Forsyth 	if(er->cchan4 != nil)
278*74a4d8c2SCharles.Forsyth 		cclose(er->cchan4);
279*74a4d8c2SCharles.Forsyth 	if(er->mchan6 != nil)
280*74a4d8c2SCharles.Forsyth 		cclose(er->mchan6);
281*74a4d8c2SCharles.Forsyth 	if(er->cchan6 != nil)
282*74a4d8c2SCharles.Forsyth 		cclose(er->cchan6);
283*74a4d8c2SCharles.Forsyth 
284*74a4d8c2SCharles.Forsyth 	free(er);
285*74a4d8c2SCharles.Forsyth }
286*74a4d8c2SCharles.Forsyth 
287*74a4d8c2SCharles.Forsyth /*
288*74a4d8c2SCharles.Forsyth  *  called by ipoput with a single block to write with ifc rlock'd
289*74a4d8c2SCharles.Forsyth  */
290*74a4d8c2SCharles.Forsyth static void
etherbwrite(Ipifc * ifc,Block * bp,int version,uchar * ip)291*74a4d8c2SCharles.Forsyth etherbwrite(Ipifc *ifc, Block *bp, int version, uchar *ip)
292*74a4d8c2SCharles.Forsyth {
293*74a4d8c2SCharles.Forsyth 	Etherhdr *eh;
294*74a4d8c2SCharles.Forsyth 	Arpent *a;
295*74a4d8c2SCharles.Forsyth 	uchar mac[6];
296*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
297*74a4d8c2SCharles.Forsyth 
298*74a4d8c2SCharles.Forsyth 	/* get mac address of destination */
299*74a4d8c2SCharles.Forsyth 	a = arpget(er->f->arp, bp, version, ifc, ip, mac);
300*74a4d8c2SCharles.Forsyth 	if(a){
301*74a4d8c2SCharles.Forsyth 		/* check for broadcast or multicast */
302*74a4d8c2SCharles.Forsyth 		bp = multicastarp(er->f, a, ifc->m, mac);
303*74a4d8c2SCharles.Forsyth 		if(bp==nil){
304*74a4d8c2SCharles.Forsyth 			switch(version){
305*74a4d8c2SCharles.Forsyth 			case V4:
306*74a4d8c2SCharles.Forsyth 				sendarp(ifc, a);
307*74a4d8c2SCharles.Forsyth 				break;
308*74a4d8c2SCharles.Forsyth 			case V6:
309*74a4d8c2SCharles.Forsyth 				resolveaddr6(ifc, a);
310*74a4d8c2SCharles.Forsyth 				break;
311*74a4d8c2SCharles.Forsyth 			default:
312*74a4d8c2SCharles.Forsyth 				panic("etherbwrite: version %d", version);
313*74a4d8c2SCharles.Forsyth 			}
314*74a4d8c2SCharles.Forsyth 			return;
315*74a4d8c2SCharles.Forsyth 		}
316*74a4d8c2SCharles.Forsyth 	}
317*74a4d8c2SCharles.Forsyth 
318*74a4d8c2SCharles.Forsyth 	/* make it a single block with space for the ether header */
319*74a4d8c2SCharles.Forsyth 	bp = padblock(bp, ifc->m->hsize);
320*74a4d8c2SCharles.Forsyth 	if(bp->next)
321*74a4d8c2SCharles.Forsyth 		bp = concatblock(bp);
322*74a4d8c2SCharles.Forsyth 	if(BLEN(bp) < ifc->mintu)
323*74a4d8c2SCharles.Forsyth 		bp = adjustblock(bp, ifc->mintu);
324*74a4d8c2SCharles.Forsyth 	eh = (Etherhdr*)bp->rp;
325*74a4d8c2SCharles.Forsyth 
326*74a4d8c2SCharles.Forsyth 	/* copy in mac addresses and ether type */
327*74a4d8c2SCharles.Forsyth 	memmove(eh->s, ifc->mac, sizeof(eh->s));
328*74a4d8c2SCharles.Forsyth 	memmove(eh->d, mac, sizeof(eh->d));
329*74a4d8c2SCharles.Forsyth 
330*74a4d8c2SCharles.Forsyth  	switch(version){
331*74a4d8c2SCharles.Forsyth 	case V4:
332*74a4d8c2SCharles.Forsyth 		eh->t[0] = 0x08;
333*74a4d8c2SCharles.Forsyth 		eh->t[1] = 0x00;
334*74a4d8c2SCharles.Forsyth 		devtab[er->mchan4->type]->bwrite(er->mchan4, bp, 0);
335*74a4d8c2SCharles.Forsyth 		break;
336*74a4d8c2SCharles.Forsyth 	case V6:
337*74a4d8c2SCharles.Forsyth 		eh->t[0] = 0x86;
338*74a4d8c2SCharles.Forsyth 		eh->t[1] = 0xDD;
339*74a4d8c2SCharles.Forsyth 		devtab[er->mchan6->type]->bwrite(er->mchan6, bp, 0);
340*74a4d8c2SCharles.Forsyth 		break;
341*74a4d8c2SCharles.Forsyth 	default:
342*74a4d8c2SCharles.Forsyth 		panic("etherbwrite2: version %d", version);
343*74a4d8c2SCharles.Forsyth 	}
344*74a4d8c2SCharles.Forsyth 	ifc->out++;
345*74a4d8c2SCharles.Forsyth }
346*74a4d8c2SCharles.Forsyth 
347*74a4d8c2SCharles.Forsyth 
348*74a4d8c2SCharles.Forsyth /*
349*74a4d8c2SCharles.Forsyth  *  process to read from the ethernet
350*74a4d8c2SCharles.Forsyth  */
351*74a4d8c2SCharles.Forsyth static void
etherread4(void * a)352*74a4d8c2SCharles.Forsyth etherread4(void *a)
353*74a4d8c2SCharles.Forsyth {
354*74a4d8c2SCharles.Forsyth 	Ipifc *ifc;
355*74a4d8c2SCharles.Forsyth 	Block *bp;
356*74a4d8c2SCharles.Forsyth 	Etherrock *er;
357*74a4d8c2SCharles.Forsyth 
358*74a4d8c2SCharles.Forsyth 	ifc = a;
359*74a4d8c2SCharles.Forsyth 	er = ifc->arg;
360*74a4d8c2SCharles.Forsyth 	er->read4p = up;	/* hide identity under a rock for unbind */
361*74a4d8c2SCharles.Forsyth 	if(waserror()){
362*74a4d8c2SCharles.Forsyth 		er->read4p = 0;
363*74a4d8c2SCharles.Forsyth 		pexit("hangup", 1);
364*74a4d8c2SCharles.Forsyth 	}
365*74a4d8c2SCharles.Forsyth 	for(;;){
366*74a4d8c2SCharles.Forsyth 		bp = devtab[er->mchan4->type]->bread(er->mchan4, ifc->maxtu, 0);
367*74a4d8c2SCharles.Forsyth 		if(!canrlock(ifc)){
368*74a4d8c2SCharles.Forsyth 			freeb(bp);
369*74a4d8c2SCharles.Forsyth 			continue;
370*74a4d8c2SCharles.Forsyth 		}
371*74a4d8c2SCharles.Forsyth 		if(waserror()){
372*74a4d8c2SCharles.Forsyth 			runlock(ifc);
373*74a4d8c2SCharles.Forsyth 			nexterror();
374*74a4d8c2SCharles.Forsyth 		}
375*74a4d8c2SCharles.Forsyth 		ifc->in++;
376*74a4d8c2SCharles.Forsyth 		bp->rp += ifc->m->hsize;
377*74a4d8c2SCharles.Forsyth 		if(ifc->lifc == nil)
378*74a4d8c2SCharles.Forsyth 			freeb(bp);
379*74a4d8c2SCharles.Forsyth 		else
380*74a4d8c2SCharles.Forsyth 			ipiput4(er->f, ifc, bp);
381*74a4d8c2SCharles.Forsyth 		runlock(ifc);
382*74a4d8c2SCharles.Forsyth 		poperror();
383*74a4d8c2SCharles.Forsyth 	}
384*74a4d8c2SCharles.Forsyth }
385*74a4d8c2SCharles.Forsyth 
386*74a4d8c2SCharles.Forsyth 
387*74a4d8c2SCharles.Forsyth /*
388*74a4d8c2SCharles.Forsyth  *  process to read from the ethernet, IPv6
389*74a4d8c2SCharles.Forsyth  */
390*74a4d8c2SCharles.Forsyth static void
etherread6(void * a)391*74a4d8c2SCharles.Forsyth etherread6(void *a)
392*74a4d8c2SCharles.Forsyth {
393*74a4d8c2SCharles.Forsyth 	Ipifc *ifc;
394*74a4d8c2SCharles.Forsyth 	Block *bp;
395*74a4d8c2SCharles.Forsyth 	Etherrock *er;
396*74a4d8c2SCharles.Forsyth 
397*74a4d8c2SCharles.Forsyth 	ifc = a;
398*74a4d8c2SCharles.Forsyth 	er = ifc->arg;
399*74a4d8c2SCharles.Forsyth 	er->read6p = up;	/* hide identity under a rock for unbind */
400*74a4d8c2SCharles.Forsyth 	if(waserror()){
401*74a4d8c2SCharles.Forsyth 		er->read6p = 0;
402*74a4d8c2SCharles.Forsyth 		pexit("hangup", 1);
403*74a4d8c2SCharles.Forsyth 	}
404*74a4d8c2SCharles.Forsyth 	for(;;){
405*74a4d8c2SCharles.Forsyth 		bp = devtab[er->mchan6->type]->bread(er->mchan6, ifc->maxtu, 0);
406*74a4d8c2SCharles.Forsyth 		if(!canrlock(ifc)){
407*74a4d8c2SCharles.Forsyth 			freeb(bp);
408*74a4d8c2SCharles.Forsyth 			continue;
409*74a4d8c2SCharles.Forsyth 		}
410*74a4d8c2SCharles.Forsyth 		if(waserror()){
411*74a4d8c2SCharles.Forsyth 			runlock(ifc);
412*74a4d8c2SCharles.Forsyth 			nexterror();
413*74a4d8c2SCharles.Forsyth 		}
414*74a4d8c2SCharles.Forsyth 		ifc->in++;
415*74a4d8c2SCharles.Forsyth 		bp->rp += ifc->m->hsize;
416*74a4d8c2SCharles.Forsyth 		if(ifc->lifc == nil)
417*74a4d8c2SCharles.Forsyth 			freeb(bp);
418*74a4d8c2SCharles.Forsyth 		else
419*74a4d8c2SCharles.Forsyth 			ipiput6(er->f, ifc, bp);
420*74a4d8c2SCharles.Forsyth 		runlock(ifc);
421*74a4d8c2SCharles.Forsyth 		poperror();
422*74a4d8c2SCharles.Forsyth 	}
423*74a4d8c2SCharles.Forsyth }
424*74a4d8c2SCharles.Forsyth 
425*74a4d8c2SCharles.Forsyth static void
etheraddmulti(Ipifc * ifc,uchar * a,uchar *)426*74a4d8c2SCharles.Forsyth etheraddmulti(Ipifc *ifc, uchar *a, uchar *)
427*74a4d8c2SCharles.Forsyth {
428*74a4d8c2SCharles.Forsyth 	uchar mac[6];
429*74a4d8c2SCharles.Forsyth 	char buf[64];
430*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
431*74a4d8c2SCharles.Forsyth 	int version;
432*74a4d8c2SCharles.Forsyth 
433*74a4d8c2SCharles.Forsyth 	version = multicastea(mac, a);
434*74a4d8c2SCharles.Forsyth 	sprint(buf, "addmulti %E", mac);
435*74a4d8c2SCharles.Forsyth 	switch(version){
436*74a4d8c2SCharles.Forsyth 	case V4:
437*74a4d8c2SCharles.Forsyth 		devtab[er->cchan4->type]->write(er->cchan4, buf, strlen(buf), 0);
438*74a4d8c2SCharles.Forsyth 		break;
439*74a4d8c2SCharles.Forsyth 	case V6:
440*74a4d8c2SCharles.Forsyth 		devtab[er->cchan6->type]->write(er->cchan6, buf, strlen(buf), 0);
441*74a4d8c2SCharles.Forsyth 		break;
442*74a4d8c2SCharles.Forsyth 	default:
443*74a4d8c2SCharles.Forsyth 		panic("etheraddmulti: version %d", version);
444*74a4d8c2SCharles.Forsyth 	}
445*74a4d8c2SCharles.Forsyth }
446*74a4d8c2SCharles.Forsyth 
447*74a4d8c2SCharles.Forsyth static void
etherremmulti(Ipifc * ifc,uchar * a,uchar *)448*74a4d8c2SCharles.Forsyth etherremmulti(Ipifc *ifc, uchar *a, uchar *)
449*74a4d8c2SCharles.Forsyth {
450*74a4d8c2SCharles.Forsyth 	uchar mac[6];
451*74a4d8c2SCharles.Forsyth 	char buf[64];
452*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
453*74a4d8c2SCharles.Forsyth 	int version;
454*74a4d8c2SCharles.Forsyth 
455*74a4d8c2SCharles.Forsyth 	version = multicastea(mac, a);
456*74a4d8c2SCharles.Forsyth 	sprint(buf, "remmulti %E", mac);
457*74a4d8c2SCharles.Forsyth 	switch(version){
458*74a4d8c2SCharles.Forsyth 	case V4:
459*74a4d8c2SCharles.Forsyth 		devtab[er->cchan4->type]->write(er->cchan4, buf, strlen(buf), 0);
460*74a4d8c2SCharles.Forsyth 		break;
461*74a4d8c2SCharles.Forsyth 	case V6:
462*74a4d8c2SCharles.Forsyth 		devtab[er->cchan6->type]->write(er->cchan6, buf, strlen(buf), 0);
463*74a4d8c2SCharles.Forsyth 		break;
464*74a4d8c2SCharles.Forsyth 	default:
465*74a4d8c2SCharles.Forsyth 		panic("etherremmulti: version %d", version);
466*74a4d8c2SCharles.Forsyth 	}
467*74a4d8c2SCharles.Forsyth }
468*74a4d8c2SCharles.Forsyth 
469*74a4d8c2SCharles.Forsyth /*
470*74a4d8c2SCharles.Forsyth  *  send an ethernet arp
471*74a4d8c2SCharles.Forsyth  *  (only v4, v6 uses the neighbor discovery, rfc1970)
472*74a4d8c2SCharles.Forsyth  */
473*74a4d8c2SCharles.Forsyth static void
sendarp(Ipifc * ifc,Arpent * a)474*74a4d8c2SCharles.Forsyth sendarp(Ipifc *ifc, Arpent *a)
475*74a4d8c2SCharles.Forsyth {
476*74a4d8c2SCharles.Forsyth 	int n;
477*74a4d8c2SCharles.Forsyth 	Block *bp;
478*74a4d8c2SCharles.Forsyth 	Etherarp *e;
479*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
480*74a4d8c2SCharles.Forsyth 
481*74a4d8c2SCharles.Forsyth 	/* don't do anything if it's been less than a second since the last */
482*74a4d8c2SCharles.Forsyth 	if(NOW - a->ctime < 1000){
483*74a4d8c2SCharles.Forsyth 		arprelease(er->f->arp, a);
484*74a4d8c2SCharles.Forsyth 		return;
485*74a4d8c2SCharles.Forsyth 	}
486*74a4d8c2SCharles.Forsyth 
487*74a4d8c2SCharles.Forsyth 	/* remove all but the last message */
488*74a4d8c2SCharles.Forsyth 	while((bp = a->hold) != nil){
489*74a4d8c2SCharles.Forsyth 		if(bp == a->last)
490*74a4d8c2SCharles.Forsyth 			break;
491*74a4d8c2SCharles.Forsyth 		a->hold = bp->list;
492*74a4d8c2SCharles.Forsyth 		freeblist(bp);
493*74a4d8c2SCharles.Forsyth 	}
494*74a4d8c2SCharles.Forsyth 
495*74a4d8c2SCharles.Forsyth 	/* try to keep it around for a second more */
496*74a4d8c2SCharles.Forsyth 	a->ctime = NOW;
497*74a4d8c2SCharles.Forsyth 	arprelease(er->f->arp, a);
498*74a4d8c2SCharles.Forsyth 
499*74a4d8c2SCharles.Forsyth 	n = sizeof(Etherarp);
500*74a4d8c2SCharles.Forsyth 	if(n < a->type->mintu)
501*74a4d8c2SCharles.Forsyth 		n = a->type->mintu;
502*74a4d8c2SCharles.Forsyth 	bp = allocb(n);
503*74a4d8c2SCharles.Forsyth 	memset(bp->rp, 0, n);
504*74a4d8c2SCharles.Forsyth 	e = (Etherarp*)bp->rp;
505*74a4d8c2SCharles.Forsyth 	memmove(e->tpa, a->ip+IPv4off, sizeof(e->tpa));
506*74a4d8c2SCharles.Forsyth 	ipv4local(ifc, e->spa);
507*74a4d8c2SCharles.Forsyth 	memmove(e->sha, ifc->mac, sizeof(e->sha));
508*74a4d8c2SCharles.Forsyth 	memset(e->d, 0xff, sizeof(e->d));		/* ethernet broadcast */
509*74a4d8c2SCharles.Forsyth 	memmove(e->s, ifc->mac, sizeof(e->s));
510*74a4d8c2SCharles.Forsyth 
511*74a4d8c2SCharles.Forsyth 	hnputs(e->type, ETARP);
512*74a4d8c2SCharles.Forsyth 	hnputs(e->hrd, 1);
513*74a4d8c2SCharles.Forsyth 	hnputs(e->pro, ETIP4);
514*74a4d8c2SCharles.Forsyth 	e->hln = sizeof(e->sha);
515*74a4d8c2SCharles.Forsyth 	e->pln = sizeof(e->spa);
516*74a4d8c2SCharles.Forsyth 	hnputs(e->op, ARPREQUEST);
517*74a4d8c2SCharles.Forsyth 	bp->wp += n;
518*74a4d8c2SCharles.Forsyth 
519*74a4d8c2SCharles.Forsyth 	n = devtab[er->achan->type]->bwrite(er->achan, bp, 0);
520*74a4d8c2SCharles.Forsyth 	if(n < 0)
521*74a4d8c2SCharles.Forsyth 		print("arp: send: %r\n");
522*74a4d8c2SCharles.Forsyth }
523*74a4d8c2SCharles.Forsyth 
524*74a4d8c2SCharles.Forsyth static void
resolveaddr6(Ipifc * ifc,Arpent * a)525*74a4d8c2SCharles.Forsyth resolveaddr6(Ipifc *ifc, Arpent *a)
526*74a4d8c2SCharles.Forsyth {
527*74a4d8c2SCharles.Forsyth 	int sflag;
528*74a4d8c2SCharles.Forsyth 	Block *bp;
529*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
530*74a4d8c2SCharles.Forsyth 	uchar ipsrc[IPaddrlen];
531*74a4d8c2SCharles.Forsyth 
532*74a4d8c2SCharles.Forsyth 	/* don't do anything if it's been less than a second since the last */
533*74a4d8c2SCharles.Forsyth 	if(NOW - a->ctime < ReTransTimer){
534*74a4d8c2SCharles.Forsyth 		arprelease(er->f->arp, a);
535*74a4d8c2SCharles.Forsyth 		return;
536*74a4d8c2SCharles.Forsyth 	}
537*74a4d8c2SCharles.Forsyth 
538*74a4d8c2SCharles.Forsyth 	/* remove all but the last message */
539*74a4d8c2SCharles.Forsyth 	while((bp = a->hold) != nil){
540*74a4d8c2SCharles.Forsyth 		if(bp == a->last)
541*74a4d8c2SCharles.Forsyth 			break;
542*74a4d8c2SCharles.Forsyth 		a->hold = bp->list;
543*74a4d8c2SCharles.Forsyth 		freeblist(bp);
544*74a4d8c2SCharles.Forsyth 	}
545*74a4d8c2SCharles.Forsyth 
546*74a4d8c2SCharles.Forsyth 	/* try to keep it around for a second more */
547*74a4d8c2SCharles.Forsyth 	a->ctime = NOW;
548*74a4d8c2SCharles.Forsyth 	a->rtime = NOW + ReTransTimer;
549*74a4d8c2SCharles.Forsyth 	if(a->rxtsrem <= 0) {
550*74a4d8c2SCharles.Forsyth 		arprelease(er->f->arp, a);
551*74a4d8c2SCharles.Forsyth 		return;
552*74a4d8c2SCharles.Forsyth 	}
553*74a4d8c2SCharles.Forsyth 
554*74a4d8c2SCharles.Forsyth 	a->rxtsrem--;
555*74a4d8c2SCharles.Forsyth 	arprelease(er->f->arp, a);
556*74a4d8c2SCharles.Forsyth 
557*74a4d8c2SCharles.Forsyth 	if(sflag = ipv6anylocal(ifc, ipsrc))
558*74a4d8c2SCharles.Forsyth 		icmpns(er->f, ipsrc, sflag, a->ip, TARG_MULTI, ifc->mac);
559*74a4d8c2SCharles.Forsyth }
560*74a4d8c2SCharles.Forsyth 
561*74a4d8c2SCharles.Forsyth /*
562*74a4d8c2SCharles.Forsyth  *  send a gratuitous arp to refresh arp caches
563*74a4d8c2SCharles.Forsyth  */
564*74a4d8c2SCharles.Forsyth static void
sendgarp(Ipifc * ifc,uchar * ip)565*74a4d8c2SCharles.Forsyth sendgarp(Ipifc *ifc, uchar *ip)
566*74a4d8c2SCharles.Forsyth {
567*74a4d8c2SCharles.Forsyth 	int n;
568*74a4d8c2SCharles.Forsyth 	Block *bp;
569*74a4d8c2SCharles.Forsyth 	Etherarp *e;
570*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
571*74a4d8c2SCharles.Forsyth 
572*74a4d8c2SCharles.Forsyth 	/* don't arp for our initial non address */
573*74a4d8c2SCharles.Forsyth 	if(ipcmp(ip, IPnoaddr) == 0)
574*74a4d8c2SCharles.Forsyth 		return;
575*74a4d8c2SCharles.Forsyth 
576*74a4d8c2SCharles.Forsyth 	n = sizeof(Etherarp);
577*74a4d8c2SCharles.Forsyth 	if(n < ifc->m->mintu)
578*74a4d8c2SCharles.Forsyth 		n = ifc->m->mintu;
579*74a4d8c2SCharles.Forsyth 	bp = allocb(n);
580*74a4d8c2SCharles.Forsyth 	memset(bp->rp, 0, n);
581*74a4d8c2SCharles.Forsyth 	e = (Etherarp*)bp->rp;
582*74a4d8c2SCharles.Forsyth 	memmove(e->tpa, ip+IPv4off, sizeof(e->tpa));
583*74a4d8c2SCharles.Forsyth 	memmove(e->spa, ip+IPv4off, sizeof(e->spa));
584*74a4d8c2SCharles.Forsyth 	memmove(e->sha, ifc->mac, sizeof(e->sha));
585*74a4d8c2SCharles.Forsyth 	memset(e->d, 0xff, sizeof(e->d));		/* ethernet broadcast */
586*74a4d8c2SCharles.Forsyth 	memmove(e->s, ifc->mac, sizeof(e->s));
587*74a4d8c2SCharles.Forsyth 
588*74a4d8c2SCharles.Forsyth 	hnputs(e->type, ETARP);
589*74a4d8c2SCharles.Forsyth 	hnputs(e->hrd, 1);
590*74a4d8c2SCharles.Forsyth 	hnputs(e->pro, ETIP4);
591*74a4d8c2SCharles.Forsyth 	e->hln = sizeof(e->sha);
592*74a4d8c2SCharles.Forsyth 	e->pln = sizeof(e->spa);
593*74a4d8c2SCharles.Forsyth 	hnputs(e->op, ARPREQUEST);
594*74a4d8c2SCharles.Forsyth 	bp->wp += n;
595*74a4d8c2SCharles.Forsyth 
596*74a4d8c2SCharles.Forsyth 	n = devtab[er->achan->type]->bwrite(er->achan, bp, 0);
597*74a4d8c2SCharles.Forsyth 	if(n < 0)
598*74a4d8c2SCharles.Forsyth 		print("garp: send: %r\n");
599*74a4d8c2SCharles.Forsyth }
600*74a4d8c2SCharles.Forsyth 
601*74a4d8c2SCharles.Forsyth static void
recvarp(Ipifc * ifc)602*74a4d8c2SCharles.Forsyth recvarp(Ipifc *ifc)
603*74a4d8c2SCharles.Forsyth {
604*74a4d8c2SCharles.Forsyth 	int n;
605*74a4d8c2SCharles.Forsyth 	Block *ebp, *rbp;
606*74a4d8c2SCharles.Forsyth 	Etherarp *e, *r;
607*74a4d8c2SCharles.Forsyth 	uchar ip[IPaddrlen];
608*74a4d8c2SCharles.Forsyth 	static uchar eprinted[4];
609*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
610*74a4d8c2SCharles.Forsyth 
611*74a4d8c2SCharles.Forsyth 	ebp = devtab[er->achan->type]->bread(er->achan, ifc->maxtu, 0);
612*74a4d8c2SCharles.Forsyth 	if(ebp == nil) {
613*74a4d8c2SCharles.Forsyth 		print("arp: rcv: %r\n");
614*74a4d8c2SCharles.Forsyth 		return;
615*74a4d8c2SCharles.Forsyth 	}
616*74a4d8c2SCharles.Forsyth 
617*74a4d8c2SCharles.Forsyth 	e = (Etherarp*)ebp->rp;
618*74a4d8c2SCharles.Forsyth 	switch(nhgets(e->op)) {
619*74a4d8c2SCharles.Forsyth 	default:
620*74a4d8c2SCharles.Forsyth 		break;
621*74a4d8c2SCharles.Forsyth 
622*74a4d8c2SCharles.Forsyth 	case ARPREPLY:
623*74a4d8c2SCharles.Forsyth 		/* check for machine using my ip address */
624*74a4d8c2SCharles.Forsyth 		v4tov6(ip, e->spa);
625*74a4d8c2SCharles.Forsyth 		if(iplocalonifc(ifc, ip) || ipproxyifc(er->f, ifc, ip)){
626*74a4d8c2SCharles.Forsyth 			if(memcmp(e->sha, ifc->mac, sizeof(e->sha)) != 0){
627*74a4d8c2SCharles.Forsyth 				print("arprep: 0x%E/0x%E also has ip addr %V\n",
628*74a4d8c2SCharles.Forsyth 					e->s, e->sha, e->spa);
629*74a4d8c2SCharles.Forsyth 				break;
630*74a4d8c2SCharles.Forsyth 			}
631*74a4d8c2SCharles.Forsyth 		}
632*74a4d8c2SCharles.Forsyth 
633*74a4d8c2SCharles.Forsyth 		/* make sure we're not entering broadcast addresses */
634*74a4d8c2SCharles.Forsyth 		if(ipcmp(ip, ipbroadcast) == 0 ||
635*74a4d8c2SCharles.Forsyth 			!memcmp(e->sha, etherbroadcast, sizeof(e->sha))){
636*74a4d8c2SCharles.Forsyth 			print("arprep: 0x%E/0x%E cannot register broadcast address %I\n",
637*74a4d8c2SCharles.Forsyth 				e->s, e->sha, e->spa);
638*74a4d8c2SCharles.Forsyth 			break;
639*74a4d8c2SCharles.Forsyth 		}
640*74a4d8c2SCharles.Forsyth 
641*74a4d8c2SCharles.Forsyth 		arpenter(er->f, V4, e->spa, e->sha, sizeof(e->sha), 0);
642*74a4d8c2SCharles.Forsyth 		break;
643*74a4d8c2SCharles.Forsyth 
644*74a4d8c2SCharles.Forsyth 	case ARPREQUEST:
645*74a4d8c2SCharles.Forsyth 		/* don't answer arps till we know who we are */
646*74a4d8c2SCharles.Forsyth 		if(ifc->lifc == 0)
647*74a4d8c2SCharles.Forsyth 			break;
648*74a4d8c2SCharles.Forsyth 
649*74a4d8c2SCharles.Forsyth 		/* check for machine using my ip or ether address */
650*74a4d8c2SCharles.Forsyth 		v4tov6(ip, e->spa);
651*74a4d8c2SCharles.Forsyth 		if(iplocalonifc(ifc, ip) || ipproxyifc(er->f, ifc, ip)){
652*74a4d8c2SCharles.Forsyth 			if(memcmp(e->sha, ifc->mac, sizeof(e->sha)) != 0){
653*74a4d8c2SCharles.Forsyth 				if (memcmp(eprinted, e->spa, sizeof(e->spa))){
654*74a4d8c2SCharles.Forsyth 					/* print only once */
655*74a4d8c2SCharles.Forsyth 					print("arpreq: 0x%E also has ip addr %V\n", e->sha, e->spa);
656*74a4d8c2SCharles.Forsyth 					memmove(eprinted, e->spa, sizeof(e->spa));
657*74a4d8c2SCharles.Forsyth 				}
658*74a4d8c2SCharles.Forsyth 			}
659*74a4d8c2SCharles.Forsyth 		} else {
660*74a4d8c2SCharles.Forsyth 			if(memcmp(e->sha, ifc->mac, sizeof(e->sha)) == 0){
661*74a4d8c2SCharles.Forsyth 				print("arpreq: %V also has ether addr %E\n", e->spa, e->sha);
662*74a4d8c2SCharles.Forsyth 				break;
663*74a4d8c2SCharles.Forsyth 			}
664*74a4d8c2SCharles.Forsyth 		}
665*74a4d8c2SCharles.Forsyth 
666*74a4d8c2SCharles.Forsyth 		/* refresh what we know about sender */
667*74a4d8c2SCharles.Forsyth 		arpenter(er->f, V4, e->spa, e->sha, sizeof(e->sha), 1);
668*74a4d8c2SCharles.Forsyth 
669*74a4d8c2SCharles.Forsyth 		/* answer only requests for our address or systems we're proxying for */
670*74a4d8c2SCharles.Forsyth 		v4tov6(ip, e->tpa);
671*74a4d8c2SCharles.Forsyth 		if(!iplocalonifc(ifc, ip))
672*74a4d8c2SCharles.Forsyth 		if(!ipproxyifc(er->f, ifc, ip))
673*74a4d8c2SCharles.Forsyth 			break;
674*74a4d8c2SCharles.Forsyth 
675*74a4d8c2SCharles.Forsyth 		n = sizeof(Etherarp);
676*74a4d8c2SCharles.Forsyth 		if(n < ifc->mintu)
677*74a4d8c2SCharles.Forsyth 			n = ifc->mintu;
678*74a4d8c2SCharles.Forsyth 		rbp = allocb(n);
679*74a4d8c2SCharles.Forsyth 		r = (Etherarp*)rbp->rp;
680*74a4d8c2SCharles.Forsyth 		memset(r, 0, sizeof(Etherarp));
681*74a4d8c2SCharles.Forsyth 		hnputs(r->type, ETARP);
682*74a4d8c2SCharles.Forsyth 		hnputs(r->hrd, 1);
683*74a4d8c2SCharles.Forsyth 		hnputs(r->pro, ETIP4);
684*74a4d8c2SCharles.Forsyth 		r->hln = sizeof(r->sha);
685*74a4d8c2SCharles.Forsyth 		r->pln = sizeof(r->spa);
686*74a4d8c2SCharles.Forsyth 		hnputs(r->op, ARPREPLY);
687*74a4d8c2SCharles.Forsyth 		memmove(r->tha, e->sha, sizeof(r->tha));
688*74a4d8c2SCharles.Forsyth 		memmove(r->tpa, e->spa, sizeof(r->tpa));
689*74a4d8c2SCharles.Forsyth 		memmove(r->sha, ifc->mac, sizeof(r->sha));
690*74a4d8c2SCharles.Forsyth 		memmove(r->spa, e->tpa, sizeof(r->spa));
691*74a4d8c2SCharles.Forsyth 		memmove(r->d, e->sha, sizeof(r->d));
692*74a4d8c2SCharles.Forsyth 		memmove(r->s, ifc->mac, sizeof(r->s));
693*74a4d8c2SCharles.Forsyth 		rbp->wp += n;
694*74a4d8c2SCharles.Forsyth 
695*74a4d8c2SCharles.Forsyth 		n = devtab[er->achan->type]->bwrite(er->achan, rbp, 0);
696*74a4d8c2SCharles.Forsyth 		if(n < 0)
697*74a4d8c2SCharles.Forsyth 			print("arp: write: %r\n");
698*74a4d8c2SCharles.Forsyth 	}
699*74a4d8c2SCharles.Forsyth 	freeb(ebp);
700*74a4d8c2SCharles.Forsyth }
701*74a4d8c2SCharles.Forsyth 
702*74a4d8c2SCharles.Forsyth static void
recvarpproc(void * v)703*74a4d8c2SCharles.Forsyth recvarpproc(void *v)
704*74a4d8c2SCharles.Forsyth {
705*74a4d8c2SCharles.Forsyth 	Ipifc *ifc = v;
706*74a4d8c2SCharles.Forsyth 	Etherrock *er = ifc->arg;
707*74a4d8c2SCharles.Forsyth 
708*74a4d8c2SCharles.Forsyth 	er->arpp = up;
709*74a4d8c2SCharles.Forsyth 	if(waserror()){
710*74a4d8c2SCharles.Forsyth 		er->arpp = 0;
711*74a4d8c2SCharles.Forsyth 		pexit("hangup", 1);
712*74a4d8c2SCharles.Forsyth 	}
713*74a4d8c2SCharles.Forsyth 	for(;;)
714*74a4d8c2SCharles.Forsyth 		recvarp(ifc);
715*74a4d8c2SCharles.Forsyth }
716*74a4d8c2SCharles.Forsyth 
717*74a4d8c2SCharles.Forsyth static int
multicastea(uchar * ea,uchar * ip)718*74a4d8c2SCharles.Forsyth multicastea(uchar *ea, uchar *ip)
719*74a4d8c2SCharles.Forsyth {
720*74a4d8c2SCharles.Forsyth 	int x;
721*74a4d8c2SCharles.Forsyth 
722*74a4d8c2SCharles.Forsyth 	switch(x = ipismulticast(ip)){
723*74a4d8c2SCharles.Forsyth 	case V4:
724*74a4d8c2SCharles.Forsyth 		ea[0] = 0x01;
725*74a4d8c2SCharles.Forsyth 		ea[1] = 0x00;
726*74a4d8c2SCharles.Forsyth 		ea[2] = 0x5e;
727*74a4d8c2SCharles.Forsyth 		ea[3] = ip[13] & 0x7f;
728*74a4d8c2SCharles.Forsyth 		ea[4] = ip[14];
729*74a4d8c2SCharles.Forsyth 		ea[5] = ip[15];
730*74a4d8c2SCharles.Forsyth 		break;
731*74a4d8c2SCharles.Forsyth  	case V6:
732*74a4d8c2SCharles.Forsyth  		ea[0] = 0x33;
733*74a4d8c2SCharles.Forsyth  		ea[1] = 0x33;
734*74a4d8c2SCharles.Forsyth  		ea[2] = ip[12];
735*74a4d8c2SCharles.Forsyth 		ea[3] = ip[13];
736*74a4d8c2SCharles.Forsyth  		ea[4] = ip[14];
737*74a4d8c2SCharles.Forsyth  		ea[5] = ip[15];
738*74a4d8c2SCharles.Forsyth  		break;
739*74a4d8c2SCharles.Forsyth 	}
740*74a4d8c2SCharles.Forsyth 	return x;
741*74a4d8c2SCharles.Forsyth }
742*74a4d8c2SCharles.Forsyth 
743*74a4d8c2SCharles.Forsyth /*
744*74a4d8c2SCharles.Forsyth  *  fill in an arp entry for broadcast or multicast
745*74a4d8c2SCharles.Forsyth  *  addresses.  Return the first queued packet for the
746*74a4d8c2SCharles.Forsyth  *  IP address.
747*74a4d8c2SCharles.Forsyth  */
748*74a4d8c2SCharles.Forsyth static Block*
multicastarp(Fs * f,Arpent * a,Medium * medium,uchar * mac)749*74a4d8c2SCharles.Forsyth multicastarp(Fs *f, Arpent *a, Medium *medium, uchar *mac)
750*74a4d8c2SCharles.Forsyth {
751*74a4d8c2SCharles.Forsyth 	/* is it broadcast? */
752*74a4d8c2SCharles.Forsyth 	switch(ipforme(f, a->ip)){
753*74a4d8c2SCharles.Forsyth 	case Runi:
754*74a4d8c2SCharles.Forsyth 		return nil;
755*74a4d8c2SCharles.Forsyth 	case Rbcast:
756*74a4d8c2SCharles.Forsyth 		memset(mac, 0xff, 6);
757*74a4d8c2SCharles.Forsyth 		return arpresolve(f->arp, a, medium, mac);
758*74a4d8c2SCharles.Forsyth 	default:
759*74a4d8c2SCharles.Forsyth 		break;
760*74a4d8c2SCharles.Forsyth 	}
761*74a4d8c2SCharles.Forsyth 
762*74a4d8c2SCharles.Forsyth 	/* if multicast, fill in mac */
763*74a4d8c2SCharles.Forsyth 	switch(multicastea(mac, a->ip)){
764*74a4d8c2SCharles.Forsyth 	case V4:
765*74a4d8c2SCharles.Forsyth 	case V6:
766*74a4d8c2SCharles.Forsyth 		return arpresolve(f->arp, a, medium, mac);
767*74a4d8c2SCharles.Forsyth 	}
768*74a4d8c2SCharles.Forsyth 
769*74a4d8c2SCharles.Forsyth 	/* let arp take care of it */
770*74a4d8c2SCharles.Forsyth 	return nil;
771*74a4d8c2SCharles.Forsyth }
772*74a4d8c2SCharles.Forsyth 
773*74a4d8c2SCharles.Forsyth void
ethermediumlink(void)774*74a4d8c2SCharles.Forsyth ethermediumlink(void)
775*74a4d8c2SCharles.Forsyth {
776*74a4d8c2SCharles.Forsyth 	addipmedium(&ethermedium);
777*74a4d8c2SCharles.Forsyth 	addipmedium(&gbemedium);
778*74a4d8c2SCharles.Forsyth }
779*74a4d8c2SCharles.Forsyth 
780*74a4d8c2SCharles.Forsyth 
781*74a4d8c2SCharles.Forsyth static void
etherpref2addr(uchar * pref,uchar * ea)782*74a4d8c2SCharles.Forsyth etherpref2addr(uchar *pref, uchar *ea)
783*74a4d8c2SCharles.Forsyth {
784*74a4d8c2SCharles.Forsyth 	pref[8]  = ea[0] | 0x2;
785*74a4d8c2SCharles.Forsyth 	pref[9]  = ea[1];
786*74a4d8c2SCharles.Forsyth 	pref[10] = ea[2];
787*74a4d8c2SCharles.Forsyth 	pref[11] = 0xFF;
788*74a4d8c2SCharles.Forsyth 	pref[12] = 0xFE;
789*74a4d8c2SCharles.Forsyth 	pref[13] = ea[3];
790*74a4d8c2SCharles.Forsyth 	pref[14] = ea[4];
791*74a4d8c2SCharles.Forsyth 	pref[15] = ea[5];
792*74a4d8c2SCharles.Forsyth }
793