dns.c 16 KB

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