# Copyright 1999-2011 Gentoo Foundation # Distributed under the terms of the GNU General Public License v2 # $Header: /var/cvsroot/gentoo-x86/eclass/ghc-package.eclass,v 1.32 2011/11/24 00:04:13 vapier Exp $ # @ECLASS: ghc-package.eclass # @MAINTAINER: # "Gentoo's Haskell Language team" <haskell@gentoo.org> # @AUTHOR: # Original Author: Andres Loeh <kosmikus@gentoo.org> # @BLURB: This eclass helps with the Glasgow Haskell Compiler's package configuration utility. # @DESCRIPTION: # Helper eclass to handle ghc installation/upgrade/deinstallation process. inherit versionator # @FUNCTION: ghc-getghc # @DESCRIPTION: # returns the name of the ghc executable ghc-getghc() { type -P ghc } # @FUNCTION: ghc-getghcpkg # @DESCRIPTION: # Internal function determines returns the name of the ghc-pkg executable ghc-getghcpkg() { type -P ghc-pkg } # @FUNCTION: ghc-getghcpkgbin # @DESCRIPTION: # returns the name of the ghc-pkg binary (ghc-pkg # itself usually is a shell script, and we have to # bypass the script under certain circumstances); # for Cabal, we add an empty global package config file, # because for some reason the global package file # must be specified ghc-getghcpkgbin() { if version_is_at_least "6.10" "$(ghc-version)"; then # the ghc-pkg executable changed name in ghc 6.10, as it no longer needs # the wrapper script with the static flags echo '[]' > "${T}/empty.conf" echo "$(ghc-libdir)/ghc-pkg" "--global-conf=${T}/empty.conf" elif ghc-cabal; then echo '[]' > "${T}/empty.conf" echo "$(ghc-libdir)/ghc-pkg.bin" "--global-conf=${T}/empty.conf" else echo "$(ghc-libdir)/ghc-pkg.bin" fi } # @FUNCTION: ghc-version # @DESCRIPTION: # returns the version of ghc _GHC_VERSION_CACHE="" ghc-version() { if [[ -z "${_GHC_VERSION_CACHE}" ]]; then _GHC_VERSION_CACHE="$($(ghc-getghc) --numeric-version)" fi echo "${_GHC_VERSION_CACHE}" } # @FUNCTION: ghc-cabal # @DESCRIPTION: # this function can be used to determine if ghc itself # uses the Cabal package format; it has nothing to do # with the Cabal libraries ... ghc uses the Cabal package # format since version 6.4 ghc-cabal() { version_is_at_least "6.4" "$(ghc-version)" } # @FUNCTION: ghc-bestcabalversion # @DESCRIPTION: # return the best version of the Cabal library that is available ghc-bestcabalversion() { local cabalversion if ghc-cabal; then # We ask portage, not ghc, so that we only pick up # portage-installed cabal versions. cabalversion="$(ghc-extractportageversion dev-haskell/cabal)" echo "Cabal-${cabalversion}" else # older ghc's don't support package versioning echo Cabal fi } # @FUNCTION: ghc-sanecabal # @DESCRIPTION: # check if a standalone Cabal version is available for the # currently used ghc; takes minimal version of Cabal as # an optional argument ghc-sanecabal() { local f local version if [[ -z "$1" ]]; then version="1.0.1"; else version="$1"; fi for f in $(ghc-confdir)/cabal-*; do [[ -f "${f}" ]] && version_is_at_least "${version}" "${f#*cabal-}" && return done return 1 } # @FUNCTION: ghc-saneghc # @DESCRIPTION: # checks if ghc and ghc-bin are installed in the same version # (if they're both installed); if this is not the case, we # unfortunately cannot trust portage's dependency resolution ghc-saneghc() { local ghcversion local ghcbinversion if [[ "${PN}" == "ghc" || "${PN}" == "ghc-bin" ]]; then return fi if has_version dev-lang/ghc && has_version dev-lang/ghc-bin; then ghcversion="$(ghc-extractportageversion dev-lang/ghc)" ghcbinversion="$(ghc-extractportageversion dev-lang/ghc-bin)" if [[ "${ghcversion}" != "${ghcbinversion}" ]]; then return 1 fi fi return } # @FUNCTION: ghc-supports-shared-libraries # @DESCRIPTION: # checks if ghc is built with support for building # shared libraries (aka '-dynamic' option) ghc-supports-shared-libraries() { $(ghc-getghc) --info | grep "RTS ways" | grep -q "dyn" } # @FUNCTION: ghc-extractportageversion # @DESCRIPTION: # extract the version of a portage-installed package ghc-extractportageversion() { local pkg local version pkg="$(best_version $1)" version="${pkg#$1-}" version="${version%-r*}" version="${version%_pre*}" echo "${version}" } # @FUNCTION: ghc-libdir # @DESCRIPTION: # returns the library directory _GHC_LIBDIR_CACHE="" ghc-libdir() { if [[ -z "${_GHC_LIBDIR_CACHE}" ]]; then _GHC_LIBDIR_CACHE="$($(ghc-getghc) --print-libdir)" fi echo "${_GHC_LIBDIR_CACHE}" } # @FUNCTION: ghc-confdir # @DESCRIPTION: # returns the (Gentoo) library configuration directory ghc-confdir() { echo "$(ghc-libdir)/gentoo" } # @FUNCTION: ghc-localpkgconf # @DESCRIPTION: # returns the name of the local (package-specific) # package configuration file ghc-localpkgconf() { echo "${PF}.conf" } # @FUNCTION: ghc-makeghcilib # @DESCRIPTION: # make a ghci foo.o file from a libfoo.a file ghc-makeghcilib() { local outfile outfile="$(dirname $1)/$(basename $1 | sed 's:^lib\?\(.*\)\.a$:\1.o:')" ld --relocatable --discard-all --output="${outfile}" --whole-archive "$1" } # @FUNCTION: ghc-package-exists # @DESCRIPTION: # tests if a ghc package exists ghc-package-exists() { local describe_flag if version_is_at_least "6.4" "$(ghc-version)"; then describe_flag="describe" else describe_flag="--show-package" fi $(ghc-getghcpkg) "${describe_flag}" "$1" > /dev/null 2>&1 } # @FUNCTION: ghc-setup-pkg # @DESCRIPTION: # creates a local (package-specific) package # configuration file; the arguments should be # uninstalled package description files, each # containing a single package description; if # no arguments are given, the resulting file is # empty ghc-setup-pkg() { local localpkgconf localpkgconf="${S}/$(ghc-localpkgconf)" echo '[]' > "${localpkgconf}" local update_flag if version_is_at_least "6.4" "$(ghc-version)"; then update_flag="update -" else update_flag="--update-package" fi for pkg in $*; do $(ghc-getghcpkgbin) -f "${localpkgconf}" ${update_flag} --force \ < "${pkg}" || die "failed to register ${pkg}" done } # @FUNCTION: ghc-fixlibpath # @DESCRIPTION: # fixes the library and import directories path # of the package configuration file ghc-fixlibpath() { sed -i "s|$1|$(ghc-libdir)|g" "${S}/$(ghc-localpkgconf)" if [[ -n "$2" ]]; then sed -i "s|$2|$(ghc-libdir)/imports|g" "${S}/$(ghc-localpkgconf)" fi } # @FUNCTION: ghc-install-pkg # @DESCRIPTION: # moves the local (package-specific) package configuration # file to its final destination ghc-install-pkg() { mkdir -p "${D}/$(ghc-confdir)" cat "${S}/$(ghc-localpkgconf)" | sed "s|${D}||g" \ > "${D}/$(ghc-confdir)/$(ghc-localpkgconf)" } # @FUNCTION: ghc-register-pkg # @DESCRIPTION: # registers all packages in the local (package-specific) # package configuration file ghc-register-pkg() { local localpkgconf localpkgconf="$(ghc-confdir)/$1" local update_flag local describe_flag if version_is_at_least "6.4" "$(ghc-version)"; then update_flag="update -" describe_flag="describe" else update_flag="--update-package" describe_flag="--show-package" fi if [[ -f "${localpkgconf}" ]]; then for pkg in $(ghc-listpkg "${localpkgconf}"); do ebegin "Registering ${pkg} " $(ghc-getghcpkgbin) -f "${localpkgconf}" "${describe_flag}" "${pkg}" \ | $(ghc-getghcpkg) ${update_flag} --force > /dev/null eend $? done fi } # @FUNCTION: ghc-reregister # @DESCRIPTION: # re-adds all available .conf files to the global # package conf file, to be used on a ghc reinstallation ghc-reregister() { has "${EAPI:-0}" 0 1 2 && ! use prefix && EPREFIX= einfo "Re-adding packages (may cause several harmless warnings) ..." PATH="${EPREFIX}/usr/bin:${PATH}" CONFDIR="$(ghc-confdir)" if [ -d "${CONFDIR}" ]; then pushd "${CONFDIR}" > /dev/null for conf in *.conf; do PATH="${EPREFIX}/usr/bin:${PATH}" ghc-register-pkg "${conf}" done popd > /dev/null fi } # @FUNCTION: ghc-unregister-pkg # @DESCRIPTION: # unregisters a package configuration file # protected are all packages that are still contained in # another package configuration file ghc-unregister-pkg() { local localpkgconf local i local pkg local unregister_flag localpkgconf="$(ghc-confdir)/$1" if version_is_at_least "6.4" "$(ghc-version)"; then unregister_flag="unregister" else unregister_flag="--remove-package" fi if [[ -f "${localpkgconf}" ]]; then for pkg in $(ghc-reverse "$(ghc-listpkg ${localpkgconf})"); do if ! ghc-package-exists "${pkg}"; then einfo "Package ${pkg} is not installed for ghc-$(ghc-version)." else ebegin "Unregistering ${pkg} " $(ghc-getghcpkg) "${unregister_flag}" "${pkg}" --force > /dev/null eend $? fi done fi } # @FUNCTION: ghc-reverse # @DESCRIPTION: # help-function: reverse a list ghc-reverse() { local result local i for i in $1; do result="${i} ${result}" done echo "${result}" } # @FUNCTION: ghc-elem # @DESCRIPTION: # help-function: element-check ghc-elem() { local i for i in $2; do [[ "$1" == "${i}" ]] && return 0 done return 1 } # @FUNCTION: ghc-listpkg # @DESCRIPTION: # show the packages in a package configuration file ghc-listpkg() { local ghcpkgcall local i local extra_flags if version_is_at_least '6.12.3' "$(ghc-version)"; then extra_flags="${extra_flags} -v0" fi for i in $*; do if ghc-cabal; then echo $($(ghc-getghcpkg) list ${extra_flags} -f "${i}") \ | sed \ -e "s|^.*${i}:\([^:]*\).*$|\1|" \ -e "s|/.*$||" \ -e "s|,| |g" -e "s|[(){}]||g" else echo $($(ghc-getghcpkgbin) -l -f "${i}") \ | cut -f2 -d':' \ | sed 's:,: :g' fi done } # @FUNCTION: ghc-package_pkg_setup # @DESCRIPTION: # exported function: check if we have a consistent ghc installation ghc-package_pkg_setup() { if ! ghc-saneghc; then eerror "You have inconsistent versions of dev-lang/ghc and dev-lang/ghc-bin" eerror "installed. Portage currently cannot work correctly with this setup." eerror "There are several possibilities to work around this problem:" eerror "(1) Up/downgrade ghc-bin to the same version as ghc." eerror "(2) Unmerge ghc-bin." eerror "(3) Unmerge ghc." eerror "You probably want option 1 or 2." die "Inconsistent versions of ghc and ghc-bin." fi } # @FUNCTION: ghc-package_pkg_postinst # @DESCRIPTION: # exported function: registers the package-specific package # configuration file ghc-package_pkg_postinst() { ghc-register-pkg "$(ghc-localpkgconf)" } # @FUNCTION: ghc-package_pkg_prerm # @DESCRIPTION: # exported function: unregisters the package-specific package # configuration file; a package contained therein is unregistered # only if it the same package is not also contained in another # package configuration file ... ghc-package_pkg_prerm() { ghc-unregister-pkg "$(ghc-localpkgconf)" } EXPORT_FUNCTIONS pkg_setup pkg_postinst pkg_prerm