dns.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <auth.h>
  4. #include <fcall.h>
  5. #include <bio.h>
  6. #include <ctype.h>
  7. #include <ip.h>
  8. #include <pool.h>
  9. #include "dns.h"
  10. enum
  11. {
  12. Maxrequest= 1024,
  13. Maxreply= 8192, /* was 512 */
  14. Maxrrr= 32, /* was 16 */
  15. Maxfdata= 8192,
  16. Defmaxage= 60*60, /* default domain name max. age */
  17. Qdir= 0,
  18. Qdns= 1,
  19. };
  20. typedef struct Mfile Mfile;
  21. typedef struct Job Job;
  22. typedef struct Network Network;
  23. int vers; /* incremented each clone/attach */
  24. static volatile int stop;
  25. struct Mfile
  26. {
  27. Mfile *next; /* next free mfile */
  28. int ref;
  29. char *user;
  30. Qid qid;
  31. int fid;
  32. int type; /* reply type */
  33. char reply[Maxreply];
  34. ushort rr[Maxrrr]; /* offset of rr's */
  35. ushort nrr; /* number of rr's */
  36. };
  37. /*
  38. * active local requests
  39. */
  40. struct Job
  41. {
  42. Job *next;
  43. int flushed;
  44. Fcall request;
  45. Fcall reply;
  46. };
  47. Lock joblock;
  48. Job *joblist;
  49. struct {
  50. Lock;
  51. Mfile *inuse; /* active mfile's */
  52. } mfalloc;
  53. Cfg cfg;
  54. int debug;
  55. uchar ipaddr[IPaddrlen]; /* my ip address */
  56. int maxage = Defmaxage;
  57. int mfd[2];
  58. int needrefresh;
  59. ulong now;
  60. vlong nowns;
  61. int sendnotifies;
  62. int testing;
  63. char *trace;
  64. int traceactivity;
  65. char *zonerefreshprogram;
  66. char *logfile = "dns"; /* or "dns.test" */
  67. char *dbfile;
  68. char mntpt[Maxpath];
  69. int fillreply(Mfile*, int);
  70. void freejob(Job*);
  71. void io(void);
  72. void mountinit(char*, char*);
  73. Job* newjob(void);
  74. void rattach(Job*, Mfile*);
  75. void rauth(Job*);
  76. void rclunk(Job*, Mfile*);
  77. void rcreate(Job*, Mfile*);
  78. void rflush(Job*);
  79. void ropen(Job*, Mfile*);
  80. void rread(Job*, Mfile*);
  81. void rremove(Job*, Mfile*);
  82. void rstat(Job*, Mfile*);
  83. void rversion(Job*);
  84. char* rwalk(Job*, Mfile*);
  85. void rwrite(Job*, Mfile*, Request*);
  86. void rwstat(Job*, Mfile*);
  87. void sendmsg(Job*, char*);
  88. void setext(char*, int, char*);
  89. void
  90. usage(void)
  91. {
  92. fprint(2, "usage: %s [-FnorRst] [-a maxage] [-f ndb-file] [-N target] "
  93. "[-x netmtpt] [-z refreshprog]\n", argv0);
  94. exits("usage");
  95. }
  96. void
  97. main(int argc, char *argv[])
  98. {
  99. int kid, pid;
  100. char servefile[Maxpath], ext[Maxpath];
  101. Dir *dir;
  102. setnetmtpt(mntpt, sizeof mntpt, nil);
  103. ext[0] = 0;
  104. ARGBEGIN{
  105. case 'a':
  106. maxage = atol(EARGF(usage()));
  107. if (maxage <= 0)
  108. maxage = Defmaxage;
  109. break;
  110. case 'd':
  111. debug = 1;
  112. traceactivity = 1;
  113. break;
  114. case 'f':
  115. dbfile = EARGF(usage());
  116. break;
  117. case 'F':
  118. cfg.justforw = cfg.resolver = 1;
  119. break;
  120. case 'n':
  121. sendnotifies = 1;
  122. break;
  123. case 'N':
  124. target = atol(EARGF(usage()));
  125. if (target < 1000)
  126. target = 1000;
  127. break;
  128. case 'o':
  129. cfg.straddle = 1; /* straddle inside & outside networks */
  130. break;
  131. case 'r':
  132. cfg.resolver = 1;
  133. break;
  134. case 'R':
  135. norecursion = 1;
  136. break;
  137. case 's':
  138. cfg.serve = 1; /* serve network */
  139. cfg.cachedb = 1;
  140. break;
  141. case 't':
  142. testing = 1;
  143. break;
  144. case 'x':
  145. setnetmtpt(mntpt, sizeof mntpt, EARGF(usage()));
  146. setext(ext, sizeof ext, mntpt);
  147. break;
  148. case 'z':
  149. zonerefreshprogram = EARGF(usage());
  150. break;
  151. }ARGEND
  152. USED(argc);
  153. USED(argv);
  154. if(testing)
  155. mainmem->flags |= POOL_NOREUSE | POOL_ANTAGONISM;
  156. mainmem->flags |= POOL_ANTAGONISM;
  157. rfork(RFREND|RFNOTEG);
  158. cfg.inside = (*mntpt == '\0' || strcmp(mntpt, "/net") == 0);
  159. /* start syslog before we fork */
  160. fmtinstall('F', fcallfmt);
  161. dninit();
  162. /* this really shouldn't be fatal */
  163. if(myipaddr(ipaddr, mntpt) < 0)
  164. sysfatal("can't read my ip address");
  165. dnslog("starting %s%sdns %s%s%son %I's %s",
  166. (cfg.straddle? "straddling ": ""),
  167. (cfg.cachedb? "caching ": ""),
  168. (cfg.serve? "udp server ": ""),
  169. (cfg.justforw? "forwarding-only ": ""),
  170. (cfg.resolver? "resolver ": ""), ipaddr, mntpt);
  171. opendatabase();
  172. now = time(nil); /* open time files before we fork */
  173. nowns = nsec();
  174. snprint(servefile, sizeof servefile, "#s/dns%s", ext);
  175. dir = dirstat(servefile);
  176. if (dir)
  177. sysfatal("%s exists; another dns instance is running",
  178. servefile);
  179. free(dir);
  180. // unmount(servefile, mntpt);
  181. // remove(servefile);
  182. mountinit(servefile, mntpt); /* forks, parent exits */
  183. srand(now*getpid());
  184. db2cache(1);
  185. dnagenever();
  186. if (cfg.straddle && !seerootns())
  187. dnslog("straddle server misconfigured; can't see root name servers");
  188. /*
  189. * fork without sharing heap.
  190. * parent waits around for child to die, then forks & restarts.
  191. * child may spawn udp server, notify procs, etc.; when it gets too
  192. * big, it kills itself and any children.
  193. * /srv/dns and /net/dns remain open and valid.
  194. */
  195. for (;;) {
  196. kid = rfork(RFPROC|RFFDG|RFNOTEG);
  197. switch (kid) {
  198. case -1:
  199. sysfatal("fork failed: %r");
  200. case 0:
  201. if(cfg.serve)
  202. dnudpserver(mntpt);
  203. if(sendnotifies)
  204. notifyproc();
  205. io();
  206. _exits("restart");
  207. default:
  208. while ((pid = waitpid()) != kid && pid != -1)
  209. continue;
  210. break;
  211. }
  212. dnslog("dns restarting");
  213. }
  214. }
  215. /*
  216. * if a mount point is specified, set the cs extension to be the mount point
  217. * with '_'s replacing '/'s
  218. */
  219. void
  220. setext(char *ext, int n, char *p)
  221. {
  222. int i, c;
  223. n--;
  224. for(i = 0; i < n; i++){
  225. c = p[i];
  226. if(c == 0)
  227. break;
  228. if(c == '/')
  229. c = '_';
  230. ext[i] = c;
  231. }
  232. ext[i] = 0;
  233. }
  234. void
  235. mountinit(char *service, char *mntpt)
  236. {
  237. int f;
  238. int p[2];
  239. char buf[32];
  240. if(pipe(p) < 0)
  241. abort(); /* "pipe failed" */;
  242. /* copy namespace to avoid a deadlock */
  243. switch(rfork(RFFDG|RFPROC|RFNAMEG)){
  244. case 0: /* child: hang around and (re)start main proc */
  245. close(p[1]);
  246. procsetname("%s restarter", mntpt);
  247. break;
  248. case -1:
  249. abort(); /* "fork failed\n" */;
  250. default: /* parent: make /srv/dns, mount it, exit */
  251. close(p[0]);
  252. /*
  253. * make a /srv/dns
  254. */
  255. f = create(service, 1, 0666);
  256. if(f < 0)
  257. abort(); /* service */;
  258. snprint(buf, sizeof buf, "%d", p[1]);
  259. if(write(f, buf, strlen(buf)) != strlen(buf))
  260. abort(); /* "write %s", service */;
  261. close(f);
  262. /*
  263. * put ourselves into the file system
  264. */
  265. if(mount(p[1], -1, mntpt, MAFTER, "") < 0)
  266. fprint(2, "dns mount failed: %r\n");
  267. _exits(0);
  268. }
  269. mfd[0] = mfd[1] = p[0];
  270. }
  271. Mfile*
  272. newfid(int fid, int needunused)
  273. {
  274. Mfile *mf;
  275. lock(&mfalloc);
  276. for(mf = mfalloc.inuse; mf != nil; mf = mf->next)
  277. if(mf->fid == fid){
  278. unlock(&mfalloc);
  279. if(needunused)
  280. return nil;
  281. return mf;
  282. }
  283. mf = emalloc(sizeof(*mf));
  284. if(mf == nil)
  285. sysfatal("out of memory");
  286. mf->fid = fid;
  287. mf->next = mfalloc.inuse;
  288. mf->user = estrdup("dummy");
  289. mfalloc.inuse = mf;
  290. unlock(&mfalloc);
  291. return mf;
  292. }
  293. void
  294. freefid(Mfile *mf)
  295. {
  296. Mfile **l;
  297. lock(&mfalloc);
  298. for(l = &mfalloc.inuse; *l != nil; l = &(*l)->next)
  299. if(*l == mf){
  300. *l = mf->next;
  301. if(mf->user)
  302. free(mf->user);
  303. memset(mf, 0, sizeof *mf); /* cause trouble */
  304. free(mf);
  305. unlock(&mfalloc);
  306. return;
  307. }
  308. sysfatal("freeing unused fid");
  309. }
  310. Mfile*
  311. copyfid(Mfile *mf, int fid)
  312. {
  313. Mfile *nmf;
  314. nmf = newfid(fid, 1);
  315. if(nmf == nil)
  316. return nil;
  317. nmf->fid = fid;
  318. nmf->user = estrdup(mf->user);
  319. nmf->qid.type = mf->qid.type;
  320. nmf->qid.path = mf->qid.path;
  321. nmf->qid.vers = vers++;
  322. return nmf;
  323. }
  324. Job*
  325. newjob(void)
  326. {
  327. Job *job;
  328. job = emalloc(sizeof *job);
  329. lock(&joblock);
  330. job->next = joblist;
  331. joblist = job;
  332. job->request.tag = -1;
  333. unlock(&joblock);
  334. return job;
  335. }
  336. void
  337. freejob(Job *job)
  338. {
  339. Job **l;
  340. lock(&joblock);
  341. for(l = &joblist; *l; l = &(*l)->next)
  342. if(*l == job){
  343. *l = job->next;
  344. memset(job, 0, sizeof *job); /* cause trouble */
  345. free(job);
  346. break;
  347. }
  348. unlock(&joblock);
  349. }
  350. void
  351. flushjob(int tag)
  352. {
  353. Job *job;
  354. lock(&joblock);
  355. for(job = joblist; job; job = job->next)
  356. if(job->request.tag == tag && job->request.type != Tflush){
  357. job->flushed = 1;
  358. break;
  359. }
  360. unlock(&joblock);
  361. }
  362. void
  363. io(void)
  364. {
  365. volatile long n;
  366. volatile uchar mdata[IOHDRSZ + Maxfdata];
  367. Job *volatile job;
  368. Mfile *volatile mf;
  369. volatile Request req;
  370. memset(&req, 0, sizeof req);
  371. /*
  372. * a slave process is sometimes forked to wait for replies from other
  373. * servers. The master process returns immediately via a longjmp
  374. * through 'mret'.
  375. */
  376. if(setjmp(req.mret))
  377. putactivity(0);
  378. // procsetname("9p server");
  379. req.isslave = 0;
  380. stop = 0;
  381. while(!stop){
  382. procsetname("%d %s/dns Twrites of %d 9p rpcs read; %d alarms",
  383. stats.qrecvd9p, mntpt, stats.qrecvd9prpc, stats.alarms);
  384. n = read9pmsg(mfd[0], mdata, sizeof mdata);
  385. if(n<=0){
  386. dnslog("error reading 9P from %s: %r", mntpt);
  387. sleep(2000); /* don't thrash */
  388. return;
  389. }
  390. stats.qrecvd9prpc++;
  391. job = newjob();
  392. if(convM2S(mdata, n, &job->request) != n){
  393. freejob(job);
  394. continue;
  395. }
  396. mf = newfid(job->request.fid, 0);
  397. if(debug)
  398. dnslog("%F", &job->request);
  399. getactivity(&req, 0);
  400. req.aborttime = time(nil) + Maxreqtm;
  401. req.from = "9p";
  402. switch(job->request.type){
  403. default:
  404. warning("unknown request type %d", job->request.type);
  405. break;
  406. case Tversion:
  407. rversion(job);
  408. break;
  409. case Tauth:
  410. rauth(job);
  411. break;
  412. case Tflush:
  413. rflush(job);
  414. break;
  415. case Tattach:
  416. rattach(job, mf);
  417. break;
  418. case Twalk:
  419. rwalk(job, mf);
  420. break;
  421. case Topen:
  422. ropen(job, mf);
  423. break;
  424. case Tcreate:
  425. rcreate(job, mf);
  426. break;
  427. case Tread:
  428. rread(job, mf);
  429. break;
  430. case Twrite:
  431. /* &req is handed to dnresolve() */
  432. rwrite(job, mf, &req);
  433. break;
  434. case Tclunk:
  435. rclunk(job, mf);
  436. break;
  437. case Tremove:
  438. rremove(job, mf);
  439. break;
  440. case Tstat:
  441. rstat(job, mf);
  442. break;
  443. case Twstat:
  444. rwstat(job, mf);
  445. break;
  446. }
  447. freejob(job);
  448. /*
  449. * slave processes die after replying
  450. */
  451. if(req.isslave){
  452. putactivity(0);
  453. _exits(0);
  454. }
  455. putactivity(0);
  456. }
  457. /* kill any udp server, notifier, etc. processes */
  458. postnote(PNGROUP, getpid(), "die");
  459. sleep(1000);
  460. }
  461. void
  462. rversion(Job *job)
  463. {
  464. if(job->request.msize > IOHDRSZ + Maxfdata)
  465. job->reply.msize = IOHDRSZ + Maxfdata;
  466. else
  467. job->reply.msize = job->request.msize;
  468. if(strncmp(job->request.version, "9P2000", 6) != 0)
  469. sendmsg(job, "unknown 9P version");
  470. else{
  471. job->reply.version = "9P2000";
  472. sendmsg(job, 0);
  473. }
  474. }
  475. void
  476. rauth(Job *job)
  477. {
  478. sendmsg(job, "dns: authentication not required");
  479. }
  480. /*
  481. * don't flush till all the slaves are done
  482. */
  483. void
  484. rflush(Job *job)
  485. {
  486. flushjob(job->request.oldtag);
  487. sendmsg(job, 0);
  488. }
  489. void
  490. rattach(Job *job, Mfile *mf)
  491. {
  492. if(mf->user != nil)
  493. free(mf->user);
  494. mf->user = estrdup(job->request.uname);
  495. mf->qid.vers = vers++;
  496. mf->qid.type = QTDIR;
  497. mf->qid.path = 0LL;
  498. job->reply.qid = mf->qid;
  499. sendmsg(job, 0);
  500. }
  501. char*
  502. rwalk(Job *job, Mfile *mf)
  503. {
  504. int i, nelems;
  505. char *err;
  506. char **elems;
  507. Mfile *nmf;
  508. Qid qid;
  509. err = 0;
  510. nmf = nil;
  511. elems = job->request.wname;
  512. nelems = job->request.nwname;
  513. job->reply.nwqid = 0;
  514. if(job->request.newfid != job->request.fid){
  515. /* clone fid */
  516. nmf = copyfid(mf, job->request.newfid);
  517. if(nmf == nil){
  518. err = "clone bad newfid";
  519. goto send;
  520. }
  521. mf = nmf;
  522. }
  523. /* else nmf will be nil */
  524. qid = mf->qid;
  525. if(nelems > 0)
  526. /* walk fid */
  527. for(i=0; i<nelems && i<MAXWELEM; i++){
  528. if((qid.type & QTDIR) == 0){
  529. err = "not a directory";
  530. break;
  531. }
  532. if (strcmp(elems[i], "..") == 0 ||
  533. strcmp(elems[i], ".") == 0){
  534. qid.type = QTDIR;
  535. qid.path = Qdir;
  536. Found:
  537. job->reply.wqid[i] = qid;
  538. job->reply.nwqid++;
  539. continue;
  540. }
  541. if(strcmp(elems[i], "dns") == 0){
  542. qid.type = QTFILE;
  543. qid.path = Qdns;
  544. goto Found;
  545. }
  546. err = "file does not exist";
  547. break;
  548. }
  549. send:
  550. if(nmf != nil && (err!=nil || job->reply.nwqid<nelems))
  551. freefid(nmf);
  552. if(err == nil)
  553. mf->qid = qid;
  554. sendmsg(job, err);
  555. return err;
  556. }
  557. void
  558. ropen(Job *job, Mfile *mf)
  559. {
  560. int mode;
  561. char *err;
  562. err = 0;
  563. mode = job->request.mode;
  564. if(mf->qid.type & QTDIR)
  565. if(mode)
  566. err = "permission denied";
  567. job->reply.qid = mf->qid;
  568. job->reply.iounit = 0;
  569. sendmsg(job, err);
  570. }
  571. void
  572. rcreate(Job *job, Mfile *mf)
  573. {
  574. USED(mf);
  575. sendmsg(job, "creation permission denied");
  576. }
  577. void
  578. rread(Job *job, Mfile *mf)
  579. {
  580. int i, n;
  581. long clock;
  582. ulong cnt;
  583. vlong off;
  584. char *err;
  585. uchar buf[Maxfdata];
  586. Dir dir;
  587. n = 0;
  588. err = nil;
  589. off = job->request.offset;
  590. cnt = job->request.count;
  591. *buf = '\0';
  592. job->reply.data = (char*)buf;
  593. if(mf->qid.type & QTDIR){
  594. clock = time(nil);
  595. if(off == 0){
  596. memset(&dir, 0, sizeof dir);
  597. dir.name = "dns";
  598. dir.qid.type = QTFILE;
  599. dir.qid.vers = vers;
  600. dir.qid.path = Qdns;
  601. dir.mode = 0666;
  602. dir.length = 0;
  603. dir.uid = dir.gid = dir.muid = mf->user;
  604. dir.atime = dir.mtime = clock; /* wrong */
  605. n = convD2M(&dir, buf, sizeof buf);
  606. }
  607. } else if (off < 0)
  608. err = "negative read offset";
  609. else {
  610. /* first offset will always be zero */
  611. for(i = 1; i <= mf->nrr; i++)
  612. if(mf->rr[i] > off)
  613. break;
  614. if(i <= mf->nrr) {
  615. if(off + cnt > mf->rr[i])
  616. n = mf->rr[i] - off;
  617. else
  618. n = cnt;
  619. assert(n >= 0);
  620. job->reply.data = mf->reply + off;
  621. }
  622. }
  623. job->reply.count = n;
  624. sendmsg(job, err);
  625. }
  626. void
  627. rwrite(Job *job, Mfile *mf, Request *req)
  628. {
  629. int rooted, status, wantsav;
  630. long n;
  631. ulong cnt;
  632. char *err, *p, *atype;
  633. RR *rp, *tp, *neg;
  634. err = nil;
  635. cnt = job->request.count;
  636. if(mf->qid.type & QTDIR){
  637. err = "can't write directory";
  638. goto send;
  639. }
  640. if (job->request.offset != 0) {
  641. err = "writing at non-zero offset";
  642. goto send;
  643. }
  644. if(cnt >= Maxrequest){
  645. err = "request too long";
  646. goto send;
  647. }
  648. job->request.data[cnt] = 0;
  649. if(cnt > 0 && job->request.data[cnt-1] == '\n')
  650. job->request.data[cnt-1] = 0;
  651. /*
  652. * special commands
  653. */
  654. dnslog("rwrite got: %s", job->request.data);
  655. if(strcmp(job->request.data, "debug")==0){
  656. debug ^= 1;
  657. goto send;
  658. } else if(strcmp(job->request.data, "dump")==0){
  659. dndump("/lib/ndb/dnsdump");
  660. goto send;
  661. } else if(strcmp(job->request.data, "poolcheck")==0){
  662. poolcheck(mainmem);
  663. goto send;
  664. } else if(strcmp(job->request.data, "refresh")==0){
  665. needrefresh = 1;
  666. goto send;
  667. } else if(strcmp(job->request.data, "restart")==0){
  668. stop = 1;
  669. goto send;
  670. } else if(strcmp(job->request.data, "stats")==0){
  671. dnstats("/lib/ndb/dnsstats");
  672. goto send;
  673. } else if(strncmp(job->request.data, "target ", 7)==0){
  674. target = atol(job->request.data + 7);
  675. dnslog("target set to %ld", target);
  676. goto send;
  677. } else if(strcmp(job->request.data, "age")==0){
  678. dnslog("dump, age & dump forced");
  679. dndump("/lib/ndb/dnsdump1");
  680. dnforceage();
  681. dndump("/lib/ndb/dnsdump2");
  682. goto send;
  683. }
  684. /*
  685. * kill previous reply
  686. */
  687. mf->nrr = 0;
  688. mf->rr[0] = 0;
  689. /*
  690. * break up request (into a name and a type)
  691. */
  692. atype = strchr(job->request.data, ' ');
  693. if(atype == 0){
  694. err = "illegal request";
  695. goto send;
  696. } else
  697. *atype++ = 0;
  698. /*
  699. * tracing request
  700. */
  701. if(strcmp(atype, "trace") == 0){
  702. if(trace)
  703. free(trace);
  704. if(*job->request.data)
  705. trace = estrdup(job->request.data);
  706. else
  707. trace = 0;
  708. goto send;
  709. }
  710. /* normal request: domain [type] */
  711. stats.qrecvd9p++;
  712. mf->type = rrtype(atype);
  713. if(mf->type < 0){
  714. err = "unknown type";
  715. goto send;
  716. }
  717. p = atype - 2;
  718. if(p >= job->request.data && *p == '.'){
  719. rooted = 1;
  720. *p = 0;
  721. } else
  722. rooted = 0;
  723. p = job->request.data;
  724. if(*p == '!'){
  725. wantsav = 1;
  726. p++;
  727. } else
  728. wantsav = 0;
  729. dncheck(0, 1);
  730. status = 0;
  731. rp = dnresolve(p, Cin, mf->type, req, 0, 0, Recurse, rooted, &status);
  732. dncheck(0, 1);
  733. neg = rrremneg(&rp);
  734. if(neg){
  735. status = neg->negrcode;
  736. rrfreelist(neg);
  737. }
  738. if(rp == nil)
  739. switch(status){
  740. case Rname:
  741. err = "name does not exist";
  742. break;
  743. case Rserver:
  744. err = "dns failure";
  745. break;
  746. default:
  747. err = "resource does not exist";
  748. break;
  749. }
  750. else {
  751. lock(&joblock);
  752. if(!job->flushed){
  753. /* format data to be read later */
  754. n = 0;
  755. mf->nrr = 0;
  756. for(tp = rp; mf->nrr < Maxrrr-1 && n < Maxreply && tp &&
  757. tsame(mf->type, tp->type); tp = tp->next){
  758. mf->rr[mf->nrr++] = n;
  759. if(wantsav)
  760. n += snprint(mf->reply+n, Maxreply-n,
  761. "%Q", tp);
  762. else
  763. n += snprint(mf->reply+n, Maxreply-n,
  764. "%R", tp);
  765. }
  766. mf->rr[mf->nrr] = n;
  767. }
  768. unlock(&joblock);
  769. rrfreelist(rp);
  770. }
  771. send:
  772. dncheck(0, 1);
  773. job->reply.count = cnt;
  774. sendmsg(job, err);
  775. }
  776. void
  777. rclunk(Job *job, Mfile *mf)
  778. {
  779. freefid(mf);
  780. sendmsg(job, 0);
  781. }
  782. void
  783. rremove(Job *job, Mfile *mf)
  784. {
  785. USED(mf);
  786. sendmsg(job, "remove permission denied");
  787. }
  788. void
  789. rstat(Job *job, Mfile *mf)
  790. {
  791. Dir dir;
  792. uchar buf[IOHDRSZ+Maxfdata];
  793. memset(&dir, 0, sizeof dir);
  794. if(mf->qid.type & QTDIR){
  795. dir.name = ".";
  796. dir.mode = DMDIR|0555;
  797. } else {
  798. dir.name = "dns";
  799. dir.mode = 0666;
  800. }
  801. dir.qid = mf->qid;
  802. dir.length = 0;
  803. dir.uid = dir.gid = dir.muid = mf->user;
  804. dir.atime = dir.mtime = time(nil);
  805. job->reply.nstat = convD2M(&dir, buf, sizeof buf);
  806. job->reply.stat = buf;
  807. sendmsg(job, 0);
  808. }
  809. void
  810. rwstat(Job *job, Mfile *mf)
  811. {
  812. USED(mf);
  813. sendmsg(job, "wstat permission denied");
  814. }
  815. void
  816. sendmsg(Job *job, char *err)
  817. {
  818. int n;
  819. uchar mdata[IOHDRSZ + Maxfdata];
  820. char ename[ERRMAX];
  821. if(err){
  822. job->reply.type = Rerror;
  823. snprint(ename, sizeof ename, "dns: %s", err);
  824. job->reply.ename = ename;
  825. }else
  826. job->reply.type = job->request.type+1;
  827. job->reply.tag = job->request.tag;
  828. n = convS2M(&job->reply, mdata, sizeof mdata);
  829. if(n == 0){
  830. warning("sendmsg convS2M of %F returns 0", &job->reply);
  831. abort();
  832. }
  833. lock(&joblock);
  834. if(job->flushed == 0)
  835. if(write(mfd[1], mdata, n)!=n)
  836. sysfatal("mount write");
  837. unlock(&joblock);
  838. if(debug)
  839. dnslog("%F %d", &job->reply, n);
  840. }
  841. /*
  842. * the following varies between dnsdebug and dns
  843. */
  844. void
  845. logreply(int id, uchar *addr, DNSmsg *mp)
  846. {
  847. RR *rp;
  848. dnslog("%d: rcvd %I flags:%s%s%s%s%s", id, addr,
  849. mp->flags & Fauth? " auth": "",
  850. mp->flags & Ftrunc? " trunc": "",
  851. mp->flags & Frecurse? " rd": "",
  852. mp->flags & Fcanrec? " ra": "",
  853. (mp->flags & (Fauth|Rmask)) == (Fauth|Rname)? " nx": "");
  854. for(rp = mp->qd; rp != nil; rp = rp->next)
  855. dnslog("%d: rcvd %I qd %s", id, addr, rp->owner->name);
  856. for(rp = mp->an; rp != nil; rp = rp->next)
  857. dnslog("%d: rcvd %I an %R", id, addr, rp);
  858. for(rp = mp->ns; rp != nil; rp = rp->next)
  859. dnslog("%d: rcvd %I ns %R", id, addr, rp);
  860. for(rp = mp->ar; rp != nil; rp = rp->next)
  861. dnslog("%d: rcvd %I ar %R", id, addr, rp);
  862. }
  863. void
  864. logsend(int id, int subid, uchar *addr, char *sname, char *rname, int type)
  865. {
  866. char buf[12];
  867. dnslog("[%d] %d.%d: sending to %I/%s %s %s",
  868. getpid(), id, subid, addr, sname, rname,
  869. rrname(type, buf, sizeof buf));
  870. }
  871. RR*
  872. getdnsservers(int class)
  873. {
  874. return dnsservers(class);
  875. }