dns.c 16 KB

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