|
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
# Copyright 2015-2022 The Meson development team
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections import defaultdict
|
|
|
|
from dataclasses import dataclass
|
|
|
|
from pathlib import PurePath
|
|
|
|
import os
|
|
|
|
import typing as T
|
|
|
|
|
|
|
|
from . import NewExtensionModule, ModuleInfo
|
|
|
|
from . import ModuleReturnValue
|
tree-wide: remove unused imports
./setup.py:17:1: F401 'os' imported but unused
import os
^
./setup.py:37:1: F401 'stat.ST_MODE' imported but unused
from stat import ST_MODE
^
./run_tests.py:17:1: F401 'os' imported but unused
import subprocess, sys, os
^
./run_tests.py:18:1: F401 'shutil' imported but unused
import shutil
^
./run_unittests.py:23:1: F401 'mesonbuild.dependencies.Qt5Dependency' imported but unused
from mesonbuild.dependencies import PkgConfigDependency, Qt5Dependency
^
./mesonbuild/build.py:15:1: F401 '.coredata' imported but unused
from . import coredata
^
./mesonbuild/interpreter.py:32:1: F401 'subprocess' imported but unused
import os, sys, subprocess, shutil, uuid, re
^
./mesonbuild/interpreter.py:32:1: F401 're' imported but unused
import os, sys, subprocess, shutil, uuid, re
^
./mesonbuild/dependencies.py:23:1: F401 'subprocess' imported but unused
import os, stat, glob, subprocess, shutil
^
./mesonbuild/mesonlib.py:17:1: F401 'sys' imported but unused
import platform, subprocess, operator, os, shutil, re, sys
^
./mesonbuild/modules/qt5.py:15:1: F401 'subprocess' imported but unused
import os, subprocess
^
./mesonbuild/modules/pkgconfig.py:15:1: F401 '..coredata' imported but unused
from .. import coredata, build
^
./mesonbuild/scripts/scanbuild.py:15:1: F401 'sys' imported but unused
import sys, os
^
./mesonbuild/scripts/meson_exe.py:20:1: F401 'subprocess' imported but unused
import subprocess
^
./mesonbuild/scripts/meson_exe.py:22:1: F401 '..mesonlib.MesonException' imported but unused
from ..mesonlib import MesonException, Popen_safe
^
./mesonbuild/scripts/symbolextractor.py:23:1: F401 'subprocess' imported but unused
import os, sys, subprocess
^
./mesonbuild/scripts/symbolextractor.py:25:1: F401 '..mesonlib.MesonException' imported but unused
from ..mesonlib import MesonException, Popen_safe
^
./mesonbuild/scripts/meson_install.py:19:1: F401 '..mesonlib.MesonException' imported but unused
from ..mesonlib import MesonException, Popen_safe
^
./mesonbuild/scripts/yelphelper.py:15:1: F401 'sys' imported but unused
import sys, os
^
./mesonbuild/scripts/yelphelper.py:20:1: F401 '..mesonlib.MesonException' imported but unused
from ..mesonlib import MesonException
^
./mesonbuild/backend/vs2010backend.py:17:1: F401 're' imported but unused
import re
^
./test cases/vala/8 generated sources/src/copy_file.py:3:1: F401 'os' imported but unused
import os
^
./test cases/common/107 postconf/postconf.py:3:1: F401 'sys' imported but unused
import sys, os
^
./test cases/common/129 object only target/obj_generator.py:5:1: F401 'shutil' imported but unused
import sys, shutil, subprocess
^
./test cases/common/57 custom target chain/usetarget/subcomp.py:3:1: F401 'os' imported but unused
import sys, os
^
./test cases/common/95 dep fallback/subprojects/boblib/genbob.py:3:1: F401 'os' imported but unused
import os
^
./test cases/common/98 gen extra/srcgen.py:4:1: F401 'os' imported but unused
import os
^
./test cases/common/113 generatorcustom/gen.py:3:1: F401 'os' imported but unused
import sys, os
^
./test cases/common/113 generatorcustom/catter.py:3:1: F401 'os' imported but unused
import sys, os
^
./test cases/common/59 object generator/obj_generator.py:5:1: F401 'shutil' imported but unused
import sys, shutil, subprocess
^
Signed-off-by: Igor Gnatenko <i.gnatenko.brain@gmail.com>
8 years ago
|
|
|
from .. import build
|
|
|
|
from .. import dependencies
|
|
|
|
from .. import mesonlib
|
|
|
|
from .. import mlog
|
|
|
|
from ..coredata import BUILTIN_DIR_OPTIONS
|
|
|
|
from ..dependencies.pkgconfig import PkgConfigDependency, PkgConfigInterface
|
|
|
|
from ..interpreter.type_checking import D_MODULE_VERSIONS_KW, INSTALL_DIR_KW, VARIABLES_KW, NoneType
|
|
|
|
from ..interpreterbase import FeatureNew, FeatureDeprecated
|
|
|
|
from ..interpreterbase.decorators import ContainerTypeInfo, KwargInfo, typed_kwargs, typed_pos_args
|
|
|
|
|
|
|
|
if T.TYPE_CHECKING:
|
|
|
|
from typing_extensions import TypedDict
|
|
|
|
|
|
|
|
from . import ModuleState
|
|
|
|
from .. import mparser
|
|
|
|
from ..interpreter import Interpreter
|
|
|
|
|
|
|
|
ANY_DEP = T.Union[dependencies.Dependency, build.BuildTargetTypes, str]
|
|
|
|
LIBS = T.Union[build.LibTypes, str]
|
|
|
|
|
|
|
|
class GenerateKw(TypedDict):
|
|
|
|
|
|
|
|
version: T.Optional[str]
|
|
|
|
name: T.Optional[str]
|
|
|
|
filebase: T.Optional[str]
|
|
|
|
description: T.Optional[str]
|
|
|
|
url: str
|
|
|
|
subdirs: T.List[str]
|
|
|
|
conflicts: T.List[str]
|
|
|
|
dataonly: bool
|
|
|
|
libraries: T.List[ANY_DEP]
|
|
|
|
libraries_private: T.List[ANY_DEP]
|
|
|
|
requires: T.List[T.Union[str, build.StaticLibrary, build.SharedLibrary, dependencies.Dependency]]
|
|
|
|
requires_private: T.List[T.Union[str, build.StaticLibrary, build.SharedLibrary, dependencies.Dependency]]
|
|
|
|
install_dir: T.Optional[str]
|
|
|
|
d_module_versions: T.List[T.Union[str, int]]
|
|
|
|
extra_cflags: T.List[str]
|
|
|
|
variables: T.Dict[str, str]
|
|
|
|
uninstalled_variables: T.Dict[str, str]
|
|
|
|
unescaped_variables: T.Dict[str, str]
|
|
|
|
unescaped_uninstalled_variables: T.Dict[str, str]
|
|
|
|
|
|
|
|
|
|
|
|
_PKG_LIBRARIES: KwargInfo[T.List[T.Union[str, dependencies.Dependency, build.SharedLibrary, build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex]]] = KwargInfo(
|
|
|
|
'libraries',
|
|
|
|
ContainerTypeInfo(list, (str, dependencies.Dependency,
|
|
|
|
build.SharedLibrary, build.StaticLibrary,
|
|
|
|
build.CustomTarget, build.CustomTargetIndex)),
|
|
|
|
default=[],
|
|
|
|
listify=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
_PKG_REQUIRES: KwargInfo[T.List[T.Union[str, build.SharedLibrary, build.StaticLibrary, dependencies.Dependency]]] = KwargInfo(
|
|
|
|
'requires',
|
|
|
|
ContainerTypeInfo(list, (str, build.SharedLibrary, build.StaticLibrary, dependencies.Dependency)),
|
|
|
|
default=[],
|
|
|
|
listify=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def _as_str(obj: object) -> str:
|
|
|
|
assert isinstance(obj, str)
|
|
|
|
return obj
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class MetaData:
|
|
|
|
|
|
|
|
filebase: str
|
|
|
|
display_name: str
|
|
|
|
location: mparser.BaseNode
|
|
|
|
warned: bool = False
|
|
|
|
|
|
|
|
|
|
|
|
class DependenciesHelper:
|
|
|
|
def __init__(self, state: ModuleState, name: str, metadata: T.Dict[str, MetaData]) -> None:
|
|
|
|
self.state = state
|
|
|
|
self.name = name
|
|
|
|
self.metadata = metadata
|
|
|
|
self.pub_libs: T.List[LIBS] = []
|
|
|
|
self.pub_reqs: T.List[str] = []
|
|
|
|
self.priv_libs: T.List[LIBS] = []
|
|
|
|
self.priv_reqs: T.List[str] = []
|
|
|
|
self.cflags: T.List[str] = []
|
|
|
|
self.version_reqs: T.DefaultDict[str, T.Set[str]] = defaultdict(set)
|
|
|
|
self.link_whole_targets: T.List[T.Union[build.CustomTarget, build.CustomTargetIndex, build.StaticLibrary]] = []
|
|
|
|
self.uninstalled_incdirs: mesonlib.OrderedSet[str] = mesonlib.OrderedSet()
|
|
|
|
|
|
|
|
def add_pub_libs(self, libs: T.List[ANY_DEP]) -> None:
|
|
|
|
p_libs, reqs, cflags = self._process_libs(libs, True)
|
|
|
|
self.pub_libs = p_libs + self.pub_libs # prepend to preserve dependencies
|
|
|
|
self.pub_reqs += reqs
|
|
|
|
self.cflags += cflags
|
|
|
|
|
|
|
|
def add_priv_libs(self, libs: T.List[ANY_DEP]) -> None:
|
|
|
|
p_libs, reqs, _ = self._process_libs(libs, False)
|
|
|
|
self.priv_libs = p_libs + self.priv_libs
|
|
|
|
self.priv_reqs += reqs
|
|
|
|
|
|
|
|
def add_pub_reqs(self, reqs: T.List[T.Union[str, build.StaticLibrary, build.SharedLibrary, dependencies.Dependency]]) -> None:
|
|
|
|
self.pub_reqs += self._process_reqs(reqs)
|
|
|
|
|
|
|
|
def add_priv_reqs(self, reqs: T.List[T.Union[str, build.StaticLibrary, build.SharedLibrary, dependencies.Dependency]]) -> None:
|
|
|
|
self.priv_reqs += self._process_reqs(reqs)
|
|
|
|
|
|
|
|
def _check_generated_pc_deprecation(self, obj: T.Union[build.CustomTarget, build.CustomTargetIndex, build.StaticLibrary, build.SharedLibrary]) -> None:
|
|
|
|
if obj.get_id() in self.metadata:
|
|
|
|
return
|
|
|
|
data = self.metadata[obj.get_id()]
|
|
|
|
if data.warned:
|
|
|
|
return
|
|
|
|
mlog.deprecation('Library', mlog.bold(obj.name), 'was passed to the '
|
|
|
|
'"libraries" keyword argument of a previous call '
|
|
|
|
'to generate() method instead of first positional '
|
|
|
|
'argument.', 'Adding', mlog.bold(data.display_name),
|
|
|
|
'to "Requires" field, but this is a deprecated '
|
|
|
|
'behaviour that will change in a future version '
|
|
|
|
'of Meson. Please report the issue if this '
|
|
|
|
'warning cannot be avoided in your case.',
|
|
|
|
location=data.location)
|
|
|
|
data.warned = True
|
|
|
|
|
|
|
|
def _process_reqs(self, reqs: T.Sequence[T.Union[str, build.StaticLibrary, build.SharedLibrary, dependencies.Dependency]]) -> T.List[str]:
|
|
|
|
'''Returns string names of requirements'''
|
|
|
|
processed_reqs: T.List[str] = []
|
|
|
|
for obj in mesonlib.listify(reqs):
|
|
|
|
if not isinstance(obj, str):
|
|
|
|
FeatureNew.single_use('pkgconfig.generate requirement from non-string object', '0.46.0', self.state.subproject)
|
|
|
|
if (isinstance(obj, (build.CustomTarget, build.CustomTargetIndex, build.SharedLibrary, build.StaticLibrary))
|
|
|
|
and obj.get_id() in self.metadata):
|
|
|
|
self._check_generated_pc_deprecation(obj)
|
|
|
|
processed_reqs.append(self.metadata[obj.get_id()].filebase)
|
|
|
|
elif isinstance(obj, PkgConfigDependency):
|
|
|
|
if obj.found():
|
|
|
|
processed_reqs.append(obj.name)
|
|
|
|
self.add_version_reqs(obj.name, obj.version_reqs)
|
|
|
|
elif isinstance(obj, str):
|
|
|
|
name, version_req = self.split_version_req(obj)
|
|
|
|
processed_reqs.append(name)
|
|
|
|
self.add_version_reqs(name, [version_req] if version_req is not None else None)
|
|
|
|
elif isinstance(obj, dependencies.Dependency) and not obj.found():
|
|
|
|
pass
|
|
|
|
elif isinstance(obj, dependencies.ExternalDependency) and obj.name == 'threads':
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise mesonlib.MesonException('requires argument not a string, '
|
|
|
|
'library with pkgconfig-generated file '
|
|
|
|
f'or pkgconfig-dependency object, got {obj!r}')
|
|
|
|
return processed_reqs
|
|
|
|
|
|
|
|
def add_cflags(self, cflags: T.List[str]) -> None:
|
|
|
|
self.cflags += mesonlib.stringlistify(cflags)
|
|
|
|
|
|
|
|
def _add_uninstalled_incdirs(self, incdirs: T.List[build.IncludeDirs], subdir: T.Optional[str] = None) -> None:
|
|
|
|
for i in incdirs:
|
|
|
|
curdir = i.get_curdir()
|
|
|
|
for d in i.get_incdirs():
|
|
|
|
path = os.path.join(curdir, d)
|
|
|
|
self.uninstalled_incdirs.add(path)
|
|
|
|
if subdir is not None:
|
|
|
|
self.uninstalled_incdirs.add(subdir)
|
|
|
|
|
|
|
|
def _process_libs(
|
|
|
|
self, libs: T.List[ANY_DEP], public: bool
|
|
|
|
) -> T.Tuple[T.List[T.Union[str, build.SharedLibrary, build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex]], T.List[str], T.List[str]]:
|
|
|
|
libs = mesonlib.listify(libs)
|
|
|
|
processed_libs: T.List[T.Union[str, build.SharedLibrary, build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex]] = []
|
|
|
|
processed_reqs: T.List[str] = []
|
|
|
|
processed_cflags: T.List[str] = []
|
|
|
|
for obj in libs:
|
|
|
|
if (isinstance(obj, (build.CustomTarget, build.CustomTargetIndex, build.SharedLibrary, build.StaticLibrary))
|
|
|
|
and obj.get_id() in self.metadata):
|
|
|
|
self._check_generated_pc_deprecation(obj)
|
|
|
|
processed_reqs.append(self.metadata[obj.get_id()].filebase)
|
|
|
|
elif isinstance(obj, dependencies.ExternalDependency) and obj.name == 'valgrind':
|
|
|
|
pass
|
|
|
|
elif isinstance(obj, PkgConfigDependency):
|
|
|
|
if obj.found():
|
|
|
|
processed_reqs.append(obj.name)
|
|
|
|
self.add_version_reqs(obj.name, obj.version_reqs)
|
|
|
|
elif isinstance(obj, dependencies.InternalDependency):
|
|
|
|
if obj.found():
|
|
|
|
if obj.objects:
|
|
|
|
raise mesonlib.MesonException('.pc file cannot refer to individual object files.')
|
|
|
|
processed_libs += obj.get_link_args()
|
|
|
|
processed_cflags += obj.get_compile_args()
|
|
|
|
self._add_lib_dependencies(obj.libraries, obj.whole_libraries, obj.ext_deps, public, private_external_deps=True)
|
|
|
|
self._add_uninstalled_incdirs(obj.get_include_dirs())
|
|
|
|
elif isinstance(obj, dependencies.Dependency):
|
|
|
|
if obj.found():
|
|
|
|
processed_libs += obj.get_link_args()
|
|
|
|
processed_cflags += obj.get_compile_args()
|
|
|
|
elif isinstance(obj, build.SharedLibrary) and obj.shared_library_only:
|
|
|
|
# Do not pull dependencies for shared libraries because they are
|
|
|
|
# only required for static linking. Adding private requires has
|
|
|
|
# the side effect of exposing their cflags, which is the
|
|
|
|
# intended behaviour of pkg-config but force Debian to add more
|
|
|
|
# than needed build deps.
|
|
|
|
# See https://bugs.freedesktop.org/show_bug.cgi?id=105572
|
|
|
|
processed_libs.append(obj)
|
|
|
|
self._add_uninstalled_incdirs(obj.get_include_dirs(), obj.get_subdir())
|
|
|
|
elif isinstance(obj, (build.SharedLibrary, build.StaticLibrary)):
|
|
|
|
processed_libs.append(obj)
|
|
|
|
self._add_uninstalled_incdirs(obj.get_include_dirs(), obj.get_subdir())
|
|
|
|
# If there is a static library in `Libs:` all its deps must be
|
|
|
|
# public too, otherwise the generated pc file will never be
|
|
|
|
# usable without --static.
|
|
|
|
self._add_lib_dependencies(obj.link_targets,
|
|
|
|
obj.link_whole_targets,
|
|
|
|
obj.external_deps,
|
|
|
|
isinstance(obj, build.StaticLibrary) and public)
|
|
|
|
elif isinstance(obj, (build.CustomTarget, build.CustomTargetIndex)):
|
|
|
|
if not obj.is_linkable_target():
|
|
|
|
raise mesonlib.MesonException('library argument contains a not linkable custom_target.')
|
|
|
|
FeatureNew.single_use('custom_target in pkgconfig.generate libraries', '0.58.0', self.state.subproject)
|
|
|
|
processed_libs.append(obj)
|
|
|
|
elif isinstance(obj, str):
|
|
|
|
processed_libs.append(obj)
|
|
|
|
else:
|
|
|
|
raise mesonlib.MesonException(f'library argument of type {type(obj).__name__} not a string, library or dependency object.')
|
|
|
|
|
|
|
|
return processed_libs, processed_reqs, processed_cflags
|
|
|
|
|
|
|
|
def _add_lib_dependencies(
|
|
|
|
self, link_targets: T.Sequence[build.BuildTargetTypes],
|
|
|
|
link_whole_targets: T.Sequence[T.Union[build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex]],
|
|
|
|
external_deps: T.List[dependencies.Dependency],
|
|
|
|
public: bool,
|
|
|
|
private_external_deps: bool = False) -> None:
|
|
|
|
add_libs = self.add_pub_libs if public else self.add_priv_libs
|
|
|
|
# Recursively add all linked libraries
|
|
|
|
for t in link_targets:
|
|
|
|
# Internal libraries (uninstalled static library) will be promoted
|
|
|
|
# to link_whole, treat them as such here.
|
|
|
|
if t.is_internal():
|
|
|
|
# `is_internal` shouldn't return True for anything but a
|
|
|
|
# StaticLibrary, or a CustomTarget that is a StaticLibrary
|
|
|
|
assert isinstance(t, (build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex)), 'for mypy'
|
|
|
|
self._add_link_whole(t, public)
|
|
|
|
else:
|
|
|
|
add_libs([t])
|
|
|
|
for t in link_whole_targets:
|
|
|
|
self._add_link_whole(t, public)
|
|
|
|
# And finally its external dependencies
|
|
|
|
if private_external_deps:
|
|
|
|
self.add_priv_libs(T.cast('T.List[ANY_DEP]', external_deps))
|
|
|
|
else:
|
|
|
|
add_libs(T.cast('T.List[ANY_DEP]', external_deps))
|
|
|
|
|
|
|
|
def _add_link_whole(self, t: T.Union[build.CustomTarget, build.CustomTargetIndex, build.StaticLibrary], public: bool) -> None:
|
|
|
|
# Don't include static libraries that we link_whole. But we still need to
|
|
|
|
# include their dependencies: a static library we link_whole
|
|
|
|
# could itself link to a shared library or an installed static library.
|
|
|
|
# Keep track of link_whole_targets so we can remove them from our
|
|
|
|
# lists in case a library is link_with and link_whole at the same time.
|
|
|
|
# See remove_dups() below.
|
|
|
|
self.link_whole_targets.append(t)
|
|
|
|
if isinstance(t, build.BuildTarget):
|
|
|
|
self._add_lib_dependencies(t.link_targets, t.link_whole_targets, t.external_deps, public)
|
|
|
|
|
|
|
|
def add_version_reqs(self, name: str, version_reqs: T.Optional[T.List[str]]) -> None:
|
|
|
|
if version_reqs:
|
|
|
|
# Note that pkg-config is picky about whitespace.
|
|
|
|
# 'foo > 1.2' is ok but 'foo>1.2' is not.
|
|
|
|
# foo, bar' is ok, but 'foo,bar' is not.
|
|
|
|
self.version_reqs[name].update(version_reqs)
|
|
|
|
|
|
|
|
def split_version_req(self, s: str) -> T.Tuple[str, T.Optional[str]]:
|
|
|
|
for op in ['>=', '<=', '!=', '==', '=', '>', '<']:
|
|
|
|
pos = s.find(op)
|
|
|
|
if pos > 0:
|
|
|
|
return s[0:pos].strip(), s[pos:].strip()
|
|
|
|
return s, None
|
|
|
|
|
|
|
|
def format_vreq(self, vreq: str) -> str:
|
|
|
|
# vreq are '>=1.0' and pkgconfig wants '>= 1.0'
|
|
|
|
for op in ['>=', '<=', '!=', '==', '=', '>', '<']:
|
|
|
|
if vreq.startswith(op):
|
|
|
|
return op + ' ' + vreq[len(op):]
|
|
|
|
return vreq
|
|
|
|
|
|
|
|
def format_reqs(self, reqs: T.List[str]) -> str:
|
|
|
|
result: T.List[str] = []
|
|
|
|
for name in reqs:
|
|
|
|
vreqs = self.version_reqs.get(name, None)
|
|
|
|
if vreqs:
|
Make the Requires.private line in generated .pkgconfig files reproducible
Whilst working on the Reproducible Builds effort, we noticed that
meson was generates .pkgconfig files that are not reproducible.
For example, here is neatvnc's pkgconfig file when built with HEAD^1:
Name: neatvnc
Description: A Neat VNC server library
Version: 0.7.0
-Requires.private: pixman-1, aml < 0.4.0, aml >= 0.3.0, zlib, libdrm, libturbojpeg, gnutls, nettle, hogweed, gmp, gbm, libavcodec, libavfilter, libavutil
+Requires.private: pixman-1, aml >= 0.3.0, aml < 0.4.0, zlib, libdrm, libturbojpeg, gnutls, nettle, hogweed, gmp, gbm, libavcodec, libavfilter, libavutil
Libs: -L${libdir} -lneatvnc
Libs.private: -lm
Cflags: -I${includedir}
This is, ultimately, due to iterating over the contents of a set within a
DefaultDict and can thus be fixed by sorting the output immediately prior to
generating the Requires.private string.
An alternative solution would be to place the sorted(…) call a few lines
down:
return ', '.join(sorted(result))
However, this changes the expected ordering of the entire line, and many users
may be unhappy with that (alternative) change as a result. By contrast, this
commit will only enforce an ordering when there are multiple version
requirements (eg. a lower and a higher version requirement, ie. a version
range). It will, additionally, order them with the lower part of the range
first.
This was originally filed (with a slightly different patch) by myself in
the the Debian bug tracker <https://bugs.debian.org/1056117>.
Signed-off-by: Chris Lamb <lamby@debian.org>
1 year ago
|
|
|
result += [name + ' ' + self.format_vreq(vreq) for vreq in sorted(vreqs)]
|
|
|
|
else:
|
|
|
|
result += [name]
|
|
|
|
return ', '.join(result)
|
|
|
|
|
|
|
|
def remove_dups(self) -> None:
|
|
|
|
# Set of ids that have already been handled and should not be added any more
|
|
|
|
exclude: T.Set[str] = set()
|
|
|
|
|
|
|
|
# We can't just check if 'x' is excluded because we could have copies of
|
|
|
|
# the same SharedLibrary object for example.
|
|
|
|
def _ids(x: T.Union[str, build.CustomTarget, build.CustomTargetIndex, build.StaticLibrary, build.SharedLibrary]) -> T.Iterable[str]:
|
|
|
|
if isinstance(x, str):
|
|
|
|
yield x
|
|
|
|
else:
|
|
|
|
if x.get_id() in self.metadata:
|
|
|
|
yield self.metadata[x.get_id()].display_name
|
|
|
|
yield x.get_id()
|
|
|
|
|
|
|
|
# Exclude 'x' in all its forms and return if it was already excluded
|
|
|
|
def _add_exclude(x: T.Union[str, build.CustomTarget, build.CustomTargetIndex, build.StaticLibrary, build.SharedLibrary]) -> bool:
|
|
|
|
was_excluded = False
|
|
|
|
for i in _ids(x):
|
|
|
|
if i in exclude:
|
|
|
|
was_excluded = True
|
|
|
|
else:
|
|
|
|
exclude.add(i)
|
|
|
|
return was_excluded
|
|
|
|
|
|
|
|
# link_whole targets are already part of other targets, exclude them all.
|
|
|
|
for t in self.link_whole_targets:
|
|
|
|
_add_exclude(t)
|
|
|
|
|
|
|
|
# Mypy thinks these overlap, but since List is invariant they don't,
|
|
|
|
# `List[str]`` is not a valid input to `List[str | BuildTarget]`.
|
|
|
|
# pylance/pyright gets this right, but for mypy we have to ignore the
|
|
|
|
# error
|
|
|
|
@T.overload
|
|
|
|
def _fn(xs: T.List[str], libs: bool = False) -> T.List[str]: ... # type: ignore
|
|
|
|
|
|
|
|
@T.overload
|
|
|
|
def _fn(xs: T.List[LIBS], libs: bool = False) -> T.List[LIBS]: ...
|
|
|
|
|
|
|
|
def _fn(xs: T.Union[T.List[str], T.List[LIBS]], libs: bool = False) -> T.Union[T.List[str], T.List[LIBS]]:
|
|
|
|
# Remove duplicates whilst preserving original order
|
|
|
|
result = []
|
|
|
|
for x in xs:
|
|
|
|
# Don't de-dup unknown strings to avoid messing up arguments like:
|
|
|
|
# ['-framework', 'CoreAudio', '-framework', 'CoreMedia']
|
|
|
|
known_flags = ['-pthread']
|
|
|
|
cannot_dedup = libs and isinstance(x, str) and \
|
|
|
|
not x.startswith(('-l', '-L')) and \
|
|
|
|
x not in known_flags
|
|
|
|
if not cannot_dedup and _add_exclude(x):
|
|
|
|
continue
|
|
|
|
result.append(x)
|
|
|
|
return result
|
|
|
|
|
|
|
|
# Handle lists in priority order: public items can be excluded from
|
|
|
|
# private and Requires can excluded from Libs.
|
|
|
|
self.pub_reqs = _fn(self.pub_reqs)
|
|
|
|
self.pub_libs = _fn(self.pub_libs, True)
|
|
|
|
self.priv_reqs = _fn(self.priv_reqs)
|
|
|
|
self.priv_libs = _fn(self.priv_libs, True)
|
|
|
|
# Reset exclude list just in case some values can be both cflags and libs.
|
|
|
|
exclude = set()
|
|
|
|
self.cflags = _fn(self.cflags)
|
|
|
|
|
|
|
|
class PkgConfigModule(NewExtensionModule):
|
|
|
|
|
|
|
|
INFO = ModuleInfo('pkgconfig')
|
|
|
|
|
|
|
|
# Track already generated pkg-config files This is stored as a class
|
|
|
|
# variable so that multiple `import()`s share metadata
|
|
|
|
devenv: T.Optional[mesonlib.EnvironmentVariables] = None
|
|
|
|
_metadata: T.ClassVar[T.Dict[str, MetaData]] = {}
|
|
|
|
|
|
|
|
def __init__(self) -> None:
|
|
|
|
super().__init__()
|
|
|
|
self.methods.update({
|
|
|
|
'generate': self.generate,
|
|
|
|
})
|
|
|
|
|
|
|
|
def postconf_hook(self, b: build.Build) -> None:
|
|
|
|
if self.devenv is not None:
|
|
|
|
b.devenv.append(self.devenv)
|
|
|
|
|
|
|
|
def _get_lname(self, l: T.Union[build.SharedLibrary, build.StaticLibrary, build.CustomTarget, build.CustomTargetIndex],
|
|
|
|
msg: str, pcfile: str) -> str:
|
|
|
|
if isinstance(l, (build.CustomTargetIndex, build.CustomTarget)):
|
|
|
|
basename = os.path.basename(l.get_filename())
|
|
|
|
name = os.path.splitext(basename)[0]
|
|
|
|
if name.startswith('lib'):
|
|
|
|
name = name[3:]
|
|
|
|
return name
|
|
|
|
# Nothing special
|
|
|
|
if not l.name_prefix_set:
|
|
|
|
return l.name
|
|
|
|
# Sometimes people want the library to start with 'lib' everywhere,
|
|
|
|
# which is achieved by setting name_prefix to '' and the target name to
|
|
|
|
# 'libfoo'. In that case, try to get the pkg-config '-lfoo' arg correct.
|
|
|
|
if l.prefix == '' and l.name.startswith('lib'):
|
|
|
|
return l.name[3:]
|
|
|
|
# If the library is imported via an import library which is always
|
|
|
|
# named after the target name, '-lfoo' is correct.
|
|
|
|
if isinstance(l, build.SharedLibrary) and l.import_filename:
|
|
|
|
return l.name
|
|
|
|
# In other cases, we can't guarantee that the compiler will be able to
|
|
|
|
# find the library via '-lfoo', so tell the user that.
|
|
|
|
mlog.warning(msg.format(l.name, 'name_prefix', l.name, pcfile))
|
|
|
|
return l.name
|
|
|
|
|
|
|
|
def _escape(self, value: T.Union[str, PurePath]) -> str:
|
|
|
|
'''
|
|
|
|
We cannot use quote_arg because it quotes with ' and " which does not
|
|
|
|
work with pkg-config and pkgconf at all.
|
|
|
|
'''
|
|
|
|
# We should always write out paths with / because pkg-config requires
|
|
|
|
# spaces to be quoted with \ and that messes up on Windows:
|
|
|
|
# https://bugs.freedesktop.org/show_bug.cgi?id=103203
|
|
|
|
if isinstance(value, PurePath):
|
|
|
|
value = value.as_posix()
|
|
|
|
return value.replace(' ', r'\ ')
|
|
|
|
|
|
|
|
def _make_relative(self, prefix: T.Union[PurePath, str], subdir: T.Union[PurePath, str]) -> str:
|
|
|
|
prefix = PurePath(prefix)
|
|
|
|
subdir = PurePath(subdir)
|
|
|
|
try:
|
|
|
|
libdir = subdir.relative_to(prefix)
|
|
|
|
except ValueError:
|
|
|
|
libdir = subdir
|
|
|
|
# pathlib joining makes sure absolute libdir is not appended to '${prefix}'
|
|
|
|
return ('${prefix}' / libdir).as_posix()
|
|
|
|
|
|
|
|
def _generate_pkgconfig_file(self, state: ModuleState, deps: DependenciesHelper,
|
|
|
|
subdirs: T.List[str], name: str,
|
|
|
|
description: str, url: str, version: str,
|
|
|
|
pcfile: str, conflicts: T.List[str],
|
|
|
|
variables: T.List[T.Tuple[str, str]],
|
|
|
|
unescaped_variables: T.List[T.Tuple[str, str]],
|
|
|
|
uninstalled: bool = False, dataonly: bool = False,
|
|
|
|
pkgroot: T.Optional[str] = None) -> None:
|
|
|
|
coredata = state.environment.get_coredata()
|
|
|
|
referenced_vars = set()
|
|
|
|
optnames = [x.name for x in BUILTIN_DIR_OPTIONS.keys()]
|
|
|
|
|
|
|
|
if not dataonly:
|
|
|
|
# includedir is always implied, although libdir may not be
|
|
|
|
# needed for header-only libraries
|
|
|
|
referenced_vars |= {'prefix', 'includedir'}
|
|
|
|
if deps.pub_libs or deps.priv_libs:
|
|
|
|
referenced_vars |= {'libdir'}
|
|
|
|
# also automatically infer variables referenced in other variables
|
|
|
|
implicit_vars_warning = False
|
|
|
|
redundant_vars_warning = False
|
|
|
|
varnames = set()
|
|
|
|
varstrings = set()
|
|
|
|
for k, v in variables + unescaped_variables:
|
|
|
|
varnames |= {k}
|
|
|
|
varstrings |= {v}
|
|
|
|
for optname in optnames:
|
|
|
|
optvar = f'${{{optname}}}'
|
|
|
|
if any(x.startswith(optvar) for x in varstrings):
|
|
|
|
if optname in varnames:
|
|
|
|
redundant_vars_warning = True
|
|
|
|
else:
|
|
|
|
# these 3 vars were always "implicit"
|
|
|
|
if dataonly or optname not in {'prefix', 'includedir', 'libdir'}:
|
|
|
|
implicit_vars_warning = True
|
|
|
|
referenced_vars |= {'prefix', optname}
|
|
|
|
if redundant_vars_warning:
|
|
|
|
FeatureDeprecated.single_use('pkgconfig.generate variable for builtin directories', '0.62.0',
|
|
|
|
state.subproject, 'They will be automatically included when referenced',
|
|
|
|
state.current_node)
|
|
|
|
if implicit_vars_warning:
|
|
|
|
FeatureNew.single_use('pkgconfig.generate implicit variable for builtin directories', '0.62.0',
|
|
|
|
state.subproject, location=state.current_node)
|
|
|
|
|
|
|
|
if uninstalled:
|
|
|
|
outdir = os.path.join(state.environment.build_dir, 'meson-uninstalled')
|
|
|
|
if not os.path.exists(outdir):
|
|
|
|
os.mkdir(outdir)
|
|
|
|
prefix = PurePath(state.environment.get_build_dir())
|
|
|
|
srcdir = PurePath(state.environment.get_source_dir())
|
|
|
|
else:
|
|
|
|
outdir = state.environment.scratch_dir
|
|
|
|
prefix = PurePath(_as_str(coredata.get_option(mesonlib.OptionKey('prefix'))))
|
|
|
|
if pkgroot:
|
|
|
|
pkgroot_ = PurePath(pkgroot)
|
|
|
|
if not pkgroot_.is_absolute():
|
|
|
|
pkgroot_ = prefix / pkgroot
|
|
|
|
elif prefix not in pkgroot_.parents:
|
|
|
|
raise mesonlib.MesonException('Pkgconfig prefix cannot be outside of the prefix '
|
|
|
|
'when pkgconfig.relocatable=true. '
|
|
|
|
f'Pkgconfig prefix is {pkgroot_.as_posix()}.')
|
|
|
|
prefix = PurePath('${pcfiledir}', os.path.relpath(prefix, pkgroot_))
|
|
|
|
fname = os.path.join(outdir, pcfile)
|
|
|
|
with open(fname, 'w', encoding='utf-8') as ofile:
|
|
|
|
for optname in optnames:
|
|
|
|
if optname in referenced_vars - varnames:
|
|
|
|
if optname == 'prefix':
|
|
|
|
ofile.write('prefix={}\n'.format(self._escape(prefix)))
|
|
|
|
else:
|
|
|
|
dirpath = PurePath(_as_str(coredata.get_option(mesonlib.OptionKey(optname))))
|
|
|
|
ofile.write('{}={}\n'.format(optname, self._escape('${prefix}' / dirpath)))
|
|
|
|
if uninstalled and not dataonly:
|
|
|
|
ofile.write('srcdir={}\n'.format(self._escape(srcdir)))
|
|
|
|
if variables or unescaped_variables:
|
|
|
|
ofile.write('\n')
|
|
|
|
for k, v in variables:
|
|
|
|
ofile.write('{}={}\n'.format(k, self._escape(v)))
|
|
|
|
for k, v in unescaped_variables:
|
|
|
|
ofile.write(f'{k}={v}\n')
|
|
|
|
ofile.write('\n')
|
|
|
|
ofile.write(f'Name: {name}\n')
|
|
|
|
if len(description) > 0:
|
|
|
|
ofile.write(f'Description: {description}\n')
|
|
|
|
if len(url) > 0:
|
|
|
|
ofile.write(f'URL: {url}\n')
|
|
|
|
ofile.write(f'Version: {version}\n')
|
|
|
|
reqs_str = deps.format_reqs(deps.pub_reqs)
|
|
|
|
if len(reqs_str) > 0:
|
|
|
|
ofile.write(f'Requires: {reqs_str}\n')
|
|
|
|
reqs_str = deps.format_reqs(deps.priv_reqs)
|
|
|
|
if len(reqs_str) > 0:
|
|
|
|
ofile.write(f'Requires.private: {reqs_str}\n')
|
|
|
|
if len(conflicts) > 0:
|
|
|
|
ofile.write('Conflicts: {}\n'.format(' '.join(conflicts)))
|
|
|
|
|
|
|
|
def generate_libs_flags(libs: T.List[LIBS]) -> T.Iterable[str]:
|
|
|
|
msg = 'Library target {0!r} has {1!r} set. Compilers ' \
|
|
|
|
'may not find it from its \'-l{2}\' linker flag in the ' \
|
|
|
|
'{3!r} pkg-config file.'
|
|
|
|
Lflags = []
|
|
|
|
for l in libs:
|
|
|
|
if isinstance(l, str):
|
|
|
|
yield l
|
|
|
|
else:
|
|
|
|
install_dir: T.Union[str, bool]
|
|
|
|
if uninstalled:
|
|
|
|
install_dir = os.path.dirname(state.backend.get_target_filename_abs(l))
|
|
|
|
else:
|
|
|
|
_i = l.get_custom_install_dir()
|
|
|
|
install_dir = _i[0] if _i else None
|
|
|
|
if install_dir is False:
|
|
|
|
continue
|
|
|
|
if isinstance(l, build.BuildTarget) and 'cs' in l.compilers:
|
|
|
|
if isinstance(install_dir, str):
|
|
|
|
Lflag = '-r{}/{}'.format(self._escape(self._make_relative(prefix, install_dir)), l.filename)
|
|
|
|
else: # install_dir is True
|
|
|
|
Lflag = '-r${libdir}/%s' % l.filename
|
|
|
|
else:
|
|
|
|
if isinstance(install_dir, str):
|
|
|
|
Lflag = '-L{}'.format(self._escape(self._make_relative(prefix, install_dir)))
|
|
|
|
else: # install_dir is True
|
|
|
|
Lflag = '-L${libdir}'
|
|
|
|
if Lflag not in Lflags:
|
|
|
|
Lflags.append(Lflag)
|
|
|
|
yield Lflag
|
|
|
|
lname = self._get_lname(l, msg, pcfile)
|
|
|
|
# If using a custom suffix, the compiler may not be able to
|
|
|
|
# find the library
|
|
|
|
if isinstance(l, build.BuildTarget) and l.name_suffix_set:
|
|
|
|
mlog.warning(msg.format(l.name, 'name_suffix', lname, pcfile))
|
|
|
|
if isinstance(l, (build.CustomTarget, build.CustomTargetIndex)) or 'cs' not in l.compilers:
|
|
|
|
yield f'-l{lname}'
|
|
|
|
|
|
|
|
if len(deps.pub_libs) > 0:
|
|
|
|
ofile.write('Libs: {}\n'.format(' '.join(generate_libs_flags(deps.pub_libs))))
|
|
|
|
if len(deps.priv_libs) > 0:
|
|
|
|
ofile.write('Libs.private: {}\n'.format(' '.join(generate_libs_flags(deps.priv_libs))))
|
|
|
|
|
|
|
|
cflags: T.List[str] = []
|
|
|
|
if uninstalled:
|
|
|
|
for d in deps.uninstalled_incdirs:
|
|
|
|
for basedir in ['${prefix}', '${srcdir}']:
|
|
|
|
path = self._escape(PurePath(basedir, d).as_posix())
|
|
|
|
cflags.append(f'-I{path}')
|
|
|
|
else:
|
|
|
|
for d in subdirs:
|
|
|
|
if d == '.':
|
|
|
|
cflags.append('-I${includedir}')
|
|
|
|
else:
|
|
|
|
cflags.append(self._escape(PurePath('-I${includedir}') / d))
|
|
|
|
cflags += [self._escape(f) for f in deps.cflags]
|
|
|
|
if cflags and not dataonly:
|
|
|
|
ofile.write('Cflags: {}\n'.format(' '.join(cflags)))
|
|
|
|
|
|
|
|
@typed_pos_args('pkgconfig.generate', optargs=[(build.SharedLibrary, build.StaticLibrary)])
|
|
|
|
@typed_kwargs(
|
|
|
|
'pkgconfig.generate',
|
|
|
|
D_MODULE_VERSIONS_KW.evolve(since='0.43.0'),
|
|
|
|
INSTALL_DIR_KW,
|
|
|
|
KwargInfo('conflicts', ContainerTypeInfo(list, str), default=[], listify=True),
|
|
|
|
KwargInfo('dataonly', bool, default=False, since='0.54.0'),
|
|
|
|
KwargInfo('description', (str, NoneType)),
|
|
|
|
KwargInfo('extra_cflags', ContainerTypeInfo(list, str), default=[], listify=True, since='0.42.0'),
|
|
|
|
KwargInfo('filebase', (str, NoneType), validator=lambda x: 'must not be an empty string' if x == '' else None),
|
|
|
|
KwargInfo('name', (str, NoneType), validator=lambda x: 'must not be an empty string' if x == '' else None),
|
|
|
|
KwargInfo('subdirs', ContainerTypeInfo(list, str), default=[], listify=True),
|
|
|
|
KwargInfo('url', str, default=''),
|
|
|
|
KwargInfo('version', (str, NoneType)),
|
|
|
|
VARIABLES_KW.evolve(name="unescaped_uninstalled_variables", since='0.59.0'),
|
|
|
|
VARIABLES_KW.evolve(name="unescaped_variables", since='0.59.0'),
|
|
|
|
VARIABLES_KW.evolve(name="uninstalled_variables", since='0.54.0', since_values={dict: '0.56.0'}),
|
|
|
|
VARIABLES_KW.evolve(since='0.41.0', since_values={dict: '0.56.0'}),
|
|
|
|
_PKG_LIBRARIES,
|
|
|
|
_PKG_LIBRARIES.evolve(name='libraries_private'),
|
|
|
|
_PKG_REQUIRES,
|
|
|
|
_PKG_REQUIRES.evolve(name='requires_private'),
|
|
|
|
)
|
|
|
|
def generate(self, state: ModuleState,
|
|
|
|
args: T.Tuple[T.Optional[T.Union[build.SharedLibrary, build.StaticLibrary]]],
|
|
|
|
kwargs: GenerateKw) -> ModuleReturnValue:
|
|
|
|
default_version = state.project_version
|
|
|
|
default_install_dir: T.Optional[str] = None
|
|
|
|
default_description: T.Optional[str] = None
|
|
|
|
default_name: T.Optional[str] = None
|
|
|
|
mainlib: T.Optional[T.Union[build.SharedLibrary, build.StaticLibrary]] = None
|
|
|
|
default_subdirs = ['.']
|
|
|
|
if args[0]:
|
|
|
|
FeatureNew.single_use('pkgconfig.generate optional positional argument', '0.46.0', state.subproject)
|
|
|
|
mainlib = args[0]
|
|
|
|
default_name = mainlib.name
|
|
|
|
default_description = state.project_name + ': ' + mainlib.name
|
|
|
|
install_dir = mainlib.get_custom_install_dir()
|
|
|
|
if install_dir and isinstance(install_dir[0], str):
|
|
|
|
default_install_dir = os.path.join(install_dir[0], 'pkgconfig')
|
|
|
|
else:
|
|
|
|
if kwargs['version'] is None:
|
|
|
|
FeatureNew.single_use('pkgconfig.generate implicit version keyword', '0.46.0', state.subproject)
|
|
|
|
msg = ('pkgconfig.generate: if a library is not passed as a '
|
|
|
|
'positional argument, the {!r} keyword argument is '
|
|
|
|
'required.')
|
|
|
|
if kwargs['name'] is None:
|
|
|
|
raise build.InvalidArguments(msg.format('name'))
|
|
|
|
if kwargs['description'] is None:
|
|
|
|
raise build.InvalidArguments(msg.format('description'))
|
|
|
|
|
|
|
|
dataonly = kwargs['dataonly']
|
|
|
|
if dataonly:
|
|
|
|
default_subdirs = []
|
|
|
|
blocked_vars = ['libraries', 'libraries_private', 'requires_private', 'extra_cflags', 'subdirs']
|
|
|
|
# Mypy can't figure out that this TypedDict index is correct, without repeating T.Literal for the entire list
|
|
|
|
if any(kwargs[k] for k in blocked_vars): # type: ignore
|
|
|
|
raise mesonlib.MesonException(f'Cannot combine dataonly with any of {blocked_vars}')
|
|
|
|
default_install_dir = os.path.join(state.environment.get_datadir(), 'pkgconfig')
|
|
|
|
|
|
|
|
subdirs = kwargs['subdirs'] or default_subdirs
|
|
|
|
version = kwargs['version'] if kwargs['version'] is not None else default_version
|
|
|
|
name = kwargs['name'] if kwargs['name'] is not None else default_name
|
|
|
|
assert isinstance(name, str), 'for mypy'
|
|
|
|
filebase = kwargs['filebase'] if kwargs['filebase'] is not None else name
|
|
|
|
description = kwargs['description'] if kwargs['description'] is not None else default_description
|
|
|
|
url = kwargs['url']
|
|
|
|
conflicts = kwargs['conflicts']
|
|
|
|
|
|
|
|
# Prepend the main library to public libraries list. This is required
|
|
|
|
# so dep.add_pub_libs() can handle dependency ordering correctly and put
|
|
|
|
# extra libraries after the main library.
|
|
|
|
libraries = kwargs['libraries'].copy()
|
|
|
|
if mainlib:
|
|
|
|
libraries.insert(0, mainlib)
|
|
|
|
|
|
|
|
deps = DependenciesHelper(state, filebase, self._metadata)
|
|
|
|
deps.add_pub_libs(libraries)
|
|
|
|
deps.add_priv_libs(kwargs['libraries_private'])
|
|
|
|
deps.add_pub_reqs(kwargs['requires'])
|
|
|
|
deps.add_priv_reqs(kwargs['requires_private'])
|
|
|
|
deps.add_cflags(kwargs['extra_cflags'])
|
|
|
|
|
|
|
|
dversions = kwargs['d_module_versions']
|
|
|
|
if dversions:
|
|
|
|
compiler = state.environment.coredata.compilers.host.get('d')
|
|
|
|
if compiler:
|
|
|
|
deps.add_cflags(compiler.get_feature_args(
|
|
|
|
{'versions': dversions, 'import_dirs': [], 'debug': [], 'unittest': False}, None))
|
|
|
|
|
|
|
|
deps.remove_dups()
|
|
|
|
|
|
|
|
def parse_variable_list(vardict: T.Dict[str, str]) -> T.List[T.Tuple[str, str]]:
|
|
|
|
reserved = ['prefix', 'libdir', 'includedir']
|
|
|
|
variables = []
|
|
|
|
for name, value in vardict.items():
|
|
|
|
if not dataonly and name in reserved:
|
|
|
|
raise mesonlib.MesonException(f'Variable "{name}" is reserved')
|
|
|
|
variables.append((name, value))
|
|
|
|
return variables
|
|
|
|
|
|
|
|
variables = parse_variable_list(kwargs['variables'])
|
|
|
|
unescaped_variables = parse_variable_list(kwargs['unescaped_variables'])
|
|
|
|
|
|
|
|
pcfile = filebase + '.pc'
|
|
|
|
pkgroot = pkgroot_name = kwargs['install_dir'] or default_install_dir
|
|
|
|
if pkgroot is None:
|
|
|
|
if mesonlib.is_freebsd():
|
|
|
|
pkgroot = os.path.join(_as_str(state.environment.coredata.get_option(mesonlib.OptionKey('prefix'))), 'libdata', 'pkgconfig')
|
|
|
|
pkgroot_name = os.path.join('{prefix}', 'libdata', 'pkgconfig')
|
|
|
|
elif mesonlib.is_haiku():
|
|
|
|
pkgroot = os.path.join(_as_str(state.environment.coredata.get_option(mesonlib.OptionKey('prefix'))), 'develop', 'lib', 'pkgconfig')
|
|
|
|
pkgroot_name = os.path.join('{prefix}', 'develop', 'lib', 'pkgconfig')
|
|
|
|
else:
|
|
|
|
pkgroot = os.path.join(_as_str(state.environment.coredata.get_option(mesonlib.OptionKey('libdir'))), 'pkgconfig')
|
|
|
|
pkgroot_name = os.path.join('{libdir}', 'pkgconfig')
|
|
|
|
relocatable = state.get_option('relocatable', module='pkgconfig')
|
|
|
|
self._generate_pkgconfig_file(state, deps, subdirs, name, description, url,
|
|
|
|
version, pcfile, conflicts, variables,
|
|
|
|
unescaped_variables, False, dataonly,
|
|
|
|
pkgroot=pkgroot if relocatable else None)
|
|
|
|
res = build.Data([mesonlib.File(True, state.environment.get_scratch_dir(), pcfile)], pkgroot, pkgroot_name, None, state.subproject, install_tag='devel')
|
|
|
|
variables = parse_variable_list(kwargs['uninstalled_variables'])
|
|
|
|
unescaped_variables = parse_variable_list(kwargs['unescaped_uninstalled_variables'])
|
|
|
|
|
|
|
|
pcfile = filebase + '-uninstalled.pc'
|
|
|
|
self._generate_pkgconfig_file(state, deps, subdirs, name, description, url,
|
|
|
|
version, pcfile, conflicts, variables,
|
|
|
|
unescaped_variables, uninstalled=True, dataonly=dataonly)
|
|
|
|
# Associate the main library with this generated pc file. If the library
|
|
|
|
# is used in any subsequent call to the generated, it will generate a
|
|
|
|
# 'Requires:' or 'Requires.private:'.
|
|
|
|
# Backward compatibility: We used to set 'generated_pc' on all public
|
|
|
|
# libraries instead of just the main one. Keep doing that but warn if
|
|
|
|
# anyone is relying on that deprecated behaviour.
|
|
|
|
if mainlib:
|
|
|
|
if mainlib.get_id() not in self._metadata:
|
|
|
|
self._metadata[mainlib.get_id()] = MetaData(
|
|
|
|
filebase, name, state.current_node)
|
|
|
|
else:
|
|
|
|
mlog.warning('Already generated a pkg-config file for', mlog.bold(mainlib.name))
|
|
|
|
else:
|
|
|
|
for lib in deps.pub_libs:
|
|
|
|
if not isinstance(lib, str) and lib.get_id() not in self._metadata:
|
|
|
|
self._metadata[lib.get_id()] = MetaData(
|
|
|
|
filebase, name, state.current_node)
|
|
|
|
if self.devenv is None:
|
|
|
|
self.devenv = PkgConfigInterface.get_env(state.environment, mesonlib.MachineChoice.HOST, uninstalled=True)
|
|
|
|
return ModuleReturnValue(res, [res])
|
|
|
|
|
|
|
|
|
|
|
|
def initialize(interp: Interpreter) -> PkgConfigModule:
|
|
|
|
return PkgConfigModule()
|