3 #include <signal.h> /* getOutputFrom() */
16 #if !defined(DT_GNU_HASH)
17 #define DT_GNU_HASH 0x6ffffef5
27 int nfiles; /*!< no. of files */
28 int fknown; /*!< no. of classified files */
29 int fwhite; /*!< no. of "white" files */
30 int ix; /*!< current file index */
31 int skipProv; /*!< Don't auto-generate Provides:? */
32 int skipReq; /*!< Don't auto-generate Requires:? */
33 int tracked; /*!< Versioned Provides: tracking dependency added? */
34 size_t brlen; /*!< strlen(spec->buildRoot) */
36 ARGV_t fn; /*!< (no. files) file names */
37 ARGI_t fcolor; /*!< (no. files) file colors */
38 ARGI_t fcdictx; /*!< (no. files) file class dictionary indices */
39 ARGI_t fddictx; /*!< (no. files) file depends dictionary start */
40 ARGI_t fddictn; /*!< (no. files) file depends dictionary no. entries */
41 ARGV_t cdict; /*!< (no. classes) file class dictionary */
42 ARGV_t ddict; /*!< (no. dependencies) file depends dictionary */
43 ARGI_t ddictx; /*!< (no. dependencies) file->dependency mapping */
45 rpmds provides; /*!< (no. provides) package provides */
46 rpmds requires; /*!< (no. requires) package requires */
48 StringBuf sb_java; /*!< concatenated list of java colored files. */
49 StringBuf sb_perl; /*!< concatenated list of perl colored files. */
50 StringBuf sb_python;/*!< concatenated list of python colored files. */
56 struct rpmfcTokens_s {
63 static int rpmfcExpandAppend(ARGV_t * argvp, const ARGV_t av)
66 int argc = argvCount(argv);
67 int ac = argvCount(av);
70 argv = xrealloc(argv, (argc + ac + 1) * sizeof(*argv));
71 for (i = 0; i < ac; i++)
72 argv[argc + i] = rpmExpand(av[i], NULL);
73 argv[argc + ac] = NULL;
79 * Return output from helper script.
80 * @todo Use poll(2) rather than select(2), if available.
81 * @param dir directory to run in (or NULL)
82 * @param argv program and arguments to run
83 * @param writePtr bytes to feed to script on stdin (or NULL)
84 * @param writeBytesLeft no. of bytes to feed to script on stdin
85 * @param failNonZero is script failure an error?
86 * @return buffered stdout from script, NULL on error
88 static StringBuf getOutputFrom(const char * dir, ARGV_t argv,
89 const char * writePtr, int writeBytesLeft,
101 oldhandler = signal(SIGPIPE, SIG_IGN);
103 toProg[0] = toProg[1] = 0;
105 fromProg[0] = fromProg[1] = 0;
106 (void) pipe(fromProg);
108 if (!(child = fork())) {
109 (void) close(toProg[1]);
110 (void) close(fromProg[0]);
112 (void) dup2(toProg[0], STDIN_FILENO); /* Make stdin the in pipe */
113 (void) dup2(fromProg[1], STDOUT_FILENO); /* Make stdout the out pipe */
115 (void) close(toProg[0]);
116 (void) close(fromProg[1]);
122 rpmlog(RPMLOG_DEBUG, _("\texecv(%s) pid %d\n"),
123 argv[0], (unsigned)getpid());
125 unsetenv("MALLOC_CHECK_");
126 (void) execvp(argv[0], (char *const *)argv);
127 /* XXX this error message is probably not seen. */
128 rpmlog(RPMLOG_ERR, _("Couldn't exec %s: %s\n"),
129 argv[0], strerror(errno));
133 rpmlog(RPMLOG_ERR, _("Couldn't fork %s: %s\n"),
134 argv[0], strerror(errno));
138 (void) close(toProg[0]);
139 (void) close(fromProg[1]);
141 /* Do not block reading or writing from/to prog. */
142 (void) fcntl(fromProg[0], F_SETFL, O_NONBLOCK);
143 (void) fcntl(toProg[1], F_SETFL, O_NONBLOCK);
145 readBuff = newStringBuf();
157 if (fromProg[0] >= 0) {
158 FD_SET(fromProg[0], &ibits);
160 if (toProg[1] >= 0) {
161 FD_SET(toProg[1], &obits);
163 /* XXX values set to limit spinning with perl doing ~100 forks/sec. */
166 nfd = ((fromProg[0] > toProg[1]) ? fromProg[0] : toProg[1]);
167 if ((rc = select(nfd, &ibits, &obits, NULL, &tv)) < 0) {
173 /* Write any data to program */
174 if (toProg[1] >= 0 && FD_ISSET(toProg[1], &obits)) {
175 if (writePtr && writeBytesLeft > 0) {
176 if ((nbw = write(toProg[1], writePtr,
177 (1024<writeBytesLeft) ? 1024 : writeBytesLeft)) < 0) {
178 if (errno != EAGAIN) {
179 perror("getOutputFrom()");
184 writeBytesLeft -= nbw;
186 } else if (toProg[1] >= 0) { /* close write fd */
187 (void) close(toProg[1]);
192 /* Read any data from prog */
193 { char buf[BUFSIZ+1];
194 while ((nbr = read(fromProg[0], buf, sizeof(buf)-1)) > 0) {
196 appendStringBuf(readBuff, buf);
200 /* terminate on (non-blocking) EOF or error */
201 done = (nbr == 0 || (nbr < 0 && errno != EAGAIN));
207 (void) close(toProg[1]);
208 if (fromProg[0] >= 0)
209 (void) close(fromProg[0]);
211 (void) signal(SIGPIPE, oldhandler);
213 /* Collect status from prog */
214 reaped = waitpid(child, &status, 0);
215 rpmlog(RPMLOG_DEBUG, _("\twaitpid(%d) rc %d status %x\n"),
216 (unsigned)child, (unsigned)reaped, status);
218 if (failNonZero && (!WIFEXITED(status) || WEXITSTATUS(status))) {
219 rpmlog(RPMLOG_ERR, _("%s failed\n"), argv[0]);
222 if (writeBytesLeft) {
223 rpmlog(RPMLOG_ERR, _("failed to write all data to %s\n"), argv[0]);
229 int rpmfcExec(ARGV_t av, StringBuf sb_stdin, StringBuf * sb_stdoutp,
232 const char * s = NULL;
238 const char * buf_stdin = NULL;
239 int buf_stdin_len = 0;
247 /* Find path to executable with (possible) args. */
248 s = rpmExpand(av[0], NULL);
252 /* Parse args buried within expanded exacutable. */
254 xx = poptParseArgvString(s, &pac, (const char ***)&pav);
255 if (!(xx == 0 && pac > 0 && pav != NULL))
258 /* Build argv, appending args to the executable args. */
260 xx = argvAppend(&xav, pav);
262 xx = rpmfcExpandAppend(&xav, av + 1);
264 if (sb_stdin != NULL) {
265 buf_stdin = getStringBuf(sb_stdin);
266 buf_stdin_len = strlen(buf_stdin);
269 /* Read output from exec'd helper. */
270 sb = getOutputFrom(NULL, xav, buf_stdin, buf_stdin_len, failnonzero);
272 if (sb_stdoutp != NULL) {
274 sb = NULL; /* XXX don't free */
280 sb = freeStringBuf(sb);
282 pav = _free(pav); /* XXX popt mallocs in single blob. */
289 static int rpmfcSaveArg(ARGV_t * argvp, const char * key)
293 if (argvSearch(*argvp, key, NULL) == NULL) {
294 rc = argvAdd(argvp, key);
295 rc = argvSort(*argvp, NULL);
300 static char * rpmfcFileDep(char * buf, int ix,
303 int32_t tagN = rpmdsTagN(ds);
308 case RPMTAG_PROVIDENAME:
311 case RPMTAG_REQUIRENAME:
316 sprintf(buf, "%08d%c %s %s 0x%08x", ix, deptype,
317 rpmdsN(ds), rpmdsEVR(ds), rpmdsFlags(ds));
322 * Run per-interpreter dependency helper.
323 * @param fc file classifier
324 * @param deptype 'P' == Provides:, 'R' == Requires:, helper
325 * @param nsdep class name for interpreter (e.g. "perl")
326 * @return 0 on success
328 static int rpmfcHelper(rpmfc fc, unsigned char deptype, const char * nsdep)
330 const char * fn = fc->fn[fc->ix];
332 StringBuf sb_stdout = NULL;
338 int32_t Flags, dsContext, tagN;
352 xx = snprintf(buf, sizeof(buf), "%%{?__%s_provides}", nsdep);
353 depsp = &fc->provides;
354 dsContext = RPMSENSE_FIND_PROVIDES;
355 tagN = RPMTAG_PROVIDENAME;
360 xx = snprintf(buf, sizeof(buf), "%%{?__%s_requires}", nsdep);
361 depsp = &fc->requires;
362 dsContext = RPMSENSE_FIND_REQUIRES;
363 tagN = RPMTAG_REQUIRENAME;
366 buf[sizeof(buf)-1] = '\0';
370 sb_stdin = newStringBuf();
371 appendLineStringBuf(sb_stdin, fn);
373 xx = rpmfcExec(av, sb_stdin, &sb_stdout, 0);
374 sb_stdin = freeStringBuf(sb_stdin);
376 if (xx == 0 && sb_stdout != NULL) {
378 xx = argvSplit(&pav, getStringBuf(sb_stdout), " \t\n\r");
379 pac = argvCount(pav);
381 for (i = 0; i < pac; i++) {
385 if (pav[i+1] && strchr("=<>", *pav[i+1])) {
387 for (s = pav[i]; *s; s++) {
393 Flags |= RPMSENSE_EQUAL;
396 Flags |= RPMSENSE_LESS;
399 Flags |= RPMSENSE_GREATER;
409 /* Add tracking dependency for versioned Provides: */
410 if (!fc->tracked && deptype == 'P' && *EVR != '\0') {
411 ds = rpmdsSingle(RPMTAG_REQUIRENAME,
412 "rpmlib(VersionedDependencies)", "3.0.3-1",
413 RPMSENSE_RPMLIB|(RPMSENSE_LESS|RPMSENSE_EQUAL));
414 xx = rpmdsMerge(&fc->requires, ds);
419 ds = rpmdsSingle(tagN, N, EVR, Flags);
421 /* Add to package dependencies. */
422 xx = rpmdsMerge(depsp, ds);
424 /* Add to file dependencies. */
425 xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(buf, fc->ix, ds));
432 sb_stdout = freeStringBuf(sb_stdout);
439 static struct rpmfcTokens_s rpmfcTokens[] = {
440 { "directory", RPMFC_DIRECTORY|RPMFC_INCLUDE },
442 { " shared object", RPMFC_LIBRARY },
443 { " executable", RPMFC_EXECUTABLE },
444 { " statically linked", RPMFC_STATIC },
445 { " not stripped", RPMFC_NOTSTRIPPED },
446 { " archive", RPMFC_ARCHIVE },
448 { "ELF 32-bit", RPMFC_ELF32|RPMFC_INCLUDE },
449 { "ELF 64-bit", RPMFC_ELF64|RPMFC_INCLUDE },
451 { " script", RPMFC_SCRIPT },
452 { " text", RPMFC_TEXT },
453 { " document", RPMFC_DOCUMENT },
455 { " compressed", RPMFC_COMPRESSED },
457 { "troff or preprocessor input", RPMFC_MANPAGE|RPMFC_INCLUDE },
458 { "GNU Info", RPMFC_MANPAGE|RPMFC_INCLUDE },
460 { "perl script text", RPMFC_PERL|RPMFC_INCLUDE },
461 { "Perl5 module source text", RPMFC_PERL|RPMFC_MODULE|RPMFC_INCLUDE },
463 { " /usr/bin/python", RPMFC_PYTHON|RPMFC_INCLUDE },
465 /* XXX "a /usr/bin/python -t script text executable" */
466 /* XXX "python 2.3 byte-compiled" */
467 { "python ", RPMFC_PYTHON|RPMFC_INCLUDE },
469 { "libtool library ", RPMFC_LIBTOOL|RPMFC_INCLUDE },
470 { "pkgconfig ", RPMFC_PKGCONFIG|RPMFC_INCLUDE },
472 /* XXX .NET executables and libraries. file(1) cannot differ from win32
473 * executables unfortunately :( */
474 { "PE executable", RPMFC_MONO|RPMFC_INCLUDE },
476 { "current ar archive", RPMFC_STATIC|RPMFC_LIBRARY|RPMFC_ARCHIVE|RPMFC_INCLUDE },
478 { "Zip archive data", RPMFC_COMPRESSED|RPMFC_ARCHIVE|RPMFC_INCLUDE },
479 { "tar archive", RPMFC_ARCHIVE|RPMFC_INCLUDE },
480 { "cpio archive", RPMFC_ARCHIVE|RPMFC_INCLUDE },
481 { "RPM v3", RPMFC_ARCHIVE|RPMFC_INCLUDE },
482 { "RPM v4", RPMFC_ARCHIVE|RPMFC_INCLUDE },
484 { " image", RPMFC_IMAGE|RPMFC_INCLUDE },
485 { " font", RPMFC_FONT|RPMFC_INCLUDE },
486 { " Font", RPMFC_FONT|RPMFC_INCLUDE },
488 { " commands", RPMFC_SCRIPT|RPMFC_INCLUDE },
489 { " script", RPMFC_SCRIPT|RPMFC_INCLUDE },
491 { "empty", RPMFC_WHITE|RPMFC_INCLUDE },
493 { "HTML", RPMFC_WHITE|RPMFC_INCLUDE },
494 { "SGML", RPMFC_WHITE|RPMFC_INCLUDE },
495 { "XML", RPMFC_WHITE|RPMFC_INCLUDE },
497 { " program text", RPMFC_WHITE|RPMFC_INCLUDE },
498 { " source", RPMFC_WHITE|RPMFC_INCLUDE },
499 { "GLS_BINARY_LSB_FIRST", RPMFC_WHITE|RPMFC_INCLUDE },
500 { " DB ", RPMFC_WHITE|RPMFC_INCLUDE },
502 { "ASCII English text", RPMFC_WHITE|RPMFC_INCLUDE },
503 { "ASCII text", RPMFC_WHITE|RPMFC_INCLUDE },
504 { "ISO-8859 text", RPMFC_WHITE|RPMFC_INCLUDE },
506 { "symbolic link to", RPMFC_SYMLINK },
507 { "socket", RPMFC_DEVICE },
508 { "special", RPMFC_DEVICE },
510 { "ASCII", RPMFC_WHITE },
511 { "ISO-8859", RPMFC_WHITE },
513 { "data", RPMFC_WHITE },
515 { "application", RPMFC_WHITE },
516 { "boot", RPMFC_WHITE },
517 { "catalog", RPMFC_WHITE },
518 { "code", RPMFC_WHITE },
519 { "file", RPMFC_WHITE },
520 { "format", RPMFC_WHITE },
521 { "message", RPMFC_WHITE },
522 { "program", RPMFC_WHITE },
524 { "broken symbolic link to ", RPMFC_WHITE|RPMFC_ERROR },
525 { "can't read", RPMFC_WHITE|RPMFC_ERROR },
526 { "can't stat", RPMFC_WHITE|RPMFC_ERROR },
527 { "executable, can't read", RPMFC_WHITE|RPMFC_ERROR },
528 { "core file", RPMFC_WHITE|RPMFC_ERROR },
530 { NULL, RPMFC_BLACK }
533 int rpmfcColoring(const char * fmstr)
536 int fcolor = RPMFC_BLACK;
538 for (fct = rpmfcTokens; fct->token != NULL; fct++) {
539 if (strstr(fmstr, fct->token) == NULL)
541 fcolor |= fct->colors;
542 if (fcolor & RPMFC_INCLUDE)
548 void rpmfcPrint(const char * msg, rpmfc fc, FILE * fp)
559 if (fp == NULL) fp = stderr;
562 fprintf(fp, "===================================== %s\n", msg);
564 nprovides = rpmdsCount(fc->provides);
565 nrequires = rpmdsCount(fc->requires);
568 for (fx = 0; fx < fc->nfiles; fx++) {
569 assert(fx < fc->fcdictx->nvals);
570 cx = fc->fcdictx->vals[fx];
571 assert(fx < fc->fcolor->nvals);
572 fcolor = fc->fcolor->vals[fx];
574 fprintf(fp, "%3d %s", fx, fc->fn[fx]);
575 if (fcolor != RPMFC_BLACK)
576 fprintf(fp, "\t0x%x", fc->fcolor->vals[fx]);
578 fprintf(fp, "\t%s", fc->cdict[cx]);
581 if (fc->fddictx == NULL || fc->fddictn == NULL)
584 assert(fx < fc->fddictx->nvals);
585 dx = fc->fddictx->vals[fx];
586 assert(fx < fc->fddictn->nvals);
587 ndx = fc->fddictn->vals[fx];
591 unsigned char deptype;
594 ix = fc->ddictx->vals[dx++];
595 deptype = ((ix >> 24) & 0xff);
600 assert(depval != NULL);
604 assert(ix < nprovides);
605 (void) rpmdsSetIx(fc->provides, ix-1);
606 if (rpmdsNext(fc->provides) >= 0)
607 depval = rpmdsDNEVR(fc->provides);
612 assert(ix < nrequires);
613 (void) rpmdsSetIx(fc->requires, ix-1);
614 if (rpmdsNext(fc->requires) >= 0)
615 depval = rpmdsDNEVR(fc->requires);
620 fprintf(fp, "\t%s\n", depval);
625 rpmfc rpmfcFree(rpmfc fc)
628 fc->fn = argvFree(fc->fn);
629 fc->fcolor = argiFree(fc->fcolor);
630 fc->fcdictx = argiFree(fc->fcdictx);
631 fc->fddictx = argiFree(fc->fddictx);
632 fc->fddictn = argiFree(fc->fddictn);
633 fc->cdict = argvFree(fc->cdict);
634 fc->ddict = argvFree(fc->ddict);
635 fc->ddictx = argiFree(fc->ddictx);
637 fc->provides = rpmdsFree(fc->provides);
638 fc->requires = rpmdsFree(fc->requires);
640 fc->sb_java = freeStringBuf(fc->sb_java);
641 fc->sb_perl = freeStringBuf(fc->sb_perl);
642 fc->sb_python = freeStringBuf(fc->sb_python);
651 rpmfc fc = xcalloc(1, sizeof(*fc));
655 rpmds rpmfcProvides(rpmfc fc)
657 return (fc != NULL ? fc->provides : NULL);
660 rpmds rpmfcRequires(rpmfc fc)
662 return (fc != NULL ? fc->requires : NULL);
667 * Extract script dependencies.
668 * @param fc file classifier
669 * @return 0 on success
671 static int rpmfcSCRIPT(rpmfc fc)
673 const char * fn = fc->fn[fc->ix];
680 struct stat sb, * st = &sb;
684 /* Only executable scripts are searched. */
685 if (stat(fn, st) < 0)
687 is_executable = (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH));
690 if (fp == NULL || ferror(fp)) {
691 if (fp) (void) fclose(fp);
695 /* Look for #! interpreter in first 10 lines. */
696 for (i = 0; i < 10; i++) {
698 s = fgets(buf, sizeof(buf) - 1, fp);
699 if (s == NULL || ferror(fp) || feof(fp))
701 s[sizeof(buf)-1] = '\0';
702 if (!(s[0] == '#' && s[1] == '!'))
706 while (*s && strchr(" \t\n\r", *s) != NULL)
713 for (se = s+1; *se; se++) {
714 if (strchr(" \t\n\r", *se) != NULL)
721 /* Add to package requires. */
722 ds = rpmdsSingle(RPMTAG_REQUIRENAME, s, "", RPMSENSE_FIND_REQUIRES);
723 xx = rpmdsMerge(&fc->requires, ds);
725 /* Add to file requires. */
726 xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(se, fc->ix, ds));
731 /* Set color based on interpreter name. */
733 if (!strcmp(bn, "perl"))
734 fc->fcolor->vals[fc->ix] |= RPMFC_PERL;
735 else if (!strncmp(bn, "python", sizeof("python")-1))
736 fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
743 if (fc->fcolor->vals[fc->ix] & RPMFC_PERL) {
744 if (fc->fcolor->vals[fc->ix] & RPMFC_MODULE)
745 xx = rpmfcHelper(fc, 'P', "perl");
746 if (is_executable || (fc->fcolor->vals[fc->ix] & RPMFC_MODULE))
747 xx = rpmfcHelper(fc, 'R', "perl");
749 if (fc->fcolor->vals[fc->ix] & RPMFC_PYTHON) {
750 xx = rpmfcHelper(fc, 'P', "python");
754 xx = rpmfcHelper(fc, 'R', "python");
756 if (fc->fcolor->vals[fc->ix] & RPMFC_LIBTOOL) {
757 xx = rpmfcHelper(fc, 'P', "libtool");
761 xx = rpmfcHelper(fc, 'R', "libtool");
763 if (fc->fcolor->vals[fc->ix] & RPMFC_PKGCONFIG) {
764 xx = rpmfcHelper(fc, 'P', "pkgconfig");
768 xx = rpmfcHelper(fc, 'R', "pkgconfig");
770 if (fc->fcolor->vals[fc->ix] & RPMFC_MONO) {
771 xx = rpmfcHelper(fc, 'P', "mono");
773 xx = rpmfcHelper(fc, 'R', "mono");
780 * Extract Elf dependencies.
781 * @param fc file classifier
782 * @return 0 on success
784 static int rpmfcELF(rpmfc fc)
786 #if HAVE_GELF_H && HAVE_LIBELF
787 const char * fn = fc->fn[fc->ix];
791 GElf_Ehdr ehdr_mem, * ehdr;
792 GElf_Shdr shdr_mem, * shdr;
793 GElf_Verdef def_mem, * def;
794 GElf_Verneed need_mem, * need;
795 GElf_Dyn dyn_mem, * dyn;
796 unsigned int auxoffset;
803 struct stat sb, * st = &sb;
804 const char * soname = NULL;
806 int32_t tagN, dsContext;
815 static int filter_GLIBC_PRIVATE = 0;
816 static int oneshot = 0;
820 filter_GLIBC_PRIVATE = rpmExpandNumeric("%{?_filter_GLIBC_PRIVATE}");
823 /* Files with executable bit set only. */
824 if (stat(fn, st) != 0)
827 fdno = open(fn, O_RDONLY);
831 (void) elf_version(EV_CURRENT);
834 if ((elf = elf_begin (fdno, ELF_C_READ, NULL)) == NULL
835 || elf_kind(elf) != ELF_K_ELF
836 || (ehdr = gelf_getehdr(elf, &ehdr_mem)) == NULL
837 || !(ehdr->e_type == ET_DYN || ehdr->e_type == ET_EXEC))
840 isElf64 = ehdr->e_ident[EI_CLASS] == ELFCLASS64;
841 isDSO = ehdr->e_type == ET_DYN;
844 while ((scn = elf_nextscn(elf, scn)) != NULL) {
845 shdr = gelf_getshdr(scn, &shdr_mem);
849 soname = _free(soname);
850 switch (shdr->sh_type) {
857 while ((data = elf_getdata (scn, data)) != NULL) {
859 for (cnt = shdr->sh_info; --cnt >= 0; ) {
861 def = gelf_getverdef (data, offset, &def_mem);
864 auxoffset = offset + def->vd_aux;
865 for (cnt2 = def->vd_cnt; --cnt2 >= 0; ) {
866 GElf_Verdaux aux_mem, * aux;
868 aux = gelf_getverdaux (data, auxoffset, &aux_mem);
872 s = elf_strptr(elf, shdr->sh_link, aux->vda_name);
875 if (def->vd_flags & VER_FLG_BASE) {
876 soname = _free(soname);
878 auxoffset += aux->vda_next;
882 && !(filter_GLIBC_PRIVATE != 0
883 && !strcmp(s, "GLIBC_PRIVATE")))
887 t = stpcpy( stpcpy( stpcpy( stpcpy(t, soname), "("), s), ")");
889 #if !defined(__alpha__)
891 t = stpcpy(t, "(64bit)");
895 /* Add to package provides. */
896 ds = rpmdsSingle(RPMTAG_PROVIDES,
897 buf, "", RPMSENSE_FIND_PROVIDES);
898 xx = rpmdsMerge(&fc->provides, ds);
900 /* Add to file dependencies. */
901 xx = rpmfcSaveArg(&fc->ddict,
902 rpmfcFileDep(t, fc->ix, ds));
906 auxoffset += aux->vda_next;
908 offset += def->vd_next;
912 case SHT_GNU_verneed:
914 /* Files with executable bit set only. */
915 if (!fc->skipReq && (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
916 while ((data = elf_getdata (scn, data)) != NULL) {
918 for (cnt = shdr->sh_info; --cnt >= 0; ) {
919 need = gelf_getverneed (data, offset, &need_mem);
923 s = elf_strptr(elf, shdr->sh_link, need->vn_file);
926 soname = _free(soname);
928 auxoffset = offset + need->vn_aux;
929 for (cnt2 = need->vn_cnt; --cnt2 >= 0; ) {
930 GElf_Vernaux aux_mem, * aux;
932 aux = gelf_getvernaux (data, auxoffset, &aux_mem);
936 s = elf_strptr(elf, shdr->sh_link, aux->vna_name);
940 /* Filter dependencies that contain GLIBC_PRIVATE */
942 && !(filter_GLIBC_PRIVATE != 0
943 && !strcmp(s, "GLIBC_PRIVATE")))
947 t = stpcpy( stpcpy( stpcpy( stpcpy(t, soname), "("), s), ")");
949 #if !defined(__alpha__)
951 t = stpcpy(t, "(64bit)");
955 /* Add to package dependencies. */
956 ds = rpmdsSingle(RPMTAG_REQUIRENAME,
957 buf, "", RPMSENSE_FIND_REQUIRES);
958 xx = rpmdsMerge(&fc->requires, ds);
960 /* Add to file dependencies. */
961 xx = rpmfcSaveArg(&fc->ddict,
962 rpmfcFileDep(t, fc->ix, ds));
965 auxoffset += aux->vna_next;
967 offset += need->vn_next;
973 while ((data = elf_getdata (scn, data)) != NULL) {
974 for (cnt = 0; cnt < (shdr->sh_size / shdr->sh_entsize); ++cnt) {
975 dyn = gelf_getdyn (data, cnt, &dyn_mem);
979 switch (dyn->d_tag) {
993 /* Files with executable bit set only. */
994 if (fc->skipReq || !(st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
996 /* Add to package requires. */
997 depsp = &fc->requires;
998 tagN = RPMTAG_REQUIRENAME;
999 dsContext = RPMSENSE_FIND_REQUIRES;
1000 s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1005 /* Add to package provides. */
1008 depsp = &fc->provides;
1009 tagN = RPMTAG_PROVIDENAME;
1010 dsContext = RPMSENSE_FIND_PROVIDES;
1011 s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1022 #if !defined(__alpha__)
1024 t = stpcpy(t, "()(64bit)");
1028 /* Add to package dependencies. */
1029 ds = rpmdsSingle(tagN, buf, "", dsContext);
1030 xx = rpmdsMerge(depsp, ds);
1032 /* Add to file dependencies. */
1033 xx = rpmfcSaveArg(&fc->ddict,
1034 rpmfcFileDep(t, fc->ix, ds));
1043 /* For DSOs which use the .gnu_hash section and don't have a .hash
1044 * section, we need to ensure that we have a new enough glibc. */
1045 if (gotGNUHASH && !gotHASH) {
1046 ds = rpmdsSingle(RPMTAG_REQUIRENAME, "rtld(GNU_HASH)", "",
1047 RPMSENSE_FIND_REQUIRES);
1048 rpmdsMerge(&fc->requires, ds);
1051 rpmfcSaveArg(&fc->ddict, rpmfcFileDep(t, fc->ix, ds));
1055 /* For DSO's, provide the basename of the file if DT_SONAME not found. */
1056 if (!fc->skipProv && isDSO && !gotDEBUG && !gotSONAME) {
1057 depsp = &fc->provides;
1058 tagN = RPMTAG_PROVIDENAME;
1059 dsContext = RPMSENSE_FIND_PROVIDES;
1061 s = strrchr(fn, '/');
1069 /* LCL: s is not null. */
1072 #if !defined(__alpha__)
1074 t = stpcpy(t, "()(64bit)");
1078 /* Add to package dependencies. */
1079 ds = rpmdsSingle(tagN, buf, "", dsContext);
1080 xx = rpmdsMerge(depsp, ds);
1082 /* Add to file dependencies. */
1083 xx = rpmfcSaveArg(&fc->ddict, rpmfcFileDep(t, fc->ix, ds));
1089 soname = _free(soname);
1090 if (elf) (void) elf_end(elf);
1098 typedef struct rpmfcApplyTbl_s {
1099 int (*func) (rpmfc fc);
1105 static struct rpmfcApplyTbl_s rpmfcApplyTable[] = {
1106 { rpmfcELF, RPMFC_ELF },
1107 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PERL) },
1108 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PYTHON) },
1109 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PKGCONFIG) },
1110 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_LIBTOOL) },
1111 { rpmfcSCRIPT, RPMFC_MONO },
1115 int rpmfcApply(rpmfc fc)
1124 unsigned char deptype;
1133 /* Generate package and per-file dependencies. */
1134 for (fc->ix = 0; fc->fn[fc->ix] != NULL; fc->ix++) {
1136 /* XXX Insure that /usr/lib{,64}/python files are marked RPMFC_PYTHON */
1137 /* XXX HACK: classification by path is intrinsically stupid. */
1138 { const char *fn = strstr(fc->fn[fc->ix], "/usr/lib");
1140 fn += sizeof("/usr/lib")-1;
1141 if (fn[0] == '6' && fn[1] == '4')
1143 if (!strncmp(fn, "/python", sizeof("/python")-1))
1144 fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
1148 if (fc->fcolor->vals[fc->ix])
1149 for (fcat = rpmfcApplyTable; fcat->func != NULL; fcat++) {
1150 if (!(fc->fcolor->vals[fc->ix] & fcat->colormask))
1152 xx = (*fcat->func) (fc);
1156 /* Generate per-file indices into package dependencies. */
1157 nddict = argvCount(fc->ddict);
1159 for (i = 0; i < nddict; i++) {
1162 /* Parse out (file#,deptype,N,EVR,Flags) */
1163 ix = strtol(s, &se, 10);
1168 while (*se && *se != ' ')
1172 while (*se && *se != ' ')
1175 Flags = strtol(se, NULL, 16);
1182 ds = rpmdsSingle(RPMTAG_PROVIDENAME, N, EVR, Flags);
1183 dix = rpmdsFind(fc->provides, ds);
1187 ds = rpmdsSingle(RPMTAG_REQUIRENAME, N, EVR, Flags);
1188 dix = rpmdsFind(fc->requires, ds);
1193 /* XXX assertion incorrect while generating -debuginfo deps. */
1201 val = (deptype << 24) | (dix & 0x00ffffff);
1202 xx = argiAdd(&fc->ddictx, -1, val);
1206 xx = argiAdd(&fc->fddictx, ix, argiCount(fc->ddictx)-1);
1208 if (fc->fddictn && fc->fddictn->vals)
1209 fc->fddictn->vals[ix]++;
1215 int rpmfcClassify(rpmfc fc, ARGV_t argv, int16_t * fmode)
1219 const char * s, * se;
1223 int msflags = MAGIC_CHECK; /* XXX MAGIC_COMPRESS flag? */
1226 if (fc == NULL || argv == NULL)
1229 fc->nfiles = argvCount(argv);
1231 /* Initialize the per-file dictionary indices. */
1232 xx = argiAdd(&fc->fddictx, fc->nfiles-1, 0);
1233 xx = argiAdd(&fc->fddictn, fc->nfiles-1, 0);
1235 /* Build (sorted) file class dictionary. */
1236 xx = argvAdd(&fc->cdict, "");
1237 xx = argvAdd(&fc->cdict, "directory");
1239 ms = magic_open(msflags);
1242 rpmlog(xx, _("magic_open(0x%x) failed: %s\n"),
1243 msflags, strerror(errno));
1244 assert(ms != NULL); /* XXX figger a proper return path. */
1247 xx = magic_load(ms, NULL);
1250 rpmlog(xx, _("magic_load failed: %s\n"), magic_error(ms));
1251 assert(xx != -1); /* XXX figger a proper return path. */
1254 for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1256 int16_t mode = (fmode ? fmode[fc->ix] : 0);
1262 switch (mode & S_IFMT) {
1263 case S_IFCHR: ftype = "character special"; break;
1264 case S_IFBLK: ftype = "block special"; break;
1265 case S_IFIFO: ftype = "fifo (named pipe)"; break;
1266 case S_IFSOCK: ftype = "socket"; break;
1271 #define _suffix(_s, _x) \
1272 (slen >= sizeof(_x) && !strcmp((_s)+slen-(sizeof(_x)-1), (_x)))
1274 /* XXX all files with extension ".pm" are perl modules for now. */
1275 if (_suffix(s, ".pm"))
1276 ftype = "Perl5 module source text";
1278 /* XXX all files with extension ".la" are libtool for now. */
1279 else if (_suffix(s, ".la"))
1280 ftype = "libtool library file";
1282 /* XXX all files with extension ".pc" are pkgconfig for now. */
1283 else if (_suffix(s, ".pc"))
1284 ftype = "pkgconfig file";
1286 /* XXX skip all files in /dev/ which are (or should be) %dev dummies. */
1287 else if (slen >= fc->brlen+sizeof("/dev/") && !strncmp(s+fc->brlen, "/dev/", sizeof("/dev/")-1))
1290 ftype = magic_file(ms, s);
1292 if (ftype == NULL) {
1294 rpmlog(xx, _("magic_file(ms, \"%s\") failed: mode %06o %s\n"),
1295 s, mode, magic_error(ms));
1296 assert(ftype != NULL); /* XXX figger a proper return path. */
1301 rpmlog(RPMLOG_DEBUG, "%s: %s\n", s, se);
1303 /* Save the path. */
1304 xx = argvAdd(&fc->fn, s);
1306 /* Save the file type string. */
1307 xx = argvAdd(&fcav, se);
1309 /* Add (filtered) entry to sorted class dictionary. */
1310 fcolor = rpmfcColoring(se);
1311 xx = argiAdd(&fc->fcolor, fc->ix, fcolor);
1313 if (fcolor != RPMFC_WHITE && (fcolor & RPMFC_INCLUDE))
1314 xx = rpmfcSaveArg(&fc->cdict, se);
1317 /* Build per-file class index array. */
1319 for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1323 dav = argvSearch(fc->cdict, se, NULL);
1325 xx = argiAdd(&fc->fcdictx, fc->ix, (dav - fc->cdict));
1328 xx = argiAdd(&fc->fcdictx, fc->ix, 0);
1333 fcav = argvFree(fcav);
1343 typedef struct DepMsg_s * DepMsg_t;
1349 const char * argv[4];
1359 static struct DepMsg_s depMsgs[] = {
1360 { "Provides", { "%{?__find_provides}", NULL, NULL, NULL },
1361 RPMTAG_PROVIDENAME, RPMTAG_PROVIDEVERSION, RPMTAG_PROVIDEFLAGS,
1363 { "Requires(interp)", { NULL, "interp", NULL, NULL },
1364 RPMTAG_REQUIRENAME, RPMTAG_REQUIREVERSION, RPMTAG_REQUIREFLAGS,
1365 _notpre(RPMSENSE_INTERP), 0 },
1366 { "Requires(rpmlib)", { NULL, "rpmlib", NULL, NULL },
1367 -1, -1, RPMTAG_REQUIREFLAGS,
1368 _notpre(RPMSENSE_RPMLIB), 0 },
1369 { "Requires(verify)", { NULL, "verify", NULL, NULL },
1370 -1, -1, RPMTAG_REQUIREFLAGS,
1371 RPMSENSE_SCRIPT_VERIFY, 0 },
1372 { "Requires(pre)", { NULL, "pre", NULL, NULL },
1373 -1, -1, RPMTAG_REQUIREFLAGS,
1374 _notpre(RPMSENSE_SCRIPT_PRE), 0 },
1375 { "Requires(post)", { NULL, "post", NULL, NULL },
1376 -1, -1, RPMTAG_REQUIREFLAGS,
1377 _notpre(RPMSENSE_SCRIPT_POST), 0 },
1378 { "Requires(preun)", { NULL, "preun", NULL, NULL },
1379 -1, -1, RPMTAG_REQUIREFLAGS,
1380 _notpre(RPMSENSE_SCRIPT_PREUN), 0 },
1381 { "Requires(postun)", { NULL, "postun", NULL, NULL },
1382 -1, -1, RPMTAG_REQUIREFLAGS,
1383 _notpre(RPMSENSE_SCRIPT_POSTUN), 0 },
1384 { "Requires", { "%{?__find_requires}", NULL, NULL, NULL },
1385 -1, -1, RPMTAG_REQUIREFLAGS, /* XXX inherit name/version arrays */
1386 RPMSENSE_PREREQ, RPMSENSE_PREREQ },
1387 { "Conflicts", { "%{?__find_conflicts}", NULL, NULL, NULL },
1388 RPMTAG_CONFLICTNAME, RPMTAG_CONFLICTVERSION, RPMTAG_CONFLICTFLAGS,
1390 { "Obsoletes", { "%{?__find_obsoletes}", NULL, NULL, NULL },
1391 RPMTAG_OBSOLETENAME, RPMTAG_OBSOLETEVERSION, RPMTAG_OBSOLETEFLAGS,
1393 { NULL, { NULL, NULL, NULL, NULL }, 0, 0, 0, 0, 0 }
1396 static DepMsg_t DepMsgs = depMsgs;
1400 static void printDeps(Header h)
1404 int flags = 0; /* XXX !scareMem */
1409 for (dm = DepMsgs; dm->msg != NULL; dm++) {
1410 if (dm->ntag != -1) {
1412 ds = rpmdsNew(h, dm->ntag, flags);
1419 continue; /* XXX can't happen */
1422 while (rpmdsNext(ds) >= 0) {
1424 Flags = rpmdsFlags(ds);
1426 if (!((Flags & dm->mask) ^ dm->xor))
1429 rpmlog(RPMLOG_NOTICE, "%s:", (dm->msg ? dm->msg : ""));
1432 if ((DNEVR = rpmdsDNEVR(ds)) == NULL)
1433 continue; /* XXX can't happen */
1434 rpmlog(RPMLOG_NOTICE, " %s", DNEVR+2);
1437 rpmlog(RPMLOG_NOTICE, "\n");
1444 static int rpmfcGenerateDependsHelper(const rpmSpec spec, Package pkg, rpmfi fi)
1447 StringBuf sb_stdout;
1449 int failnonzero = 0;
1453 * Create file manifest buffer to deliver to dependency finder.
1455 sb_stdin = newStringBuf();
1456 fi = rpmfiInit(fi, 0);
1458 while (rpmfiNext(fi) >= 0)
1459 appendLineStringBuf(sb_stdin, rpmfiFN(fi));
1461 for (dm = DepMsgs; dm->msg != NULL; dm++) {
1466 tag = (dm->ftag > 0) ? dm->ftag : dm->ntag;
1471 case RPMTAG_PROVIDEFLAGS:
1475 tagflags = RPMSENSE_FIND_PROVIDES;
1477 case RPMTAG_REQUIREFLAGS:
1481 tagflags = RPMSENSE_FIND_REQUIRES;
1488 xx = rpmfcExec(dm->argv, sb_stdin, &sb_stdout, failnonzero);
1492 s = rpmExpand(dm->argv[0], NULL);
1493 rpmlog(RPMLOG_NOTICE, _("Finding %s: %s\n"), dm->msg,
1497 if (sb_stdout == NULL) {
1499 rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1503 /* Parse dependencies into header */
1504 rc = parseRCPOT(spec, pkg, getStringBuf(sb_stdout), tag, 0, tagflags);
1505 sb_stdout = freeStringBuf(sb_stdout);
1508 rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1513 sb_stdin = freeStringBuf(sb_stdin);
1518 int rpmfcGenerateDepends(const rpmSpec spec, Package pkg)
1520 rpmfi fi = pkg->cpioList;
1523 int flags = 0; /* XXX !scareMem */
1526 int ac = rpmfiFC(fi);
1536 /* Skip packages with no files. */
1540 /* Skip packages that have dependency generation disabled. */
1541 if (! (pkg->autoReq || pkg->autoProv))
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);
1552 /* Extract absolute file paths in argv format. */
1553 av = xcalloc(ac+1, sizeof(*av));
1554 fmode = xcalloc(ac+1, sizeof(*fmode));
1557 fi = rpmfiInit(fi, 0);
1559 while ((c = rpmfiNext(fi)) >= 0) {
1560 rpmfileAttrs fileAttrs;
1562 /* Does package have any %config files? */
1563 fileAttrs = rpmfiFFlags(fi);
1564 genConfigDeps |= (fileAttrs & RPMFILE_CONFIG);
1566 av[c] = xstrdup(rpmfiFN(fi));
1567 fmode[c] = rpmfiFMode(fi);
1572 fc->skipProv = !pkg->autoProv;
1573 fc->skipReq = !pkg->autoReq;
1575 fc->brlen = (spec->buildRootURL ? strlen(spec->buildRootURL) : 0);
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);
1582 xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDENAME);
1583 xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDEVERSION);
1584 xx = headerRemoveEntry(pkg->header, RPMTAG_PROVIDEFLAGS);
1586 /* Add config dependency, Provides: config(N) = EVR */
1587 if (genConfigDeps) {
1588 N = rpmdsN(pkg->ds);
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);
1601 ds = rpmdsNew(pkg->header, RPMTAG_REQUIRENAME, flags);
1602 xx = rpmdsMerge(&fc->requires, ds);
1604 xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIRENAME);
1605 xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIREVERSION);
1606 xx = headerRemoveEntry(pkg->header, RPMTAG_REQUIREFLAGS);
1608 /* Add config dependency, Requires: config(N) = EVR */
1609 if (genConfigDeps) {
1610 N = rpmdsN(pkg->ds);
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);
1622 /* Build file class dictionary. */
1623 xx = rpmfcClassify(fc, av, fmode);
1625 /* Build file/package dependency dictionary. */
1626 xx = rpmfcApply(fc);
1628 /* Add per-file colors(#files) */
1629 p = (const void **) argiData(fc->fcolor);
1630 c = argiCount(fc->fcolor);
1632 if (p != NULL && c > 0) {
1633 int32_t * fcolors = (int32_t *)p;
1636 /* XXX Make sure only primary (i.e. Elf32/Elf64) colors are added. */
1637 for (i = 0; i < c; i++)
1639 xx = headerAddEntry(pkg->header, RPMTAG_FILECOLORS, RPM_INT32_TYPE,
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,
1650 /* Add per-file classes(#files) */
1651 p = (const void **) argiData(fc->fcdictx);
1652 c = argiCount(fc->fcdictx);
1654 if (p != NULL && c > 0)
1655 xx = headerAddEntry(pkg->header, RPMTAG_FILECLASS, RPM_INT32_TYPE,
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 int32_t *flags = xcalloc(c, sizeof(int32_t *));
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);
1671 assert(names != NULL);
1672 xx = headerAddEntry(pkg->header,
1674 RPM_STRING_ARRAY_TYPE,
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,
1682 assert(flags != NULL);
1683 xx = headerAddEntry(pkg->header,
1684 RPMTAG_PROVIDEFLAGS,
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 int32_t *flags = xcalloc(c, sizeof(int32_t *));
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);
1705 assert(names != NULL);
1706 xx = headerAddEntry(pkg->header,
1708 RPM_STRING_ARRAY_TYPE,
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,
1716 assert(flags != NULL);
1717 xx = headerAddEntry(pkg->header,
1718 RPMTAG_REQUIREFLAGS,
1726 /* Add dependency dictionary(#dependencies) */
1727 p = (const void **) argiData(fc->ddictx);
1728 c = argiCount(fc->ddictx);
1730 xx = headerAddEntry(pkg->header, RPMTAG_DEPENDSDICT, RPM_INT32_TYPE,
1733 /* Add per-file dependency (start,number) pairs (#files) */
1734 p = (const void **) argiData(fc->fddictx);
1735 c = argiCount(fc->fddictx);
1738 xx = headerAddEntry(pkg->header, RPMTAG_FILEDEPENDSX, RPM_INT32_TYPE,
1741 p = (const void **) argiData(fc->fddictn);
1742 c = argiCount(fc->fddictn);
1745 xx = headerAddEntry(pkg->header, RPMTAG_FILEDEPENDSN, RPM_INT32_TYPE,
1748 printDeps(pkg->header);
1750 if (fc != NULL && _rpmfc_debug) {
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);
1757 fmode = _free(fmode);