#include "system.h" #include #include #include "build.h" #include "install.h" static int checkSpec(Header h) { char *rootdir = NULL; rpmdb db = NULL; int mode = O_RDONLY; rpmTransactionSet ts; struct rpmDependencyConflict * conflicts; int numConflicts; int rc; if (!headerIsEntry(h, RPMTAG_REQUIREFLAGS)) return 0; if (rpmdbOpen(rootdir, &db, mode, 0644)) { const char *dn; dn = rpmGetPath( (rootdir ? rootdir : ""), "%{_dbpath}", NULL); rpmMessage(RPMMESS_ERROR, _("cannot open %s/packages.rpm\n"), dn); xfree(dn); exit(EXIT_FAILURE); } ts = rpmtransCreateSet(db, rootdir); rc = rpmtransAddPackage(ts, h, NULL, NULL, 0, NULL); rc = rpmdepCheck(ts, &conflicts, &numConflicts); if (rc == 0 && conflicts) { rpmMessage(RPMMESS_ERROR, _("failed build dependencies:\n")); printDepProblems(stderr, conflicts, numConflicts); rpmdepFreeConflicts(conflicts, numConflicts); rc = 1; } if (ts) rpmtransFree(ts); if (db) rpmdbClose(db); return rc; } /* * Kurwa, durni ameryka?ce sobe zawsze my?l?, ?e ca?y ?wiat mówi po * angielsku... */ /* XXX this is still a dumb test but at least it's i18n aware */ static int isSpecFile(const char *specfile) { char buf[256]; const char * s; FD_t fd; int count; int checking; fd = Fopen(specfile, "r.ufdio"); if (fd == NULL || Ferror(fd)) { fprintf(stderr, _("Unable to open spec file %s: %s\n"), specfile, Fstrerror(fd)); return 0; } count = Fread(buf, sizeof(buf[0]), sizeof(buf), fd); Fclose(fd); checking = 1; for (s = buf; count--; s++) { switch (*s) { case '\r': case '\n': checking = 1; break; case ':': checking = 0; break; default: if (checking && !(isprint(*s) || isspace(*s))) return 0; break; } } return 1; } static int buildForTarget(const char *arg, struct rpmBuildArguments *ba, const char *passPhrase, int fromTarball, char *cookie, int force, int nodeps) { int buildAmount = ba->buildAmount; const char *buildRootURL = NULL; int test = ba->noBuild; const char * specFile; const char * specURL; int specut; char buf[BUFSIZ]; Spec spec = NULL; int rc; rpmSetTables(RPM_MACHTABLE_BUILDARCH, RPM_MACHTABLE_BUILDOS); if (ba->buildRootOverride) buildRootURL = rpmGenPath(NULL, ba->buildRootOverride, NULL); if (fromTarball) { FILE *fp; const char *specDir; const char * tmpSpecFile; char * cmd, *s; int res; static const char *zcmds[] = { "cat", "gunzip", "bunzip2", "cat" }; specDir = rpmGetPath("%{_specdir}", NULL); { char tfn[64]; strcpy(tfn, "rpm-spec.XXXXXX"); tmpSpecFile = rpmGetPath("%{_specdir}/", mktemp(tfn), NULL); } isCompressed(arg, &res); cmd = alloca(strlen(arg) + 50 + strlen(tmpSpecFile)); sprintf(cmd, "%s < %s | tar xOvf - Specfile 2>&1 > %s", zcmds[res & 0x3], arg, tmpSpecFile); if (!(fp = popen(cmd, "r"))) { fprintf(stderr, _("Failed to open tar pipe: %s\n"), strerror(errno)); xfree(specDir); xfree(tmpSpecFile); return 1; } if ((!fgets(buf, sizeof(buf) - 1, fp)) || !strchr(buf, '/')) { /* Try again */ pclose(fp); sprintf(cmd, "%s < %s | tar xOvf - \\*.spec 2>&1 > %s", zcmds[res & 0x3], arg, tmpSpecFile); if (!(fp = popen(cmd, "r"))) { fprintf(stderr, _("Failed to open tar pipe: %s\n"), strerror(errno)); xfree(specDir); xfree(tmpSpecFile); return 1; } if (!fgets(buf, sizeof(buf) - 1, fp)) { /* Give up */ fprintf(stderr, _("Failed to read spec file from %s\n"), arg); unlink(tmpSpecFile); xfree(specDir); xfree(tmpSpecFile); return 1; } } pclose(fp); cmd = s = buf; while (*cmd) { if (*cmd == '/') s = cmd + 1; cmd++; } cmd = s; /* remove trailing \n */ s = cmd + strlen(cmd) - 1; *s = '\0'; specURL = s = alloca(strlen(specDir) + strlen(cmd) + 5); sprintf(s, "%s/%s", specDir, cmd); res = rename(tmpSpecFile, s); xfree(specDir); xfree(tmpSpecFile); if (res) { fprintf(stderr, _("Failed to rename %s to %s: %s\n"), tmpSpecFile, s, strerror(errno)); unlink(tmpSpecFile); return 1; } /* Make the directory which contains the tarball the source directory for this run */ if (*arg != '/') { (void)getcwd(buf, BUFSIZ); strcat(buf, "/"); strcat(buf, arg); } else strcpy(buf, arg); cmd = buf + strlen(buf) - 1; while (*cmd != '/') cmd--; *cmd = '\0'; addMacro(NULL, "_sourcedir", NULL, buf, RMIL_TARBALL); } else { specURL = arg; } specut = urlPath(specURL, &specFile); if (*specFile != '/') { char *s = alloca(BUFSIZ); (void)getcwd(s, BUFSIZ); strcat(s, "/"); strcat(s, arg); specURL = s; } if (specut != URL_IS_DASH) { struct stat st; Stat(specURL, &st); if (! S_ISREG(st.st_mode)) { fprintf(stderr, _("File is not a regular file: %s\n"), specURL); rc = 1; goto exit; } /* Try to verify that the file is actually a specfile */ if (!isSpecFile(specURL)) { fprintf(stderr, _("File %s does not appear to be a specfile.\n"), specURL); rc = 1; goto exit; } } /* Parse the spec file */ #define _anyarch(_f) \ (((_f)&(RPMBUILD_PREP|RPMBUILD_BUILD|RPMBUILD_INSTALL|RPMBUILD_PACKAGEBINARY)) == 0) if (parseSpec(&spec, specURL, ba->rootdir, buildRootURL, 0, passPhrase, cookie, _anyarch(buildAmount), force)) { rc = 1; goto exit; } #undef _anyarch /* Assemble source header from parsed components */ initSourceHeader(spec); /* Check build prerequisites */ if (!nodeps && checkSpec(spec->sourceHeader)) { rc = 1; goto exit; } if (buildSpec(spec, buildAmount, test)) { rc = 1; goto exit; } if (fromTarball) Unlink(specURL); rc = 0; exit: if (spec) freeSpec(spec); if (buildRootURL) xfree(buildRootURL); return rc; } /** */ int build(const char * arg, struct rpmBuildArguments * ba, const char * passPhrase, int fromTarball, char * cookie, const char * rcfile, int force, int nodeps) { char *t, *te; int rc = 0; char *targets = ba->targets; #define buildCleanMask (RPMBUILD_RMSOURCE|RPMBUILD_RMSPEC) int cleanFlags = ba->buildAmount & buildCleanMask; if (targets == NULL) { rc = buildForTarget(arg, ba, passPhrase, fromTarball, cookie, force, nodeps); goto exit; } /* parse up the build operators */ printf(_("Building target platforms: %s\n"), targets); ba->buildAmount &= ~buildCleanMask; for (t = targets; *t != '\0'; t = te) { char *target; if ((te = strchr(t, ',')) == NULL) te = t + strlen(t); target = alloca(te-t+1); strncpy(target, t, (te-t)); target[te-t] = '\0'; if (*te) te++; else /* XXX Perform clean-up after last target build. */ ba->buildAmount |= cleanFlags; printf(_("Building for target %s\n"), target); /* Read in configuration for target. */ rpmFreeMacros(NULL); rpmReadConfigFiles(rcfile, target); rc = buildForTarget(arg, ba, passPhrase, fromTarball, cookie, force, nodeps); if (rc) break; } exit: /* Restore original configuration. */ rpmFreeMacros(NULL); rpmReadConfigFiles(rcfile, NULL); return rc; } #define POPT_USECATALOG 1000 #define POPT_NOLANG 1001 #define POPT_RMSOURCE 1002 #define POPT_RMBUILD 1003 #define POPT_BUILDROOT 1004 #define POPT_BUILDARCH 1005 #define POPT_BUILDOS 1006 #define POPT_TARGETPLATFORM 1007 #define POPT_NOBUILD 1008 #define POPT_SHORTCIRCUIT 1009 #define POPT_RMSPEC 1010 extern int noLang; static int noBuild = 0; static int useCatalog = 0; static void buildArgCallback( /*@unused@*/ poptContext con, /*@unused@*/ enum poptCallbackReason reason, const struct poptOption * opt, const char * arg, const void * data) { struct rpmBuildArguments * rba = (struct rpmBuildArguments *) data; switch (opt->val) { case POPT_USECATALOG: rba->useCatalog = 1; break; case POPT_NOBUILD: rba->noBuild = 1; break; case POPT_NOLANG: rba->noLang = 1; break; case POPT_SHORTCIRCUIT: rba->shortCircuit = 1; break; case POPT_RMSOURCE: rba->buildAmount |= RPMBUILD_RMSOURCE; break; case POPT_RMSPEC: rba->buildAmount |= RPMBUILD_RMSPEC; break; case POPT_RMBUILD: rba->buildAmount |= RPMBUILD_RMBUILD; break; case POPT_BUILDROOT: if (rba->buildRootOverride) { fprintf(stderr, _("buildroot already specified")); exit(EXIT_FAILURE); /*@notreached@*/ } rba->buildRootOverride = xstrdup(arg); break; case POPT_BUILDARCH: fprintf(stderr, _("--buildarch has been obsoleted. Use the --target option instead.\n")); exit(EXIT_FAILURE); /*@notreached@*/ break; case POPT_BUILDOS: fprintf(stderr, _("--buildos has been obsoleted. Use the --target option instead.\n")); exit(EXIT_FAILURE); /*@notreached@*/ break; case POPT_TARGETPLATFORM: if (rba->targets) { int len = strlen(rba->targets) + 1 + strlen(arg) + 1; rba->targets = xrealloc(rba->targets, len); strcat(rba->targets, ","); } else { rba->targets = xmalloc(strlen(arg) + 1); rba->targets[0] = '\0'; } strcat(rba->targets, arg); break; } } /** */ struct poptOption rpmBuildPoptTable[] = { { NULL, '\0', POPT_ARG_CALLBACK | POPT_CBFLAG_INC_DATA, buildArgCallback, 0, NULL, NULL }, { "buildarch", '\0', POPT_ARG_STRING, 0, POPT_BUILDARCH, N_("override build architecture"), "ARCH" }, { "buildos", '\0', POPT_ARG_STRING, 0, POPT_BUILDOS, N_("override build operating system"), "OS" }, { "buildroot", '\0', POPT_ARG_STRING, 0, POPT_BUILDROOT, N_("override build root"), "DIRECTORY" }, { "clean", '\0', 0, 0, POPT_RMBUILD, N_("remove build tree when done"), NULL}, { "nobuild", '\0', 0, &noBuild, POPT_NOBUILD, N_("do not execute any stages of the build"), NULL }, { "nolang", '\0', 0, &noLang, POPT_NOLANG, N_("do not accept I18N msgstr's from specfile"), NULL}, { "rmsource", '\0', 0, 0, POPT_RMSOURCE, N_("remove sources when done"), NULL}, { "rmspec", '\0', 0, 0, POPT_RMSPEC, N_("remove specfile when done"), NULL}, { "short-circuit", '\0', 0, 0, POPT_SHORTCIRCUIT, N_("skip straight to specified stage (only for c,i)"), NULL }, { "target", '\0', POPT_ARG_STRING, 0, POPT_TARGETPLATFORM, N_("override target platform"), "CPU-VENDOR-OS" }, { "usecatalog", '\0', 0, &useCatalog, POPT_USECATALOG, N_("lookup I18N strings in specfile catalog"), NULL}, { 0, 0, 0, 0, 0, NULL, NULL } };