cs.c 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842
  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 <ndb.h>
  8. #include <ip.h>
  9. enum
  10. {
  11. Nreply= 20,
  12. Maxreply= 256,
  13. Maxrequest= 128,
  14. Maxpath= 128,
  15. Maxfdata= 8192,
  16. Qdir= 0,
  17. Qcs= 1,
  18. };
  19. typedef struct Mfile Mfile;
  20. typedef struct Mlist Mlist;
  21. typedef struct Network Network;
  22. typedef struct Flushreq Flushreq;
  23. typedef struct Job Job;
  24. int vers; /* incremented each clone/attach */
  25. struct Mfile
  26. {
  27. int busy;
  28. char *user;
  29. Qid qid;
  30. int fid;
  31. /*
  32. * current request
  33. */
  34. char *net;
  35. char *host;
  36. char *serv;
  37. char *rem;
  38. /*
  39. * result of the last lookup
  40. */
  41. Network *nextnet;
  42. int nreply;
  43. char *reply[Nreply];
  44. int replylen[Nreply];
  45. };
  46. struct Mlist
  47. {
  48. Mlist *next;
  49. Mfile mf;
  50. };
  51. //
  52. // active requests
  53. //
  54. struct Job
  55. {
  56. Job *next;
  57. int flushed;
  58. Fcall request;
  59. Fcall reply;
  60. };
  61. Lock joblock;
  62. Job *joblist;
  63. Mlist *mlist;
  64. int mfd[2];
  65. int debug;
  66. int paranoia;
  67. jmp_buf masterjmp; /* return through here after a slave process has been created */
  68. int *isslave; /* *isslave non-zero means this is a slave process */
  69. char *dbfile;
  70. Ndb *db, *netdb;
  71. void rversion(Job*);
  72. void rflush(Job*);
  73. void rattach(Job*, Mfile*);
  74. char* rwalk(Job*, Mfile*);
  75. void ropen(Job*, Mfile*);
  76. void rcreate(Job*, Mfile*);
  77. void rread(Job*, Mfile*);
  78. void rwrite(Job*, Mfile*);
  79. void rclunk(Job*, Mfile*);
  80. void rremove(Job*, Mfile*);
  81. void rstat(Job*, Mfile*);
  82. void rwstat(Job*, Mfile*);
  83. void rauth(Job*);
  84. void sendmsg(Job*, char*);
  85. void error(char*);
  86. void mountinit(char*, char*);
  87. void io(void);
  88. void ndbinit(void);
  89. void netinit(int);
  90. void netadd(char*);
  91. char *genquery(Mfile*, char*);
  92. char* ipinfoquery(Mfile*, char**, int);
  93. int needproto(Network*, Ndbtuple*);
  94. int lookup(Mfile*);
  95. Ndbtuple* reorder(Ndbtuple*, Ndbtuple*);
  96. void ipid(void);
  97. void readipinterfaces(void);
  98. void* emalloc(int);
  99. char* estrdup(char*);
  100. Job* newjob(void);
  101. void freejob(Job*);
  102. void setext(char*, int, char*);
  103. void cleanmf(Mfile*);
  104. extern void paralloc(void);
  105. Lock dblock; /* mutex on database operations */
  106. Lock netlock; /* mutex for netinit() */
  107. char *logfile = "cs";
  108. char *paranoiafile = "cs.paranoia";
  109. char mntpt[Maxpath];
  110. char netndb[Maxpath];
  111. /*
  112. * Network specific translators
  113. */
  114. Ndbtuple* iplookup(Network*, char*, char*, int);
  115. char* iptrans(Ndbtuple*, Network*, char*, char*, int);
  116. Ndbtuple* telcolookup(Network*, char*, char*, int);
  117. char* telcotrans(Ndbtuple*, Network*, char*, char*, int);
  118. Ndbtuple* dnsiplookup(char*, Ndbs*);
  119. struct Network
  120. {
  121. char *net;
  122. Ndbtuple *(*lookup)(Network*, char*, char*, int);
  123. char *(*trans)(Ndbtuple*, Network*, char*, char*, int);
  124. int considered;
  125. int fasttimeouthack;
  126. Network *next;
  127. };
  128. enum
  129. {
  130. Nilfast,
  131. Ntcp,
  132. Nil,
  133. Nudp,
  134. Nicmp,
  135. Nrudp,
  136. Ntelco,
  137. };
  138. /*
  139. * net doesn't apply to udp, icmp, or telco (for speed)
  140. */
  141. Network network[] = {
  142. [Nilfast] { "il", iplookup, iptrans, 0, 1 },
  143. [Ntcp] { "tcp", iplookup, iptrans, 0, 0 },
  144. [Nil] { "il", iplookup, iptrans, 0, 0 },
  145. [Nudp] { "udp", iplookup, iptrans, 1, 0 },
  146. [Nicmp] { "icmp", iplookup, iptrans, 1, 0 },
  147. [Nrudp] { "rudp", iplookup, iptrans, 1, 0 },
  148. [Ntelco] { "telco", telcolookup, telcotrans, 1, 0 },
  149. { 0 },
  150. };
  151. Lock ipifclock;
  152. Ipifc *ipifcs;
  153. char eaddr[Ndbvlen]; /* ascii ethernet address */
  154. char ipaddr[Ndbvlen]; /* ascii internet address */
  155. uchar ipa[IPaddrlen]; /* binary internet address */
  156. char mysysname[Ndbvlen];
  157. Network *netlist; /* networks ordered by preference */
  158. Network *last;
  159. void
  160. usage(void)
  161. {
  162. fprint(2, "usage: %s [-d] [-f ndb-file] [-x netmtpt] [-n]\n", argv0);
  163. exits("usage");
  164. }
  165. void
  166. main(int argc, char *argv[])
  167. {
  168. char servefile[Maxpath];
  169. int justsetname;
  170. char *p;
  171. char ext[Maxpath];
  172. justsetname = 0;
  173. setnetmtpt(mntpt, sizeof(mntpt), nil);
  174. ext[0] = 0;
  175. ARGBEGIN{
  176. case 'd':
  177. debug = 1;
  178. break;
  179. case 'f':
  180. p = ARGF();
  181. if(p == nil)
  182. usage();
  183. dbfile = p;
  184. break;
  185. case 'x':
  186. p = ARGF();
  187. if(p == nil)
  188. usage();
  189. setnetmtpt(mntpt, sizeof(mntpt), p);
  190. setext(ext, sizeof(ext), mntpt);
  191. break;
  192. case 'n':
  193. justsetname = 1;
  194. break;
  195. }ARGEND
  196. USED(argc);
  197. USED(argv);
  198. rfork(RFREND|RFNOTEG);
  199. snprint(servefile, sizeof(servefile), "#s/cs%s", ext);
  200. snprint(netndb, sizeof(netndb), "%s/ndb", mntpt);
  201. unmount(servefile, mntpt);
  202. remove(servefile);
  203. fmtinstall('E', eipfmt);
  204. fmtinstall('I', eipfmt);
  205. fmtinstall('M', eipfmt);
  206. fmtinstall('F', fcallfmt);
  207. ndbinit();
  208. netinit(0);
  209. if(!justsetname){
  210. mountinit(servefile, mntpt);
  211. io();
  212. }
  213. exits(0);
  214. }
  215. /*
  216. * if a mount point is specified, set the cs extention 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. error("pipe failed");
  242. switch(rfork(RFFDG|RFPROC|RFNAMEG)){
  243. case 0:
  244. close(p[1]);
  245. break;
  246. case -1:
  247. error("fork failed\n");
  248. default:
  249. /*
  250. * make a /srv/cs
  251. */
  252. f = create(service, 1, 0666);
  253. if(f < 0)
  254. error(service);
  255. snprint(buf, sizeof(buf), "%d", p[1]);
  256. if(write(f, buf, strlen(buf)) != strlen(buf))
  257. error("write /srv/cs");
  258. close(f);
  259. /*
  260. * put ourselves into the file system
  261. */
  262. close(p[0]);
  263. if(mount(p[1], -1, mntpt, MAFTER, "") < 0)
  264. error("mount failed\n");
  265. _exits(0);
  266. }
  267. mfd[0] = mfd[1] = p[0];
  268. }
  269. void
  270. ndbinit(void)
  271. {
  272. db = ndbopen(dbfile);
  273. if(db == nil)
  274. error("can't open network database");
  275. netdb = ndbopen(netndb);
  276. if(netdb != nil){
  277. netdb->nohash = 1;
  278. db = ndbcat(netdb, db);
  279. }
  280. }
  281. Mfile*
  282. newfid(int fid)
  283. {
  284. Mlist *f, *ff;
  285. Mfile *mf;
  286. ff = 0;
  287. for(f = mlist; f; f = f->next)
  288. if(f->mf.busy && f->mf.fid == fid)
  289. return &f->mf;
  290. else if(!ff && !f->mf.busy)
  291. ff = f;
  292. if(ff == 0){
  293. ff = emalloc(sizeof *f);
  294. ff->next = mlist;
  295. mlist = ff;
  296. }
  297. mf = &ff->mf;
  298. memset(mf, 0, sizeof *mf);
  299. mf->fid = fid;
  300. return mf;
  301. }
  302. Job*
  303. newjob(void)
  304. {
  305. Job *job;
  306. job = mallocz(sizeof(Job), 1);
  307. lock(&joblock);
  308. job->next = joblist;
  309. joblist = job;
  310. job->request.tag = -1;
  311. unlock(&joblock);
  312. return job;
  313. }
  314. void
  315. freejob(Job *job)
  316. {
  317. Job **l;
  318. lock(&joblock);
  319. for(l = &joblist; *l; l = &(*l)->next){
  320. if((*l) == job){
  321. *l = job->next;
  322. free(job);
  323. break;
  324. }
  325. }
  326. unlock(&joblock);
  327. }
  328. void
  329. flushjob(int tag)
  330. {
  331. Job *job;
  332. lock(&joblock);
  333. for(job = joblist; job; job = job->next){
  334. if(job->request.tag == tag && job->request.type != Tflush){
  335. job->flushed = 1;
  336. break;
  337. }
  338. }
  339. unlock(&joblock);
  340. }
  341. void
  342. io(void)
  343. {
  344. long n;
  345. Mfile *mf;
  346. int slaveflag;
  347. uchar mdata[IOHDRSZ + Maxfdata];
  348. Job *job;
  349. /*
  350. * if we ask dns to fulfill requests,
  351. * a slave process is created to wait for replies. The
  352. * master process returns immediately via a longjmp
  353. * through 'masterjmp'.
  354. *
  355. * *isslave is a pointer into the call stack to a variable
  356. * that tells whether or not the current process is a slave.
  357. */
  358. slaveflag = 0; /* init slave variable */
  359. isslave = &slaveflag;
  360. setjmp(masterjmp);
  361. for(;;){
  362. n = read9pmsg(mfd[0], mdata, sizeof mdata);
  363. if(n<=0)
  364. error("mount read");
  365. job = newjob();
  366. if(convM2S(mdata, n, &job->request) != n){
  367. syslog(1, logfile, "format error %ux %ux %ux %ux %ux",
  368. mdata[0], mdata[1], mdata[2], mdata[3], mdata[4]);
  369. freejob(job);
  370. continue;
  371. }
  372. if(job->request.fid<0)
  373. error("fid out of range");
  374. lock(&dblock);
  375. mf = newfid(job->request.fid);
  376. if(debug)
  377. syslog(0, logfile, "%F", &job->request);
  378. switch(job->request.type){
  379. default:
  380. syslog(1, logfile, "unknown request type %d", job->request.type);
  381. break;
  382. case Tversion:
  383. rversion(job);
  384. break;
  385. case Tauth:
  386. rauth(job);
  387. break;
  388. case Tflush:
  389. rflush(job);
  390. break;
  391. case Tattach:
  392. rattach(job, mf);
  393. break;
  394. case Twalk:
  395. rwalk(job, mf);
  396. break;
  397. case Topen:
  398. ropen(job, mf);
  399. break;
  400. case Tcreate:
  401. rcreate(job, mf);
  402. break;
  403. case Tread:
  404. rread(job, mf);
  405. break;
  406. case Twrite:
  407. rwrite(job, mf);
  408. break;
  409. case Tclunk:
  410. rclunk(job, mf);
  411. break;
  412. case Tremove:
  413. rremove(job, mf);
  414. break;
  415. case Tstat:
  416. rstat(job, mf);
  417. break;
  418. case Twstat:
  419. rwstat(job, mf);
  420. break;
  421. }
  422. unlock(&dblock);
  423. freejob(job);
  424. /*
  425. * slave processes die after replying
  426. */
  427. if(*isslave){
  428. if(debug)
  429. syslog(0, logfile, "slave death %d", getpid());
  430. _exits(0);
  431. }
  432. }
  433. }
  434. void
  435. rversion(Job *job)
  436. {
  437. if(job->request.msize > IOHDRSZ + Maxfdata)
  438. job->reply.msize = IOHDRSZ + Maxfdata;
  439. else
  440. job->reply.msize = job->request.msize;
  441. if(strncmp(job->request.version, "9P2000", 6) != 0)
  442. sendmsg(job, "unknown 9P version");
  443. else{
  444. job->reply.version = "9P2000";
  445. sendmsg(job, 0);
  446. }
  447. }
  448. void
  449. rauth(Job *job)
  450. {
  451. sendmsg(job, "cs: authentication not required");
  452. }
  453. /*
  454. * don't flush till all the slaves are done
  455. */
  456. void
  457. rflush(Job *job)
  458. {
  459. flushjob(job->request.oldtag);
  460. sendmsg(job, 0);
  461. }
  462. void
  463. rattach(Job *job, Mfile *mf)
  464. {
  465. if(mf->busy == 0){
  466. mf->busy = 1;
  467. mf->user = estrdup(job->request.uname);
  468. }
  469. mf->qid.vers = vers++;
  470. mf->qid.type = QTDIR;
  471. mf->qid.path = 0LL;
  472. job->reply.qid = mf->qid;
  473. sendmsg(job, 0);
  474. }
  475. char*
  476. rwalk(Job *job, Mfile *mf)
  477. {
  478. char *err;
  479. char **elems;
  480. int nelems;
  481. int i;
  482. Mfile *nmf;
  483. Qid qid;
  484. err = 0;
  485. nmf = nil;
  486. elems = job->request.wname;
  487. nelems = job->request.nwname;
  488. job->reply.nwqid = 0;
  489. if(job->request.newfid != job->request.fid){
  490. /* clone fid */
  491. if(job->request.newfid<0){
  492. err = "clone newfid out of range";
  493. goto send;
  494. }
  495. nmf = newfid(job->request.newfid);
  496. if(nmf->busy){
  497. nmf = nil;
  498. err = "clone to used channel";
  499. goto send;
  500. }
  501. *nmf = *mf;
  502. nmf->user = estrdup(mf->user);
  503. nmf->fid = job->request.newfid;
  504. nmf->qid.vers = vers++;
  505. mf = nmf;
  506. }
  507. /* else nmf will be nil */
  508. qid = mf->qid;
  509. if(nelems > 0){
  510. /* walk fid */
  511. for(i=0; i<nelems && i<MAXWELEM; i++){
  512. if((qid.type & QTDIR) == 0){
  513. err = "not a directory";
  514. break;
  515. }
  516. if(strcmp(elems[i], "..") == 0 || strcmp(elems[i], ".") == 0){
  517. qid.type = QTDIR;
  518. qid.path = Qdir;
  519. Found:
  520. job->reply.wqid[i] = qid;
  521. job->reply.nwqid++;
  522. continue;
  523. }
  524. if(strcmp(elems[i], "cs") == 0){
  525. qid.type = QTFILE;
  526. qid.path = Qcs;
  527. goto Found;
  528. }
  529. err = "file does not exist";
  530. break;
  531. }
  532. }
  533. send:
  534. if(nmf != nil && (err!=nil || job->reply.nwqid<nelems)){
  535. nmf->busy = 0;
  536. nmf->fid = 0;
  537. }
  538. if(err == nil)
  539. mf->qid = qid;
  540. sendmsg(job, err);
  541. return err;
  542. }
  543. void
  544. ropen(Job *job, Mfile *mf)
  545. {
  546. int mode;
  547. char *err;
  548. err = 0;
  549. mode = job->request.mode;
  550. if(mf->qid.type & QTDIR){
  551. if(mode)
  552. err = "permission denied";
  553. }
  554. job->reply.qid = mf->qid;
  555. job->reply.iounit = 0;
  556. sendmsg(job, err);
  557. }
  558. void
  559. rcreate(Job *job, Mfile *mf)
  560. {
  561. USED(mf);
  562. sendmsg(job, "creation permission denied");
  563. }
  564. void
  565. rread(Job *job, Mfile *mf)
  566. {
  567. int i, n, cnt;
  568. long off, toff, clock;
  569. Dir dir;
  570. uchar buf[Maxfdata];
  571. char *err;
  572. n = 0;
  573. err = 0;
  574. off = job->request.offset;
  575. cnt = job->request.count;
  576. if(mf->qid.type & QTDIR){
  577. clock = time(0);
  578. if(off == 0){
  579. dir.name = "cs";
  580. dir.qid.type = QTFILE;
  581. dir.qid.vers = vers;
  582. dir.qid.path = Qcs;
  583. dir.mode = 0666;
  584. dir.length = 0;
  585. dir.uid = mf->user;
  586. dir.gid = mf->user;
  587. dir.muid = mf->user;
  588. dir.atime = clock; /* wrong */
  589. dir.mtime = clock; /* wrong */
  590. n = convD2M(&dir, buf, sizeof buf);
  591. }
  592. job->reply.data = (char*)buf;
  593. } else {
  594. for(;;){
  595. /* look for an answer at the right offset */
  596. toff = 0;
  597. for(i = 0; mf->reply[i] && i < mf->nreply; i++){
  598. n = mf->replylen[i];
  599. if(off < toff + n)
  600. break;
  601. toff += n;
  602. }
  603. if(i < mf->nreply)
  604. break; /* got something to return */
  605. /* try looking up more answers */
  606. if(lookup(mf) == 0){
  607. /* no more */
  608. n = 0;
  609. goto send;
  610. }
  611. }
  612. /* give back a single reply (or part of one) */
  613. job->reply.data = mf->reply[i] + (off - toff);
  614. if(cnt > toff - off + n)
  615. n = toff - off + n;
  616. else
  617. n = cnt;
  618. }
  619. send:
  620. job->reply.count = n;
  621. sendmsg(job, err);
  622. }
  623. void
  624. cleanmf(Mfile *mf)
  625. {
  626. int i;
  627. if(mf->net != nil){
  628. free(mf->net);
  629. mf->net = nil;
  630. }
  631. if(mf->host != nil){
  632. free(mf->host);
  633. mf->host = nil;
  634. }
  635. if(mf->serv != nil){
  636. free(mf->serv);
  637. mf->serv = nil;
  638. }
  639. if(mf->rem != nil){
  640. free(mf->rem);
  641. mf->rem = nil;
  642. }
  643. if(mf->user != nil){
  644. free(mf->user);
  645. mf->user = nil;
  646. }
  647. for(i = 0; i < mf->nreply; i++){
  648. free(mf->reply[i]);
  649. mf->reply[i] = nil;
  650. mf->replylen[i] = 0;
  651. }
  652. mf->nreply = 0;
  653. mf->nextnet = netlist;
  654. }
  655. void
  656. rwrite(Job *job, Mfile *mf)
  657. {
  658. int cnt, n;
  659. char *err;
  660. char *field[4];
  661. char curerr[64];
  662. err = 0;
  663. cnt = job->request.count;
  664. if(mf->qid.type & QTDIR){
  665. err = "can't write directory";
  666. goto send;
  667. }
  668. if(cnt >= Maxrequest){
  669. err = "request too long";
  670. goto send;
  671. }
  672. job->request.data[cnt] = 0;
  673. /*
  674. * toggle debugging
  675. */
  676. if(strncmp(job->request.data, "debug", 5)==0){
  677. debug ^= 1;
  678. syslog(1, logfile, "debug %d", debug);
  679. goto send;
  680. }
  681. /*
  682. * toggle debugging
  683. */
  684. if(strncmp(job->request.data, "paranoia", 8)==0){
  685. paranoia ^= 1;
  686. syslog(1, logfile, "paranoia %d", paranoia);
  687. goto send;
  688. }
  689. /*
  690. * add networks to the default list
  691. */
  692. if(strncmp(job->request.data, "add ", 4)==0){
  693. if(job->request.data[cnt-1] == '\n')
  694. job->request.data[cnt-1] = 0;
  695. netadd(job->request.data+4);
  696. readipinterfaces();
  697. goto send;
  698. }
  699. /*
  700. * refresh all state
  701. */
  702. if(strncmp(job->request.data, "refresh", 7)==0){
  703. netinit(1);
  704. goto send;
  705. }
  706. /* start transaction with a clean slate */
  707. cleanmf(mf);
  708. /*
  709. * look for a general query
  710. */
  711. if(*job->request.data == '!'){
  712. err = genquery(mf, job->request.data+1);
  713. goto send;
  714. }
  715. if(debug)
  716. syslog(0, logfile, "write %s", job->request.data);
  717. if(paranoia)
  718. syslog(0, paranoiafile, "write %s by %s", job->request.data, mf->user);
  719. /*
  720. * break up name
  721. */
  722. n = getfields(job->request.data, field, 4, 1, "!");
  723. switch(n){
  724. case 1:
  725. mf->net = strdup("net");
  726. mf->host = strdup(field[0]);
  727. break;
  728. case 4:
  729. mf->rem = strdup(field[3]);
  730. /* fall through */
  731. case 3:
  732. mf->serv = strdup(field[2]);
  733. /* fall through */
  734. case 2:
  735. mf->host = strdup(field[1]);
  736. mf->net = strdup(field[0]);
  737. break;
  738. }
  739. /*
  740. * do the first net worth of lookup
  741. */
  742. if(lookup(mf) == 0){
  743. rerrstr(curerr, sizeof curerr);
  744. err = curerr;
  745. }
  746. send:
  747. job->reply.count = cnt;
  748. sendmsg(job, err);
  749. }
  750. void
  751. rclunk(Job *job, Mfile *mf)
  752. {
  753. cleanmf(mf);
  754. mf->busy = 0;
  755. mf->fid = 0;
  756. sendmsg(job, 0);
  757. }
  758. void
  759. rremove(Job *job, Mfile *mf)
  760. {
  761. USED(mf);
  762. sendmsg(job, "remove permission denied");
  763. }
  764. void
  765. rstat(Job *job, Mfile *mf)
  766. {
  767. Dir dir;
  768. uchar buf[IOHDRSZ+Maxfdata];
  769. if(mf->qid.type & QTDIR){
  770. dir.name = ".";
  771. dir.mode = DMDIR|0555;
  772. } else {
  773. dir.name = "cs";
  774. dir.mode = 0666;
  775. }
  776. dir.qid = mf->qid;
  777. dir.length = 0;
  778. dir.uid = mf->user;
  779. dir.gid = mf->user;
  780. dir.muid = mf->user;
  781. dir.atime = dir.mtime = time(0);
  782. job->reply.nstat = convD2M(&dir, buf, sizeof buf);
  783. job->reply.stat = buf;
  784. sendmsg(job, 0);
  785. }
  786. void
  787. rwstat(Job *job, Mfile *mf)
  788. {
  789. USED(mf);
  790. sendmsg(job, "wstat permission denied");
  791. }
  792. void
  793. sendmsg(Job *job, char *err)
  794. {
  795. int n;
  796. uchar mdata[IOHDRSZ + Maxfdata];
  797. char ename[ERRMAX];
  798. if(err){
  799. job->reply.type = Rerror;
  800. snprint(ename, sizeof(ename), "cs: %s", err);
  801. job->reply.ename = ename;
  802. }else{
  803. job->reply.type = job->request.type+1;
  804. }
  805. job->reply.tag = job->request.tag;
  806. n = convS2M(&job->reply, mdata, sizeof mdata);
  807. if(n == 0){
  808. syslog(1, logfile, "sendmsg convS2M of %F returns 0", &job->reply);
  809. abort();
  810. }
  811. lock(&joblock);
  812. if(job->flushed == 0)
  813. if(write(mfd[1], mdata, n)!=n)
  814. error("mount write");
  815. unlock(&joblock);
  816. if(debug)
  817. syslog(0, logfile, "%F %d", &job->reply, n);
  818. }
  819. void
  820. error(char *s)
  821. {
  822. syslog(1, "cs", "%s: %r", s);
  823. _exits(0);
  824. }
  825. static int
  826. isvalidip(uchar *ip)
  827. {
  828. return ipcmp(ip, IPnoaddr) != 0 && ipcmp(ip, v4prefix) != 0;
  829. }
  830. static uchar loopbacknet[IPaddrlen] = {
  831. 0, 0, 0, 0,
  832. 0, 0, 0, 0,
  833. 0, 0, 0xff, 0xff,
  834. 127, 0, 0, 0
  835. };
  836. static uchar loopbackmask[IPaddrlen] = {
  837. 0xff, 0xff, 0xff, 0xff,
  838. 0xff, 0xff, 0xff, 0xff,
  839. 0xff, 0xff, 0xff, 0xff,
  840. 0xff, 0, 0, 0
  841. };
  842. void
  843. readipinterfaces(void)
  844. {
  845. if(myipaddr(ipa, mntpt) != 0)
  846. ipmove(ipa, IPnoaddr);
  847. sprint(ipaddr, "%I", ipa);
  848. if (debug)
  849. syslog(0, "dns", "ipaddr is %s\n", ipaddr);
  850. }
  851. /*
  852. * get the system name
  853. */
  854. void
  855. ipid(void)
  856. {
  857. uchar addr[6];
  858. Ndbtuple *t;
  859. char *p, *attr;
  860. Ndbs s;
  861. int f;
  862. char buf[Maxpath];
  863. /* use environment, ether addr, or ipaddr to get system name */
  864. if(*mysysname == 0){
  865. /*
  866. * environment has priority.
  867. *
  868. * on the sgi power the default system name
  869. * is the ip address. ignore that.
  870. *
  871. */
  872. p = getenv("sysname");
  873. if(p){
  874. attr = ipattr(p);
  875. if(strcmp(attr, "ip") != 0)
  876. strcpy(mysysname, p);
  877. }
  878. /*
  879. * the /net/ndb contains what the network
  880. * figured out from DHCP. use that name if
  881. * there is one.
  882. */
  883. if(*mysysname == 0 && netdb != nil){
  884. ndbreopen(netdb);
  885. for(t = ndbparse(netdb); t != nil; t = t->entry){
  886. if(strcmp(t->attr, "sys") == 0){
  887. strcpy(mysysname, t->val);
  888. break;
  889. }
  890. }
  891. ndbfree(t);
  892. }
  893. /* next network database, ip address, and ether address to find a name */
  894. if(*mysysname == 0){
  895. t = nil;
  896. if(isvalidip(ipa))
  897. t = ndbgetval(db, &s, "ip", ipaddr, "sys", mysysname);
  898. else {
  899. for(f = 0; f < 3; f++){
  900. snprint(buf, sizeof buf, "%s/ether%d", mntpt, f);
  901. if(myetheraddr(addr, buf) >= 0){
  902. snprint(eaddr, sizeof(eaddr), "%E", addr);
  903. t = ndbgetval(db, &s, "ether", eaddr, "sys",
  904. mysysname);
  905. if(t != nil)
  906. break;
  907. }
  908. }
  909. }
  910. ndbfree(t);
  911. }
  912. /* nothing else worked, use the ip address */
  913. if(*mysysname == 0 && isvalidip(ipa))
  914. strcpy(mysysname, ipaddr);
  915. /* set /dev/sysname if we now know it */
  916. if(*mysysname){
  917. f = open("/dev/sysname", OWRITE);
  918. if(f >= 0){
  919. write(f, mysysname, strlen(mysysname));
  920. close(f);
  921. }
  922. }
  923. }
  924. }
  925. /*
  926. * Set up a list of default networks by looking for
  927. * /net/ * /clone.
  928. */
  929. void
  930. netinit(int background)
  931. {
  932. char clone[Maxpath];
  933. Network *np;
  934. static int working;
  935. if(background){
  936. switch(rfork(RFPROC|RFNOTEG|RFMEM|RFNOWAIT)){
  937. case 0:
  938. break;
  939. default:
  940. return;
  941. }
  942. lock(&netlock);
  943. }
  944. /* add the mounted networks to the default list */
  945. for(np = network; np->net; np++){
  946. if(np->considered)
  947. continue;
  948. snprint(clone, sizeof(clone), "%s/%s/clone", mntpt, np->net);
  949. if(access(clone, AEXIST) < 0)
  950. continue;
  951. if(netlist)
  952. last->next = np;
  953. else
  954. netlist = np;
  955. last = np;
  956. np->next = 0;
  957. np->considered = 1;
  958. }
  959. /* find out what our ip address is */
  960. readipinterfaces();
  961. /* set the system name if we need to, these says ip is all we have */
  962. ipid();
  963. if(debug)
  964. syslog(0, logfile, "mysysname %s eaddr %s ipaddr %s ipa %I\n",
  965. mysysname, eaddr, ipaddr, ipa);
  966. if(background){
  967. unlock(&netlock);
  968. _exits(0);
  969. }
  970. }
  971. /*
  972. * add networks to the standard list
  973. */
  974. void
  975. netadd(char *p)
  976. {
  977. Network *np;
  978. char *field[12];
  979. int i, n;
  980. n = getfields(p, field, 12, 1, " ");
  981. for(i = 0; i < n; i++){
  982. for(np = network; np->net; np++){
  983. if(strcmp(field[i], np->net) != 0)
  984. continue;
  985. if(np->considered)
  986. break;
  987. if(netlist)
  988. last->next = np;
  989. else
  990. netlist = np;
  991. last = np;
  992. np->next = 0;
  993. np->considered = 1;
  994. }
  995. }
  996. }
  997. /*
  998. * make a tuple
  999. */
  1000. Ndbtuple*
  1001. mktuple(char *attr, char *val)
  1002. {
  1003. Ndbtuple *t;
  1004. t = emalloc(sizeof(Ndbtuple));
  1005. strcpy(t->attr, attr);
  1006. strncpy(t->val, val, sizeof(t->val));
  1007. t->val[sizeof(t->val)-1] = 0;
  1008. t->line = t;
  1009. t->entry = 0;
  1010. return t;
  1011. }
  1012. int
  1013. lookforproto(Ndbtuple *t, char *proto)
  1014. {
  1015. for(; t != nil; t = t->entry)
  1016. if(strcmp(t->attr, "proto") == 0 && strcmp(t->val, proto) == 0)
  1017. return 1;
  1018. return 0;
  1019. }
  1020. /*
  1021. * lookup a request. the network "net" means we should pick the
  1022. * best network to get there.
  1023. */
  1024. int
  1025. lookup(Mfile *mf)
  1026. {
  1027. Network *np;
  1028. char *cp;
  1029. Ndbtuple *nt, *t;
  1030. char reply[Maxreply];
  1031. int i, rv;
  1032. int hack;
  1033. /* open up the standard db files */
  1034. if(db == 0)
  1035. ndbinit();
  1036. if(db == 0)
  1037. error("can't open mf->network database\n");
  1038. rv = 0;
  1039. if(mf->net == nil)
  1040. return 0; /* must have been a genquery */
  1041. if(strcmp(mf->net, "net") == 0){
  1042. /*
  1043. * go through set of default nets
  1044. */
  1045. for(np = mf->nextnet; np; np = np->next){
  1046. nt = (*np->lookup)(np, mf->host, mf->serv, 1);
  1047. if(nt == nil)
  1048. continue;
  1049. hack = np->fasttimeouthack && !lookforproto(nt, np->net);
  1050. for(t = nt; mf->nreply < Nreply && t; t = t->entry){
  1051. cp = (*np->trans)(t, np, mf->serv, mf->rem, hack);
  1052. if(cp){
  1053. /* avoid duplicates */
  1054. for(i = 0; i < mf->nreply; i++)
  1055. if(strcmp(mf->reply[i], cp) == 0)
  1056. break;
  1057. if(i == mf->nreply){
  1058. /* save the reply */
  1059. mf->replylen[mf->nreply] = strlen(cp);
  1060. mf->reply[mf->nreply++] = cp;
  1061. rv++;
  1062. }
  1063. }
  1064. }
  1065. ndbfree(nt);
  1066. np = np->next;
  1067. break;
  1068. }
  1069. mf->nextnet = np;
  1070. return rv;
  1071. }
  1072. /*
  1073. * if not /net, we only get one lookup
  1074. */
  1075. if(mf->nreply != 0)
  1076. return 0;
  1077. /*
  1078. * look for a specific network
  1079. */
  1080. for(np = netlist; np->net != nil; np++){
  1081. if(np->fasttimeouthack)
  1082. continue;
  1083. if(strcmp(np->net, mf->net) == 0)
  1084. break;
  1085. }
  1086. if(np->net != nil){
  1087. /*
  1088. * known network
  1089. */
  1090. nt = (*np->lookup)(np, mf->host, mf->serv, 1);
  1091. for(t = nt; mf->nreply < Nreply && t; t = t->entry){
  1092. cp = (*np->trans)(t, np, mf->serv, mf->rem, 0);
  1093. if(cp){
  1094. mf->replylen[mf->nreply] = strlen(cp);
  1095. mf->reply[mf->nreply++] = cp;
  1096. rv++;
  1097. }
  1098. }
  1099. ndbfree(nt);
  1100. return rv;
  1101. } else {
  1102. /*
  1103. * not a known network, don't translate host or service
  1104. */
  1105. if(mf->serv)
  1106. snprint(reply, sizeof(reply), "%s/%s/clone %s!%s",
  1107. mntpt, mf->net, mf->host, mf->serv);
  1108. else
  1109. snprint(reply, sizeof(reply), "%s/%s/clone %s",
  1110. mntpt, mf->net, mf->host);
  1111. mf->reply[0] = strdup(reply);
  1112. mf->replylen[0] = strlen(reply);
  1113. mf->nreply = 1;
  1114. return 1;
  1115. }
  1116. }
  1117. /*
  1118. * translate an ip service name into a port number. If it's a numeric port
  1119. * number, look for restricted access.
  1120. *
  1121. * the service '*' needs no translation.
  1122. */
  1123. char*
  1124. ipserv(Network *np, char *name, char *buf)
  1125. {
  1126. char *p;
  1127. int alpha = 0;
  1128. int restr = 0;
  1129. char port[Ndbvlen];
  1130. Ndbtuple *t, *nt;
  1131. Ndbs s;
  1132. /* '*' means any service */
  1133. if(strcmp(name, "*")==0){
  1134. strcpy(buf, name);
  1135. return buf;
  1136. }
  1137. /* see if it's numeric or symbolic */
  1138. port[0] = 0;
  1139. for(p = name; *p; p++){
  1140. if(isdigit(*p))
  1141. {}
  1142. else if(isalpha(*p) || *p == '-' || *p == '$')
  1143. alpha = 1;
  1144. else
  1145. return 0;
  1146. }
  1147. if(alpha){
  1148. t = ndbgetval(db, &s, np->net, name, "port", port);
  1149. if(t == 0)
  1150. return 0;
  1151. } else {
  1152. t = ndbgetval(db, &s, "port", name, "port", port);
  1153. if(t == 0){
  1154. strncpy(port, name, sizeof(port));
  1155. port[sizeof(port)-1] = 0;
  1156. }
  1157. }
  1158. if(t){
  1159. for(nt = t; nt; nt = nt->entry)
  1160. if(strcmp(nt->attr, "restricted") == 0)
  1161. restr = 1;
  1162. ndbfree(t);
  1163. }
  1164. sprint(buf, "%s%s", port, restr ? "!r" : "");
  1165. return buf;
  1166. }
  1167. /*
  1168. * lookup an ip attribute
  1169. */
  1170. int
  1171. ipattrlookup(Ndb *db, char *ipa, char *attr, char *val)
  1172. {
  1173. Ndbtuple *t, *nt;
  1174. char *alist[2];
  1175. alist[0] = attr;
  1176. t = ndbipinfo(db, "ip", ipa, alist, 1);
  1177. if(t == nil)
  1178. return 0;
  1179. for(nt = t; nt != nil; nt = nt->entry){
  1180. if(strcmp(nt->attr, attr) == 0){
  1181. strcpy(val, nt->val);
  1182. ndbfree(t);
  1183. return 1;
  1184. }
  1185. }
  1186. /* we shouldn't get here */
  1187. ndbfree(t);
  1188. return 0;
  1189. }
  1190. /*
  1191. * lookup (and translate) an ip destination
  1192. */
  1193. Ndbtuple*
  1194. iplookup(Network *np, char *host, char *serv, int nolookup)
  1195. {
  1196. char *attr;
  1197. Ndbtuple *t, *nt;
  1198. Ndbs s;
  1199. char ts[Ndbvlen+1];
  1200. char th[Ndbvlen+1];
  1201. char dollar[Ndbvlen+1];
  1202. uchar ip[IPaddrlen];
  1203. uchar net[IPaddrlen];
  1204. uchar tnet[IPaddrlen];
  1205. Ipifc *ifc;
  1206. Iplifc *lifc;
  1207. USED(nolookup);
  1208. /*
  1209. * start with the service since it's the most likely to fail
  1210. * and costs the least
  1211. */
  1212. werrstr("can't translate address");
  1213. if(serv==0 || ipserv(np, serv, ts) == 0){
  1214. werrstr("can't translate service");
  1215. return 0;
  1216. }
  1217. /* for dial strings with no host */
  1218. if(strcmp(host, "*") == 0)
  1219. return mktuple("ip", "*");
  1220. /*
  1221. * hack till we go v6 :: = 0.0.0.0
  1222. */
  1223. if(strcmp("::", host) == 0)
  1224. return mktuple("ip", "*");
  1225. /*
  1226. * '$' means the rest of the name is an attribute that we
  1227. * need to search for
  1228. */
  1229. if(*host == '$'){
  1230. if(ipattrlookup(db, ipaddr, host+1, dollar))
  1231. host = dollar;
  1232. }
  1233. /*
  1234. * turn '[ip address]' into just 'ip address'
  1235. */
  1236. if(*host == '[' && host[strlen(host)-1] == ']'){
  1237. host++;
  1238. host[strlen(host)-1] = 0;
  1239. }
  1240. /*
  1241. * just accept addresses
  1242. */
  1243. attr = ipattr(host);
  1244. if(strcmp(attr, "ip") == 0)
  1245. return mktuple("ip", host);
  1246. /*
  1247. * give the domain name server the first opportunity to
  1248. * resolve domain names. if that fails try the database.
  1249. */
  1250. t = 0;
  1251. werrstr("can't translate address");
  1252. if(strcmp(attr, "dom") == 0)
  1253. t = dnsiplookup(host, &s);
  1254. if(t == 0)
  1255. t = ndbgetval(db, &s, attr, host, "ip", th);
  1256. if(t == 0)
  1257. t = dnsiplookup(host, &s);
  1258. if(t == 0 && strcmp(attr, "dom") != 0)
  1259. t = dnsiplookup(host, &s);
  1260. if(t == 0)
  1261. return 0;
  1262. /*
  1263. * reorder the tuple to have the matched line first and
  1264. * save that in the request structure.
  1265. */
  1266. t = reorder(t, s.t);
  1267. /*
  1268. * reorder according to our interfaces
  1269. */
  1270. lock(&ipifclock);
  1271. for(ifc = ipifcs; ifc != nil; ifc = ifc->next){
  1272. for(lifc = ifc->lifc; lifc != nil; lifc = lifc->next){
  1273. maskip(lifc->ip, lifc->mask, net);
  1274. for(nt = t; nt; nt = nt->entry){
  1275. if(strcmp(nt->attr, "ip") != 0)
  1276. continue;
  1277. parseip(ip, nt->val);
  1278. maskip(ip, lifc->mask, tnet);
  1279. if(memcmp(net, tnet, IPaddrlen) == 0){
  1280. t = reorder(t, nt);
  1281. unlock(&ipifclock);
  1282. return t;
  1283. }
  1284. }
  1285. }
  1286. }
  1287. unlock(&ipifclock);
  1288. return t;
  1289. }
  1290. /*
  1291. * translate an ip address
  1292. */
  1293. char*
  1294. iptrans(Ndbtuple *t, Network *np, char *serv, char *rem, int hack)
  1295. {
  1296. char ts[Ndbvlen+1];
  1297. char reply[Maxreply];
  1298. char x[Ndbvlen+1];
  1299. if(strcmp(t->attr, "ip") != 0)
  1300. return 0;
  1301. if(serv == 0 || ipserv(np, serv, ts) == 0){
  1302. werrstr("can't translate service");
  1303. return 0;
  1304. }
  1305. if(rem != nil)
  1306. snprint(x, sizeof(x), "!%s", rem);
  1307. else
  1308. *x = 0;
  1309. if(*t->val == '*')
  1310. snprint(reply, sizeof(reply), "%s/%s/clone %s%s",
  1311. mntpt, np->net, ts, x);
  1312. else
  1313. snprint(reply, sizeof(reply), "%s/%s/clone %s!%s%s%s",
  1314. mntpt, np->net, t->val, ts, x, hack?"!fasttimeout":"");
  1315. return strdup(reply);
  1316. }
  1317. /*
  1318. * lookup a telephone number
  1319. */
  1320. Ndbtuple*
  1321. telcolookup(Network *np, char *host, char *serv, int nolookup)
  1322. {
  1323. Ndbtuple *t;
  1324. Ndbs s;
  1325. char th[Ndbvlen+1];
  1326. USED(np, nolookup, serv);
  1327. werrstr("can't translate address");
  1328. t = ndbgetval(db, &s, "sys", host, "telco", th);
  1329. if(t == 0)
  1330. return mktuple("telco", host);
  1331. return reorder(t, s.t);
  1332. }
  1333. /*
  1334. * translate a telephone address
  1335. */
  1336. char*
  1337. telcotrans(Ndbtuple *t, Network *np, char *serv, char *rem, int)
  1338. {
  1339. char reply[Maxreply];
  1340. char x[Ndbvlen+1];
  1341. if(strcmp(t->attr, "telco") != 0)
  1342. return 0;
  1343. if(rem != nil)
  1344. snprint(x, sizeof(x), "!%s", rem);
  1345. else
  1346. *x = 0;
  1347. if(serv)
  1348. snprint(reply, sizeof(reply), "%s/%s/clone %s!%s%s", mntpt, np->net,
  1349. t->val, serv, x);
  1350. else
  1351. snprint(reply, sizeof(reply), "%s/%s/clone %s%s", mntpt, np->net,
  1352. t->val, x);
  1353. return strdup(reply);
  1354. }
  1355. /*
  1356. * reorder the tuple to put x's line first in the entry
  1357. */
  1358. Ndbtuple*
  1359. reorder(Ndbtuple *t, Ndbtuple *x)
  1360. {
  1361. Ndbtuple *nt;
  1362. Ndbtuple *line;
  1363. /* find start of this entry's line */
  1364. for(line = x; line->entry == line->line; line = line->line)
  1365. ;
  1366. line = line->line;
  1367. if(line == t)
  1368. return t; /* already the first line */
  1369. /* remove this line and everything after it from the entry */
  1370. for(nt = t; nt->entry != line; nt = nt->entry)
  1371. ;
  1372. nt->entry = 0;
  1373. /* make that the start of the entry */
  1374. for(nt = line; nt->entry; nt = nt->entry)
  1375. ;
  1376. nt->entry = t;
  1377. return line;
  1378. }
  1379. /*
  1380. * create a slave process to handle a request to avoid one request blocking
  1381. * another. parent returns to job loop.
  1382. */
  1383. void
  1384. slave(void)
  1385. {
  1386. if(*isslave)
  1387. return; /* we're already a slave process */
  1388. switch(rfork(RFPROC|RFNOTEG|RFMEM|RFNOWAIT)){
  1389. case -1:
  1390. break;
  1391. case 0:
  1392. if(debug)
  1393. syslog(0, logfile, "slave %d", getpid());
  1394. *isslave = 1;
  1395. break;
  1396. default:
  1397. longjmp(masterjmp, 1);
  1398. }
  1399. }
  1400. /*
  1401. * call the dns process and have it try to translate a name
  1402. */
  1403. Ndbtuple*
  1404. dnsiplookup(char *host, Ndbs *s)
  1405. {
  1406. char buf[Ndbvlen + 4];
  1407. Ndbtuple *t;
  1408. unlock(&dblock);
  1409. /* save the name before starting a slave */
  1410. snprint(buf, sizeof(buf), "%s", host);
  1411. slave();
  1412. if(strcmp(ipattr(buf), "ip") == 0)
  1413. t = dnsquery(mntpt, buf, "ptr");
  1414. else
  1415. t = dnsquery(mntpt, buf, "ip");
  1416. s->t = t;
  1417. if(t == nil){
  1418. rerrstr(buf, sizeof buf);
  1419. if(strstr(buf, "exist"))
  1420. werrstr("can't translate address: %s", buf);
  1421. else if(strstr(buf, "dns failure"))
  1422. werrstr("temporary problem: %s", buf);
  1423. }
  1424. lock(&dblock);
  1425. return t;
  1426. }
  1427. int
  1428. qmatch(Ndbtuple *t, char **attr, char **val, int n)
  1429. {
  1430. int i, found;
  1431. Ndbtuple *nt;
  1432. for(i = 1; i < n; i++){
  1433. found = 0;
  1434. for(nt = t; nt; nt = nt->entry)
  1435. if(strcmp(attr[i], nt->attr) == 0)
  1436. if(strcmp(val[i], "*") == 0
  1437. || strcmp(val[i], nt->val) == 0){
  1438. found = 1;
  1439. break;
  1440. }
  1441. if(found == 0)
  1442. break;
  1443. }
  1444. return i == n;
  1445. }
  1446. void
  1447. qreply(Mfile *mf, Ndbtuple *t)
  1448. {
  1449. int i;
  1450. Ndbtuple *nt;
  1451. char buf[2048];
  1452. buf[0] = 0;
  1453. for(nt = t; mf->nreply < Nreply && nt; nt = nt->entry){
  1454. strcat(buf, nt->attr);
  1455. strcat(buf, "=");
  1456. strcat(buf, nt->val);
  1457. i = strlen(buf);
  1458. if(nt->line != nt->entry || sizeof(buf) - i < 2*Ndbvlen+2){
  1459. mf->replylen[mf->nreply] = strlen(buf);
  1460. mf->reply[mf->nreply++] = strdup(buf);
  1461. buf[0] = 0;
  1462. } else
  1463. strcat(buf, " ");
  1464. }
  1465. }
  1466. enum
  1467. {
  1468. Maxattr= 32,
  1469. };
  1470. /*
  1471. * generic query lookup. The query is of one of the following
  1472. * forms:
  1473. *
  1474. * attr1=val1 attr2=val2 attr3=val3 ...
  1475. *
  1476. * returns the matching tuple
  1477. *
  1478. * ipinfo attr=val attr1 attr2 attr3 ...
  1479. *
  1480. * is like ipinfo and returns the attr{1-n}
  1481. * associated with the ip address.
  1482. */
  1483. char*
  1484. genquery(Mfile *mf, char *query)
  1485. {
  1486. int i, n;
  1487. char *p;
  1488. char *attr[Maxattr];
  1489. char *val[Maxattr];
  1490. Ndbtuple *t;
  1491. Ndbs s;
  1492. n = getfields(query, attr, 32, 1, " ");
  1493. if(n == 0)
  1494. return "bad query";
  1495. if(strcmp(attr[0], "ipinfo") == 0)
  1496. return ipinfoquery(mf, attr, n);
  1497. /* parse pairs */
  1498. for(i = 0; i < n; i++){
  1499. p = strchr(attr[i], '=');
  1500. if(p == 0)
  1501. return "bad query";
  1502. *p++ = 0;
  1503. val[i] = p;
  1504. }
  1505. /* give dns a chance */
  1506. if((strcmp(attr[0], "dom") == 0 || strcmp(attr[0], "ip") == 0) && val[0]){
  1507. t = dnsiplookup(val[0], &s);
  1508. if(t){
  1509. if(qmatch(t, attr, val, n)){
  1510. qreply(mf, t);
  1511. ndbfree(t);
  1512. return 0;
  1513. }
  1514. ndbfree(t);
  1515. }
  1516. }
  1517. /* first pair is always the key. It can't be a '*' */
  1518. t = ndbsearch(db, &s, attr[0], val[0]);
  1519. /* search is the and of all the pairs */
  1520. while(t){
  1521. if(qmatch(t, attr, val, n)){
  1522. qreply(mf, t);
  1523. ndbfree(t);
  1524. return 0;
  1525. }
  1526. ndbfree(t);
  1527. t = ndbsnext(&s, attr[0], val[0]);
  1528. }
  1529. return "no match";
  1530. }
  1531. /*
  1532. * resolve an ip address
  1533. */
  1534. static Ndbtuple*
  1535. ipresolve(char *attr, char *host)
  1536. {
  1537. Ndbtuple *t, *nt, **l;
  1538. t = iplookup(&network[Ntcp], host, "*", 0);
  1539. for(l = &t; *l != nil; ){
  1540. nt = *l;
  1541. if(strcmp(nt->attr, "ip") != 0){
  1542. *l = nt->entry;
  1543. nt->entry = nil;
  1544. ndbfree(nt);
  1545. continue;
  1546. }
  1547. strcpy(nt->attr, attr);
  1548. l = &nt->entry;
  1549. }
  1550. return t;
  1551. }
  1552. char*
  1553. ipinfoquery(Mfile *mf, char **list, int n)
  1554. {
  1555. int i, nresolve;
  1556. int resolve[Maxattr];
  1557. Ndbtuple *t, *nt, **l;
  1558. char *attr, *val;
  1559. /* skip 'ipinfo' */
  1560. list++; n--;
  1561. if(n < 2)
  1562. return "bad query";
  1563. /* get search attribute=value */
  1564. attr = *list++; n--;
  1565. val = strchr(attr, '=');
  1566. if(val == nil)
  1567. return "bad query";
  1568. *val++ = 0;
  1569. /*
  1570. * don't let ndbipinfo resolve the addresses, we're
  1571. * better at it.
  1572. */
  1573. nresolve = 0;
  1574. for(i = 0; i < n; i++)
  1575. if(*list[i] == '@'){
  1576. list[i]++;
  1577. resolve[i] = 1;
  1578. nresolve++;
  1579. } else
  1580. resolve[i] = 0;
  1581. t = ndbipinfo(db, attr, val, list, n);
  1582. if(t == nil)
  1583. return "no match";
  1584. if(nresolve != 0){
  1585. for(l = &t; *l != nil;){
  1586. nt = *l;
  1587. /* already an address? */
  1588. if(strcmp(ipattr(nt->val), "ip") == 0){
  1589. l = &(*l)->entry;
  1590. continue;
  1591. }
  1592. /* user wants it resolved? */
  1593. for(i = 0; i < n; i++)
  1594. if(strcmp(list[i], nt->attr) == 0)
  1595. break;
  1596. if(i >= n || resolve[i] == 0){
  1597. l = &(*l)->entry;
  1598. continue;
  1599. }
  1600. /* resolve address and replace entry */
  1601. *l = ipresolve(nt->attr, nt->val);
  1602. while(*l != nil)
  1603. l = &(*l)->entry;
  1604. *l = nt->entry;
  1605. nt->entry = nil;
  1606. ndbfree(nt);
  1607. }
  1608. }
  1609. /* make it all one line */
  1610. for(nt = t; nt != nil; nt = nt->entry){
  1611. if(nt->entry == nil)
  1612. nt->line = t;
  1613. else
  1614. nt->line = nt->entry;
  1615. }
  1616. qreply(mf, t);
  1617. return nil;
  1618. }
  1619. void*
  1620. emalloc(int size)
  1621. {
  1622. void *x;
  1623. x = malloc(size);
  1624. if(x == nil)
  1625. abort();
  1626. memset(x, 0, size);
  1627. return x;
  1628. }
  1629. char*
  1630. estrdup(char *s)
  1631. {
  1632. int size;
  1633. char *p;
  1634. size = strlen(s)+1;
  1635. p = malloc(size);
  1636. if(p == nil)
  1637. abort();
  1638. memmove(p, s, size);
  1639. return p;
  1640. }