From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Date: Tue, 28 Aug 2007 21:11:00 +0400 From: "Alexey M. Tourbin" To: devel@lists.altlinux.org Message-ID: <20070828171100.GE24207@solemn.turbinal> Mail-Followup-To: devel@lists.altlinux.org Mime-Version: 1.0 Content-Type: text/plain; charset=koi8-r Content-Disposition: inline Content-Transfer-Encoding: 8bit Subject: [devel] pkgconfiglib.req X-BeenThere: devel@lists.altlinux.org X-Mailman-Version: 2.1.9rc1 Precedence: list Reply-To: ALT Devel discussion list List-Id: ALT Devel discussion list List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Tue, 28 Aug 2007 17:11:00 -0000 Archived-At: List-Archive: List-Post: Иногда бывает так, что с новым devel-пакетом перестает что-то собираться из-за добавившихся библиотек для линковки в *.pc файле. Здесь есть два подхода. Первый подход -- это, если добавленные библиотеки на самом деле излишни, то мы получаем ценную информацию, чтобы дать кому-то подзатыльник (точнее, чтобы убрать ненужные библиотеки из Libs в новом пакете). С другой стороны, раздача подзатыльников методом поломки репозитария не кажется мне вполне технологичным развлечением. Второй подход -- это, если добавленные библиотеки на самом деле не лишние, добить новых *-devel зависимостей на соответствующие пакеты. Кроме всего прочего, этот подход относительно легко перепоручить автоматике. Поэтому предлагаю замыкать зависимости между *-devel пакетами по содержимому поля Libs в *.pc файлах. Алгоритм, который реализован ниже, не слишком точно мимикрирует перебор путей в ld (в частности, *.a библиотеки не просматриваются). Тем не мене, мне пока не приходит в голову конфигурация, даже, допустим, патологическая, при которой может получиться неправильный результат. ДАННОЕ ИЗМЕНЕНИЕ, и не одно оно, ОКОНЧАТЕЛЬНО ПЕРЕВОДИТ *.pc ФАЙЛЫ В СТАТУС "ДЛЯ *-devel ПАКЕТОВ". Уважаемые товарищи maintaner'ы! Кладите *.pc файл в *-devel подпакет, либо не пакуйте его вообще, до тех пор, пока он кому-нибудь не понадобится. В своей хост-системе я обнаружил 2 *.pc файла из не-devel пакетов, которые имеют некоторые проблемы при попытке их обработки: $ rpm -qf /usr/lib/pkgconfig/libgdiplus.pc libgdiplus-1.2.4-alt1 $ rpm -qf /usr/lib/pkgconfig/avahi-qt3.pc libavahi-qt3-0.6.20-alt2 $ Некоторые подробности на этот счет приведены ниже. Changelog since `4.0.4-alt77-67-g63ac11f' follows: commit 0e085c0a8388be44d06afe1a1f4b88aa8895ad2e Author: Alexey Tourbin Date: Tue Aug 28 20:10:35 2007 +0400 pkgconfiglib.req: new pkgconfig.req mode (makes dependencies on Libs) This will grab libraries from ^Libs: clause and map each library to rpm dependency, which is typically lib*-devel package. $ grep ^Libs: /usr/lib/pkgconfig/directfb.pc Libs: -ldirectfb -lpthread -ldl -lz $ It works like this: $ ln -s pkgconfig.req.in scripts/pkgconfiglib.req.in $ scripts/pkgconfiglib.req.in -v /usr/lib/pkgconfig/directfb.pc pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libdirectfb.so -> libdirectfb-devel libdirectfb-devel pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libz.so -> zlib-devel zlib-devel pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libfusion.so -> libdirectfb-devel libdirectfb-devel pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libdirect.so -> libdirectfb-devel libdirectfb-devel pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libpthread.so -> glibc-devel (skip) pkgconfiglib.req.in: /usr/lib/pkgconfig/directfb.pc: libdl.so -> glibc-devel (skip) $ Some minor problems: $ scripts/pkgconfiglib.req.in /usr/lib/pkgconfig/*.pc >/dev/null pkgconfiglib.req.in: /usr/lib/pkgconfig/avahi-qt3.pc: cannot find libavahi-qt3.so library path (skip) pkgconfiglib.req.in: /usr/lib/pkgconfig/libgdiplus.pc: cannot find libexif.so library path (skip) pkgconfiglib.req.in: /usr/lib/pkgconfig/valgrind.pc: cannot find libcoregrind.so library path (skip) pkgconfiglib.req.in: /usr/lib/pkgconfig/valgrind.pc: cannot find libvex.so library path (skip) pkgconfiglib.req.in: /usr/lib/pkgconfig/valgrind.pc: cannot find libgcc.so library path (skip) $ Full diff since `4.0.4-alt77-67-g63ac11f' follows: diff --git a/rpm-4_0.spec b/rpm-4_0.spec index a7ace2a..c167300 100644 --- a/rpm-4_0.spec +++ b/rpm-4_0.spec @@ -474,6 +474,7 @@ fi %rpmattr %_rpmlibdir/lib.* %rpmattr %_rpmlibdir/pam.* %rpmattr %_rpmlibdir/pkgconfig.* +%rpmattr %_rpmlibdir/pkgconfiglib.* %rpmattr %_rpmlibdir/shell.* %rpmattr %_rpmlibdir/shebang.* %rpmattr %_rpmlibdir/static.* diff --git a/scripts/Makefile.am b/scripts/Makefile.am index d93d477..e58108a 100644 --- a/scripts/Makefile.am +++ b/scripts/Makefile.am @@ -51,4 +51,6 @@ config_SCRIPTS = \ install-data-local: @LN_S@ pkgconfig.req $(DESTDIR)$(configdir)/pkgconfig.prov + @LN_S@ pkgconfig.req $(DESTDIR)$(configdir)/pkgconfiglib.req @LN_S@ pkgconfig.req.files $(DESTDIR)$(configdir)/pkgconfig.prov.files + @LN_S@ pkgconfig.req.files $(DESTDIR)$(configdir)/pkgconfiglib.req.files diff --git a/scripts/pkgconfig.req.in b/scripts/pkgconfig.req.in index 37391d4..95d0210 100755 --- a/scripts/pkgconfig.req.in +++ b/scripts/pkgconfig.req.in @@ -58,8 +58,55 @@ PkgconfigProv() done } +PkgconfigLibReq() +{ + local f="$1" l L; shift + l=$(pkg-config --print-errors --libs-only-l "$f") || Fatal "failed to process $f" + L=$(pkg-config --print-errors --libs-only-L "$f") || Fatal "failed to process $f" + l=$(echo '' $l |sed -e 's/ -l/ /g') + L=$(echo '' $L |sed -e 's/ -L/ /g') + local lib libdir + for lib in $l; do + lib=lib$lib.so + if [ -n "${RPM_BUILD_ROOT-}" ]; then + for libdir in $L "$RPM_LIBDIR"; do + libdir=${libdir%/} + [ -f "$RPM_BUILD_ROOT$libdir/$lib" ] || continue + # The library is under RPM_BUILD_ROOT. + # Nothing is required. Do next lib. + Verbose "$f: $lib -> \$RPM_BUILD_ROOT$libdir/$lib (skip)" + continue 2 + done + fi + for libdir in $L "$RPM_LIBDIR"; do + libdir=${libdir%/} + [ -f "$libdir/$lib" ] || continue + # The library is found in the host system. + # Generate rpm dependency and do next lib. + local pkg n + pkg=$(rpmquery --whatprovides --queryformat='%{NAME}\n' "$libdir/$lib" |sort -u) + n=$(set -- $pkg; echo $#) + if [ "$pkg" = glibc-devel ]; then + Verbose "$f: $lib -> $pkg (skip)" + elif [ $n -eq 1 ]; then + Verbose "$f: $lib -> $pkg" + printf '%s\n' "$pkg" + elif [ $n -gt 1 ]; then + Info "$f: $libdir/$lib provided by:$(echo '' $pkg)" + Info "$f: $lib -> $libdir/$lib (raw, ambiguous)" + printf '%s\n' "$libdir/$lib" + else + Info "$f: cannot map $lib to rpm dependency (skip)" + fi + continue 2 + done + Info "$f: cannot find $lib library path (skip)" + done +} + case "${0##*/}" in pkgconfig.req*) ArgvFileAction PkgconfigReq "$@" ;; pkgconfig.prov*) ArgvFileAction PkgconfigProv "$@" ;; + pkgconfiglib.req*) ArgvFileAction PkgconfigLibReq "$@" ;; *) Fatal "req/prov method not recognized" ;; esac