From d06308ca0bc9cdc504bf468dc1c1e08c13fd34ab Mon Sep 17 00:00:00 2001 From: Panu Matilainen <pmatilai@redhat.com> Date: Mar 07 2012 10:08:22 +0000 Subject: - fix memory corruption on rpmdb size estimation (#766260) - fix couple of memleaks in python bindings (#782147) - fix regression in verify output formatting (#797964) - dont process spec include in false branch of if (#782970) - only warn on missing excluded files on build (#745629) - dont free up file info sets on test transactions --- diff --git a/rpm-4.9.x-exclude-warn.patch b/rpm-4.9.x-exclude-warn.patch new file mode 100644 index 0000000..d11bf76 --- /dev/null +++ b/rpm-4.9.x-exclude-warn.patch @@ -0,0 +1,57 @@ +commit cce686b2129e4e8dc27f1a640f7c4746f9ffb032 +Author: Panu Matilainen <pmatilai@redhat.com> +Date: Sun Oct 23 13:59:46 2011 +0300 + + Warn but don't fail the build on missing excluded files (RhBug:745629) + + - If a file/directory is not to be packaged, there's not a whole lot + point making the build fail if its missing. In case exclude is + used to leave certain files to sub-packages, the sub-package file + lists will catch out missing files that are really missing as a + result of actual build failure or such (except perhaps for some + glob cases but missing files can go unnoticed in those cases anyway) + - backported from commit 084a00bf51a941ec85c094a436bda401fccf7d3a + +diff --git a/build/files.c b/build/files.c +index e0747f8..a520410 100644 +--- a/build/files.c ++++ b/build/files.c +@@ -1393,12 +1393,17 @@ static rpmRC addFile(FileList fl, const char * diskPath, + statp->st_mtime = now; + statp->st_ctime = now; + } else { ++ int rc = RPMRC_FAIL; ++ int lvl = RPMLOG_ERR; + const char *msg = fl->isDir ? + _("Directory not found: %s\n") : + _("File not found: %s\n"); +- rpmlog(RPMLOG_ERR, msg, diskPath); +- fl->processingFailed = 1; +- return RPMRC_FAIL; ++ if (fl->currentFlags & RPMFILE_EXCLUDE) { ++ lvl = RPMLOG_WARNING; ++ rc = RPMRC_OK; ++ } ++ rpmlog(lvl, msg, diskPath); ++ return rc; + } + } + } +@@ -1702,11 +1707,15 @@ static rpmRC processBinaryFile(Package pkg, FileList fl, const char * fileName) + } + argvFree(argv); + } else { ++ int lvl = RPMLOG_WARNING; + const char *msg = (fl->isDir) ? + _("Directory not found by glob: %s\n") : + _("File not found by glob: %s\n"); +- rpmlog(RPMLOG_ERR, msg, diskPath); +- rc = RPMRC_FAIL; ++ if (!(fl->currentFlags & RPMFILE_EXCLUDE)) { ++ lvl = RPMLOG_ERR; ++ rc = RPMRC_FAIL; ++ } ++ rpmlog(lvl, msg, diskPath); + goto exit; + } + } else { diff --git a/rpm-4.9.x-include-cond.patch b/rpm-4.9.x-include-cond.patch new file mode 100644 index 0000000..50be6f3 --- /dev/null +++ b/rpm-4.9.x-include-cond.patch @@ -0,0 +1,21 @@ +commit bf92b843fabd6c9881b19efb0cae1578d16e4f7b +Author: Panu Matilainen <pmatilai@redhat.com> +Date: Tue Feb 28 12:18:10 2012 +0200 + + Don't process spec %include in false branch of %if clauses (RhBug:782970) + (backported from commit 9defc922e971d98203890f1557ab951ec94f2a3f) + +diff --git a/build/parseSpec.c b/build/parseSpec.c +index 01620bd..01c3c08 100644 +--- a/build/parseSpec.c ++++ b/build/parseSpec.c +@@ -374,7 +374,8 @@ int readLine(rpmSpec spec, int strip) + spec->readStack = spec->readStack->next; + free(rl); + spec->line[0] = '\0'; +- } else if (rstreqn("%include", s, sizeof("%include")-1)) { ++ } else if (spec->readStack->reading && ++ rstreqn("%include", s, sizeof("%include")-1)) { + char *fileName, *endFileName, *p; + + s += 8; diff --git a/rpm-4.9.x-python-memleaks.patch b/rpm-4.9.x-python-memleaks.patch new file mode 100644 index 0000000..795294a --- /dev/null +++ b/rpm-4.9.x-python-memleaks.patch @@ -0,0 +1,66 @@ +commit 59807943af5c10c892b239f11b8fafb209254a4a +Author: David Malcolm <dmalcolm@redhat.com> +Date: Thu Dec 15 22:22:56 2011 -0500 + + fix memory leaks in invocations of PyObject_Call + + - Various functions in the Python bindings have expressions of the form: + + PyObject_Call(callable, + Py_BuildValue(fmtstring, ...), NULL); + + This leaks memory for the case when Py_BuildValue succeeds (it returns a + new reference, which is never freed; PyObject_Call doesn't steal the + reference): the argument tuple and all of its components will not be + freed (until the process exits). + + Signed-off-by: Ales Kozumplik <akozumpl@redhat.com> + +diff --git a/python/header-py.c b/python/header-py.c +index 96cf200..cef457b 100644 +--- a/python/header-py.c ++++ b/python/header-py.c +@@ -295,8 +295,7 @@ static PyObject * hdrWrite(hdrObject *s, PyObject *args, PyObject *kwds) + */ + static PyObject * hdr_fiFromHeader(PyObject * s, PyObject * args, PyObject * kwds) + { +- return PyObject_Call((PyObject *) &rpmfi_Type, +- Py_BuildValue("(O)", s), NULL); ++ return PyObject_CallFunctionObjArgs((PyObject *) &rpmfi_Type, s, NULL); + } + + /* Backwards compatibility. Flags argument is just a dummy and discarded. */ +@@ -309,14 +308,14 @@ static PyObject * hdr_dsFromHeader(PyObject * s, PyObject * args, PyObject * kwd + tagNumFromPyObject, &tag, &flags)) + return NULL; + +- return PyObject_Call((PyObject *) &rpmds_Type, +- Py_BuildValue("(Oi)", s, tag), NULL); ++ return PyObject_CallFunction((PyObject *) &rpmds_Type, ++ "(Oi)", s, tag); + } + + static PyObject * hdr_dsOfHeader(PyObject * s) + { +- return PyObject_Call((PyObject *) &rpmds_Type, +- Py_BuildValue("(Oi)", s, RPMTAG_NEVR), NULL); ++ return PyObject_CallFunction((PyObject *) &rpmds_Type, ++ "(Oi)", s, RPMTAG_NEVR); + } + + static long hdr_hash(PyObject * h) +diff --git a/python/rpmfd-py.c b/python/rpmfd-py.c +index 89a70cd..1150aa1 100644 +--- a/python/rpmfd-py.c ++++ b/python/rpmfd-py.c +@@ -23,8 +23,8 @@ int rpmfdFromPyObject(PyObject *obj, rpmfdObject **fdop) + Py_INCREF(obj); + fdo = (rpmfdObject *) obj; + } else { +- fdo = (rpmfdObject *) PyObject_Call((PyObject *)&rpmfd_Type, +- Py_BuildValue("(O)", obj), NULL); ++ fdo = (rpmfdObject *) PyObject_CallFunctionObjArgs((PyObject *)&rpmfd_Type, ++ obj, NULL); + } + if (fdo == NULL) return 0; + diff --git a/rpm-4.9.x-rpmdb-dsi.patch b/rpm-4.9.x-rpmdb-dsi.patch new file mode 100644 index 0000000..8157a86 --- /dev/null +++ b/rpm-4.9.x-rpmdb-dsi.patch @@ -0,0 +1,98 @@ +diff --git a/lib/transaction.c b/lib/transaction.c +index 7adf60b..5acc08e 100644 +--- a/lib/transaction.c ++++ b/lib/transaction.c +@@ -227,46 +227,12 @@ static void rpmtsUpdateDSI(const rpmts ts, dev_t dev, const char *dirName, + if (dsi->ineeded < dsi->oineeded) dsi->oineeded = dsi->ineeded; + } + +-/* return DSI of the device the rpmdb lives on */ +-static rpmDiskSpaceInfo rpmtsDbDSI(const rpmts ts) { +- const char *dbhome = rpmdbHome(rpmtsGetRdb(ts)); +- struct stat sb; +- int rc; +- +- rc = stat(dbhome, &sb); +- if (rc) { +- return NULL; +- } +- return rpmtsGetDSI(ts, sb.st_dev, dbhome); +-} +- +-/* Update DSI for changing size of the rpmdb */ +-static void rpmtsUpdateDSIrpmDBSize(const rpmte p, +- rpmDiskSpaceInfo dsi) { +- rpm_loff_t headerSize; +- int64_t bneeded; +- +- /* XXX somehow we can end up here with bsize 0 (RhBug:671056) */ +- if (dsi == NULL || dsi->bsize == 0) return; +- +- headerSize = rpmteHeaderSize(p); +- bneeded = BLOCK_ROUND(headerSize, dsi->bsize); +- /* REMOVE doesn't neccessarily shrink the database */ +- if (rpmteType(p) == TR_ADDED) { +- /* guessing that db grows 4 times more than the header size */ +- dsi->bneeded += (bneeded * 4); +- } +-} +- +- + static void rpmtsCheckDSIProblems(const rpmts ts, const rpmte te) + { + rpmDiskSpaceInfo dsi = ts->dsi; + + if (dsi == NULL || !dsi->bsize) + return; +- if (rpmfiFC(rpmteFI(te)) <= 0) +- return; + + for (; dsi->bsize; dsi++) { + +@@ -1294,11 +1260,12 @@ static int rpmtsPrepare(rpmts ts) + rpmfi fi; + int rc = 0; + uint64_t fileCount = countFiles(ts); ++ const char *dbhome = NULL; ++ struct stat dbstat; + + fingerPrintCache fpc = fpCacheCreate(fileCount/2 + 10001); + rpmFpHash ht = rpmFpHashCreate(fileCount/2+1, fpHashFunction, fpEqual, + NULL, NULL); +- rpmDiskSpaceInfo dsi; + + rpmlog(RPMLOG_DEBUG, "computing %" PRIu64 " file fingerprints\n", fileCount); + +@@ -1326,7 +1293,10 @@ static int rpmtsPrepare(rpmts ts) + /* check against files in the rpmdb */ + checkInstalledFiles(ts, fileCount, ht, fpc); + +- dsi = rpmtsDbDSI(ts); ++ dbhome = rpmdbHome(rpmtsGetRdb(ts)); ++ /* If we can't stat, ignore db growth. Probably not right but... */ ++ if (dbhome && stat(dbhome, &dbstat)) ++ dbhome = NULL; + + pi = rpmtsiInit(ts); + while ((p = rpmtsiNext(pi, 0)) != NULL) { +@@ -1338,10 +1308,18 @@ static int rpmtsPrepare(rpmts ts) + needs on each partition for this package. */ + handleOverlappedFiles(ts, ht, p, fi); + +- rpmtsUpdateDSIrpmDBSize(p, dsi); +- + /* Check added package has sufficient space on each partition used. */ + if (rpmteType(p) == TR_ADDED) { ++ /* ++ * Try to estimate space needed for rpmdb growth: guess that the ++ * db grows 4 times the header size (indexes and all). ++ */ ++ if (dbhome) { ++ int64_t hsize = rpmteHeaderSize(p) * 4; ++ rpmtsUpdateDSI(ts, dbstat.st_dev, dbhome, ++ hsize, 0, 0, FA_CREATE); ++ } ++ + rpmtsCheckDSIProblems(ts, p); + } + (void) rpmswExit(rpmtsOp(ts, RPMTS_OP_FINGERPRINT), 0); diff --git a/rpm-4.9.x-tstest-fileinfo.patch b/rpm-4.9.x-tstest-fileinfo.patch new file mode 100644 index 0000000..60250d4 --- /dev/null +++ b/rpm-4.9.x-tstest-fileinfo.patch @@ -0,0 +1,43 @@ +commit 7d88d5e54a9c3fa2b7fb443921c8adb799051784 +Author: Panu Matilainen <pmatilai@redhat.com> +Date: Wed Feb 1 17:48:20 2012 +0200 + + Don't free up file info sets on transaction test-runs + + - We'd like to get rid of the potentially huge amounts of memory + eaten by file info sets as early as possible, but when there's a + chance that we'll get called again with either added transacation + elements or on-disk changes, such as %pretrans changing something + underneath us, we need to (be able to) recalculate everything + from scratch. Only free up the memory when we know we dont need + it anymore, ie on an actual transaction run. + - This doesn't change anything for rpm itself, for yum and others + which do a separate test-transaction first, it means %pretrans + directory<->symlink replacement hacks and the like have a chance + of working again. I'm sure there's a bug filed on this somewhere but... + (cherry picked from commit cef18c94807af0935b7796c462aab8ed39f0f376) + +diff --git a/lib/transaction.c b/lib/transaction.c +index 56ebc17..8d29ab9 100644 +--- a/lib/transaction.c ++++ b/lib/transaction.c +@@ -1355,12 +1355,14 @@ static int rpmtsPrepare(rpmts ts) + if (rpmChrootOut()) + rc = -1; + +- /* File info sets, fp caches etc not needed beyond here, free 'em up. */ +- pi = rpmtsiInit(ts); +- while ((p = rpmtsiNext(pi, 0)) != NULL) { +- rpmteSetFI(p, NULL); ++ /* On actual transaction, file info sets are not needed after this */ ++ if (!(rpmtsFlags(ts) & (RPMTRANS_FLAG_TEST|RPMTRANS_FLAG_BUILD_PROBS))) { ++ pi = rpmtsiInit(ts); ++ while ((p = rpmtsiNext(pi, 0)) != NULL) { ++ rpmteSetFI(p, NULL); ++ } ++ rpmtsiFree(pi); + } +- pi = rpmtsiFree(pi); + + exit: + ht = rpmFpHashFree(ht); diff --git a/rpm-4.9.x-verify-output.patch b/rpm-4.9.x-verify-output.patch new file mode 100644 index 0000000..a06cf51 --- /dev/null +++ b/rpm-4.9.x-verify-output.patch @@ -0,0 +1,50 @@ +commit 4e207bfdfce434a6484babc14fe86aeadeec5329 +Author: Panu Matilainen <pmatilai@redhat.com> +Date: Tue Feb 28 10:31:28 2012 +0200 + + Unbreak rpm -V output (RhBug:797964) + + - Commit ac0ab016a5ec31e65eb0c0910a5a6f1199aae3e7 unintentionally + changed the order of the problems shown in verify strings due to + a dumb oversight (greetings to self, duh). In other words, this + fixes a verify output regression in rpm >= 4.9.x by restoring + the long-standing (and documented) order of the verify output chars. + - Also fix the testcase which unfortunately was only added after + the output-changing commit so it didn't catch the breakage either :-/ + +diff --git a/lib/verify.c b/lib/verify.c +index 35612fe..1edb27f 100644 +--- a/lib/verify.c ++++ b/lib/verify.c +@@ -292,14 +292,14 @@ char * rpmVerifyString(uint32_t verifyResult, const char *pad) + { + char *fmt = NULL; + rasprintf(&fmt, "%s%s%s%s%s%s%s%s%s", +- _verifyfile(RPMVERIFY_FILEDIGEST, "5", pad), + _verify(RPMVERIFY_FILESIZE, "S", pad), +- _verifylink(RPMVERIFY_LINKTO, "L", pad), +- _verify(RPMVERIFY_MTIME, "T", pad), ++ _verify(RPMVERIFY_MODE, "M", pad), ++ _verifyfile(RPMVERIFY_FILEDIGEST, "5", pad), + _verify(RPMVERIFY_RDEV, "D", pad), ++ _verifylink(RPMVERIFY_LINKTO, "L", pad), + _verify(RPMVERIFY_USER, "U", pad), + _verify(RPMVERIFY_GROUP, "G", pad), +- _verify(RPMVERIFY_MODE, "M", pad), ++ _verify(RPMVERIFY_MTIME, "T", pad), + _verify(RPMVERIFY_CAPS, "P", pad)); + + return fmt; +diff --git a/tests/rpmverify.at b/tests/rpmverify.at +index dd23a4a..77d6bfe 100644 +--- a/tests/rpmverify.at ++++ b/tests/rpmverify.at +@@ -79,7 +79,7 @@ dd if=/dev/zero of="${RPMTEST}"/usr/local/bin/hello \ + runroot rpm -Va --nodeps --nouser --nogroup + ], + [1], +-[5..T...M. /usr/local/bin/hello ++[.M5....T. /usr/local/bin/hello + missing d /usr/share/doc/hello-1.0/FAQ + ], + []) diff --git a/rpm.spec b/rpm.spec index 49a2276..821fd39 100644 --- a/rpm.spec +++ b/rpm.spec @@ -21,7 +21,7 @@ Summary: The RPM package management system Name: rpm Version: %{rpmver} -Release: %{?snapver:0.%{snapver}.}12%{?dist} +Release: %{?snapver:0.%{snapver}.}13%{?dist} Group: System Environment/Base Url: http://www.rpm.org/ Source0: http://rpm.org/releases/rpm-4.9.x/%{name}-%{srcver}.tar.bz2 @@ -47,6 +47,12 @@ Patch100: rpm-4.9.x-fontattr.patch Patch101: rpm-4.9.x-elfattr.patch Patch102: rpm-4.9.1.2-perl-python-attr.patch Patch103: rpm-4.9.x-mpsize.patch +Patch104: rpm-4.9.x-rpmdb-dsi.patch +Patch105: rpm-4.9.x-python-memleaks.patch +Patch106: rpm-4.9.x-verify-output.patch +Patch107: rpm-4.9.x-include-cond.patch +Patch108: rpm-4.9.x-exclude-warn.patch +Patch109: rpm-4.9.x-tstest-fileinfo.patch # These are not yet upstream Patch301: rpm-4.6.0-niagara.patch @@ -228,6 +234,12 @@ packages on a system. %patch101 -p1 -b .elfattr %patch102 -p1 -b .perl-python-attr %patch103 -p1 -b .mpsize +%patch104 -p1 -b .rpmdb-dsi +%patch105 -p1 -b .python-memleaks +%patch106 -p1 -b .verify-output +%patch107 -p1 -b .include-cond +%patch108 -p1 -b .exclude-warn +%patch109 -p1 -b .tstest-fileinfo %patch301 -p1 -b .niagara %patch302 -p1 -b .geode @@ -457,6 +469,14 @@ exit 0 %doc COPYING doc/librpm/html/* %changelog +* Wed Mar 07 2012 Panu Matilainen <pmatilai@redhat.com> - 4.9.1.2-13 +- fix memory corruption on rpmdb size estimation (#766260) +- fix couple of memleaks in python bindings (#782147) +- fix regression in verify output formatting (#797964) +- dont process spec include in false branch of if (#782970) +- only warn on missing excluded files on build (#745629) +- dont free up file info sets on test transactions + * Thu Feb 09 2012 Panu Matilainen <pmatilai@redhat.com> - 4.9.1.2-12 - switch back to smaller BDB cache default (#752897)