Plan 9 from Bell Labs’s /usr/web/sources/patch/sorry/dns-no-recurse/dnserver.c

Copyright © 2021 Plan 9 Foundation.
Distributed under the MIT License.
Download the Plan 9 distribution.


#include <u.h>
#include <libc.h>
#include <ip.h>
#include "dns.h"

static RR*	doextquery(DNSmsg*, Request*, int);
static void	hint(RR**, RR*);

extern char *logfile;

/*
 *  answer a dns request
 */
void
dnserver(DNSmsg *reqp, DNSmsg *repp, Request *req)
{
	RR *tp, *neg;
	char *cp;
	DN *nsdp, *dp;
	Area *myarea;
	char tname[32];

	dncheck(nil, 1);

	memset(repp, 0, sizeof(*repp));
	repp->id = reqp->id;
	repp->flags = Fresp | Oquery;
	if(!public)
		repp->flags |= Fcanrec;

	/* move one question from reqp to repp */
	tp = reqp->qd;
	reqp->qd = tp->next;
	tp->next = 0;
	repp->qd = tp;

	if((reqp->flags & Frecurse) != 0 && !public){
		syslog(0, logfile, "server: recursive lookup refused");
		repp->flags = Rrefused | Fresp | Oquery;
		return;
	}

	if(!rrsupported(repp->qd->type)){
		syslog(0, logfile, "server: request %s", rrname(repp->qd->type, tname, sizeof tname));
		repp->flags = Runimplimented | Fresp | Oquery;
		if(!public)
			repp->flags |= Fcanrec;

		return;
	}

	if(repp->qd->owner->class != Cin){
		syslog(0, logfile, "server: class %d", repp->qd->owner->class);
		repp->flags = Runimplimented | Fresp | Oquery;
		if(!public)
			repp->flags |= Fcanrec;
		return;
	}

	myarea = inmyarea(repp->qd->owner->name);
	if(myarea != nil && (repp->qd->type == Tixfr || repp->qd->type == Taxfr)){
		syslog(0, logfile, "server: request %s", rrname(repp->qd->type, tname, sizeof tname));
		repp->flags = Runimplimented | Fresp | Oquery;
		if(!public)
			repp->flags |= Fcanrec;
		return;
	}

	/*
	 *  get the answer if we can
	 */
	if((reqp->flags & Frecurse) && !public)
		neg = doextquery(repp, req, Recurse);
	else
		neg = doextquery(repp, req, Dontrecurse);

	/* authority is transitive */
	if(myarea != nil || (repp->an && repp->an->auth))
		repp->flags |= Fauth;

	/* pass on error codes */
	if(repp->an == 0){
		dp = dnlookup(repp->qd->owner->name, repp->qd->owner->class, 0);
		if(dp->rr == 0)
			if((reqp->flags & Frecurse) && !public)
				repp->flags |= dp->nonexistent|Fauth;
	}

	if(myarea == nil){
		/*
		 *  add name server if we know
		 */
		for(cp = repp->qd->owner->name; cp; cp = walkup(cp)){
			nsdp = dnlookup(cp, repp->qd->owner->class, 0);
			if(nsdp == 0)
				continue;
	
			repp->ns = rrlookup(nsdp, Tns, OKneg);
			if(repp->ns){
				/* don't pass on anything we know is wrong */
				if(repp->ns->negative){
					rrfreelist(repp->ns);
					repp->ns = nil;
				}
				break;
			}
	
			repp->ns = dblookup(cp, repp->qd->owner->class, Tns, 0, 0);
			if(repp->ns)
				break;
		}
	}

	/*
	 *  add ip addresses as hints
	 */
	if(repp->qd->type != Taxfr && repp->qd->type != Tixfr){
		for(tp = repp->ns; tp; tp = tp->next)
			hint(&repp->ar, tp);
		for(tp = repp->an; tp; tp = tp->next)
			hint(&repp->ar, tp);
	}

	/*
	 *  add an soa to the authority section to help client with negative caching
	 */
	if(repp->an == nil){
		if(myarea != nil){
			rrcopy(myarea->soarr, &tp);
			rrcat(&repp->ns, tp);
		} else if(neg != nil) {
			if(neg->negsoaowner != nil)
				rrcat(&repp->ns, rrlookup(neg->negsoaowner, Tsoa, NOneg));
			repp->flags |= neg->negrcode;
		}
	}

	/*
	 *  get rid of duplicates
	 */
	unique(repp->an);
	unique(repp->ns);
	unique(repp->ar);

	rrfreelist(neg);

	dncheck(nil, 1);
}

/*
 *  satisfy a recursive request.  dnlookup will handle cnames.
 */
static RR*
doextquery(DNSmsg *mp, Request *req, int recurse)
{
	int type;
	char *name;
	RR *rp, *neg;

	name = mp->qd->owner->name;
	type = mp->qd->type;
	rp = dnresolve(name, Cin, type, req, &mp->an, 0, recurse, 1, 0);

	/* don't return soa hints as answers, it's wrong */
	if(rp && rp->db && !rp->auth && rp->type == Tsoa)
		rrfreelist(rp);

	/* don't let negative cached entries escape */
	neg = rrremneg(&rp);
	rrcat(&mp->an, rp);
	return neg;
}

static void
hint(RR **last, RR *rp)
{
	RR *hp;

	switch(rp->type){
	case Tns:
	case Tmx:
	case Tmb:
	case Tmf:
	case Tmd:
		hp = rrlookup(rp->host, Ta, NOneg);
		if(hp == nil)
			hp = dblookup(rp->host->name, Cin, Ta, 0, 0);
		rrcat(last, hp);
		break;
	}
}

Bell Labs OSI certified Powered by Plan 9

(Return to Plan 9 Home Page)

Copyright © 2021 Plan 9 Foundation. All Rights Reserved.
Comments to webmaster@9p.io.