1*17056Sbloom /*	raw_pup.c	6.2	84/08/29	*/
25671Ssam 
3*17056Sbloom #include "param.h"
4*17056Sbloom #include "mbuf.h"
5*17056Sbloom #include "socket.h"
6*17056Sbloom #include "protosw.h"
7*17056Sbloom #include "socketvar.h"
8*17056Sbloom #include "errno.h"
910898Ssam 
1010898Ssam #include "../net/if.h"
1113456Ssam #include "../net/route.h"
1213456Ssam #include "../net/raw_cb.h"
1310898Ssam 
148397Swnj #include "../netpup/pup.h"
155671Ssam 
165671Ssam /*
175671Ssam  * Raw PUP protocol interface.
185671Ssam  */
195671Ssam 
2013456Ssam struct	sockaddr_pup pupsrc = { AF_PUP };
2113456Ssam struct	sockaddr_pup pupdst = { AF_PUP };
2213456Ssam struct	sockproto pupproto = { PF_PUP };
235671Ssam /*
2413456Ssam  * Raw PUP input.
2513456Ssam  */
2613456Ssam rpup_input(m)
2713456Ssam 	struct mbuf *m;
2813456Ssam {
2913456Ssam 	register struct pup_header *pup = mtod(m, struct pup_header *);
3013456Ssam 
3113456Ssam 	pupproto.sp_protocol = pup->pup_type;
3213456Ssam 	bcopy((caddr_t)&pup->pup_dnet, (caddr_t)&pupdst.spup_net,
3313456Ssam 	    sizeof (struct pupport));
3413456Ssam 	bcopy((caddr_t)&pup->pup_snet, (caddr_t)&pupsrc.spup_net,
3513456Ssam 	    sizeof (struct pupport));
3613456Ssam 	raw_input(m, &pupproto, (struct sockaddr *)&pupsrc,
3713456Ssam 	  (struct sockaddr *)&pupdst);
3813456Ssam }
3913456Ssam 
4013456Ssam /*
415671Ssam  * Encapsulate packet in PUP header which is supplied by the
425671Ssam  * user.  This is done to allow user to specify PUP identifier.
435671Ssam  */
rpup_output(m,so)446045Swnj rpup_output(m, so)
456045Swnj 	register struct mbuf *m;
465671Ssam 	struct socket *so;
475671Ssam {
485671Ssam 	register struct rawcb *rp = sotorawcb(so);
495671Ssam 	register struct pup_header *pup;
506509Ssam 	int len, error = 0;
5112826Ssam 	register struct mbuf *n, *last;
526509Ssam 	struct sockaddr_pup *dst;
536027Ssam 	struct ifnet *ifp;
5413456Ssam 	u_short *pc;
555671Ssam 
565671Ssam 	/*
575671Ssam 	 * Verify user has supplied necessary space
585671Ssam 	 * for the header and check parameters in it.
595671Ssam 	 */
605671Ssam 	if ((m->m_off > MMAXOFF || m->m_len < sizeof(struct pup_header)) &&
616509Ssam 	    (m = m_pullup(m, sizeof(struct pup_header))) == 0) {
6212826Ssam 		error = EINVAL;
635671Ssam 		goto bad;
646509Ssam 	}
655671Ssam 	pup = mtod(m, struct pup_header *);
6612826Ssam 	if (pup->pup_type == 0 || (pup->pup_tcontrol &~ PUP_TRACE)) {
6712826Ssam 		error = EINVAL;
685671Ssam 		goto bad;
696509Ssam 	}
7012826Ssam 	for (len = 0, n = last = m; n; last = n, n = n->m_next)
7112826Ssam 		len += n->m_len;
7212826Ssam 	/* assume user leaves space for checksum */
7312826Ssam 	if ((len & 1) || len < MINPUPSIZ || len > MAXPUPSIZ) {
7412826Ssam 		error = EMSGSIZE;
755671Ssam 		goto bad;
766509Ssam 	}
7713096Ssam 	pup->pup_length = htons((u_short)len);
786509Ssam 	dst = (struct sockaddr_pup *)&rp->rcb_faddr;
7913456Ssam 	bcopy((caddr_t)&dst->spup_net, (caddr_t)&pup->pup_dnet,
8012826Ssam 	    sizeof (struct pupport));
8113456Ssam 	if (rp->rcb_route.ro_rt == 0)
8213456Ssam 		ifp = if_ifonnetof(dst->spup_net);
8313456Ssam 	else {
8413456Ssam 		rp->rcb_route.ro_rt->rt_use++;
8513456Ssam 		ifp = rp->rcb_route.ro_rt->rt_ifp;
8613456Ssam 	}
8712826Ssam 	if (ifp == 0) {
8812826Ssam 		error = ENETUNREACH;
8912826Ssam 		goto bad;
9012826Ssam 	}
9112826Ssam 	if (rp->rcb_flags & RAW_LADDR) {
9212826Ssam 		register struct sockaddr_pup *src;
935671Ssam 
9412826Ssam 		src = (struct sockaddr_pup *)&rp->rcb_laddr;
9513456Ssam 		bcopy((caddr_t)&src->spup_net, (caddr_t)&pup->pup_snet,
9612826Ssam 		    sizeof (struct pupport));
9712826Ssam 	} else {
9812826Ssam 		pup->pup_snet = ifp->if_net;
9912826Ssam 		pup->pup_shost = ifp->if_host[0];
10012826Ssam 		/* socket is specified by user */
1016509Ssam 	}
10212826Ssam 	/*
10312826Ssam 	 * Fill in checksum unless user indicates none should be specified.
10412826Ssam 	 */
10513456Ssam 	pc = (u_short *)(mtod(last, caddr_t) + last->m_len - sizeof (short));
10613456Ssam 	if (*pc != PUP_NOCKSUM)
10713468Ssam 		*pc = htons((u_short)pup_cksum(m, len - sizeof (short)));
10812826Ssam 	return ((*ifp->if_output)(ifp, m, (struct sockaddr *)dst));
1095671Ssam bad:
1105671Ssam 	m_freem(m);
1116509Ssam 	return (error);
1125671Ssam }
113