[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