proto.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <bio.h>
  4. #include <auth.h>
  5. #include <fcall.h>
  6. #include <disk.h>
  7. enum {
  8. LEN = 8*1024,
  9. HUNKS = 128,
  10. };
  11. typedef struct File File;
  12. struct File{
  13. char *new;
  14. char *elem;
  15. char *old;
  16. char *uid;
  17. char *gid;
  18. ulong mode;
  19. };
  20. typedef void Mkfserr(char*, void*);
  21. typedef void Mkfsenum(char*, char*, Dir*, void*);
  22. typedef struct Name Name;
  23. struct Name {
  24. int n;
  25. char *s;
  26. };
  27. typedef struct Mkaux Mkaux;
  28. struct Mkaux {
  29. Mkfserr *warn;
  30. Mkfsenum *mkenum;
  31. char *root;
  32. char *proto;
  33. jmp_buf jmp;
  34. Biobuf *b;
  35. Name oldfile;
  36. Name fullname;
  37. int lineno;
  38. int indent;
  39. void *a;
  40. };
  41. static void domkfs(Mkaux *mkaux, File *me, int level);
  42. static int copyfile(Mkaux*, File*, Dir*, int);
  43. static void freefile(File*);
  44. static File* getfile(Mkaux*, File*);
  45. static char* getmode(Mkaux*, char*, ulong*);
  46. static char* getname(Mkaux*, char*, char**);
  47. static char* getpath(Mkaux*, char*);
  48. static int mkfile(Mkaux*, File*);
  49. static char* mkpath(Mkaux*, char*, char*);
  50. static void mktree(Mkaux*, File*, int);
  51. static void setnames(Mkaux*, File*);
  52. static void skipdir(Mkaux*);
  53. static void warn(Mkaux*, char *, ...);
  54. //static void
  55. //mprint(char *new, char *old, Dir *d, void*)
  56. //{
  57. // print("%s %s %D\n", new, old, d);
  58. //}
  59. int
  60. rdproto(char *proto, char *root, Mkfsenum *mkenum, Mkfserr *mkerr, void *a)
  61. {
  62. Mkaux mx, *m;
  63. File file;
  64. int rv;
  65. m = &mx;
  66. memset(&mx, 0, sizeof mx);
  67. if(root == nil)
  68. root = "/";
  69. m->root = root;
  70. m->warn = mkerr;
  71. m->mkenum = mkenum;
  72. m->a = a;
  73. m->proto = proto;
  74. m->lineno = 0;
  75. m->indent = 0;
  76. if((m->b = Bopen(proto, OREAD)) == nil) {
  77. werrstr("open %q: %r", proto);
  78. return -1;
  79. }
  80. memset(&file, 0, sizeof file);
  81. file.new = "";
  82. file.old = nil;
  83. rv = 0;
  84. if(setjmp(m->jmp) == 0)
  85. domkfs(m, &file, -1);
  86. else
  87. rv = -1;
  88. free(m->oldfile.s);
  89. free(m->fullname.s);
  90. return rv;
  91. }
  92. static void*
  93. emalloc(Mkaux *mkaux, ulong n)
  94. {
  95. void *v;
  96. v = malloc(n);
  97. if(v == nil)
  98. longjmp(mkaux->jmp, 1); /* memory leak */
  99. memset(v, 0, n);
  100. return v;
  101. }
  102. static char*
  103. estrdup(Mkaux *mkaux, char *s)
  104. {
  105. s = strdup(s);
  106. if(s == nil)
  107. longjmp(mkaux->jmp, 1); /* memory leak */
  108. return s;
  109. }
  110. static void
  111. domkfs(Mkaux *mkaux, File *me, int level)
  112. {
  113. File *child;
  114. int rec;
  115. child = getfile(mkaux, me);
  116. if(!child)
  117. return;
  118. if((child->elem[0] == '+' || child->elem[0] == '*') && child->elem[1] == '\0'){
  119. rec = child->elem[0] == '+';
  120. free(child->new);
  121. child->new = estrdup(mkaux, me->new);
  122. setnames(mkaux, child);
  123. mktree(mkaux, child, rec);
  124. freefile(child);
  125. child = getfile(mkaux, me);
  126. }
  127. while(child && mkaux->indent > level){
  128. if(mkfile(mkaux, child))
  129. domkfs(mkaux, child, mkaux->indent);
  130. freefile(child);
  131. child = getfile(mkaux, me);
  132. }
  133. if(child){
  134. freefile(child);
  135. Bseek(mkaux->b, -Blinelen(mkaux->b), 1);
  136. mkaux->lineno--;
  137. }
  138. }
  139. static void
  140. mktree(Mkaux *mkaux, File *me, int rec)
  141. {
  142. File child;
  143. Dir *d;
  144. int i, n, fd;
  145. fd = open(mkaux->oldfile.s, OREAD);
  146. if(fd < 0){
  147. warn(mkaux, "can't open %q: %r", mkaux->oldfile.s);
  148. return;
  149. }
  150. child = *me;
  151. while((n = dirread(fd, &d)) > 0){
  152. for(i = 0; i < n; i++){
  153. child.new = mkpath(mkaux, me->new, d[i].name);
  154. if(me->old)
  155. child.old = mkpath(mkaux, me->old, d[i].name);
  156. child.elem = d[i].name;
  157. setnames(mkaux, &child);
  158. if(
  159. if(copyfile(mkaux, &child, &d[i], 1) && rec)
  160. mktree(mkaux, &child, rec);
  161. free(child.new);
  162. if(child.old)
  163. free(child.old);
  164. }
  165. }
  166. close(fd);
  167. }
  168. static int
  169. mkfile(Mkaux *mkaux, File *f)
  170. {
  171. Dir *d;
  172. if((d = dirstat(mkaux->oldfile.s)) == nil){
  173. warn(mkaux, "can't stat file %q: %r", mkaux->oldfile.s);
  174. skipdir(mkaux);
  175. return 0;
  176. }
  177. return copyfile(mkaux, f, d, 0);
  178. }
  179. enum {
  180. SLOP = 30
  181. };
  182. static void
  183. setname(Mkaux *mkaux, Name *name, char *s1, char *s2)
  184. {
  185. int l;
  186. l = strlen(s1)+strlen(s2)+1;
  187. if(name->n < l) {
  188. free(name->s);
  189. name->s = emalloc(mkaux, l+SLOP);
  190. name->n = l+SLOP;
  191. }
  192. snprint(name->s, name->n, "%s%s%s", s1, s1[0]==0 || s1[strlen(s1)-1]!='/' ? "/" : "", s2);
  193. }
  194. static int
  195. copyfile(Mkaux *mkaux, File *f, Dir *d, int permonly)
  196. {
  197. Dir *nd;
  198. ulong xmode;
  199. char *p;
  200. setname(mkaux, &mkaux->fullname, mkaux->root, f->old ? f->old : f->new);
  201. /*
  202. * Extra stat here is inefficient but accounts for binds.
  203. */
  204. if((nd = dirstat(mkaux->fullname.s)) != nil)
  205. d = nd;
  206. d->name = f->elem;
  207. if(d->type != 'M'){
  208. d->uid = "sys";
  209. d->gid = "sys";
  210. xmode = (d->mode >> 6) & 7;
  211. d->mode |= xmode | (xmode << 3);
  212. }
  213. if(strcmp(f->uid, "-") != 0)
  214. d->uid = f->uid;
  215. if(strcmp(f->gid, "-") != 0)
  216. d->gid = f->gid;
  217. if(f->mode != ~0){
  218. if(permonly)
  219. d->mode = (d->mode & ~0666) | (f->mode & 0666);
  220. else if((d->mode&DMDIR) != (f->mode&DMDIR))
  221. warn(mkaux, "inconsistent mode for %s", f->new);
  222. else
  223. d->mode = f->mode;
  224. }
  225. if(p = strrchr(f->new, '/'))
  226. d->name = p+1;
  227. else
  228. d->name = f->new;
  229. mkaux->mkenum(f->new, mkaux->fullname.s, d, mkaux->a);
  230. xmode = d->mode;
  231. free(nd);
  232. return (xmode&DMDIR) != 0;
  233. }
  234. static char *
  235. mkpath(Mkaux *mkaux, char *prefix, char *elem)
  236. {
  237. char *p;
  238. int n;
  239. n = strlen(prefix) + strlen(elem) + 2;
  240. p = emalloc(mkaux, n);
  241. strcpy(p, prefix);
  242. strcat(p, "/");
  243. strcat(p, elem);
  244. return p;
  245. }
  246. static void
  247. setnames(Mkaux *mkaux, File *f)
  248. {
  249. if(f->old){
  250. if(f->old[0] == '/')
  251. setname(mkaux, &mkaux->oldfile, f->old, "");
  252. else
  253. setname(mkaux, &mkaux->oldfile, mkaux->root, f->old);
  254. } else
  255. setname(mkaux, &mkaux->oldfile, mkaux->root, f->new);
  256. }
  257. static void
  258. freefile(File *f)
  259. {
  260. if(f->old)
  261. free(f->old);
  262. if(f->new)
  263. free(f->new);
  264. free(f);
  265. }
  266. /*
  267. * skip all files in the proto that
  268. * could be in the current dir
  269. */
  270. static void
  271. skipdir(Mkaux *mkaux)
  272. {
  273. char *p, c;
  274. int level;
  275. if(mkaux->indent < 0)
  276. return;
  277. level = mkaux->indent;
  278. for(;;){
  279. mkaux->indent = 0;
  280. p = Brdline(mkaux->b, '\n');
  281. mkaux->lineno++;
  282. if(!p){
  283. mkaux->indent = -1;
  284. return;
  285. }
  286. while((c = *p++) != '\n')
  287. if(c == ' ')
  288. mkaux->indent++;
  289. else if(c == '\t')
  290. mkaux->indent += 8;
  291. else
  292. break;
  293. if(mkaux->indent <= level){
  294. Bseek(mkaux->b, -Blinelen(mkaux->b), 1);
  295. mkaux->lineno--;
  296. return;
  297. }
  298. }
  299. }
  300. static File*
  301. getfile(Mkaux *mkaux, File *old)
  302. {
  303. File *f;
  304. char *elem;
  305. char *p;
  306. int c;
  307. if(mkaux->indent < 0)
  308. return 0;
  309. loop:
  310. mkaux->indent = 0;
  311. p = Brdline(mkaux->b, '\n');
  312. mkaux->lineno++;
  313. if(!p){
  314. mkaux->indent = -1;
  315. return 0;
  316. }
  317. while((c = *p++) != '\n')
  318. if(c == ' ')
  319. mkaux->indent++;
  320. else if(c == '\t')
  321. mkaux->indent += 8;
  322. else
  323. break;
  324. if(c == '\n' || c == '#')
  325. goto loop;
  326. p--;
  327. f = emalloc(mkaux, sizeof *f);
  328. p = getname(mkaux, p, &elem);
  329. if(p == nil)
  330. return nil;
  331. f->new = mkpath(mkaux, old->new, elem);
  332. free(elem);
  333. f->elem = utfrrune(f->new, L'/') + 1;
  334. p = getmode(mkaux, p, &f->mode);
  335. p = getname(mkaux, p, &f->uid); /* LEAK */
  336. if(p == nil)
  337. return nil;
  338. if(!*f->uid)
  339. f->uid = "-"; /* LEAK */
  340. p = getname(mkaux, p, &f->gid); /* LEAK */
  341. if(p == nil)
  342. return nil;
  343. if(!*f->gid)
  344. f->gid = "-": /* LEAK */
  345. f->old = getpath(mkaux, p);
  346. if(f->old && strcmp(f->old, "-") == 0){
  347. free(f->old);
  348. f->old = 0;
  349. }
  350. setnames(mkaux, f);
  351. return f;
  352. }
  353. static char*
  354. getpath(Mkaux *mkaux, char *p)
  355. {
  356. char *q, *new;
  357. int c, n;
  358. while((c = *p) == ' ' || c == '\t')
  359. p++;
  360. q = p;
  361. while((c = *q) != '\n' && c != ' ' && c != '\t')
  362. q++;
  363. if(q == p)
  364. return 0;
  365. n = q - p;
  366. new = emalloc(mkaux, n + 1);
  367. memcpy(new, p, n);
  368. new[n] = 0;
  369. return new;
  370. }
  371. static char*
  372. getname(Mkaux *mkaux, char *p, char **buf)
  373. {
  374. char *s, *start;
  375. int c;
  376. while((c = *p) == ' ' || c == '\t')
  377. p++;
  378. start = p;
  379. while((c = *p) != '\n' && c != ' ' && c != '\t')
  380. p++;
  381. *buf = malloc(p+1-start);
  382. if(*buf == nil)
  383. return nil;
  384. memmove(*buf, start, p-start);
  385. (*buf)[p-start] = '\0';
  386. if(**buf == '$'){
  387. s = getenv(*buf+1);
  388. if(s == 0){
  389. warn(mkaux, "can't read environment variable %s", *buf+1);
  390. skipdir(mkaux);
  391. free(*buf);
  392. return nil;
  393. }
  394. free(*buf);
  395. *buf = s;
  396. }
  397. return p;
  398. }
  399. static char*
  400. getmode(Mkaux *mkaux, char *p, ulong *xmode)
  401. {
  402. char *buf, *s;
  403. ulong m;
  404. *xmode = ~0;
  405. p = getname(mkaux, p, &buf);
  406. if(p == nil)
  407. return nil;
  408. s = buf;
  409. if(!*s || strcmp(s, "-") == 0)
  410. return p;
  411. m = 0;
  412. if(*s == 'd'){
  413. m |= DMDIR;
  414. s++;
  415. }
  416. if(*s == 'a'){
  417. m |= DMAPPEND;
  418. s++;
  419. }
  420. if(*s == 'l'){
  421. m |= DMEXCL;
  422. s++;
  423. }
  424. if(s[0] < '0' || s[0] > '7'
  425. || s[1] < '0' || s[1] > '7'
  426. || s[2] < '0' || s[2] > '7'
  427. || s[3]){
  428. warn(mkaux, "bad mode specification %s", buf);
  429. free(buf);
  430. return p;
  431. }
  432. *xmode = m | strtoul(s, 0, 8);
  433. free(buf);
  434. return p;
  435. }
  436. static void
  437. warn(Mkaux *mkaux, char *fmt, ...)
  438. {
  439. char buf[256];
  440. va_list va;
  441. va_start(va, fmt);
  442. vseprint(buf, buf+sizeof(buf), fmt, va);
  443. va_end(va);
  444. if(mkaux->warn)
  445. mkaux->warn(buf, mkaux->a);
  446. else
  447. fprint(2, "warning: %s\n", buf);
  448. }