dns.c 16 KB

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