Include spring-cleaning
[platform/upstream/rpm.git] / build / rpmfc.c
1 #include "system.h"
2
3 #include <signal.h>
4 #if HAVE_GELF_H
5 #include <gelf.h>
6
7 #if !defined(DT_GNU_HASH)
8 #define DT_GNU_HASH             0x6ffffef5
9 #endif
10
11 #endif
12
13 #include <rpm/rpmtag.h>
14 #include <rpm/rpmbuild.h>
15 #include <rpm/argv.h>
16 #include <rpm/rpmfc.h>
17 #include <rpm/rpmlog.h>
18 #include <rpm/rpmfileutil.h>
19
20 #include <rpm/rpmds.h>
21 #include <rpm/rpmfi.h>
22
23 #include "debug.h"
24
25 /**
26  */
27 struct rpmfc_s {
28     int nfiles;         /*!< no. of files */
29     int fknown;         /*!< no. of classified files */
30     int fwhite;         /*!< no. of "white" files */
31     int ix;             /*!< current file index */
32     int skipProv;       /*!< Don't auto-generate Provides:? */
33     int skipReq;        /*!< Don't auto-generate Requires:? */
34     int tracked;        /*!< Versioned Provides: tracking dependency added? */
35     size_t brlen;       /*!< strlen(spec->buildRoot) */
36
37     ARGV_t fn;          /*!< (no. files) file names */
38     ARGI_t fcolor;      /*!< (no. files) file colors */
39     ARGI_t fcdictx;     /*!< (no. files) file class dictionary indices */
40     ARGI_t fddictx;     /*!< (no. files) file depends dictionary start */
41     ARGI_t fddictn;     /*!< (no. files) file depends dictionary no. entries */
42     ARGV_t cdict;       /*!< (no. classes) file class dictionary */
43     ARGV_t ddict;       /*!< (no. dependencies) file depends dictionary */
44     ARGI_t ddictx;      /*!< (no. dependencies) file->dependency mapping */
45
46     rpmds provides;     /*!< (no. provides) package provides */
47     rpmds requires;     /*!< (no. requires) package requires */
48
49     StringBuf sb_java;  /*!< concatenated list of java colored files. */
50     StringBuf sb_perl;  /*!< concatenated list of perl colored files. */
51     StringBuf sb_python;/*!< concatenated list of python colored files. */
52
53 };
54
55 /**
56  */
57 struct rpmfcTokens_s {
58     const char * token;
59     int colors;
60 };  
61
62 /**
63  */
64 static int rpmfcExpandAppend(ARGV_t * argvp, const ARGV_t av)
65 {
66     ARGV_t argv = *argvp;
67     int argc = argvCount(argv);
68     int ac = argvCount(av);
69     int i;
70
71     argv = xrealloc(argv, (argc + ac + 1) * sizeof(*argv));
72     for (i = 0; i < ac; i++)
73         argv[argc + i] = rpmExpand(av[i], NULL);
74     argv[argc + ac] = NULL;
75     *argvp = argv;
76     return 0;
77 }
78
79 /** \ingroup rpmbuild
80  * Return output from helper script.
81  * @todo Use poll(2) rather than select(2), if available.
82  * @param dir           directory to run in (or NULL)
83  * @param argv          program and arguments to run
84  * @param writePtr      bytes to feed to script on stdin (or NULL)
85  * @param writeBytesLeft no. of bytes to feed to script on stdin
86  * @param failNonZero   is script failure an error?
87  * @return              buffered stdout from script, NULL on error
88  */     
89 static StringBuf getOutputFrom(const char * dir, ARGV_t argv,
90                         const char * writePtr, size_t writeBytesLeft,
91                         int failNonZero)
92 {
93     pid_t child, reaped;
94     int toProg[2];
95     int fromProg[2];
96     int status;
97     void *oldhandler;
98     StringBuf readBuff;
99     int done;
100
101     /* FIX: cast? */
102     oldhandler = signal(SIGPIPE, SIG_IGN);
103
104     toProg[0] = toProg[1] = 0;
105     (void) pipe(toProg);
106     fromProg[0] = fromProg[1] = 0;
107     (void) pipe(fromProg);
108     
109     if (!(child = fork())) {
110         (void) close(toProg[1]);
111         (void) close(fromProg[0]);
112         
113         (void) dup2(toProg[0], STDIN_FILENO);   /* Make stdin the in pipe */
114         (void) dup2(fromProg[1], STDOUT_FILENO); /* Make stdout the out pipe */
115
116         (void) close(toProg[0]);
117         (void) close(fromProg[1]);
118
119         if (dir) {
120             (void) chdir(dir);
121         }
122         
123         rpmlog(RPMLOG_DEBUG, _("\texecv(%s) pid %d\n"),
124                         argv[0], (unsigned)getpid());
125
126         unsetenv("MALLOC_CHECK_");
127         (void) execvp(argv[0], (char *const *)argv);
128         /* XXX this error message is probably not seen. */
129         rpmlog(RPMLOG_ERR, _("Couldn't exec %s: %s\n"),
130                 argv[0], strerror(errno));
131         _exit(EXIT_FAILURE);
132     }
133     if (child < 0) {
134         rpmlog(RPMLOG_ERR, _("Couldn't fork %s: %s\n"),
135                 argv[0], strerror(errno));
136         return NULL;
137     }
138
139     (void) close(toProg[0]);
140     (void) close(fromProg[1]);
141
142     /* Do not block reading or writing from/to prog. */
143     (void) fcntl(fromProg[0], F_SETFL, O_NONBLOCK);
144     (void) fcntl(toProg[1], F_SETFL, O_NONBLOCK);
145     
146     readBuff = newStringBuf();
147
148     do {
149         fd_set ibits, obits;
150         struct timeval tv;
151         int nfd, nbw, nbr;
152         int rc;
153
154         done = 0;
155 top:
156         FD_ZERO(&ibits);
157         FD_ZERO(&obits);
158         if (fromProg[0] >= 0) {
159             FD_SET(fromProg[0], &ibits);
160         }
161         if (toProg[1] >= 0) {
162             FD_SET(toProg[1], &obits);
163         }
164         /* XXX values set to limit spinning with perl doing ~100 forks/sec. */
165         tv.tv_sec = 0;
166         tv.tv_usec = 10000;
167         nfd = ((fromProg[0] > toProg[1]) ? fromProg[0] : toProg[1]);
168         if ((rc = select(nfd, &ibits, &obits, NULL, &tv)) < 0) {
169             if (errno == EINTR)
170                 goto top;
171             break;
172         }
173
174         /* Write any data to program */
175         if (toProg[1] >= 0 && FD_ISSET(toProg[1], &obits)) {
176           if (writePtr && writeBytesLeft > 0) {
177             if ((nbw = write(toProg[1], writePtr,
178                     (1024<writeBytesLeft) ? 1024 : writeBytesLeft)) < 0) {
179                 if (errno != EAGAIN) {
180                     perror("getOutputFrom()");
181                     exit(EXIT_FAILURE);
182                 }
183                 nbw = 0;
184             }
185             writeBytesLeft -= nbw;
186             writePtr += nbw;
187           } else if (toProg[1] >= 0) {  /* close write fd */
188             (void) close(toProg[1]);
189             toProg[1] = -1;
190           }
191         }
192         
193         /* Read any data from prog */
194         {   char buf[BUFSIZ+1];
195             while ((nbr = read(fromProg[0], buf, sizeof(buf)-1)) > 0) {
196                 buf[nbr] = '\0';
197                 appendStringBuf(readBuff, buf);
198             }
199         }
200
201         /* terminate on (non-blocking) EOF or error */
202         done = (nbr == 0 || (nbr < 0 && errno != EAGAIN));
203
204     } while (!done);
205
206     /* Clean up */
207     if (toProg[1] >= 0)
208         (void) close(toProg[1]);
209     if (fromProg[0] >= 0)
210         (void) close(fromProg[0]);
211     /* FIX: cast? */
212     (void) signal(SIGPIPE, oldhandler);
213
214     /* Collect status from prog */
215     reaped = waitpid(child, &status, 0);
216     rpmlog(RPMLOG_DEBUG, _("\twaitpid(%d) rc %d status %x\n"),
217         (unsigned)child, (unsigned)reaped, status);
218
219     if (failNonZero && (!WIFEXITED(status) || WEXITSTATUS(status))) {
220         rpmlog(RPMLOG_ERR, _("%s failed\n"), argv[0]);
221         return NULL;
222     }
223     if (writeBytesLeft) {
224         rpmlog(RPMLOG_ERR, _("failed to write all data to %s\n"), argv[0]);
225         return NULL;
226     }
227     return readBuff;
228 }
229
230 int rpmfcExec(ARGV_t av, StringBuf sb_stdin, StringBuf * sb_stdoutp,
231                 int failnonzero)
232 {
233     char * s = NULL;
234     ARGV_t xav = NULL;
235     ARGV_t pav = NULL;
236     int pac = 0;
237     int ec = -1;
238     StringBuf sb = NULL;
239     const char * buf_stdin = NULL;
240     size_t buf_stdin_len = 0;
241     int xx;
242
243     if (sb_stdoutp)
244         *sb_stdoutp = NULL;
245     if (!(av && *av))
246         goto exit;
247
248     /* Find path to executable with (possible) args. */
249     s = rpmExpand(av[0], NULL);
250     if (!(s && *s))
251         goto exit;
252
253     /* Parse args buried within expanded exacutable. */
254     pac = 0;
255     xx = poptParseArgvString(s, &pac, (const char ***)&pav);
256     if (!(xx == 0 && pac > 0 && pav != NULL))
257         goto exit;
258
259     /* Build argv, appending args to the executable args. */
260     xav = NULL;
261     xx = argvAppend(&xav, pav);
262     if (av[1])
263         xx = rpmfcExpandAppend(&xav, av + 1);
264
265     if (sb_stdin != NULL) {
266         buf_stdin = getStringBuf(sb_stdin);
267         buf_stdin_len = strlen(buf_stdin);
268     }
269
270     /* Read output from exec'd helper. */
271     sb = getOutputFrom(NULL, xav, buf_stdin, buf_stdin_len, failnonzero);
272
273     if (sb_stdoutp != NULL) {
274         *sb_stdoutp = sb;
275         sb = NULL;      /* XXX don't free */
276     }
277
278     ec = 0;
279
280 exit:
281     sb = freeStringBuf(sb);
282     xav = argvFree(xav);
283     pav = _free(pav);   /* XXX popt mallocs in single blob. */
284     s = _free(s);
285     return ec;
286 }
287
288 /**
289  */
290 static int rpmfcSaveArg(ARGV_t * argvp, const char * key)
291 {
292     int rc = 0;
293
294     if (argvSearch(*argvp, key, NULL) == NULL) {
295         rc = argvAdd(argvp, key);
296         rc = argvSort(*argvp, NULL);
297     }
298     return rc;
299 }
300
301 static char * rpmfcFileDep(char * buf, int ix,
302                 rpmds ds)
303 {
304     rpm_tag_t tagN = rpmdsTagN(ds);
305     char deptype = 'X';
306
307     buf[0] = '\0';
308     switch (tagN) {
309     case RPMTAG_PROVIDENAME:
310         deptype = 'P';
311         break;
312     case RPMTAG_REQUIRENAME:
313         deptype = 'R';
314         break;
315     }
316     if (ds != NULL)
317         sprintf(buf, "%08d%c %s %s 0x%08x", ix, deptype,
318                 rpmdsN(ds), rpmdsEVR(ds), rpmdsFlags(ds));
319     return buf;
320 };
321
322 /**
323  * Run per-interpreter dependency helper.
324  * @param fc            file classifier
325  * @param deptype       'P' == Provides:, 'R' == Requires:, helper
326  * @param nsdep         class name for interpreter (e.g. "perl")
327  * @return              0 on success
328  */
329 static int rpmfcHelper(rpmfc fc, unsigned char deptype, const char * nsdep)
330 {
331     const char * fn = fc->fn[fc->ix];
332     char buf[BUFSIZ];
333     StringBuf sb_stdout = NULL;
334     StringBuf sb_stdin;
335     const char *av[2];
336     rpmds * depsp, ds;
337     const char * N;
338     const char * EVR;
339     rpmsenseFlags Flags;
340     int32_t dsContext, tagN;
341     ARGV_t pav;
342     const char * s;
343     int pac;
344     int xx;
345     int i;
346
347     switch (deptype) {
348     default:
349         return -1;
350         break;
351     case 'P':
352         if (fc->skipProv)
353             return 0;
354         xx = snprintf(buf, sizeof(buf), "%%{?__%s_provides}", nsdep);
355         depsp = &fc->provides;
356         dsContext = RPMSENSE_FIND_PROVIDES;
357         tagN = RPMTAG_PROVIDENAME;
358         break;
359     case 'R':
360         if (fc->skipReq)
361             return 0;
362         xx = snprintf(buf, sizeof(buf), "%%{?__%s_requires}", nsdep);
363         depsp = &fc->requires;
364         dsContext = RPMSENSE_FIND_REQUIRES;
365         tagN = RPMTAG_REQUIRENAME;
366         break;
367     }
368     buf[sizeof(buf)-1] = '\0';
369     av[0] = buf;
370     av[1] = NULL;
371
372     sb_stdin = newStringBuf();
373     appendLineStringBuf(sb_stdin, fn);
374     sb_stdout = NULL;
375     xx = rpmfcExec(av, sb_stdin, &sb_stdout, 0);
376     sb_stdin = freeStringBuf(sb_stdin);
377
378     if (xx == 0 && sb_stdout != NULL) {
379         pav = NULL;
380         xx = argvSplit(&pav, getStringBuf(sb_stdout), " \t\n\r");
381         pac = argvCount(pav);
382         if (pav)
383         for (i = 0; i < pac; i++) {
384             N = pav[i];
385             EVR = "";
386             Flags = dsContext;
387             if (pav[i+1] && strchr("=<>", *pav[i+1])) {
388                 i++;
389                 for (s = pav[i]; *s; s++) {
390                     switch(*s) {
391                     default:
392 assert(*s != '\0');
393                         break;
394                     case '=':
395                         Flags |= RPMSENSE_EQUAL;
396                         break;
397                     case '<':
398                         Flags |= RPMSENSE_LESS;
399                         break;
400                     case '>':
401                         Flags |= RPMSENSE_GREATER;
402                         break;
403                     }
404                 }
405                 i++;
406                 EVR = pav[i];
407 assert(EVR != NULL);
408             }
409
410
411             /* Add tracking dependency for versioned Provides: */
412             if (!fc->tracked && deptype == 'P' && *EVR != '\0') {
413                 ds = rpmdsSingle(RPMTAG_REQUIRENAME,
414                         "rpmlib(VersionedDependencies)", "3.0.3-1",
415                         RPMSENSE_RPMLIB|(RPMSENSE_LESS|RPMSENSE_EQUAL));
416                 xx = rpmdsMerge(&fc->requires, ds);
417                 ds = rpmdsFree(ds);
418                 fc->tracked = 1;
419             }
420
421             ds = rpmdsSingle(tagN, N, EVR, Flags);
422
423             /* Add to package dependencies. */
424             xx = rpmdsMerge(depsp, ds);
425
426             /* Add to file dependencies. */
427             xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(buf, fc->ix, ds));
428
429             ds = rpmdsFree(ds);
430         }
431
432         pav = argvFree(pav);
433     }
434     sb_stdout = freeStringBuf(sb_stdout);
435
436     return 0;
437 }
438
439 /**
440  */
441 static struct rpmfcTokens_s rpmfcTokens[] = {
442   { "directory",                RPMFC_DIRECTORY|RPMFC_INCLUDE },
443
444   { " shared object",           RPMFC_LIBRARY },
445   { " executable",              RPMFC_EXECUTABLE },
446   { " statically linked",       RPMFC_STATIC },
447   { " not stripped",            RPMFC_NOTSTRIPPED },
448   { " archive",                 RPMFC_ARCHIVE },
449
450   { "ELF 32-bit",               RPMFC_ELF32|RPMFC_INCLUDE },
451   { "ELF 64-bit",               RPMFC_ELF64|RPMFC_INCLUDE },
452
453   { " script",                  RPMFC_SCRIPT },
454   { " text",                    RPMFC_TEXT },
455   { " document",                RPMFC_DOCUMENT },
456
457   { " compressed",              RPMFC_COMPRESSED },
458
459   { "troff or preprocessor input",      RPMFC_MANPAGE|RPMFC_INCLUDE },
460   { "GNU Info",                 RPMFC_MANPAGE|RPMFC_INCLUDE },
461
462   { "perl script text",         RPMFC_PERL|RPMFC_INCLUDE },
463   { "Perl5 module source text", RPMFC_PERL|RPMFC_MODULE|RPMFC_INCLUDE },
464
465   { " /usr/bin/python",         RPMFC_PYTHON|RPMFC_INCLUDE },
466
467   /* XXX "a /usr/bin/python -t script text executable" */
468   /* XXX "python 2.3 byte-compiled" */
469   { "python ",                  RPMFC_PYTHON|RPMFC_INCLUDE },
470
471   { "libtool library ",         RPMFC_LIBTOOL|RPMFC_INCLUDE },
472   { "pkgconfig ",               RPMFC_PKGCONFIG|RPMFC_INCLUDE },
473
474   /* XXX .NET executables and libraries.  file(1) cannot differ from win32 
475    * executables unfortunately :( */
476   { "PE executable",            RPMFC_MONO|RPMFC_INCLUDE },
477
478   { "current ar archive",       RPMFC_STATIC|RPMFC_LIBRARY|RPMFC_ARCHIVE|RPMFC_INCLUDE },
479
480   { "Zip archive data",         RPMFC_COMPRESSED|RPMFC_ARCHIVE|RPMFC_INCLUDE },
481   { "tar archive",              RPMFC_ARCHIVE|RPMFC_INCLUDE },
482   { "cpio archive",             RPMFC_ARCHIVE|RPMFC_INCLUDE },
483   { "RPM v3",                   RPMFC_ARCHIVE|RPMFC_INCLUDE },
484   { "RPM v4",                   RPMFC_ARCHIVE|RPMFC_INCLUDE },
485
486   { " image",                   RPMFC_IMAGE|RPMFC_INCLUDE },
487   { " font",                    RPMFC_FONT|RPMFC_INCLUDE },
488   { " Font",                    RPMFC_FONT|RPMFC_INCLUDE },
489
490   { " commands",                RPMFC_SCRIPT|RPMFC_INCLUDE },
491   { " script",                  RPMFC_SCRIPT|RPMFC_INCLUDE },
492
493   { "empty",                    RPMFC_WHITE|RPMFC_INCLUDE },
494
495   { "HTML",                     RPMFC_WHITE|RPMFC_INCLUDE },
496   { "SGML",                     RPMFC_WHITE|RPMFC_INCLUDE },
497   { "XML",                      RPMFC_WHITE|RPMFC_INCLUDE },
498
499   { " program text",            RPMFC_WHITE|RPMFC_INCLUDE },
500   { " source",                  RPMFC_WHITE|RPMFC_INCLUDE },
501   { "GLS_BINARY_LSB_FIRST",     RPMFC_WHITE|RPMFC_INCLUDE },
502   { " DB ",                     RPMFC_WHITE|RPMFC_INCLUDE },
503
504   { "ASCII English text",       RPMFC_WHITE|RPMFC_INCLUDE },
505   { "ASCII text",               RPMFC_WHITE|RPMFC_INCLUDE },
506   { "ISO-8859 text",            RPMFC_WHITE|RPMFC_INCLUDE },
507
508   { "symbolic link to",         RPMFC_SYMLINK },
509   { "socket",                   RPMFC_DEVICE },
510   { "special",                  RPMFC_DEVICE },
511
512   { "ASCII",                    RPMFC_WHITE },
513   { "ISO-8859",                 RPMFC_WHITE },
514
515   { "data",                     RPMFC_WHITE },
516
517   { "application",              RPMFC_WHITE },
518   { "boot",                     RPMFC_WHITE },
519   { "catalog",                  RPMFC_WHITE },
520   { "code",                     RPMFC_WHITE },
521   { "file",                     RPMFC_WHITE },
522   { "format",                   RPMFC_WHITE },
523   { "message",                  RPMFC_WHITE },
524   { "program",                  RPMFC_WHITE },
525
526   { "broken symbolic link to ", RPMFC_WHITE|RPMFC_ERROR },
527   { "can't read",               RPMFC_WHITE|RPMFC_ERROR },
528   { "can't stat",               RPMFC_WHITE|RPMFC_ERROR },
529   { "executable, can't read",   RPMFC_WHITE|RPMFC_ERROR },
530   { "core file",                RPMFC_WHITE|RPMFC_ERROR },
531
532   { NULL,                       RPMFC_BLACK }
533 };
534
535 int rpmfcColoring(const char * fmstr)
536 {
537     rpmfcToken fct;
538     int fcolor = RPMFC_BLACK;
539
540     for (fct = rpmfcTokens; fct->token != NULL; fct++) {
541         if (strstr(fmstr, fct->token) == NULL)
542             continue;
543         fcolor |= fct->colors;
544         if (fcolor & RPMFC_INCLUDE)
545             return fcolor;
546     }
547     return fcolor;
548 }
549
550 void rpmfcPrint(const char * msg, rpmfc fc, FILE * fp)
551 {
552     int fcolor;
553     int ndx;
554     int cx;
555     int dx;
556     int fx;
557
558 int nprovides;
559 int nrequires;
560
561     if (fp == NULL) fp = stderr;
562
563     if (msg)
564         fprintf(fp, "===================================== %s\n", msg);
565
566 nprovides = rpmdsCount(fc->provides);
567 nrequires = rpmdsCount(fc->requires);
568
569     if (fc)
570     for (fx = 0; fx < fc->nfiles; fx++) {
571 assert(fx < fc->fcdictx->nvals);
572         cx = fc->fcdictx->vals[fx];
573 assert(fx < fc->fcolor->nvals);
574         fcolor = fc->fcolor->vals[fx];
575
576         fprintf(fp, "%3d %s", fx, fc->fn[fx]);
577         if (fcolor != RPMFC_BLACK)
578                 fprintf(fp, "\t0x%x", fc->fcolor->vals[fx]);
579         else
580                 fprintf(fp, "\t%s", fc->cdict[cx]);
581         fprintf(fp, "\n");
582
583         if (fc->fddictx == NULL || fc->fddictn == NULL)
584             continue;
585
586 assert(fx < fc->fddictx->nvals);
587         dx = fc->fddictx->vals[fx];
588 assert(fx < fc->fddictn->nvals);
589         ndx = fc->fddictn->vals[fx];
590
591         while (ndx-- > 0) {
592             const char * depval;
593             unsigned char deptype;
594             unsigned ix;
595
596             ix = fc->ddictx->vals[dx++];
597             deptype = ((ix >> 24) & 0xff);
598             ix &= 0x00ffffff;
599             depval = NULL;
600             switch (deptype) {
601             default:
602 assert(depval != NULL);
603                 break;
604             case 'P':
605                 if (nprovides > 0) {
606 assert(ix < nprovides);
607                     (void) rpmdsSetIx(fc->provides, ix-1);
608                     if (rpmdsNext(fc->provides) >= 0)
609                         depval = rpmdsDNEVR(fc->provides);
610                 }
611                 break;
612             case 'R':
613                 if (nrequires > 0) {
614 assert(ix < nrequires);
615                     (void) rpmdsSetIx(fc->requires, ix-1);
616                     if (rpmdsNext(fc->requires) >= 0)
617                         depval = rpmdsDNEVR(fc->requires);
618                 }
619                 break;
620             }
621             if (depval)
622                 fprintf(fp, "\t%s\n", depval);
623         }
624     }
625 }
626
627 rpmfc rpmfcFree(rpmfc fc)
628 {
629     if (fc) {
630         fc->fn = argvFree(fc->fn);
631         fc->fcolor = argiFree(fc->fcolor);
632         fc->fcdictx = argiFree(fc->fcdictx);
633         fc->fddictx = argiFree(fc->fddictx);
634         fc->fddictn = argiFree(fc->fddictn);
635         fc->cdict = argvFree(fc->cdict);
636         fc->ddict = argvFree(fc->ddict);
637         fc->ddictx = argiFree(fc->ddictx);
638
639         fc->provides = rpmdsFree(fc->provides);
640         fc->requires = rpmdsFree(fc->requires);
641
642         fc->sb_java = freeStringBuf(fc->sb_java);
643         fc->sb_perl = freeStringBuf(fc->sb_perl);
644         fc->sb_python = freeStringBuf(fc->sb_python);
645
646     }
647     fc = _free(fc);
648     return NULL;
649 }
650
651 rpmfc rpmfcNew(void)
652 {
653     rpmfc fc = xcalloc(1, sizeof(*fc));
654     return fc;
655 }
656
657 rpmds rpmfcProvides(rpmfc fc)
658 {
659     return (fc != NULL ? fc->provides : NULL);
660 }
661
662 rpmds rpmfcRequires(rpmfc fc)
663 {
664     return (fc != NULL ? fc->requires : NULL);
665 }
666
667
668 /**
669  * Extract script dependencies.
670  * @param fc            file classifier
671  * @return              0 on success
672  */
673 static int rpmfcSCRIPT(rpmfc fc)
674 {
675     const char * fn = fc->fn[fc->ix];
676     const char * bn;
677     rpmds ds;
678     char buf[BUFSIZ];
679     FILE * fp;
680     char * s, * se;
681     int i;
682     struct stat sb, * st = &sb;
683     int is_executable;
684     int xx;
685
686     /* Only executable scripts are searched. */
687     if (stat(fn, st) < 0)
688         return -1;
689     is_executable = (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH));
690
691     fp = fopen(fn, "r");
692     if (fp == NULL || ferror(fp)) {
693         if (fp) (void) fclose(fp);
694         return -1;
695     }
696
697     /* Look for #! interpreter in first 10 lines. */
698     for (i = 0; i < 10; i++) {
699
700         s = fgets(buf, sizeof(buf) - 1, fp);
701         if (s == NULL || ferror(fp) || feof(fp))
702             break;
703         s[sizeof(buf)-1] = '\0';
704         if (!(s[0] == '#' && s[1] == '!'))
705             continue;
706         s += 2;
707
708         while (*s && strchr(" \t\n\r", *s) != NULL)
709             s++;
710         if (*s == '\0')
711             continue;
712         if (*s != '/')
713             continue;
714
715         for (se = s+1; *se; se++) {
716             if (strchr(" \t\n\r", *se) != NULL)
717                 break;
718         }
719         *se = '\0';
720         se++;
721
722         if (is_executable) {
723             /* Add to package requires. */
724             ds = rpmdsSingle(RPMTAG_REQUIRENAME, s, "", RPMSENSE_FIND_REQUIRES);
725             xx = rpmdsMerge(&fc->requires, ds);
726
727             /* Add to file requires. */
728             xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(se, fc->ix, ds));
729
730             ds = rpmdsFree(ds);
731         }
732
733         /* Set color based on interpreter name. */
734         bn = basename(s);
735         if (!strcmp(bn, "perl"))
736             fc->fcolor->vals[fc->ix] |= RPMFC_PERL;
737         else if (!strncmp(bn, "python", sizeof("python")-1))
738             fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
739
740         break;
741     }
742
743     (void) fclose(fp);
744
745     if (fc->fcolor->vals[fc->ix] & RPMFC_PERL) {
746         if (fc->fcolor->vals[fc->ix] & RPMFC_MODULE)
747             xx = rpmfcHelper(fc, 'P', "perl");
748         if (is_executable || (fc->fcolor->vals[fc->ix] & RPMFC_MODULE))
749             xx = rpmfcHelper(fc, 'R', "perl");
750     }
751     if (fc->fcolor->vals[fc->ix] & RPMFC_PYTHON) {
752         xx = rpmfcHelper(fc, 'P', "python");
753 #ifdef  NOTYET
754         if (is_executable)
755 #endif
756             xx = rpmfcHelper(fc, 'R', "python");
757     } else 
758     if (fc->fcolor->vals[fc->ix] & RPMFC_LIBTOOL) {
759         xx = rpmfcHelper(fc, 'P', "libtool");
760 #ifdef  NOTYET
761         if (is_executable)
762 #endif
763             xx = rpmfcHelper(fc, 'R', "libtool");
764     } else
765     if (fc->fcolor->vals[fc->ix] & RPMFC_PKGCONFIG) {
766         xx = rpmfcHelper(fc, 'P', "pkgconfig");
767 #ifdef  NOTYET
768         if (is_executable)
769 #endif
770             xx = rpmfcHelper(fc, 'R', "pkgconfig");
771     } else 
772     if (fc->fcolor->vals[fc->ix] & RPMFC_MONO) {
773         xx = rpmfcHelper(fc, 'P', "mono");
774         if (is_executable)
775             xx = rpmfcHelper(fc, 'R', "mono");
776     }
777
778     return 0;
779 }
780
781 /**
782  * Extract Elf dependencies.
783  * @param fc            file classifier
784  * @return              0 on success
785  */
786 static int rpmfcELF(rpmfc fc)
787 {
788 #if HAVE_GELF_H && HAVE_LIBELF
789     const char * fn = fc->fn[fc->ix];
790     Elf * elf;
791     Elf_Scn * scn;
792     Elf_Data * data;
793     GElf_Ehdr ehdr_mem, * ehdr;
794     GElf_Shdr shdr_mem, * shdr;
795     GElf_Verdef def_mem, * def;
796     GElf_Verneed need_mem, * need;
797     GElf_Dyn dyn_mem, * dyn;
798     unsigned int auxoffset;
799     unsigned int offset;
800     int fdno;
801     int cnt2;
802     int cnt;
803     char buf[BUFSIZ];
804     const char * s;
805     struct stat sb, * st = &sb;
806     char * soname = NULL;
807     rpmds * depsp, ds;
808     rpm_tag_t tagN;
809     int32_t dsContext;
810     char * t;
811     int xx;
812     int isElf64;
813     int isDSO;
814     int gotSONAME = 0;
815     int gotDEBUG = 0;
816     int gotHASH = 0;
817     int gotGNUHASH = 0;
818     static int filter_GLIBC_PRIVATE = 0;
819     static int oneshot = 0;
820
821     if (oneshot == 0) {
822         oneshot = 1;
823         filter_GLIBC_PRIVATE = rpmExpandNumeric("%{?_filter_GLIBC_PRIVATE}");
824     }
825
826     /* Files with executable bit set only. */
827     if (stat(fn, st) != 0)
828         return(-1);
829
830     fdno = open(fn, O_RDONLY);
831     if (fdno < 0)
832         return fdno;
833
834     (void) elf_version(EV_CURRENT);
835
836     elf = NULL;
837     if ((elf = elf_begin (fdno, ELF_C_READ, NULL)) == NULL
838      || elf_kind(elf) != ELF_K_ELF
839      || (ehdr = gelf_getehdr(elf, &ehdr_mem)) == NULL
840      || !(ehdr->e_type == ET_DYN || ehdr->e_type == ET_EXEC))
841         goto exit;
842
843     isElf64 = ehdr->e_ident[EI_CLASS] == ELFCLASS64;
844     isDSO = ehdr->e_type == ET_DYN;
845
846     scn = NULL;
847     while ((scn = elf_nextscn(elf, scn)) != NULL) {
848         shdr = gelf_getshdr(scn, &shdr_mem);
849         if (shdr == NULL)
850             break;
851
852         soname = _free(soname);
853         switch (shdr->sh_type) {
854         default:
855             continue;
856             break;
857         case SHT_GNU_verdef:
858             data = NULL;
859             if (!fc->skipProv)
860             while ((data = elf_getdata (scn, data)) != NULL) {
861                 offset = 0;
862                 for (cnt = shdr->sh_info; --cnt >= 0; ) {
863                 
864                     def = gelf_getverdef (data, offset, &def_mem);
865                     if (def == NULL)
866                         break;
867                     auxoffset = offset + def->vd_aux;
868                     for (cnt2 = def->vd_cnt; --cnt2 >= 0; ) {
869                         GElf_Verdaux aux_mem, * aux;
870
871                         aux = gelf_getverdaux (data, auxoffset, &aux_mem);
872                         if (aux == NULL)
873                             break;
874
875                         s = elf_strptr(elf, shdr->sh_link, aux->vda_name);
876                         if (s == NULL)
877                             break;
878                         if (def->vd_flags & VER_FLG_BASE) {
879                             soname = _free(soname);
880                             soname = xstrdup(s);
881                             auxoffset += aux->vda_next;
882                             continue;
883                         } else
884                         if (soname != NULL
885                          && !(filter_GLIBC_PRIVATE != 0
886                                 && !strcmp(s, "GLIBC_PRIVATE")))
887                         {
888                             buf[0] = '\0';
889                             t = buf;
890                             t = stpcpy( stpcpy( stpcpy( stpcpy(t, soname), "("), s), ")");
891
892 #if !defined(__alpha__)
893                             if (isElf64)
894                                 t = stpcpy(t, "(64bit)");
895 #endif
896                             t++;
897
898                             /* Add to package provides. */
899                             ds = rpmdsSingle(RPMTAG_PROVIDES,
900                                         buf, "", RPMSENSE_FIND_PROVIDES);
901                             xx = rpmdsMerge(&fc->provides, ds);
902
903                             /* Add to file dependencies. */
904                             xx = rpmfcSaveArg(&fc->ddict,
905                                         rpmfcFileDep(t, fc->ix, ds));
906
907                             ds = rpmdsFree(ds);
908                         }
909                         auxoffset += aux->vda_next;
910                     }
911                     offset += def->vd_next;
912                 }
913             }
914             break;
915         case SHT_GNU_verneed:
916             data = NULL;
917             /* Files with executable bit set only. */
918             if (!fc->skipReq && (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
919             while ((data = elf_getdata (scn, data)) != NULL) {
920                 offset = 0;
921                 for (cnt = shdr->sh_info; --cnt >= 0; ) {
922                     need = gelf_getverneed (data, offset, &need_mem);
923                     if (need == NULL)
924                         break;
925
926                     s = elf_strptr(elf, shdr->sh_link, need->vn_file);
927                     if (s == NULL)
928                         break;
929                     soname = _free(soname);
930                     soname = xstrdup(s);
931                     auxoffset = offset + need->vn_aux;
932                     for (cnt2 = need->vn_cnt; --cnt2 >= 0; ) {
933                         GElf_Vernaux aux_mem, * aux;
934
935                         aux = gelf_getvernaux (data, auxoffset, &aux_mem);
936                         if (aux == NULL)
937                             break;
938
939                         s = elf_strptr(elf, shdr->sh_link, aux->vna_name);
940                         if (s == NULL)
941                             break;
942
943                         /* Filter dependencies that contain GLIBC_PRIVATE */
944                         if (soname != NULL
945                          && !(filter_GLIBC_PRIVATE != 0
946                                 && !strcmp(s, "GLIBC_PRIVATE")))
947                         {
948                             buf[0] = '\0';
949                             t = buf;
950                             t = stpcpy( stpcpy( stpcpy( stpcpy(t, soname), "("), s), ")");
951
952 #if !defined(__alpha__)
953                             if (isElf64)
954                                 t = stpcpy(t, "(64bit)");
955 #endif
956                             t++;
957
958                             /* Add to package dependencies. */
959                             ds = rpmdsSingle(RPMTAG_REQUIRENAME,
960                                         buf, "", RPMSENSE_FIND_REQUIRES);
961                             xx = rpmdsMerge(&fc->requires, ds);
962
963                             /* Add to file dependencies. */
964                             xx = rpmfcSaveArg(&fc->ddict,
965                                         rpmfcFileDep(t, fc->ix, ds));
966                             ds = rpmdsFree(ds);
967                         }
968                         auxoffset += aux->vna_next;
969                     }
970                     offset += need->vn_next;
971                 }
972             }
973             break;
974         case SHT_DYNAMIC:
975             data = NULL;
976             while ((data = elf_getdata (scn, data)) != NULL) {
977                 for (cnt = 0; cnt < (shdr->sh_size / shdr->sh_entsize); ++cnt) {
978                     dyn = gelf_getdyn (data, cnt, &dyn_mem);
979                     if (dyn == NULL)
980                         break;
981                     s = NULL;
982                     switch (dyn->d_tag) {
983                     default:
984                         continue;
985                         break;
986                     case DT_HASH:    
987                         gotHASH= 1;
988                         continue;
989                     case DT_GNU_HASH:
990                         gotGNUHASH= 1;
991                         continue;
992                     case DT_DEBUG:    
993                         gotDEBUG = 1;
994                         continue;
995                     case DT_NEEDED:
996                         /* Files with executable bit set only. */
997                         if (fc->skipReq || !(st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
998                             continue;
999                         /* Add to package requires. */
1000                         depsp = &fc->requires;
1001                         tagN = RPMTAG_REQUIRENAME;
1002                         dsContext = RPMSENSE_FIND_REQUIRES;
1003                         s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1004 assert(s != NULL);
1005                         break;
1006                     case DT_SONAME:
1007                         gotSONAME = 1;
1008                         /* Add to package provides. */
1009                         if (fc->skipProv)
1010                             continue;
1011                         depsp = &fc->provides;
1012                         tagN = RPMTAG_PROVIDENAME;
1013                         dsContext = RPMSENSE_FIND_PROVIDES;
1014                         s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1015 assert(s != NULL);
1016                         break;
1017                     }
1018                     if (s == NULL)
1019                         continue;
1020
1021                     buf[0] = '\0';
1022                     t = buf;
1023                     t = stpcpy(t, s);
1024
1025 #if !defined(__alpha__)
1026                     if (isElf64)
1027                         t = stpcpy(t, "()(64bit)");
1028 #endif
1029                     t++;
1030
1031                     /* Add to package dependencies. */
1032                     ds = rpmdsSingle(tagN, buf, "", dsContext);
1033                     xx = rpmdsMerge(depsp, ds);
1034
1035                     /* Add to file dependencies. */
1036                     xx = rpmfcSaveArg(&fc->ddict,
1037                                         rpmfcFileDep(t, fc->ix, ds));
1038
1039                     ds = rpmdsFree(ds);
1040                 }
1041             }
1042             break;
1043         }
1044     }
1045
1046     /* For DSOs which use the .gnu_hash section and don't have a .hash
1047      * section, we need to ensure that we have a new enough glibc. */ 
1048     if (gotGNUHASH && !gotHASH) {
1049         ds = rpmdsSingle(RPMTAG_REQUIRENAME, "rtld(GNU_HASH)", "", 
1050                          RPMSENSE_FIND_REQUIRES);
1051         rpmdsMerge(&fc->requires, ds);
1052         buf[0] = '\0';
1053         t = buf;
1054         rpmfcSaveArg(&fc->ddict, rpmfcFileDep(t, fc->ix, ds));
1055         ds = rpmdsFree(ds);
1056     }
1057
1058     /* For DSO's, provide the basename of the file if DT_SONAME not found. */
1059     if (!fc->skipProv && isDSO && !gotDEBUG && !gotSONAME) {
1060         depsp = &fc->provides;
1061         tagN = RPMTAG_PROVIDENAME;
1062         dsContext = RPMSENSE_FIND_PROVIDES;
1063
1064         s = strrchr(fn, '/');
1065         if (s)
1066             s++;
1067         else
1068             s = fn;
1069
1070         buf[0] = '\0';
1071         t = buf;
1072 /* LCL: s is not null. */
1073         t = stpcpy(t, s);
1074
1075 #if !defined(__alpha__)
1076         if (isElf64)
1077             t = stpcpy(t, "()(64bit)");
1078 #endif
1079         t++;
1080
1081         /* Add to package dependencies. */
1082         ds = rpmdsSingle(tagN, buf, "", dsContext);
1083         xx = rpmdsMerge(depsp, ds);
1084
1085         /* Add to file dependencies. */
1086         xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(t, fc->ix, ds));
1087
1088         ds = rpmdsFree(ds);
1089     }
1090
1091 exit:
1092     soname = _free(soname);
1093     if (elf) (void) elf_end(elf);
1094     xx = close(fdno);
1095     return 0;
1096 #else
1097     return -1;
1098 #endif
1099 }
1100
1101 typedef struct rpmfcApplyTbl_s {
1102     int (*func) (rpmfc fc);
1103     int colormask;
1104 } * rpmfcApplyTbl;
1105
1106 /**
1107  */
1108 static struct rpmfcApplyTbl_s rpmfcApplyTable[] = {
1109     { rpmfcELF,         RPMFC_ELF },
1110     { rpmfcSCRIPT,      (RPMFC_SCRIPT|RPMFC_PERL) },
1111     { rpmfcSCRIPT,      (RPMFC_SCRIPT|RPMFC_PYTHON) },
1112     { rpmfcSCRIPT,      (RPMFC_SCRIPT|RPMFC_PKGCONFIG) },
1113     { rpmfcSCRIPT,      (RPMFC_SCRIPT|RPMFC_LIBTOOL) },
1114     { rpmfcSCRIPT,      RPMFC_MONO },
1115     { NULL, 0 }
1116 };
1117
1118 rpmRC rpmfcApply(rpmfc fc)
1119 {
1120     rpmfcApplyTbl fcat;
1121     const char * s;
1122     char * se;
1123     rpmds ds;
1124     const char * N;
1125     const char * EVR;
1126     rpmsenseFlags Flags;
1127     unsigned char deptype;
1128     int nddict;
1129     int previx;
1130     unsigned int val;
1131     int dix;
1132     int ix;
1133     int i;
1134     int xx;
1135
1136     /* Generate package and per-file dependencies. */
1137     for (fc->ix = 0; fc->fn[fc->ix] != NULL; fc->ix++) {
1138
1139         /* XXX Insure that /usr/lib{,64}/python files are marked RPMFC_PYTHON */
1140         /* XXX HACK: classification by path is intrinsically stupid. */
1141         {   const char *fn = strstr(fc->fn[fc->ix], "/usr/lib");
1142             if (fn) {
1143                 fn += sizeof("/usr/lib")-1;
1144                 if (fn[0] == '6' && fn[1] == '4')
1145                     fn += 2;
1146                 if (!strncmp(fn, "/python", sizeof("/python")-1))
1147                     fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
1148             }
1149         }
1150
1151         if (fc->fcolor->vals[fc->ix])
1152         for (fcat = rpmfcApplyTable; fcat->func != NULL; fcat++) {
1153             if (!(fc->fcolor->vals[fc->ix] & fcat->colormask))
1154                 continue;
1155             xx = (*fcat->func) (fc);
1156         }
1157     }
1158
1159     /* Generate per-file indices into package dependencies. */
1160     nddict = argvCount(fc->ddict);
1161     previx = -1;
1162     for (i = 0; i < nddict; i++) {
1163         s = fc->ddict[i];
1164
1165         /* Parse out (file#,deptype,N,EVR,Flags) */
1166         ix = strtol(s, &se, 10);
1167 assert(se != NULL);
1168         deptype = *se++;
1169         se++;
1170         N = se;
1171         while (*se && *se != ' ')
1172             se++;
1173         *se++ = '\0';
1174         EVR = se;
1175         while (*se && *se != ' ')
1176             se++;
1177         *se++ = '\0';
1178         Flags = strtol(se, NULL, 16);
1179
1180         dix = -1;
1181         switch (deptype) {
1182         default:
1183             break;
1184         case 'P':       
1185             ds = rpmdsSingle(RPMTAG_PROVIDENAME, N, EVR, Flags);
1186             dix = rpmdsFind(fc->provides, ds);
1187             ds = rpmdsFree(ds);
1188             break;
1189         case 'R':
1190             ds = rpmdsSingle(RPMTAG_REQUIRENAME, N, EVR, Flags);
1191             dix = rpmdsFind(fc->requires, ds);
1192             ds = rpmdsFree(ds);
1193             break;
1194         }
1195
1196 /* XXX assertion incorrect while generating -debuginfo deps. */
1197 #if 0
1198 assert(dix >= 0);
1199 #else
1200         if (dix < 0)
1201             continue;
1202 #endif
1203
1204         val = (deptype << 24) | (dix & 0x00ffffff);
1205         xx = argiAdd(&fc->ddictx, -1, val);
1206
1207         if (previx != ix) {
1208             previx = ix;
1209             xx = argiAdd(&fc->fddictx, ix, argiCount(fc->ddictx)-1);
1210         }
1211         if (fc->fddictn && fc->fddictn->vals)
1212             fc->fddictn->vals[ix]++;
1213     }
1214
1215     return 0;
1216 }
1217
1218 rpmRC rpmfcClassify(rpmfc fc, ARGV_t argv, int16_t * fmode)
1219 {
1220     ARGV_t fcav = NULL;
1221     ARGV_t dav;
1222     const char * s, * se;
1223     size_t slen;
1224     int fcolor;
1225     int xx;
1226     int msflags = MAGIC_CHECK;  /* XXX MAGIC_COMPRESS flag? */
1227     magic_t ms = NULL;
1228
1229     if (fc == NULL || argv == NULL)
1230         return 0;
1231
1232     fc->nfiles = argvCount(argv);
1233
1234     /* Initialize the per-file dictionary indices. */
1235     xx = argiAdd(&fc->fddictx, fc->nfiles-1, 0);
1236     xx = argiAdd(&fc->fddictn, fc->nfiles-1, 0);
1237
1238     /* Build (sorted) file class dictionary. */
1239     xx = argvAdd(&fc->cdict, "");
1240     xx = argvAdd(&fc->cdict, "directory");
1241
1242     ms = magic_open(msflags);
1243     if (ms == NULL) {
1244         rpmlog(RPMLOG_ERR, _("magic_open(0x%x) failed: %s\n"),
1245                 msflags, strerror(errno));
1246 assert(ms != NULL);     /* XXX figger a proper return path. */
1247     }
1248
1249     xx = magic_load(ms, NULL);
1250     if (xx == -1) {
1251         rpmlog(RPMLOG_ERR, _("magic_load failed: %s\n"), magic_error(ms));
1252 assert(xx != -1);       /* XXX figger a proper return path. */
1253     }
1254
1255     for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1256         const char * ftype;
1257         int16_t mode = (fmode ? fmode[fc->ix] : 0);
1258
1259         s = argv[fc->ix];
1260 assert(s != NULL);
1261         slen = strlen(s);
1262
1263         switch (mode & S_IFMT) {
1264         case S_IFCHR:   ftype = "character special";    break;
1265         case S_IFBLK:   ftype = "block special";        break;
1266         case S_IFIFO:   ftype = "fifo (named pipe)";    break;
1267         case S_IFSOCK:  ftype = "socket";               break;
1268         case S_IFDIR:
1269         case S_IFLNK:
1270         case S_IFREG:
1271         default:
1272             /* XXX all files with extension ".pm" are perl modules for now. */
1273             if (rpmFileHasSuffix(s, ".pm"))
1274                 ftype = "Perl5 module source text";
1275
1276             /* XXX all files with extension ".la" are libtool for now. */
1277             else if (rpmFileHasSuffix(s, ".la"))
1278                 ftype = "libtool library file";
1279
1280             /* XXX all files with extension ".pc" are pkgconfig for now. */
1281             else if (rpmFileHasSuffix(s, ".pc"))
1282                 ftype = "pkgconfig file";
1283
1284             /* XXX skip all files in /dev/ which are (or should be) %dev dummies. */
1285             else if (slen >= fc->brlen+sizeof("/dev/") && !strncmp(s+fc->brlen, "/dev/", sizeof("/dev/")-1))
1286                 ftype = "";
1287             else
1288                 ftype = magic_file(ms, s);
1289
1290             if (ftype == NULL) {
1291                 rpmlog(RPMLOG_ERR, 
1292                        _("magic_file(ms, \"%s\") failed: mode %06o %s\n"),
1293                        s, mode, magic_error(ms));
1294 assert(ftype != NULL);  /* XXX figger a proper return path. */
1295             }
1296         }
1297
1298         se = ftype;
1299         rpmlog(RPMLOG_DEBUG, "%s: %s\n", s, se);
1300
1301         /* Save the path. */
1302         xx = argvAdd(&fc->fn, s);
1303
1304         /* Save the file type string. */
1305         xx = argvAdd(&fcav, se);
1306
1307         /* Add (filtered) entry to sorted class dictionary. */
1308         fcolor = rpmfcColoring(se);
1309         xx = argiAdd(&fc->fcolor, fc->ix, fcolor);
1310
1311         if (fcolor != RPMFC_WHITE && (fcolor & RPMFC_INCLUDE))
1312             xx = rpmfcSaveArg(&fc->cdict, se);
1313     }
1314
1315     /* Build per-file class index array. */
1316     fc->fknown = 0;
1317     for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1318         se = fcav[fc->ix];
1319 assert(se != NULL);
1320
1321         dav = argvSearch(fc->cdict, se, NULL);
1322         if (dav) {
1323             xx = argiAdd(&fc->fcdictx, fc->ix, (dav - fc->cdict));
1324             fc->fknown++;
1325         } else {
1326             xx = argiAdd(&fc->fcdictx, fc->ix, 0);
1327             fc->fwhite++;
1328         }
1329     }
1330
1331     fcav = argvFree(fcav);
1332
1333     if (ms != NULL)
1334         magic_close(ms);
1335
1336     return 0;
1337 }
1338
1339 /**
1340  */
1341 typedef struct DepMsg_s * DepMsg_t;
1342
1343 /**
1344  */
1345 struct DepMsg_s {
1346     const char * msg;
1347     const char * argv[4];
1348     rpm_tag_t ntag;
1349     rpm_tag_t vtag;
1350     rpm_tag_t ftag;
1351     int mask;
1352     int xor;
1353 };
1354
1355 /**
1356  */
1357 static struct DepMsg_s depMsgs[] = {
1358   { "Provides",         { "%{?__find_provides}", NULL, NULL, NULL },
1359         RPMTAG_PROVIDENAME, RPMTAG_PROVIDEVERSION, RPMTAG_PROVIDEFLAGS,
1360         0, -1 },
1361   { "Requires(interp)", { NULL, "interp", NULL, NULL },
1362         RPMTAG_REQUIRENAME, RPMTAG_REQUIREVERSION, RPMTAG_REQUIREFLAGS,
1363         _notpre(RPMSENSE_INTERP), 0 },
1364   { "Requires(rpmlib)", { NULL, "rpmlib", NULL, NULL },
1365         -1, -1, RPMTAG_REQUIREFLAGS,
1366         _notpre(RPMSENSE_RPMLIB), 0 },
1367   { "Requires(verify)", { NULL, "verify", NULL, NULL },
1368         -1, -1, RPMTAG_REQUIREFLAGS,
1369         RPMSENSE_SCRIPT_VERIFY, 0 },
1370   { "Requires(pre)",    { NULL, "pre", NULL, NULL },
1371         -1, -1, RPMTAG_REQUIREFLAGS,
1372         _notpre(RPMSENSE_SCRIPT_PRE), 0 },
1373   { "Requires(post)",   { NULL, "post", NULL, NULL },
1374         -1, -1, RPMTAG_REQUIREFLAGS,
1375         _notpre(RPMSENSE_SCRIPT_POST), 0 },
1376   { "Requires(preun)",  { NULL, "preun", NULL, NULL },
1377         -1, -1, RPMTAG_REQUIREFLAGS,
1378         _notpre(RPMSENSE_SCRIPT_PREUN), 0 },
1379   { "Requires(postun)", { NULL, "postun", NULL, NULL },
1380         -1, -1, RPMTAG_REQUIREFLAGS,
1381         _notpre(RPMSENSE_SCRIPT_POSTUN), 0 },
1382   { "Requires",         { "%{?__find_requires}", NULL, NULL, NULL },
1383         -1, -1, RPMTAG_REQUIREFLAGS,    /* XXX inherit name/version arrays */
1384         RPMSENSE_FIND_REQUIRES|RPMSENSE_TRIGGERIN|RPMSENSE_TRIGGERUN|RPMSENSE_TRIGGERPOSTUN|RPMSENSE_TRIGGERPREIN, 0 },
1385   { "Conflicts",        { "%{?__find_conflicts}", NULL, NULL, NULL },
1386         RPMTAG_CONFLICTNAME, RPMTAG_CONFLICTVERSION, RPMTAG_CONFLICTFLAGS,
1387         0, -1 },
1388   { "Obsoletes",        { "%{?__find_obsoletes}", NULL, NULL, NULL },
1389         RPMTAG_OBSOLETENAME, RPMTAG_OBSOLETEVERSION, RPMTAG_OBSOLETEFLAGS,
1390         0, -1 },
1391   { NULL,               { NULL, NULL, NULL, NULL },     0, 0, 0, 0, 0 }
1392 };
1393
1394 static DepMsg_t DepMsgs = depMsgs;
1395
1396 /**
1397  */
1398 static void printDeps(Header h)
1399 {
1400     DepMsg_t dm;
1401     rpmds ds = NULL;
1402     int flags = 0;      /* XXX !scareMem */
1403     const char * DNEVR;
1404     rpmsenseFlags Flags;
1405     int bingo = 0;
1406
1407     for (dm = DepMsgs; dm->msg != NULL; dm++) {
1408         if (dm->ntag != -1) {
1409             ds = rpmdsFree(ds);
1410             ds = rpmdsNew(h, dm->ntag, flags);
1411         }
1412         if (dm->ftag == 0)
1413             continue;
1414
1415         ds = rpmdsInit(ds);
1416         if (ds == NULL)
1417             continue;   /* XXX can't happen */
1418
1419         bingo = 0;
1420         while (rpmdsNext(ds) >= 0) {
1421
1422             Flags = rpmdsFlags(ds);
1423         
1424             if (!((Flags & dm->mask) ^ dm->xor))
1425                 continue;
1426             if (bingo == 0) {
1427                 rpmlog(RPMLOG_NOTICE, "%s:", (dm->msg ? dm->msg : ""));
1428                 bingo = 1;
1429             }
1430             if ((DNEVR = rpmdsDNEVR(ds)) == NULL)
1431                 continue;       /* XXX can't happen */
1432             rpmlog(RPMLOG_NOTICE, " %s", DNEVR+2);
1433         }
1434         if (bingo)
1435             rpmlog(RPMLOG_NOTICE, "\n");
1436     }
1437     ds = rpmdsFree(ds);
1438 }
1439
1440 /**
1441  */
1442 static int rpmfcGenerateDependsHelper(const rpmSpec spec, Package pkg, rpmfi fi)
1443 {
1444     StringBuf sb_stdin;
1445     StringBuf sb_stdout;
1446     DepMsg_t dm;
1447     int failnonzero = 0;
1448     int rc = 0;
1449
1450     /*
1451      * Create file manifest buffer to deliver to dependency finder.
1452      */
1453     sb_stdin = newStringBuf();
1454     fi = rpmfiInit(fi, 0);
1455     if (fi != NULL)
1456     while (rpmfiNext(fi) >= 0)
1457         appendLineStringBuf(sb_stdin, rpmfiFN(fi));
1458
1459     for (dm = DepMsgs; dm->msg != NULL; dm++) {
1460         rpm_tag_t tag;
1461         rpmsenseFlags tagflags;
1462         char * s;
1463         int xx;
1464
1465         tag = (dm->ftag > 0) ? dm->ftag : dm->ntag;
1466         tagflags = 0;
1467         s = NULL;
1468
1469         switch(tag) {
1470         case RPMTAG_PROVIDEFLAGS:
1471             if (!pkg->autoProv)
1472                 continue;
1473             failnonzero = 1;
1474             tagflags = RPMSENSE_FIND_PROVIDES;
1475             break;
1476         case RPMTAG_REQUIREFLAGS:
1477             if (!pkg->autoReq)
1478                 continue;
1479             failnonzero = 0;
1480             tagflags = RPMSENSE_FIND_REQUIRES;
1481             break;
1482         default:
1483             continue;
1484             break;
1485         }
1486
1487         xx = rpmfcExec(dm->argv, sb_stdin, &sb_stdout, failnonzero);
1488         if (xx == -1)
1489             continue;
1490
1491         s = rpmExpand(dm->argv[0], NULL);
1492         rpmlog(RPMLOG_NOTICE, _("Finding  %s: %s\n"), dm->msg,
1493                 (s ? s : ""));
1494         s = _free(s);
1495
1496         if (sb_stdout == NULL) {
1497             rc = RPMRC_FAIL;
1498             rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1499             break;
1500         }
1501
1502         /* Parse dependencies into header */
1503         rc = parseRCPOT(spec, pkg, getStringBuf(sb_stdout), tag, 0, tagflags);
1504         sb_stdout = freeStringBuf(sb_stdout);
1505
1506         if (rc) {
1507             rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1508             break;
1509         }
1510     }
1511
1512     sb_stdin = freeStringBuf(sb_stdin);
1513
1514     return rc;
1515 }
1516
1517 rpmRC rpmfcGenerateDepends(const rpmSpec spec, Package pkg)
1518 {
1519     rpmfi fi = pkg->cpioList;
1520     rpmfc fc = NULL;
1521     rpmds ds;
1522     int flags = 0;      /* XXX !scareMem */
1523     ARGV_t av;
1524     int16_t * fmode;
1525     int ac = rpmfiFC(fi);
1526     const void ** p;
1527     char buf[BUFSIZ];
1528     const char * N;
1529     const char * EVR;
1530     int genConfigDeps;
1531     rpm_count_t c;
1532     int rc = 0;
1533     int xx;
1534     int idx;
1535
1536     /* Skip packages with no files. */
1537     if (ac <= 0)
1538         return 0;
1539
1540     /* Skip packages that have dependency generation disabled. */
1541     if (! (pkg->autoReq || pkg->autoProv))
1542         return 0;
1543
1544     /* If new-fangled dependency generation is disabled ... */
1545     if (!rpmExpandNumeric("%{?_use_internal_dependency_generator}")) {
1546         /* ... then generate dependencies using %{__find_requires} et al. */
1547         rc = rpmfcGenerateDependsHelper(spec, pkg, fi);
1548         printDeps(pkg->header);
1549         return rc;
1550     }
1551
1552     /* Extract absolute file paths in argv format. */
1553     av = xcalloc(ac+1, sizeof(*av));
1554     fmode = xcalloc(ac+1, sizeof(*fmode));
1555
1556     genConfigDeps = 0;
1557     fi = rpmfiInit(fi, 0);
1558     if (fi != NULL)
1559     while ((idx = rpmfiNext(fi)) >= 0) {
1560         rpmfileAttrs fileAttrs;
1561
1562         /* Does package have any %config files? */
1563         fileAttrs = rpmfiFFlags(fi);
1564         genConfigDeps |= (fileAttrs & RPMFILE_CONFIG);
1565
1566         av[idx] = xstrdup(rpmfiFN(fi));
1567         fmode[idx] = rpmfiFMode(fi);
1568     }
1569     av[ac] = NULL;
1570
1571     fc = rpmfcNew();
1572     fc->skipProv = !pkg->autoProv;
1573     fc->skipReq = !pkg->autoReq;
1574     fc->tracked = 0;
1575     fc->brlen = (spec->buildRootURL ? strlen(spec->buildRootURL) : 0);
1576
1577     /* Copy (and delete) manually generated dependencies to dictionary. */
1578     if (!fc->skipProv) {
1579         ds = rpmdsNew(pkg->header, RPMTAG_PROVIDENAME, flags);
1580         xx = rpmdsMerge(&fc->provides, ds);
1581         ds = rpmdsFree(ds);
1582         xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDENAME);
1583         xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDEVERSION);
1584         xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDEFLAGS);
1585
1586         /* Add config dependency, Provides: config(N) = EVR */
1587         if (genConfigDeps) {
1588             N = rpmdsN(pkg->ds);
1589 assert(N != NULL);
1590             EVR = rpmdsEVR(pkg->ds);
1591 assert(EVR != NULL);
1592             sprintf(buf, "config(%s)", N);
1593             ds = rpmdsSingle(RPMTAG_PROVIDENAME, buf, EVR,
1594                         (RPMSENSE_EQUAL|RPMSENSE_CONFIG));
1595             xx = rpmdsMerge(&fc->provides, ds);
1596             ds = rpmdsFree(ds);
1597         }
1598     }
1599
1600     if (!fc->skipReq) {
1601         ds = rpmdsNew(pkg->header, RPMTAG_REQUIRENAME, flags);
1602         xx = rpmdsMerge(&fc->requires, ds);
1603         ds = rpmdsFree(ds);
1604         xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIRENAME);
1605         xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIREVERSION);
1606         xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIREFLAGS);
1607
1608         /* Add config dependency,  Requires: config(N) = EVR */
1609         if (genConfigDeps) {
1610             N = rpmdsN(pkg->ds);
1611 assert(N != NULL);
1612             EVR = rpmdsEVR(pkg->ds);
1613 assert(EVR != NULL);
1614             sprintf(buf, "config(%s)", N);
1615             ds = rpmdsSingle(RPMTAG_REQUIRENAME, buf, EVR,
1616                         (RPMSENSE_EQUAL|RPMSENSE_CONFIG));
1617             xx = rpmdsMerge(&fc->requires, ds);
1618             ds = rpmdsFree(ds);
1619         }
1620     }
1621
1622     /* Build file class dictionary. */
1623     xx = rpmfcClassify(fc, av, fmode);
1624
1625     /* Build file/package dependency dictionary. */
1626     xx = rpmfcApply(fc);
1627
1628     /* Add per-file colors(#files) */
1629     p = (const void **) argiData(fc->fcolor);
1630     c = argiCount(fc->fcolor);
1631 assert(ac == c);
1632     if (p != NULL && c > 0) {
1633         int32_t * fcolors = (int32_t *)p;
1634         int i;
1635
1636         /* XXX Make sure only primary (i.e. Elf32/Elf64) colors are added. */
1637         for (i = 0; i < c; i++)
1638             fcolors[i] &= 0x0f;
1639         xx = headerAddEntry(pkg->header, RPMTAG_FILECOLORS, RPM_INT32_TYPE,
1640                         p, c);
1641     }
1642
1643     /* Add classes(#classes) */
1644     p = (const void **) argvData(fc->cdict);
1645     c = argvCount(fc->cdict);
1646     if (p != NULL && c > 0)
1647         xx = headerAddEntry(pkg->header, RPMTAG_CLASSDICT, RPM_STRING_ARRAY_TYPE,
1648                         p, c);
1649
1650     /* Add per-file classes(#files) */
1651     p = (const void **) argiData(fc->fcdictx);
1652     c = argiCount(fc->fcdictx);
1653 assert(ac == c);
1654     if (p != NULL && c > 0)
1655         xx = headerAddEntry(pkg->header, RPMTAG_FILECLASS, RPM_INT32_TYPE,
1656                         p, c);
1657
1658     /* Add Provides: */
1659     if (fc->provides != NULL && (c = rpmdsCount(fc->provides)) > 0 && !fc->skipProv) {
1660         const char **names = xcalloc(c, sizeof(char *));        
1661         const char **evrs = xcalloc(c, sizeof(char *)); 
1662         rpmsenseFlags *flags = xcalloc(c, sizeof(rpmsenseFlags *));     
1663         int i;
1664         rpmds pi = rpmdsInit(fc->provides);
1665         while ((i = rpmdsNext(pi)) >= 0) {
1666             names[i] = rpmdsN(pi);
1667             evrs[i] = rpmdsEVR(pi);
1668             flags[i] = rpmdsFlags(pi);
1669         }
1670          
1671 assert(names != NULL);
1672         xx = headerAddEntry(pkg->header, 
1673                             RPMTAG_PROVIDENAME, 
1674                             RPM_STRING_ARRAY_TYPE,
1675                             names, c);
1676     /* XXX rpm prior to 3.0.2 did not always supply EVR and Flags. */
1677 assert(evrs != NULL);
1678         xx = headerAddEntry(pkg->header, 
1679                             RPMTAG_PROVIDEVERSION, 
1680                             RPM_STRING_ARRAY_TYPE,
1681                             evrs, c);
1682 assert(flags != NULL);
1683         xx = headerAddEntry(pkg->header, 
1684                              RPMTAG_PROVIDEFLAGS, 
1685                              RPM_INT32_TYPE,
1686                              flags, c);
1687         free(names);
1688         free(evrs);
1689         free(flags);
1690     }
1691
1692     /* Add Requires: */
1693     if (fc->requires != NULL && (c = rpmdsCount(fc->requires)) > 0 && !fc->skipReq) {
1694         const char **names = xcalloc(c, sizeof(char *));        
1695         const char **evrs = xcalloc(c, sizeof(char *)); 
1696         rpmsenseFlags *flags = xcalloc(c, sizeof(rpmsenseFlags *));     
1697         int i;
1698         rpmds ri = rpmdsInit(fc->requires);
1699         while ((i = rpmdsNext(ri)) >= 0) {
1700             names[i] = rpmdsN(ri);
1701             evrs[i] = rpmdsEVR(ri);
1702             flags[i] = rpmdsFlags(ri);
1703         }
1704          
1705 assert(names != NULL);
1706         xx = headerAddEntry(pkg->header, 
1707                             RPMTAG_REQUIRENAME, 
1708                             RPM_STRING_ARRAY_TYPE,
1709                             names, c);
1710     /* XXX rpm prior to 3.0.2 did not always supply EVR and Flags. */
1711 assert(evrs != NULL);
1712         xx = headerAddEntry(pkg->header, 
1713                             RPMTAG_REQUIREVERSION, 
1714                             RPM_STRING_ARRAY_TYPE,
1715                             evrs, c);
1716 assert(flags != NULL);
1717         xx = headerAddEntry(pkg->header, 
1718                              RPMTAG_REQUIREFLAGS, 
1719                              RPM_INT32_TYPE,
1720                              flags, c);
1721         free(names);
1722         free(evrs);
1723         free(flags);
1724     }
1725
1726     /* Add dependency dictionary(#dependencies) */
1727     p = (const void **) argiData(fc->ddictx);
1728     c = argiCount(fc->ddictx);
1729     if (p != NULL)
1730         xx = headerAddEntry(pkg->header, RPMTAG_DEPENDSDICT, RPM_INT32_TYPE,
1731                         p, c);
1732
1733     /* Add per-file dependency (start,number) pairs (#files) */
1734     p = (const void **) argiData(fc->fddictx);
1735     c = argiCount(fc->fddictx);
1736 assert(ac == c);
1737     if (p != NULL)
1738         xx = headerAddEntry(pkg->header, RPMTAG_FILEDEPENDSX, RPM_INT32_TYPE,
1739                         p, c);
1740
1741     p = (const void **) argiData(fc->fddictn);
1742     c = argiCount(fc->fddictn);
1743 assert(ac == c);
1744     if (p != NULL)
1745         xx = headerAddEntry(pkg->header, RPMTAG_FILEDEPENDSN, RPM_INT32_TYPE,
1746                         p, c);
1747
1748     printDeps(pkg->header);
1749
1750 if (fc != NULL && _rpmfc_debug) {
1751 char msg[BUFSIZ];
1752 sprintf(msg, "final: files %d cdict[%d] %d%% ddictx[%d]", fc->nfiles, argvCount(fc->cdict), ((100 * fc->fknown)/fc->nfiles), argiCount(fc->ddictx));
1753 rpmfcPrint(msg, fc, NULL);
1754 }
1755
1756     /* Clean up. */
1757     fmode = _free(fmode);
1758     fc = rpmfcFree(fc);
1759     av = argvFree(av);
1760
1761     return rc;
1762 }