1*13468Ssam /* raw_pup.c 4.20 83/06/30 */ 25671Ssam 35671Ssam #include "../h/param.h" 45671Ssam #include "../h/mbuf.h" 55671Ssam #include "../h/socket.h" 65671Ssam #include "../h/protosw.h" 75671Ssam #include "../h/socketvar.h" 810898Ssam #include "../h/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 */ 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) 107*13468Ssam *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