123206Smckusick /*
263231Sbostic * Copyright (c) 1984, 1988, 1993
363231Sbostic * The Regents of the University of California. All rights reserved.
423206Smckusick *
544502Sbostic * %sccs.include.redist.c%
633371Ssklower *
7*68253Ssklower * @(#)ns_error.c 8.2 (Berkeley) 02/09/95
823206Smckusick */
921483Ssklower
1056534Sbostic #include <sys/param.h>
1156534Sbostic #include <sys/systm.h>
1256534Sbostic #include <sys/malloc.h>
1356534Sbostic #include <sys/mbuf.h>
1456534Sbostic #include <sys/protosw.h>
1556534Sbostic #include <sys/socket.h>
16*68253Ssklower #include <sys/socketvar.h>
1756534Sbostic #include <sys/time.h>
1856534Sbostic #include <sys/kernel.h>
1921483Ssklower
2056534Sbostic #include <net/route.h>
2121483Ssklower
2256534Sbostic #include <netns/ns.h>
2356534Sbostic #include <netns/ns_pcb.h>
2456534Sbostic #include <netns/idp.h>
2556534Sbostic #include <netns/ns_error.h>
2621483Ssklower
2724226Ssklower #ifdef lint
2824226Ssklower #define NS_ERRPRINTFS 1
2924226Ssklower #endif
3024226Ssklower
3121483Ssklower #ifdef NS_ERRPRINTFS
3221483Ssklower /*
3321483Ssklower * NS_ERR routines: error generation, receive packet processing, and
3421746Ssklower * routines to turnaround packets back to the originator.
3521483Ssklower */
3621483Ssklower int ns_errprintfs = 0;
3721483Ssklower #endif
3821483Ssklower
ns_err_x(c)3933429Smckusick ns_err_x(c)
4033411Ssklower {
4133411Ssklower register u_short *w, *lim, *base = ns_errstat.ns_es_codes;
4233411Ssklower u_short x = c;
4333411Ssklower
4433411Ssklower /*
4533411Ssklower * zero is a legit error code, handle specially
4633411Ssklower */
4733411Ssklower if (x == 0)
4833411Ssklower return (0);
4933411Ssklower lim = base + NS_ERR_MAX - 1;
5033411Ssklower for (w = base + 1; w < lim; w++) {
5133411Ssklower if (*w == 0)
5233411Ssklower *w = x;
5333411Ssklower if (*w == x)
5433411Ssklower break;
5533411Ssklower }
5633411Ssklower return (w - base);
5733411Ssklower }
5833411Ssklower
5921483Ssklower /*
6021483Ssklower * Generate an error packet of type error
6121483Ssklower * in response to bad packet.
6221483Ssklower */
6321483Ssklower
6421483Ssklower ns_error(om, type, param)
6521483Ssklower struct mbuf *om;
6621483Ssklower int type;
6721483Ssklower {
6821483Ssklower register struct ns_epidp *ep;
6921483Ssklower struct mbuf *m;
7021483Ssklower struct idp *nip;
7121483Ssklower register struct idp *oip = mtod(om, struct idp *);
7221483Ssklower extern int idpcksum;
7321483Ssklower
7421746Ssklower /*
7521746Ssklower * If this packet was sent to the echo port,
7621746Ssklower * and nobody was there, just echo it.
7721746Ssklower * (Yes, this is a wart!)
7821746Ssklower */
7937473Ssklower if (type == NS_ERR_NOSOCK &&
8037473Ssklower oip->idp_dna.x_port == htons(2) &&
8137473Ssklower (type = ns_echo(om))==0)
8221746Ssklower return;
8321746Ssklower
8421483Ssklower #ifdef NS_ERRPRINTFS
8521483Ssklower if (ns_errprintfs)
8621483Ssklower printf("ns_err_error(%x, %d, %d)\n", oip, type, param);
8721483Ssklower #endif
8823980Ssklower /*
8923980Ssklower * Don't Generate error packets in response to multicasts.
9023980Ssklower */
9123980Ssklower if (oip->idp_dna.x_host.c_host[0] & 1)
9237473Ssklower goto freeit;
9323980Ssklower
9421483Ssklower ns_errstat.ns_es_error++;
9521483Ssklower /*
9621483Ssklower * Make sure that the old IDP packet had 30 bytes of data to return;
9721483Ssklower * if not, don't bother. Also don't EVER error if the old
9821483Ssklower * packet protocol was NS_ERR.
9921483Ssklower */
10021483Ssklower if (oip->idp_len < sizeof(struct idp)) {
10121483Ssklower ns_errstat.ns_es_oldshort++;
10237473Ssklower goto freeit;
10321483Ssklower }
10421483Ssklower if (oip->idp_pt == NSPROTO_ERROR) {
10521483Ssklower ns_errstat.ns_es_oldns_err++;
10637473Ssklower goto freeit;
10721483Ssklower }
10821483Ssklower
10921483Ssklower /*
11021483Ssklower * First, formulate ns_err message
11121483Ssklower */
11237473Ssklower m = m_gethdr(M_DONTWAIT, MT_HEADER);
11321483Ssklower if (m == NULL)
11437473Ssklower goto freeit;
11521483Ssklower m->m_len = sizeof(*ep);
11637473Ssklower MH_ALIGN(m, m->m_len);
11721483Ssklower ep = mtod(m, struct ns_epidp *);
11821483Ssklower if ((u_int)type > NS_ERR_TOO_BIG)
11921483Ssklower panic("ns_err_error");
12021483Ssklower ns_errstat.ns_es_outhist[ns_err_x(type)]++;
12124226Ssklower ep->ns_ep_errp.ns_err_num = htons((u_short)type);
12224226Ssklower ep->ns_ep_errp.ns_err_param = htons((u_short)param);
12321483Ssklower bcopy((caddr_t)oip, (caddr_t)&ep->ns_ep_errp.ns_err_idp, 42);
12421483Ssklower nip = &ep->ns_ep_idp;
12521483Ssklower nip->idp_len = sizeof(*ep);
12621483Ssklower nip->idp_len = htons((u_short)nip->idp_len);
12721483Ssklower nip->idp_pt = NSPROTO_ERROR;
12821483Ssklower nip->idp_tc = 0;
12921483Ssklower nip->idp_dna = oip->idp_sna;
13021483Ssklower nip->idp_sna = oip->idp_dna;
13121483Ssklower if (idpcksum) {
13221483Ssklower nip->idp_sum = 0;
13337473Ssklower nip->idp_sum = ns_cksum(m, sizeof(*ep));
13421483Ssklower } else
13521483Ssklower nip->idp_sum = 0xffff;
13637473Ssklower (void) ns_output(m, (struct route *)0, 0);
13721483Ssklower
13837473Ssklower freeit:
13937473Ssklower m_freem(om);
14021483Ssklower }
14121483Ssklower
ns_printhost(p)14221483Ssklower ns_printhost(p)
14321483Ssklower register struct ns_addr *p;
14421483Ssklower {
14521483Ssklower
14621483Ssklower printf("<net:%x%x,host:%x%x%x,port:%x>",
14721483Ssklower p->x_net.s_net[0],
14821483Ssklower p->x_net.s_net[1],
14921483Ssklower p->x_host.s_host[0],
15021483Ssklower p->x_host.s_host[1],
15121483Ssklower p->x_host.s_host[2],
15221483Ssklower p->x_port);
15321483Ssklower
15421483Ssklower }
15521483Ssklower
15621483Ssklower /*
15721483Ssklower * Process a received NS_ERR message.
15821483Ssklower */
15921483Ssklower ns_err_input(m)
16021483Ssklower struct mbuf *m;
16121483Ssklower {
16221483Ssklower register struct ns_errp *ep;
16321483Ssklower register struct ns_epidp *epidp = mtod(m, struct ns_epidp *);
16421483Ssklower register int i;
16521483Ssklower int type, code, param;
16621483Ssklower
16721483Ssklower /*
16821483Ssklower * Locate ns_err structure in mbuf, and check
16921483Ssklower * that not corrupted and of at least minimum length.
17021483Ssklower */
17121483Ssklower #ifdef NS_ERRPRINTFS
17221483Ssklower if (ns_errprintfs) {
17321483Ssklower printf("ns_err_input from ");
17421483Ssklower ns_printhost(&epidp->ns_ep_idp.idp_sna);
17521483Ssklower printf("len %d\n", ntohs(epidp->ns_ep_idp.idp_len));
17621483Ssklower }
17721483Ssklower #endif
17821483Ssklower i = sizeof (struct ns_epidp);
17937473Ssklower if (((m->m_flags & M_EXT) || m->m_len < i) &&
18021483Ssklower (m = m_pullup(m, i)) == 0) {
18121483Ssklower ns_errstat.ns_es_tooshort++;
18221483Ssklower return;
18321483Ssklower }
18421483Ssklower ep = &(mtod(m, struct ns_epidp *)->ns_ep_errp);
18521483Ssklower type = ntohs(ep->ns_err_num);
18621483Ssklower param = ntohs(ep->ns_err_param);
18723980Ssklower ns_errstat.ns_es_inhist[ns_err_x(type)]++;
18821483Ssklower
18921483Ssklower #ifdef NS_ERRPRINTFS
19021483Ssklower /*
19121483Ssklower * Message type specific processing.
19221483Ssklower */
19321483Ssklower if (ns_errprintfs)
19421483Ssklower printf("ns_err_input, type %d param %d\n", type, param);
19521483Ssklower #endif
19621483Ssklower if (type >= NS_ERR_TOO_BIG) {
19721483Ssklower goto badcode;
19821483Ssklower }
19921483Ssklower ns_errstat.ns_es_outhist[ns_err_x(type)]++;
20021483Ssklower switch (type) {
20121483Ssklower
20221483Ssklower case NS_ERR_UNREACH_HOST:
20321483Ssklower code = PRC_UNREACH_NET;
20421483Ssklower goto deliver;
20521483Ssklower
20621483Ssklower case NS_ERR_TOO_OLD:
20721483Ssklower code = PRC_TIMXCEED_INTRANS;
20821483Ssklower goto deliver;
20921483Ssklower
21021483Ssklower case NS_ERR_TOO_BIG:
21121483Ssklower code = PRC_MSGSIZE;
21221483Ssklower goto deliver;
21321483Ssklower
21421483Ssklower case NS_ERR_FULLUP:
21521483Ssklower code = PRC_QUENCH;
21621483Ssklower goto deliver;
21721483Ssklower
21821483Ssklower case NS_ERR_NOSOCK:
21921483Ssklower code = PRC_UNREACH_PORT;
22021483Ssklower goto deliver;
22121483Ssklower
22221483Ssklower case NS_ERR_UNSPEC_T:
22321483Ssklower case NS_ERR_BADSUM_T:
22421483Ssklower case NS_ERR_BADSUM:
22521483Ssklower case NS_ERR_UNSPEC:
22621483Ssklower code = PRC_PARAMPROB;
22721483Ssklower goto deliver;
22821483Ssklower
22921483Ssklower deliver:
23021483Ssklower /*
23121483Ssklower * Problem with datagram; advise higher level routines.
23221483Ssklower */
23321483Ssklower #ifdef NS_ERRPRINTFS
23421483Ssklower if (ns_errprintfs)
23521483Ssklower printf("deliver to protocol %d\n",
23621483Ssklower ep->ns_err_idp.idp_pt);
23721483Ssklower #endif
23821483Ssklower switch(ep->ns_err_idp.idp_pt) {
23921483Ssklower case NSPROTO_SPP:
24021483Ssklower spp_ctlinput(code, (caddr_t)ep);
24121483Ssklower break;
24221483Ssklower
24321483Ssklower default:
24421483Ssklower idp_ctlinput(code, (caddr_t)ep);
24521483Ssklower }
24621483Ssklower
24737473Ssklower goto freeit;
24821483Ssklower
24921483Ssklower default:
25021483Ssklower badcode:
25121483Ssklower ns_errstat.ns_es_badcode++;
25237473Ssklower goto freeit;
25321483Ssklower
25421483Ssklower }
25537473Ssklower freeit:
25621483Ssklower m_freem(m);
25721483Ssklower }
25821746Ssklower
25928956Skarels #ifdef notdef
26021483Ssklower u_long
nstime()26121483Ssklower nstime()
26221483Ssklower {
26329924Skarels int s = splclock();
26421483Ssklower u_long t;
26521483Ssklower
26621483Ssklower t = (time.tv_sec % (24*60*60)) * 1000 + time.tv_usec / 1000;
26721483Ssklower splx(s);
26821483Ssklower return (htonl(t));
26921483Ssklower }
27028956Skarels #endif
27121746Ssklower
27237473Ssklower ns_echo(m)
27337473Ssklower struct mbuf *m;
27421746Ssklower {
27537473Ssklower register struct idp *idp = mtod(m, struct idp *);
27621746Ssklower register struct echo {
27721746Ssklower struct idp ec_idp;
27821746Ssklower u_short ec_op; /* Operation, 1 = request, 2 = reply */
27921746Ssklower } *ec = (struct echo *)idp;
28021746Ssklower struct ns_addr temp;
28121746Ssklower
28221746Ssklower if (idp->idp_pt!=NSPROTO_ECHO) return(NS_ERR_NOSOCK);
28321746Ssklower if (ec->ec_op!=htons(1)) return(NS_ERR_UNSPEC);
28421746Ssklower
28521746Ssklower ec->ec_op = htons(2);
28621746Ssklower
28721746Ssklower temp = idp->idp_dna;
28821746Ssklower idp->idp_dna = idp->idp_sna;
28921746Ssklower idp->idp_sna = temp;
29021746Ssklower
29121746Ssklower if (idp->idp_sum != 0xffff) {
29221746Ssklower idp->idp_sum = 0;
29328956Skarels idp->idp_sum = ns_cksum(m,
29428956Skarels (int)(((ntohs(idp->idp_len) - 1)|1)+1));
29521746Ssklower }
29626389Skarels (void) ns_output(m, (struct route *)0, NS_FORWARDING);
29721746Ssklower return(0);
29821746Ssklower }
299