[devel] pkgconfiglib.req
Alexey M. Tourbin
=?iso-8859-1?q?at_=CE=C1_altlinux=2Eorg?=
Вт Авг 28 21:11:00 MSD 2007
Иногда бывает так, что с новым 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 <at на altlinux>
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
Подробная информация о списке рассылки Devel