/* * checkFiles.c - check packaged file list. * Written by Alexey Tourbin . * License: GPLv2+. */ #include "system.h" #include "rpmio_internal.h" #include "rpmbuild.h" #include "psm.h" #include "checkFiles.h" void fiIntersect(const TFI_t fi1, const TFI_t fi2, void (*cb)(const TFI_t fi1, const TFI_t fi2, const char *f, int i1, int i2, void *data), void *data) { if (!fi1 || !fi2) return; int i1 = 0, i2 = 0; while (i1 < fi1->fc && i2 < fi2->fc) { char f1[PATH_MAX], f2[PATH_MAX]; strcpy(f1, fi1->dnl[fi1->dil[i1]] + fi1->astriplen); strcpy(f2, fi2->dnl[fi2->dil[i2]] + fi2->astriplen); strcat(f1, fi1->bnl[i1]); strcat(f2, fi2->bnl[i2]); int cmp = strcmp(f1, f2); if (cmp < 0) { i1++; continue; } if (cmp > 0) { i2++; continue; } cb(fi1, fi2, f1, i1, i2, data); i1++; i2++; } } static void fiIntersect_cb(const TFI_t fi1, const TFI_t fi2, const char *f, int i1, int i2, void *data) { if (S_ISDIR(fi1->fmodes[i1]) && S_ISDIR(fi2->fmodes[i2])) return; const char src[] = "/usr/src/debug/"; if (strncmp(f, src, sizeof(src) - 1) == 0) return; int *once = data; if (!*once) { rpmlog(RPMLOG_WARNING, "File(s) packaged into both %s-%s-%s and %s-%s-%s:\n", fi1->name, fi1->version, fi1->release, fi2->name, fi2->version, fi2->release); *once = 1; } rpmlog(RPMLOG_INFO, " %s\n", f); } static void checkPkgIntersect(Package pkg1, Package pkg2) { const TFI_t fi1 = pkg1->cpioList; const TFI_t fi2 = pkg2->cpioList; if (!fi1 || !fi2) return; int once = 0; fiIntersect(fi1, fi2, fiIntersect_cb, (void *) &once); } static void checkIntersect(Spec spec) { Package pkg1, pkg2; for (pkg1 = spec->packages; pkg1; pkg1 = pkg1->next) for (pkg2 = pkg1->next; pkg2; pkg2 = pkg2->next) checkPkgIntersect(pkg1, pkg2); } static int fiSearch(TFI_t fi, const char *path) { if (fi == NULL || fi->fc == 0) return -1; int l = 0; int u = fi->fc; while (l < u) { int i = (l + u) / 2; const char *d = fi->dnl[fi->dil[i]]; int dlen = strlen(d); int cmp = strncmp(path, d, dlen); if (cmp == 0) { const char *b = fi->bnl[i]; cmp = strcmp(path + dlen, b); } if (cmp < 0) u = i; else if (cmp > 0) l = i + 1; else return i; } return -1; } #include static int avcmp(const void *sptr1, const void *sptr2) { const char *s1 = *(const char **) sptr1; const char *s2 = *(const char **) sptr2; return strcmp(s1, s2); } static int check(const char *path, const Spec spec, unsigned int *uc, char ***uv) { Package pkg; for (pkg = spec->packages; pkg; pkg = pkg->next) if (fiSearch(pkg->cpioList, path) >= 0) return 0; if (spec->exclude) { const char *key = path + strlen(spec->buildRootURL); if (bsearch(&key, spec->exclude, spec->excludeCount, sizeof(char *), avcmp)) return 0; } AUTO_REALLOC(*uv, *uc, 8); (*uv)[(*uc)++] = xstrdup(path + strlen(spec->buildRootURL)); return 1; } static int checkUnpackaged(Spec spec) { int rc = 0; unsigned int uc = 0; char **uv = NULL; if (spec->exclude) qsort(spec->exclude, spec->excludeCount, sizeof(char *), avcmp); char *paths[] = { (char *) spec->buildRootURL, 0 }; int options = FTS_COMFOLLOW | FTS_PHYSICAL; FTS *ftsp = fts_open(paths, options, NULL); FTSENT *fts; while ((fts = fts_read(ftsp)) != NULL) { // buildroot may not exist if (fts->fts_level == 0 && fts->fts_info == FTS_NS) { Package pkg; for (pkg = spec->packages; pkg; pkg = pkg->next) { TFI_t fi = pkg->cpioList; if (fi && fi->fc > 0) break; } if (pkg == NULL) continue; } // skip dotfiles in buildroot if (fts->fts_level == 1 && *fts->fts_name == '.') { if (fts->fts_info == FTS_D) fts_set(ftsp, fts, FTS_SKIP); continue; } // skip debuginfo if (fts->fts_level == 3 && fts->fts_info == FTS_D) { const char *dir = fts->fts_path + strlen(spec->buildRootURL); if (strcmp(dir, "/usr/lib/debug") == 0 || strcmp(dir, "/usr/src/debug") == 0) { fts_set(ftsp, fts, FTS_SKIP); continue; } } switch (fts->fts_info) { // do not check for unpackaged directories case FTS_D: case FTS_DP: break; case FTS_F: case FTS_SL: case FTS_SLNONE: case FTS_DEFAULT: rc |= check(fts->fts_path, spec, &uc, &uv); break; default: rpmlog(RPMLOG_WARNING, "%s: fts error\n", fts->fts_path); rc |= 2; break; } } if (uv) { rpmlog(RPMLOG_WARNING, "Installed (but unpackaged) file(s) found:\n"); qsort(uv, uc, sizeof(*uv), avcmp); unsigned int i; for (i = 0; i < uc; i++) rpmlog(RPMLOG_INFO, " %s\n", uv[i]); free(uv); } return rc; } int checkFiles(Spec spec) { checkIntersect(spec); int rc = checkUnpackaged(spec); if (rc && rpmExpandNumeric("%{?_unpackaged_files_terminate_build}")) { rpmlog(RPMLOG_ERR, "File list check failed, terminating build\n"); return rc; } return 0; } // ex: set ts=8 sts=4 sw=4 noet: