7 #if !defined(DT_GNU_HASH)
8 #define DT_GNU_HASH 0x6ffffef5
13 #include <rpm/header.h>
14 #include <rpm/rpmbuild.h>
16 #include <rpm/rpmfc.h>
17 #include <rpm/rpmlog.h>
18 #include <rpm/rpmfileutil.h>
20 #include <rpm/rpmds.h>
21 #include <rpm/rpmfi.h>
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) */
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 */
46 rpmds provides; /*!< (no. provides) package provides */
47 rpmds requires; /*!< (no. requires) package requires */
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. */
57 struct rpmfcTokens_s {
64 static int rpmfcExpandAppend(ARGV_t * argvp, ARGV_const_t av)
67 int argc = argvCount(argv);
68 int ac = argvCount(av);
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;
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
89 static StringBuf getOutputFrom(const char * dir, ARGV_t argv,
90 const char * writePtr, size_t writeBytesLeft,
102 oldhandler = signal(SIGPIPE, SIG_IGN);
104 toProg[0] = toProg[1] = 0;
105 fromProg[0] = fromProg[1] = 0;
106 if (pipe(toProg) < 0 || pipe(fromProg) < 0) {
107 rpmlog(RPMLOG_ERR, _("Couldn't create pipe for %s: %m\n"), argv[0]);
111 if (!(child = fork())) {
112 (void) close(toProg[1]);
113 (void) close(fromProg[0]);
115 (void) dup2(toProg[0], STDIN_FILENO); /* Make stdin the in pipe */
116 (void) dup2(fromProg[1], STDOUT_FILENO); /* Make stdout the out pipe */
118 (void) close(toProg[0]);
119 (void) close(fromProg[1]);
121 if (dir && chdir(dir)) {
122 rpmlog(RPMLOG_ERR, _("Couldn't chdir to %s: %s\n"),
123 dir, strerror(errno));
127 rpmlog(RPMLOG_DEBUG, "\texecv(%s) pid %d\n",
128 argv[0], (unsigned)getpid());
130 unsetenv("MALLOC_CHECK_");
131 (void) execvp(argv[0], (char *const *)argv);
132 /* XXX this error message is probably not seen. */
133 rpmlog(RPMLOG_ERR, _("Couldn't exec %s: %s\n"),
134 argv[0], strerror(errno));
138 rpmlog(RPMLOG_ERR, _("Couldn't fork %s: %s\n"),
139 argv[0], strerror(errno));
143 (void) close(toProg[0]);
144 (void) close(fromProg[1]);
146 /* Do not block reading or writing from/to prog. */
147 (void) fcntl(fromProg[0], F_SETFL, O_NONBLOCK);
148 (void) fcntl(toProg[1], F_SETFL, O_NONBLOCK);
150 readBuff = newStringBuf();
162 if (fromProg[0] >= 0) {
163 FD_SET(fromProg[0], &ibits);
165 if (toProg[1] >= 0) {
166 FD_SET(toProg[1], &obits);
168 /* XXX values set to limit spinning with perl doing ~100 forks/sec. */
171 nfd = ((fromProg[0] > toProg[1]) ? fromProg[0] : toProg[1]);
172 if ((rc = select(nfd, &ibits, &obits, NULL, &tv)) < 0) {
178 /* Write any data to program */
179 if (toProg[1] >= 0 && FD_ISSET(toProg[1], &obits)) {
180 if (writePtr && writeBytesLeft > 0) {
181 if ((nbw = write(toProg[1], writePtr,
182 (1024<writeBytesLeft) ? 1024 : writeBytesLeft)) < 0) {
183 if (errno != EAGAIN) {
184 perror("getOutputFrom()");
189 writeBytesLeft -= nbw;
191 } else if (toProg[1] >= 0) { /* close write fd */
192 (void) close(toProg[1]);
197 /* Read any data from prog */
198 { char buf[BUFSIZ+1];
199 while ((nbr = read(fromProg[0], buf, sizeof(buf)-1)) > 0) {
201 appendStringBuf(readBuff, buf);
205 /* terminate on (non-blocking) EOF or error */
206 done = (nbr == 0 || (nbr < 0 && errno != EAGAIN));
212 (void) close(toProg[1]);
213 if (fromProg[0] >= 0)
214 (void) close(fromProg[0]);
216 (void) signal(SIGPIPE, oldhandler);
218 /* Collect status from prog */
219 reaped = waitpid(child, &status, 0);
220 rpmlog(RPMLOG_DEBUG, "\twaitpid(%d) rc %d status %x\n",
221 (unsigned)child, (unsigned)reaped, status);
223 if (failNonZero && (!WIFEXITED(status) || WEXITSTATUS(status))) {
224 rpmlog(RPMLOG_ERR, _("%s failed\n"), argv[0]);
227 if (writeBytesLeft) {
228 rpmlog(RPMLOG_ERR, _("failed to write all data to %s\n"), argv[0]);
234 int rpmfcExec(ARGV_const_t av, StringBuf sb_stdin, StringBuf * sb_stdoutp,
243 const char * buf_stdin = NULL;
244 size_t buf_stdin_len = 0;
252 /* Find path to executable with (possible) args. */
253 s = rpmExpand(av[0], NULL);
257 /* Parse args buried within expanded exacutable. */
259 xx = poptParseArgvString(s, &pac, (const char ***)&pav);
260 if (!(xx == 0 && pac > 0 && pav != NULL))
263 /* Build argv, appending args to the executable args. */
265 xx = argvAppend(&xav, pav);
267 xx = rpmfcExpandAppend(&xav, av + 1);
269 if (sb_stdin != NULL) {
270 buf_stdin = getStringBuf(sb_stdin);
271 buf_stdin_len = strlen(buf_stdin);
274 /* Read output from exec'd helper. */
275 sb = getOutputFrom(NULL, xav, buf_stdin, buf_stdin_len, failnonzero);
277 if (sb_stdoutp != NULL) {
279 sb = NULL; /* XXX don't free */
285 sb = freeStringBuf(sb);
287 pav = _free(pav); /* XXX popt mallocs in single blob. */
294 static int rpmfcSaveArg(ARGV_t * argvp, const char * key)
298 if (argvSearch(*argvp, key, NULL) == NULL) {
299 rc = argvAdd(argvp, key);
300 rc = argvSort(*argvp, NULL);
305 static char * rpmfcFileDep(int ix, rpmds ds)
307 rpmTag tagN = rpmdsTagN(ds);
315 assert(tagN == RPMTAG_PROVIDENAME || tagN == RPMTAG_REQUIRENAME);
317 switch ((rpm_tag_t) tagN) {
318 case RPMTAG_PROVIDENAME:
321 case RPMTAG_REQUIRENAME:
326 rasprintf(&buf, "%08d%c %s %s 0x%08x", ix, deptype,
327 rpmdsN(ds), rpmdsEVR(ds), rpmdsFlags(ds));
332 static void rpmfcAddFileDep(ARGV_t * argvp, int ix, rpmds ds)
334 char *key = rpmfcFileDep(ix, ds);
335 rpmfcSaveArg(argvp, key);
340 * Run per-interpreter dependency helper.
341 * @param fc file classifier
342 * @param deptype 'P' == Provides:, 'R' == Requires:, helper
343 * @param nsdep class name for interpreter (e.g. "perl")
344 * @return 0 on success
346 static int rpmfcHelper(rpmfc fc, unsigned char deptype, const char * nsdep)
348 const char * fn = fc->fn[fc->ix];
350 StringBuf sb_stdout = NULL;
357 int32_t dsContext, tagN;
371 rasprintf(&buf, "%%{?__%s_provides}", nsdep);
372 depsp = &fc->provides;
373 dsContext = RPMSENSE_FIND_PROVIDES;
374 tagN = RPMTAG_PROVIDENAME;
379 rasprintf(&buf, "%%{?__%s_requires}", nsdep);
380 depsp = &fc->requires;
381 dsContext = RPMSENSE_FIND_REQUIRES;
382 tagN = RPMTAG_REQUIRENAME;
388 sb_stdin = newStringBuf();
389 appendLineStringBuf(sb_stdin, fn);
391 xx = rpmfcExec(av, sb_stdin, &sb_stdout, 0);
392 sb_stdin = freeStringBuf(sb_stdin);
394 if (xx == 0 && sb_stdout != NULL) {
396 xx = argvSplit(&pav, getStringBuf(sb_stdout), " \t\n\r");
397 pac = argvCount(pav);
399 for (i = 0; i < pac; i++) {
403 if (pav[i+1] && strchr("=<>", *pav[i+1])) {
405 for (s = pav[i]; *s; s++) {
411 Flags |= RPMSENSE_EQUAL;
414 Flags |= RPMSENSE_LESS;
417 Flags |= RPMSENSE_GREATER;
427 /* Add tracking dependency for versioned Provides: */
428 if (!fc->tracked && deptype == 'P' && *EVR != '\0') {
429 ds = rpmdsSingle(RPMTAG_REQUIRENAME,
430 "rpmlib(VersionedDependencies)", "3.0.3-1",
431 RPMSENSE_RPMLIB|(RPMSENSE_LESS|RPMSENSE_EQUAL));
432 xx = rpmdsMerge(&fc->requires, ds);
437 ds = rpmdsSingle(tagN, N, EVR, Flags);
439 /* Add to package dependencies. */
440 xx = rpmdsMerge(depsp, ds);
442 /* Add to file dependencies. */
443 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
450 sb_stdout = freeStringBuf(sb_stdout);
458 static const struct rpmfcTokens_s const rpmfcTokens[] = {
459 { "directory", RPMFC_DIRECTORY|RPMFC_INCLUDE },
461 { " shared object", RPMFC_LIBRARY },
462 { " executable", RPMFC_EXECUTABLE },
463 { " statically linked", RPMFC_STATIC },
464 { " not stripped", RPMFC_NOTSTRIPPED },
465 { " archive", RPMFC_ARCHIVE },
467 { "ELF 32-bit", RPMFC_ELF32|RPMFC_INCLUDE },
468 { "ELF 64-bit", RPMFC_ELF64|RPMFC_INCLUDE },
470 { " script", RPMFC_SCRIPT },
471 { " text", RPMFC_TEXT },
472 { " document", RPMFC_DOCUMENT },
474 { " compressed", RPMFC_COMPRESSED },
476 { "troff or preprocessor input", RPMFC_MANPAGE|RPMFC_INCLUDE },
477 { "GNU Info", RPMFC_MANPAGE|RPMFC_INCLUDE },
479 { "perl script text", RPMFC_PERL|RPMFC_INCLUDE },
480 { "Perl5 module source text", RPMFC_PERL|RPMFC_MODULE|RPMFC_INCLUDE },
482 { " /usr/bin/python", RPMFC_PYTHON|RPMFC_INCLUDE },
484 /* XXX "a /usr/bin/python -t script text executable" */
485 /* XXX "python 2.3 byte-compiled" */
486 { "python ", RPMFC_PYTHON|RPMFC_INCLUDE },
488 { "libtool library ", RPMFC_LIBTOOL|RPMFC_INCLUDE },
489 { "pkgconfig ", RPMFC_PKGCONFIG|RPMFC_INCLUDE },
491 /* XXX .NET executables and libraries. file(1) cannot differ from win32
492 * executables unfortunately :( */
493 { "PE executable", RPMFC_MONO|RPMFC_INCLUDE },
495 { "current ar archive", RPMFC_STATIC|RPMFC_LIBRARY|RPMFC_ARCHIVE|RPMFC_INCLUDE },
497 { "Zip archive data", RPMFC_COMPRESSED|RPMFC_ARCHIVE|RPMFC_INCLUDE },
498 { "tar archive", RPMFC_ARCHIVE|RPMFC_INCLUDE },
499 { "cpio archive", RPMFC_ARCHIVE|RPMFC_INCLUDE },
500 { "RPM v3", RPMFC_ARCHIVE|RPMFC_INCLUDE },
501 { "RPM v4", RPMFC_ARCHIVE|RPMFC_INCLUDE },
503 { " image", RPMFC_IMAGE|RPMFC_INCLUDE },
504 { " font", RPMFC_FONT|RPMFC_INCLUDE },
505 { " Font", RPMFC_FONT|RPMFC_INCLUDE },
507 { " commands", RPMFC_SCRIPT|RPMFC_INCLUDE },
508 { " script", RPMFC_SCRIPT|RPMFC_INCLUDE },
510 { "empty", RPMFC_WHITE|RPMFC_INCLUDE },
512 { "HTML", RPMFC_WHITE|RPMFC_INCLUDE },
513 { "SGML", RPMFC_WHITE|RPMFC_INCLUDE },
514 { "XML", RPMFC_WHITE|RPMFC_INCLUDE },
516 { " program text", RPMFC_WHITE|RPMFC_INCLUDE },
517 { " source", RPMFC_WHITE|RPMFC_INCLUDE },
518 { "GLS_BINARY_LSB_FIRST", RPMFC_WHITE|RPMFC_INCLUDE },
519 { " DB ", RPMFC_WHITE|RPMFC_INCLUDE },
521 { "ASCII English text", RPMFC_WHITE|RPMFC_INCLUDE },
522 { "ASCII text", RPMFC_WHITE|RPMFC_INCLUDE },
523 { "ISO-8859 text", RPMFC_WHITE|RPMFC_INCLUDE },
525 { "symbolic link to", RPMFC_SYMLINK },
526 { "socket", RPMFC_DEVICE },
527 { "special", RPMFC_DEVICE },
529 { "ASCII", RPMFC_WHITE },
530 { "ISO-8859", RPMFC_WHITE },
532 { "data", RPMFC_WHITE },
534 { "application", RPMFC_WHITE },
535 { "boot", RPMFC_WHITE },
536 { "catalog", RPMFC_WHITE },
537 { "code", RPMFC_WHITE },
538 { "file", RPMFC_WHITE },
539 { "format", RPMFC_WHITE },
540 { "message", RPMFC_WHITE },
541 { "program", RPMFC_WHITE },
543 { "broken symbolic link to ", RPMFC_WHITE|RPMFC_ERROR },
544 { "can't read", RPMFC_WHITE|RPMFC_ERROR },
545 { "can't stat", RPMFC_WHITE|RPMFC_ERROR },
546 { "executable, can't read", RPMFC_WHITE|RPMFC_ERROR },
547 { "core file", RPMFC_WHITE|RPMFC_ERROR },
549 { NULL, RPMFC_BLACK }
552 int rpmfcColoring(const char * fmstr)
555 rpm_color_t fcolor = RPMFC_BLACK;
557 for (fct = rpmfcTokens; fct->token != NULL; fct++) {
558 if (strstr(fmstr, fct->token) == NULL)
560 fcolor |= fct->colors;
561 if (fcolor & RPMFC_INCLUDE)
567 void rpmfcPrint(const char * msg, rpmfc fc, FILE * fp)
578 if (fp == NULL) fp = stderr;
581 fprintf(fp, "===================================== %s\n", msg);
583 nprovides = rpmdsCount(fc->provides);
584 nrequires = rpmdsCount(fc->requires);
587 for (fx = 0; fx < fc->nfiles; fx++) {
588 assert(fx < fc->fcdictx->nvals);
589 cx = fc->fcdictx->vals[fx];
590 assert(fx < fc->fcolor->nvals);
591 fcolor = fc->fcolor->vals[fx];
593 fprintf(fp, "%3d %s", fx, fc->fn[fx]);
594 if (fcolor != RPMFC_BLACK)
595 fprintf(fp, "\t0x%x", fc->fcolor->vals[fx]);
597 fprintf(fp, "\t%s", fc->cdict[cx]);
600 if (fc->fddictx == NULL || fc->fddictn == NULL)
603 assert(fx < fc->fddictx->nvals);
604 dx = fc->fddictx->vals[fx];
605 assert(fx < fc->fddictn->nvals);
606 ndx = fc->fddictn->vals[fx];
610 unsigned char deptype;
613 ix = fc->ddictx->vals[dx++];
614 deptype = ((ix >> 24) & 0xff);
619 assert(depval != NULL);
623 assert(ix < nprovides);
624 (void) rpmdsSetIx(fc->provides, ix-1);
625 if (rpmdsNext(fc->provides) >= 0)
626 depval = rpmdsDNEVR(fc->provides);
631 assert(ix < nrequires);
632 (void) rpmdsSetIx(fc->requires, ix-1);
633 if (rpmdsNext(fc->requires) >= 0)
634 depval = rpmdsDNEVR(fc->requires);
639 fprintf(fp, "\t%s\n", depval);
644 rpmfc rpmfcFree(rpmfc fc)
647 fc->fn = argvFree(fc->fn);
648 fc->fcolor = argiFree(fc->fcolor);
649 fc->fcdictx = argiFree(fc->fcdictx);
650 fc->fddictx = argiFree(fc->fddictx);
651 fc->fddictn = argiFree(fc->fddictn);
652 fc->cdict = argvFree(fc->cdict);
653 fc->ddict = argvFree(fc->ddict);
654 fc->ddictx = argiFree(fc->ddictx);
656 fc->provides = rpmdsFree(fc->provides);
657 fc->requires = rpmdsFree(fc->requires);
659 fc->sb_java = freeStringBuf(fc->sb_java);
660 fc->sb_perl = freeStringBuf(fc->sb_perl);
661 fc->sb_python = freeStringBuf(fc->sb_python);
670 rpmfc fc = xcalloc(1, sizeof(*fc));
674 rpmds rpmfcProvides(rpmfc fc)
676 return (fc != NULL ? fc->provides : NULL);
679 rpmds rpmfcRequires(rpmfc fc)
681 return (fc != NULL ? fc->requires : NULL);
686 * Extract script dependencies.
687 * @param fc file classifier
688 * @return 0 on success
690 static int rpmfcSCRIPT(rpmfc fc)
692 const char * fn = fc->fn[fc->ix];
699 struct stat sb, * st = &sb;
703 /* Only executable scripts are searched. */
704 if (stat(fn, st) < 0)
706 is_executable = (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH));
709 if (fp == NULL || ferror(fp)) {
710 if (fp) (void) fclose(fp);
714 /* Look for #! interpreter in first 10 lines. */
715 for (i = 0; i < 10; i++) {
717 s = fgets(buf, sizeof(buf) - 1, fp);
718 if (s == NULL || ferror(fp) || feof(fp))
720 s[sizeof(buf)-1] = '\0';
721 if (!(s[0] == '#' && s[1] == '!'))
725 while (*s && strchr(" \t\n\r", *s) != NULL)
732 for (se = s+1; *se; se++) {
733 if (strchr(" \t\n\r", *se) != NULL)
740 /* Add to package requires. */
741 ds = rpmdsSingle(RPMTAG_REQUIRENAME, s, "", RPMSENSE_FIND_REQUIRES);
742 xx = rpmdsMerge(&fc->requires, ds);
744 /* Add to file requires. */
745 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
750 /* Set color based on interpreter name. */
752 if (!strcmp(bn, "perl"))
753 fc->fcolor->vals[fc->ix] |= RPMFC_PERL;
754 else if (!strncmp(bn, "python", sizeof("python")-1))
755 fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
762 if (fc->fcolor->vals[fc->ix] & RPMFC_PERL) {
763 if (fc->fcolor->vals[fc->ix] & RPMFC_MODULE)
764 xx = rpmfcHelper(fc, 'P', "perl");
765 if (is_executable || (fc->fcolor->vals[fc->ix] & RPMFC_MODULE))
766 xx = rpmfcHelper(fc, 'R', "perl");
768 if (fc->fcolor->vals[fc->ix] & RPMFC_PYTHON) {
769 xx = rpmfcHelper(fc, 'P', "python");
773 xx = rpmfcHelper(fc, 'R', "python");
775 if (fc->fcolor->vals[fc->ix] & RPMFC_LIBTOOL) {
776 xx = rpmfcHelper(fc, 'P', "libtool");
780 xx = rpmfcHelper(fc, 'R', "libtool");
782 if (fc->fcolor->vals[fc->ix] & RPMFC_PKGCONFIG) {
783 xx = rpmfcHelper(fc, 'P', "pkgconfig");
787 xx = rpmfcHelper(fc, 'R', "pkgconfig");
789 if (fc->fcolor->vals[fc->ix] & RPMFC_MONO) {
790 xx = rpmfcHelper(fc, 'P', "mono");
792 xx = rpmfcHelper(fc, 'R', "mono");
799 * Extract Elf dependencies.
800 * @param fc file classifier
801 * @return 0 on success
803 static int rpmfcELF(rpmfc fc)
805 #if HAVE_GELF_H && HAVE_LIBELF
806 const char * fn = fc->fn[fc->ix];
810 GElf_Ehdr ehdr_mem, * ehdr;
811 GElf_Shdr shdr_mem, * shdr;
812 GElf_Verdef def_mem, * def;
813 GElf_Verneed need_mem, * need;
814 GElf_Dyn dyn_mem, * dyn;
815 unsigned int auxoffset;
822 struct stat sb, * st = &sb;
823 char * soname = NULL;
834 static int filter_GLIBC_PRIVATE = 0;
835 static int oneshot = 0;
839 filter_GLIBC_PRIVATE = rpmExpandNumeric("%{?_filter_GLIBC_PRIVATE}");
842 /* Files with executable bit set only. */
843 if (stat(fn, st) != 0)
846 fdno = open(fn, O_RDONLY);
850 (void) elf_version(EV_CURRENT);
853 if ((elf = elf_begin (fdno, ELF_C_READ, NULL)) == NULL
854 || elf_kind(elf) != ELF_K_ELF
855 || (ehdr = gelf_getehdr(elf, &ehdr_mem)) == NULL
856 || !(ehdr->e_type == ET_DYN || ehdr->e_type == ET_EXEC))
859 isElf64 = ehdr->e_ident[EI_CLASS] == ELFCLASS64;
860 isDSO = ehdr->e_type == ET_DYN;
863 while ((scn = elf_nextscn(elf, scn)) != NULL) {
864 shdr = gelf_getshdr(scn, &shdr_mem);
868 soname = _free(soname);
869 switch (shdr->sh_type) {
876 while ((data = elf_getdata (scn, data)) != NULL) {
878 for (cnt = shdr->sh_info; --cnt >= 0; ) {
880 def = gelf_getverdef (data, offset, &def_mem);
883 auxoffset = offset + def->vd_aux;
884 for (cnt2 = def->vd_cnt; --cnt2 >= 0; ) {
885 GElf_Verdaux aux_mem, * aux;
887 aux = gelf_getverdaux (data, auxoffset, &aux_mem);
891 s = elf_strptr(elf, shdr->sh_link, aux->vda_name);
894 if (def->vd_flags & VER_FLG_BASE) {
895 soname = _free(soname);
897 auxoffset += aux->vda_next;
901 && !(filter_GLIBC_PRIVATE != 0
902 && !strcmp(s, "GLIBC_PRIVATE")))
904 rasprintf(&buf, "%s(%s)%s", soname, s,
905 #if !defined(__alpha__)
906 isElf64 ? "(64bit)" : "");
911 /* Add to package provides. */
912 ds = rpmdsSingle(RPMTAG_PROVIDES,
913 buf, "", RPMSENSE_FIND_PROVIDES);
914 xx = rpmdsMerge(&fc->provides, ds);
916 /* Add to file dependencies. */
917 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
922 auxoffset += aux->vda_next;
924 offset += def->vd_next;
928 case SHT_GNU_verneed:
930 /* Files with executable bit set only. */
931 if (!fc->skipReq && (st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
932 while ((data = elf_getdata (scn, data)) != NULL) {
934 for (cnt = shdr->sh_info; --cnt >= 0; ) {
935 need = gelf_getverneed (data, offset, &need_mem);
939 s = elf_strptr(elf, shdr->sh_link, need->vn_file);
942 soname = _free(soname);
944 auxoffset = offset + need->vn_aux;
945 for (cnt2 = need->vn_cnt; --cnt2 >= 0; ) {
946 GElf_Vernaux aux_mem, * aux;
948 aux = gelf_getvernaux (data, auxoffset, &aux_mem);
952 s = elf_strptr(elf, shdr->sh_link, aux->vna_name);
956 /* Filter dependencies that contain GLIBC_PRIVATE */
958 && !(filter_GLIBC_PRIVATE != 0
959 && !strcmp(s, "GLIBC_PRIVATE")))
961 rasprintf(&buf, "%s(%s)%s", soname, s,
962 #if !defined(__alpha__)
963 isElf64 ? "(64bit)" : "");
968 /* Add to package dependencies. */
969 ds = rpmdsSingle(RPMTAG_REQUIRENAME,
970 buf, "", RPMSENSE_FIND_REQUIRES);
971 xx = rpmdsMerge(&fc->requires, ds);
973 /* Add to file dependencies. */
974 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
978 auxoffset += aux->vna_next;
980 offset += need->vn_next;
986 while ((data = elf_getdata (scn, data)) != NULL) {
987 for (cnt = 0; cnt < (shdr->sh_size / shdr->sh_entsize); ++cnt) {
988 dyn = gelf_getdyn (data, cnt, &dyn_mem);
992 switch (dyn->d_tag) {
1006 /* Files with executable bit set only. */
1007 if (fc->skipReq || !(st->st_mode & (S_IXUSR|S_IXGRP|S_IXOTH)))
1009 /* Add to package requires. */
1010 depsp = &fc->requires;
1011 tagN = RPMTAG_REQUIRENAME;
1012 dsContext = RPMSENSE_FIND_REQUIRES;
1013 s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1018 /* Add to package provides. */
1021 depsp = &fc->provides;
1022 tagN = RPMTAG_PROVIDENAME;
1023 dsContext = RPMSENSE_FIND_PROVIDES;
1024 s = elf_strptr(elf, shdr->sh_link, dyn->d_un.d_val);
1031 rasprintf(&buf, "%s%s", s,
1032 #if !defined(__alpha__)
1033 isElf64 ? "()(64bit)" : "");
1038 /* Add to package dependencies. */
1039 ds = rpmdsSingle(tagN, buf, "", dsContext);
1040 xx = rpmdsMerge(depsp, ds);
1042 /* Add to file dependencies. */
1043 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
1053 /* For DSOs which use the .gnu_hash section and don't have a .hash
1054 * section, we need to ensure that we have a new enough glibc. */
1055 if (gotGNUHASH && !gotHASH) {
1056 ds = rpmdsSingle(RPMTAG_REQUIRENAME, "rtld(GNU_HASH)", "",
1057 RPMSENSE_FIND_REQUIRES);
1058 rpmdsMerge(&fc->requires, ds);
1059 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
1063 /* For DSO's, provide the basename of the file if DT_SONAME not found. */
1064 if (!fc->skipProv && isDSO && !gotDEBUG && !gotSONAME) {
1065 depsp = &fc->provides;
1066 tagN = RPMTAG_PROVIDENAME;
1067 dsContext = RPMSENSE_FIND_PROVIDES;
1069 s = strrchr(fn, '/');
1075 /* LCL: s is not null. */
1076 rasprintf(&buf, "%s%s", s,
1077 #if !defined(__alpha__)
1078 isElf64 ? "()(64bit)" : "");
1083 /* Add to package dependencies. */
1084 ds = rpmdsSingle(tagN, buf, "", dsContext);
1085 xx = rpmdsMerge(depsp, ds);
1087 /* Add to file dependencies. */
1088 rpmfcAddFileDep(&fc->ddict, fc->ix, ds);
1095 soname = _free(soname);
1096 if (elf) (void) elf_end(elf);
1104 typedef const struct rpmfcApplyTbl_s {
1105 int (*func) (rpmfc fc);
1111 static const struct rpmfcApplyTbl_s const rpmfcApplyTable[] = {
1112 { rpmfcELF, RPMFC_ELF },
1113 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PERL) },
1114 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PYTHON) },
1115 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_PKGCONFIG) },
1116 { rpmfcSCRIPT, (RPMFC_SCRIPT|RPMFC_LIBTOOL) },
1117 { rpmfcSCRIPT, RPMFC_MONO },
1121 rpmRC rpmfcApply(rpmfc fc)
1129 rpmsenseFlags Flags;
1130 unsigned char deptype;
1139 /* Generate package and per-file dependencies. */
1140 for (fc->ix = 0; fc->fn[fc->ix] != NULL; fc->ix++) {
1142 /* XXX Insure that /usr/lib{,64}/python files are marked RPMFC_PYTHON */
1143 /* XXX HACK: classification by path is intrinsically stupid. */
1144 { const char *fn = strstr(fc->fn[fc->ix], "/usr/lib");
1146 fn += sizeof("/usr/lib")-1;
1147 if (fn[0] == '6' && fn[1] == '4')
1149 if (!strncmp(fn, "/python", sizeof("/python")-1))
1150 fc->fcolor->vals[fc->ix] |= RPMFC_PYTHON;
1154 if (fc->fcolor->vals[fc->ix])
1155 for (fcat = rpmfcApplyTable; fcat->func != NULL; fcat++) {
1156 if (!(fc->fcolor->vals[fc->ix] & fcat->colormask))
1158 xx = (*fcat->func) (fc);
1162 /* Generate per-file indices into package dependencies. */
1163 nddict = argvCount(fc->ddict);
1165 for (i = 0; i < nddict; i++) {
1168 /* Parse out (file#,deptype,N,EVR,Flags) */
1169 ix = strtol(s, &se, 10);
1174 while (*se && *se != ' ')
1178 while (*se && *se != ' ')
1181 Flags = strtol(se, NULL, 16);
1188 ds = rpmdsSingle(RPMTAG_PROVIDENAME, N, EVR, Flags);
1189 dix = rpmdsFind(fc->provides, ds);
1193 ds = rpmdsSingle(RPMTAG_REQUIRENAME, N, EVR, Flags);
1194 dix = rpmdsFind(fc->requires, ds);
1199 /* XXX assertion incorrect while generating -debuginfo deps. */
1207 val = (deptype << 24) | (dix & 0x00ffffff);
1208 xx = argiAdd(&fc->ddictx, -1, val);
1212 xx = argiAdd(&fc->fddictx, ix, argiCount(fc->ddictx)-1);
1214 if (fc->fddictn && fc->fddictn->vals)
1215 fc->fddictn->vals[ix]++;
1221 rpmRC rpmfcClassify(rpmfc fc, ARGV_t argv, rpm_mode_t * fmode)
1225 const char * s, * se;
1229 int msflags = MAGIC_CHECK; /* XXX MAGIC_COMPRESS flag? */
1232 if (fc == NULL || argv == NULL)
1235 fc->nfiles = argvCount(argv);
1237 /* Initialize the per-file dictionary indices. */
1238 xx = argiAdd(&fc->fddictx, fc->nfiles-1, 0);
1239 xx = argiAdd(&fc->fddictn, fc->nfiles-1, 0);
1241 /* Build (sorted) file class dictionary. */
1242 xx = argvAdd(&fc->cdict, "");
1243 xx = argvAdd(&fc->cdict, "directory");
1245 ms = magic_open(msflags);
1247 rpmlog(RPMLOG_ERR, _("magic_open(0x%x) failed: %s\n"),
1248 msflags, strerror(errno));
1249 assert(ms != NULL); /* XXX figger a proper return path. */
1252 xx = magic_load(ms, NULL);
1254 rpmlog(RPMLOG_ERR, _("magic_load failed: %s\n"), magic_error(ms));
1255 assert(xx != -1); /* XXX figger a proper return path. */
1258 for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1260 rpm_mode_t mode = (fmode ? fmode[fc->ix] : 0);
1266 switch (mode & S_IFMT) {
1267 case S_IFCHR: ftype = "character special"; break;
1268 case S_IFBLK: ftype = "block special"; break;
1269 case S_IFIFO: ftype = "fifo (named pipe)"; break;
1270 case S_IFSOCK: ftype = "socket"; break;
1275 /* XXX all files with extension ".pm" are perl modules for now. */
1276 if (rpmFileHasSuffix(s, ".pm"))
1277 ftype = "Perl5 module source text";
1279 /* XXX all files with extension ".la" are libtool for now. */
1280 else if (rpmFileHasSuffix(s, ".la"))
1281 ftype = "libtool library file";
1283 /* XXX all files with extension ".pc" are pkgconfig for now. */
1284 else if (rpmFileHasSuffix(s, ".pc"))
1285 ftype = "pkgconfig file";
1287 /* XXX skip all files in /dev/ which are (or should be) %dev dummies. */
1288 else if (slen >= fc->brlen+sizeof("/dev/") && !strncmp(s+fc->brlen, "/dev/", sizeof("/dev/")-1))
1291 ftype = magic_file(ms, s);
1293 if (ftype == NULL) {
1295 _("magic_file(ms, \"%s\") failed: mode %06o %s\n"),
1296 s, mode, magic_error(ms));
1297 assert(ftype != NULL); /* XXX figger a proper return path. */
1302 rpmlog(RPMLOG_DEBUG, "%s: %s\n", s, se);
1304 /* Save the path. */
1305 xx = argvAdd(&fc->fn, s);
1307 /* Save the file type string. */
1308 xx = argvAdd(&fcav, se);
1310 /* Add (filtered) entry to sorted class dictionary. */
1311 fcolor = rpmfcColoring(se);
1312 xx = argiAdd(&fc->fcolor, fc->ix, fcolor);
1314 if (fcolor != RPMFC_WHITE && (fcolor & RPMFC_INCLUDE))
1315 xx = rpmfcSaveArg(&fc->cdict, se);
1318 /* Build per-file class index array. */
1320 for (fc->ix = 0; fc->ix < fc->nfiles; fc->ix++) {
1324 dav = argvSearch(fc->cdict, se, NULL);
1326 xx = argiAdd(&fc->fcdictx, fc->ix, (dav - fc->cdict));
1329 xx = argiAdd(&fc->fcdictx, fc->ix, 0);
1334 fcav = argvFree(fcav);
1344 typedef struct DepMsg_s * DepMsg_t;
1350 char * const argv[4];
1360 static struct DepMsg_s depMsgs[] = {
1361 { "Provides", { "%{?__find_provides}", NULL, NULL, NULL },
1362 RPMTAG_PROVIDENAME, RPMTAG_PROVIDEVERSION, RPMTAG_PROVIDEFLAGS,
1364 { "Requires(interp)", { NULL, "interp", NULL, NULL },
1365 RPMTAG_REQUIRENAME, RPMTAG_REQUIREVERSION, RPMTAG_REQUIREFLAGS,
1366 _notpre(RPMSENSE_INTERP), 0 },
1367 { "Requires(rpmlib)", { NULL, "rpmlib", NULL, NULL },
1368 -1, -1, RPMTAG_REQUIREFLAGS,
1369 _notpre(RPMSENSE_RPMLIB), 0 },
1370 { "Requires(verify)", { NULL, "verify", NULL, NULL },
1371 -1, -1, RPMTAG_REQUIREFLAGS,
1372 RPMSENSE_SCRIPT_VERIFY, 0 },
1373 { "Requires(pre)", { NULL, "pre", NULL, NULL },
1374 -1, -1, RPMTAG_REQUIREFLAGS,
1375 _notpre(RPMSENSE_SCRIPT_PRE), 0 },
1376 { "Requires(post)", { NULL, "post", NULL, NULL },
1377 -1, -1, RPMTAG_REQUIREFLAGS,
1378 _notpre(RPMSENSE_SCRIPT_POST), 0 },
1379 { "Requires(preun)", { NULL, "preun", NULL, NULL },
1380 -1, -1, RPMTAG_REQUIREFLAGS,
1381 _notpre(RPMSENSE_SCRIPT_PREUN), 0 },
1382 { "Requires(postun)", { NULL, "postun", NULL, NULL },
1383 -1, -1, RPMTAG_REQUIREFLAGS,
1384 _notpre(RPMSENSE_SCRIPT_POSTUN), 0 },
1385 { "Requires", { "%{?__find_requires}", NULL, NULL, NULL },
1386 -1, -1, RPMTAG_REQUIREFLAGS, /* XXX inherit name/version arrays */
1387 RPMSENSE_FIND_REQUIRES|RPMSENSE_TRIGGERIN|RPMSENSE_TRIGGERUN|RPMSENSE_TRIGGERPOSTUN|RPMSENSE_TRIGGERPREIN, 0 },
1388 { "Conflicts", { "%{?__find_conflicts}", NULL, NULL, NULL },
1389 RPMTAG_CONFLICTNAME, RPMTAG_CONFLICTVERSION, RPMTAG_CONFLICTFLAGS,
1391 { "Obsoletes", { "%{?__find_obsoletes}", NULL, NULL, NULL },
1392 RPMTAG_OBSOLETENAME, RPMTAG_OBSOLETEVERSION, RPMTAG_OBSOLETEFLAGS,
1394 { NULL, { NULL, NULL, NULL, NULL }, 0, 0, 0, 0, 0 }
1397 static DepMsg_t DepMsgs = depMsgs;
1401 static void printDeps(Header h)
1405 int flags = 0; /* XXX !scareMem */
1407 rpmsenseFlags Flags;
1410 for (dm = DepMsgs; dm->msg != NULL; dm++) {
1411 if (dm->ntag != -1) {
1413 ds = rpmdsNew(h, dm->ntag, flags);
1420 continue; /* XXX can't happen */
1423 while (rpmdsNext(ds) >= 0) {
1425 Flags = rpmdsFlags(ds);
1427 if (!((Flags & dm->mask) ^ dm->xor))
1430 rpmlog(RPMLOG_NOTICE, "%s:", (dm->msg ? dm->msg : ""));
1433 if ((DNEVR = rpmdsDNEVR(ds)) == NULL)
1434 continue; /* XXX can't happen */
1435 rpmlog(RPMLOG_NOTICE, " %s", DNEVR+2);
1438 rpmlog(RPMLOG_NOTICE, "\n");
1445 static int rpmfcGenerateDependsHelper(const rpmSpec spec, Package pkg, rpmfi fi)
1448 StringBuf sb_stdout;
1450 int failnonzero = 0;
1454 * Create file manifest buffer to deliver to dependency finder.
1456 sb_stdin = newStringBuf();
1457 fi = rpmfiInit(fi, 0);
1459 while (rpmfiNext(fi) >= 0)
1460 appendLineStringBuf(sb_stdin, rpmfiFN(fi));
1462 for (dm = DepMsgs; dm->msg != NULL; dm++) {
1464 rpmsenseFlags tagflags;
1468 tag = (dm->ftag > 0) ? dm->ftag : dm->ntag;
1473 case RPMTAG_PROVIDEFLAGS:
1477 tagflags = RPMSENSE_FIND_PROVIDES;
1479 case RPMTAG_REQUIREFLAGS:
1483 tagflags = RPMSENSE_FIND_REQUIRES;
1490 xx = rpmfcExec(dm->argv, sb_stdin, &sb_stdout, failnonzero);
1494 s = rpmExpand(dm->argv[0], NULL);
1495 rpmlog(RPMLOG_NOTICE, _("Finding %s: %s\n"), dm->msg,
1499 if (sb_stdout == NULL) {
1501 rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1505 /* Parse dependencies into header */
1506 rc = parseRCPOT(spec, pkg, getStringBuf(sb_stdout), tag, 0, tagflags);
1507 sb_stdout = freeStringBuf(sb_stdout);
1510 rpmlog(rc, _("Failed to find %s:\n"), dm->msg);
1515 sb_stdin = freeStringBuf(sb_stdin);
1520 rpmRC rpmfcGenerateDepends(const rpmSpec spec, Package pkg)
1522 rpmfi fi = pkg->cpioList;
1525 int flags = 0; /* XXX !scareMem */
1528 int ac = rpmfiFC(fi);
1539 /* Skip packages with no files. */
1543 /* Skip packages that have dependency generation disabled. */
1544 if (! (pkg->autoReq || pkg->autoProv))
1547 /* If new-fangled dependency generation is disabled ... */
1548 if (!rpmExpandNumeric("%{?_use_internal_dependency_generator}")) {
1549 /* ... then generate dependencies using %{__find_requires} et al. */
1550 rc = rpmfcGenerateDependsHelper(spec, pkg, fi);
1551 printDeps(pkg->header);
1555 /* Extract absolute file paths in argv format. */
1556 av = xcalloc(ac+1, sizeof(*av));
1557 fmode = xcalloc(ac+1, sizeof(*fmode));
1560 fi = rpmfiInit(fi, 0);
1562 while ((idx = rpmfiNext(fi)) >= 0) {
1563 rpmfileAttrs fileAttrs;
1565 /* Does package have any %config files? */
1566 fileAttrs = rpmfiFFlags(fi);
1567 genConfigDeps |= (fileAttrs & RPMFILE_CONFIG);
1569 av[idx] = xstrdup(rpmfiFN(fi));
1570 fmode[idx] = rpmfiFMode(fi);
1575 fc->skipProv = !pkg->autoProv;
1576 fc->skipReq = !pkg->autoReq;
1578 fc->brlen = (spec->buildRoot ? strlen(spec->buildRoot) : 0);
1580 /* Copy (and delete) manually generated dependencies to dictionary. */
1581 if (!fc->skipProv) {
1582 ds = rpmdsNew(pkg->header, RPMTAG_PROVIDENAME, flags);
1583 xx = rpmdsMerge(&fc->provides, ds);
1585 xx = headerDel(pkg->header, RPMTAG_PROVIDENAME);
1586 xx = headerDel(pkg->header, RPMTAG_PROVIDEVERSION);
1587 xx = headerDel(pkg->header, RPMTAG_PROVIDEFLAGS);
1589 /* Add config dependency, Provides: config(N) = EVR */
1590 if (genConfigDeps) {
1591 N = rpmdsN(pkg->ds);
1593 EVR = rpmdsEVR(pkg->ds);
1594 assert(EVR != NULL);
1595 rasprintf(&buf, "config(%s)", N);
1596 ds = rpmdsSingle(RPMTAG_PROVIDENAME, buf, EVR,
1597 (RPMSENSE_EQUAL|RPMSENSE_CONFIG));
1599 xx = rpmdsMerge(&fc->provides, ds);
1605 ds = rpmdsNew(pkg->header, RPMTAG_REQUIRENAME, flags);
1606 xx = rpmdsMerge(&fc->requires, ds);
1608 xx = headerDel(pkg->header, RPMTAG_REQUIRENAME);
1609 xx = headerDel(pkg->header, RPMTAG_REQUIREVERSION);
1610 xx = headerDel(pkg->header, RPMTAG_REQUIREFLAGS);
1612 /* Add config dependency, Requires: config(N) = EVR */
1613 if (genConfigDeps) {
1614 N = rpmdsN(pkg->ds);
1616 EVR = rpmdsEVR(pkg->ds);
1617 assert(EVR != NULL);
1618 rasprintf(&buf, "config(%s)", N);
1619 ds = rpmdsSingle(RPMTAG_REQUIRENAME, buf, EVR,
1620 (RPMSENSE_EQUAL|RPMSENSE_CONFIG));
1622 xx = rpmdsMerge(&fc->requires, ds);
1627 /* Build file class dictionary. */
1628 xx = rpmfcClassify(fc, av, fmode);
1630 /* Build file/package dependency dictionary. */
1631 xx = rpmfcApply(fc);
1633 /* Add per-file colors(#files) */
1634 if (rpmtdFromArgi(&td, RPMTAG_FILECOLORS, fc->fcolor)) {
1635 rpm_color_t *fcolor;
1636 assert(ac == rpmtdCount(&td));
1637 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1638 /* XXX Make sure only primary (i.e. Elf32/Elf64) colors are added. */
1639 while ((fcolor = rpmtdNextUint32(&td))) {
1642 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1645 /* Add classes(#classes) */
1646 if (rpmtdFromArgv(&td, RPMTAG_CLASSDICT, fc->cdict)) {
1647 assert(rpmtdType(&td) == RPM_STRING_ARRAY_TYPE);
1648 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1651 /* Add per-file classes(#files) */
1652 if (rpmtdFromArgi(&td, RPMTAG_FILECLASS, fc->fcdictx)) {
1653 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1654 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1658 if (fc->provides != NULL && (c = rpmdsCount(fc->provides)) > 0 && !fc->skipProv) {
1659 const char **names = xcalloc(c, sizeof(char *));
1660 const char **evrs = xcalloc(c, sizeof(char *));
1661 rpmsenseFlags *flags = xcalloc(c, sizeof(rpmsenseFlags *));
1663 rpmds pi = rpmdsInit(fc->provides);
1664 while ((i = rpmdsNext(pi)) >= 0) {
1665 names[i] = rpmdsN(pi);
1666 evrs[i] = rpmdsEVR(pi);
1667 flags[i] = rpmdsFlags(pi);
1670 assert(names != NULL);
1671 if (rpmtdFromStringArray(&td, RPMTAG_PROVIDENAME, names, c))
1672 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1673 assert(rpmtdType(&td) == RPM_STRING_ARRAY_TYPE);
1675 assert(evrs != NULL);
1676 if (rpmtdFromStringArray(&td, RPMTAG_PROVIDEVERSION, evrs, c))
1677 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1678 assert(rpmtdType(&td) == RPM_STRING_ARRAY_TYPE);
1680 assert(flags != NULL);
1681 if (rpmtdFromUint32(&td, RPMTAG_PROVIDEFLAGS, flags, c))
1682 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1683 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1691 if (fc->requires != NULL && (c = rpmdsCount(fc->requires)) > 0 && !fc->skipReq) {
1692 const char **names = xcalloc(c, sizeof(char *));
1693 const char **evrs = xcalloc(c, sizeof(char *));
1694 rpmsenseFlags *flags = xcalloc(c, sizeof(rpmsenseFlags *));
1696 rpmds ri = rpmdsInit(fc->requires);
1697 while ((i = rpmdsNext(ri)) >= 0) {
1698 names[i] = rpmdsN(ri);
1699 evrs[i] = rpmdsEVR(ri);
1700 flags[i] = rpmdsFlags(ri);
1703 assert(names != NULL);
1704 if (rpmtdFromStringArray(&td, RPMTAG_REQUIRENAME, names, c))
1705 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1706 assert(rpmtdType(&td) == RPM_STRING_ARRAY_TYPE);
1708 assert(evrs != NULL);
1709 if (rpmtdFromStringArray(&td, RPMTAG_REQUIREVERSION, evrs, c))
1710 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1711 assert(rpmtdType(&td) == RPM_STRING_ARRAY_TYPE);
1713 assert(flags != NULL);
1714 if (rpmtdFromUint32(&td, RPMTAG_REQUIREFLAGS, flags, c))
1715 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1716 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1723 /* Add dependency dictionary(#dependencies) */
1724 if (rpmtdFromArgi(&td, RPMTAG_DEPENDSDICT, fc->ddictx)) {
1725 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1726 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1729 /* Add per-file dependency (start,number) pairs (#files) */
1730 if (rpmtdFromArgi(&td, RPMTAG_FILEDEPENDSX, fc->fddictx)) {
1731 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1732 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1735 if (rpmtdFromArgi(&td, RPMTAG_FILEDEPENDSN, fc->fddictn)) {
1736 assert(rpmtdType(&td) == RPM_INT32_TYPE);
1737 headerPut(pkg->header, &td, HEADERPUT_DEFAULT);
1740 printDeps(pkg->header);
1742 if (fc != NULL && _rpmfc_debug) {
1744 rasprintf(&msg, "final: files %d cdict[%d] %d%% ddictx[%d]", fc->nfiles, argvCount(fc->cdict), ((100 * fc->fknown)/fc->nfiles), argiCount(fc->ddictx));
1745 rpmfcPrint(msg, fc, NULL);
1750 fmode = _free(fmode);