mirror of
https://github.com/pkgconf/pkgconf.git
synced 2026-01-29 02:44:26 +00:00
If PKG_CONFIG_SYSROOT_DIR is not set, then pc_sysrootdir would be defined as / which causes problems on Windows. Fixes: https://github.com/pkgconf/pkgconf/issues/437 Signed-off-by: Ariadne Conill <ariadne@ariadne.space>
897 lines
27 KiB
C
897 lines
27 KiB
C
/*
|
|
* client.c
|
|
* libpkgconf consumer lifecycle management
|
|
*
|
|
* Copyright (c) 2016 pkgconf authors (see AUTHORS).
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* This software is provided 'as is' and without any warranty, express or
|
|
* implied. In no event shall the authors be liable for any damages arising
|
|
* from the use of this software.
|
|
*/
|
|
|
|
#include <libpkgconf/config.h>
|
|
#include <libpkgconf/stdinc.h>
|
|
#include <libpkgconf/libpkgconf.h>
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* libpkgconf `client` module
|
|
* ==========================
|
|
*
|
|
* The libpkgconf `client` module implements the `pkgconf_client_t` "client" object.
|
|
* Client objects store all necessary state for libpkgconf allowing for multiple instances to run
|
|
* in parallel.
|
|
*
|
|
* Client objects are not thread safe, in other words, a client object should not be shared across
|
|
* thread boundaries.
|
|
*/
|
|
|
|
static void
|
|
trace_path_list(const pkgconf_client_t *client, const char *desc, pkgconf_list_t *list)
|
|
{
|
|
const pkgconf_node_t *n;
|
|
|
|
PKGCONF_TRACE(client, "%s:", desc);
|
|
PKGCONF_FOREACH_LIST_ENTRY(list->head, n)
|
|
{
|
|
const pkgconf_path_t *p = n->data;
|
|
|
|
PKGCONF_TRACE(client, " - '%s'", p->path);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_dir_list_build(pkgconf_client_t *client)
|
|
*
|
|
* Bootstraps the package search paths. If the ``PKGCONF_PKG_PKGF_ENV_ONLY`` `flag` is set on the client,
|
|
* then only the ``PKG_CONFIG_PATH`` environment variable will be used, otherwise both the
|
|
* ``PKG_CONFIG_PATH`` and ``PKG_CONFIG_LIBDIR`` environment variables will be used.
|
|
*
|
|
* :param pkgconf_client_t* client: The pkgconf client object to bootstrap.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_dir_list_build(pkgconf_client_t *client, const pkgconf_cross_personality_t *personality)
|
|
{
|
|
pkgconf_path_build_from_environ(client, "PKG_CONFIG_PATH", NULL, &client->dir_list, true);
|
|
|
|
if (!(client->flags & PKGCONF_PKG_PKGF_ENV_ONLY))
|
|
{
|
|
pkgconf_list_t dir_list = PKGCONF_LIST_INITIALIZER;
|
|
const pkgconf_list_t *prepend_list = &personality->dir_list;
|
|
|
|
#ifdef _WIN32
|
|
(void) pkgconf_path_build_from_registry(HKEY_CURRENT_USER, &client->dir_list, true);
|
|
(void) pkgconf_path_build_from_registry(HKEY_LOCAL_MACHINE, &client->dir_list, true);
|
|
#endif
|
|
|
|
if (pkgconf_client_getenv(client, "PKG_CONFIG_LIBDIR") != NULL)
|
|
{
|
|
/* PKG_CONFIG_LIBDIR= should empty the search path entirely. */
|
|
(void) pkgconf_path_build_from_environ(client, "PKG_CONFIG_LIBDIR", NULL, &dir_list, true);
|
|
prepend_list = &dir_list;
|
|
}
|
|
|
|
pkgconf_path_copy_list(&client->dir_list, prepend_list);
|
|
pkgconf_path_free(&dir_list);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_init(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality, void *client_data, pkgconf_environ_lookup_handler_func_t environ_lookup_handler)
|
|
*
|
|
* Initialise a pkgconf client object.
|
|
*
|
|
* :param pkgconf_client_t* client: The client to initialise.
|
|
* :param pkgconf_error_handler_func_t error_handler: An optional error handler to use for logging errors.
|
|
* :param void* error_handler_data: user data passed to optional error handler
|
|
* :param pkgconf_cross_personality_t* personality: the cross-compile personality to use for defaults
|
|
* :param void* client_data: user data associated with the client
|
|
* :param pkgconf_environ_lookup_handler_func_t environ_lookup_handler: the lookup handler to use for environment variables
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_init(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality, void *client_data, pkgconf_environ_lookup_handler_func_t environ_lookup_handler)
|
|
{
|
|
client->personality = personality;
|
|
client->client_data = client_data;
|
|
client->environ_lookup_handler = environ_lookup_handler;
|
|
client->error_handler_data = error_handler_data;
|
|
client->error_handler = error_handler;
|
|
client->auditf = NULL;
|
|
client->cache_table = NULL;
|
|
client->cache_count = 0;
|
|
|
|
#ifndef PKGCONF_LITE
|
|
if (client->trace_handler == NULL)
|
|
pkgconf_client_set_trace_handler(client, NULL, NULL);
|
|
#endif
|
|
|
|
if (client->unveil_handler == NULL)
|
|
pkgconf_client_set_unveil_handler(client, NULL);
|
|
|
|
pkgconf_client_set_error_handler(client, error_handler, error_handler_data);
|
|
pkgconf_client_set_warn_handler(client, NULL, NULL);
|
|
|
|
pkgconf_client_set_sysroot_dir(client, personality->sysroot_dir);
|
|
pkgconf_client_set_buildroot_dir(client, NULL);
|
|
pkgconf_client_set_prefix_varname(client, NULL);
|
|
|
|
if(pkgconf_client_getenv(client, "PKG_CONFIG_SYSTEM_LIBRARY_PATH") == NULL)
|
|
pkgconf_path_copy_list(&client->filter_libdirs, &personality->filter_libdirs);
|
|
else
|
|
pkgconf_path_build_from_environ(client, "PKG_CONFIG_SYSTEM_LIBRARY_PATH", NULL, &client->filter_libdirs, false);
|
|
|
|
if(pkgconf_client_getenv(client, "PKG_CONFIG_SYSTEM_INCLUDE_PATH") == NULL)
|
|
pkgconf_path_copy_list(&client->filter_includedirs, &personality->filter_includedirs);
|
|
else
|
|
pkgconf_path_build_from_environ(client, "PKG_CONFIG_SYSTEM_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
|
|
|
|
/* GCC uses these environment variables to define system include paths, so we should check them. */
|
|
#ifdef __HAIKU__
|
|
pkgconf_path_build_from_environ(client, "BELIBRARIES", NULL, &client->filter_libdirs, false);
|
|
#else
|
|
pkgconf_path_build_from_environ(client, "LIBRARY_PATH", NULL, &client->filter_libdirs, false);
|
|
#endif
|
|
pkgconf_path_build_from_environ(client, "CPATH", NULL, &client->filter_includedirs, false);
|
|
pkgconf_path_build_from_environ(client, "C_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
|
|
pkgconf_path_build_from_environ(client, "CPLUS_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
|
|
pkgconf_path_build_from_environ(client, "OBJC_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
|
|
|
|
#ifdef _WIN32
|
|
/* also use the path lists that MSVC uses on windows */
|
|
pkgconf_path_build_from_environ(client, "INCLUDE", NULL, &client->filter_includedirs, false);
|
|
#endif
|
|
|
|
PKGCONF_TRACE(client, "initialized client @%p", client);
|
|
|
|
trace_path_list(client, "filtered library paths", &client->filter_libdirs);
|
|
trace_path_list(client, "filtered include paths", &client->filter_includedirs);
|
|
|
|
client->output = pkgconf_output_default();
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_t* pkgconf_client_new(pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality)
|
|
*
|
|
* Allocate and initialise a pkgconf client object.
|
|
*
|
|
* :param pkgconf_error_handler_func_t error_handler: An optional error handler to use for logging errors.
|
|
* :param void* error_handler_data: user data passed to optional error handler
|
|
* :param pkgconf_cross_personality_t* personality: cross-compile personality to use
|
|
* :param void* client_data: user data associated with the client
|
|
* :param pkgconf_environ_lookup_handler_func_t environ_lookup_handler: the lookup handler to use for environment variables
|
|
* :return: A pkgconf client object.
|
|
* :rtype: pkgconf_client_t*
|
|
*/
|
|
pkgconf_client_t *
|
|
pkgconf_client_new(pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality, void *client_data, pkgconf_environ_lookup_handler_func_t environ_lookup_handler)
|
|
{
|
|
pkgconf_client_t *out = calloc(1, sizeof(pkgconf_client_t));
|
|
if (out == NULL)
|
|
return NULL;
|
|
|
|
pkgconf_client_init(out, error_handler, error_handler_data, personality, client_data, environ_lookup_handler);
|
|
return out;
|
|
}
|
|
|
|
static void
|
|
unref_preload_list(pkgconf_client_t *client)
|
|
{
|
|
pkgconf_node_t *n, *tn;
|
|
|
|
PKGCONF_FOREACH_LIST_ENTRY_SAFE(client->preloaded_pkgs.head, tn, n)
|
|
{
|
|
pkgconf_pkg_t *pkg = n->data;
|
|
pkgconf_pkg_unref(client, pkg);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_deinit(pkgconf_client_t *client)
|
|
*
|
|
* Release resources belonging to a pkgconf client object.
|
|
*
|
|
* :param pkgconf_client_t* client: The client to deinitialise.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_deinit(pkgconf_client_t *client)
|
|
{
|
|
PKGCONF_TRACE(client, "deinit @%p", client);
|
|
|
|
unref_preload_list(client);
|
|
|
|
if (client->prefix_varname != NULL)
|
|
free(client->prefix_varname);
|
|
|
|
if (client->sysroot_dir != NULL)
|
|
free(client->sysroot_dir);
|
|
|
|
if (client->buildroot_dir != NULL)
|
|
free(client->buildroot_dir);
|
|
|
|
pkgconf_path_free(&client->filter_libdirs);
|
|
pkgconf_path_free(&client->filter_includedirs);
|
|
|
|
pkgconf_tuple_free_global(client);
|
|
pkgconf_path_free(&client->dir_list);
|
|
pkgconf_cache_free(client);
|
|
|
|
memset(client, '\0', sizeof(*client));
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_free(pkgconf_client_t *client)
|
|
*
|
|
* Release resources belonging to a pkgconf client object and then free the client object itself.
|
|
*
|
|
* :param pkgconf_client_t* client: The client to deinitialise and free.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_free(pkgconf_client_t *client)
|
|
{
|
|
pkgconf_client_deinit(client);
|
|
free(client);
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: const char *pkgconf_client_get_sysroot_dir(const pkgconf_client_t *client)
|
|
*
|
|
* Retrieves the client's sysroot directory (if any).
|
|
*
|
|
* :param pkgconf_client_t* client: The client object being accessed.
|
|
* :return: A string containing the sysroot directory or NULL.
|
|
* :rtype: const char *
|
|
*/
|
|
const char *
|
|
pkgconf_client_get_sysroot_dir(const pkgconf_client_t *client)
|
|
{
|
|
return client->sysroot_dir;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_set_sysroot_dir(pkgconf_client_t *client, const char *sysroot_dir)
|
|
*
|
|
* Sets or clears the sysroot directory on a client object. Any previous sysroot directory setting is
|
|
* automatically released if one was previously set.
|
|
*
|
|
* Additionally, the global tuple ``$(pc_sysrootdir)`` is set as appropriate based on the new setting.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object being modified.
|
|
* :param char* sysroot_dir: The sysroot directory to set or NULL to unset.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_sysroot_dir(pkgconf_client_t *client, const char *sysroot_dir)
|
|
{
|
|
if (client->sysroot_dir != NULL)
|
|
free(client->sysroot_dir);
|
|
|
|
client->sysroot_dir = sysroot_dir != NULL ? strdup(sysroot_dir) : NULL;
|
|
|
|
PKGCONF_TRACE(client, "set sysroot_dir to: %s", client->sysroot_dir != NULL ? client->sysroot_dir : "<default>");
|
|
|
|
pkgconf_tuple_add_global(client, "pc_sysrootdir", client->sysroot_dir != NULL ? client->sysroot_dir : "");
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: const char *pkgconf_client_get_buildroot_dir(const pkgconf_client_t *client)
|
|
*
|
|
* Retrieves the client's buildroot directory (if any).
|
|
*
|
|
* :param pkgconf_client_t* client: The client object being accessed.
|
|
* :return: A string containing the buildroot directory or NULL.
|
|
* :rtype: const char *
|
|
*/
|
|
const char *
|
|
pkgconf_client_get_buildroot_dir(const pkgconf_client_t *client)
|
|
{
|
|
return client->buildroot_dir;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_set_buildroot_dir(pkgconf_client_t *client, const char *buildroot_dir)
|
|
*
|
|
* Sets or clears the buildroot directory on a client object. Any previous buildroot directory setting is
|
|
* automatically released if one was previously set.
|
|
*
|
|
* Additionally, the global tuple ``$(pc_top_builddir)`` is set as appropriate based on the new setting.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object being modified.
|
|
* :param char* buildroot_dir: The buildroot directory to set or NULL to unset.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_buildroot_dir(pkgconf_client_t *client, const char *buildroot_dir)
|
|
{
|
|
if (client->buildroot_dir != NULL)
|
|
free(client->buildroot_dir);
|
|
|
|
client->buildroot_dir = buildroot_dir != NULL ? strdup(buildroot_dir) : NULL;
|
|
|
|
PKGCONF_TRACE(client, "set buildroot_dir to: %s", client->buildroot_dir != NULL ? client->buildroot_dir : "<default>");
|
|
|
|
pkgconf_tuple_add_global(client, "pc_top_builddir", client->buildroot_dir != NULL ? client->buildroot_dir : "$(top_builddir)");
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_error(const pkgconf_client_t *client, const char *format, ...)
|
|
*
|
|
* Report an error to a client-registered error handler.
|
|
*
|
|
* :param pkgconf_client_t* client: The pkgconf client object to report the error to.
|
|
* :param char* format: A printf-style format string to use for formatting the error message.
|
|
* :return: true if the error handler processed the message, else false.
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_error(const pkgconf_client_t *client, const char *format, ...)
|
|
{
|
|
char *errbuf;
|
|
ssize_t msgsize = 0;
|
|
bool ret;
|
|
va_list va;
|
|
|
|
va_start(va, format);
|
|
msgsize = vsnprintf(NULL, 0, format, va);
|
|
va_end(va);
|
|
|
|
if (msgsize < 0)
|
|
return false;
|
|
|
|
msgsize++;
|
|
|
|
errbuf = calloc(1, msgsize);
|
|
if (errbuf == NULL)
|
|
return false;
|
|
|
|
va_start(va, format);
|
|
vsnprintf(errbuf, msgsize, format, va);
|
|
va_end(va);
|
|
|
|
ret = client->error_handler(errbuf, client, client->error_handler_data);
|
|
free(errbuf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_warn(const pkgconf_client_t *client, const char *format, ...)
|
|
*
|
|
* Report an error to a client-registered warn handler.
|
|
*
|
|
* :param pkgconf_client_t* client: The pkgconf client object to report the error to.
|
|
* :param char* format: A printf-style format string to use for formatting the warning message.
|
|
* :return: true if the warn handler processed the message, else false.
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_warn(const pkgconf_client_t *client, const char *format, ...)
|
|
{
|
|
char *errbuf;
|
|
ssize_t msgsize = 0;
|
|
bool ret;
|
|
va_list va;
|
|
|
|
va_start(va, format);
|
|
msgsize = vsnprintf(NULL, 0, format, va);
|
|
va_end(va);
|
|
|
|
if (msgsize < 0)
|
|
return false;
|
|
|
|
msgsize++;
|
|
|
|
errbuf = calloc(1, msgsize);
|
|
if (errbuf == NULL)
|
|
return false;
|
|
|
|
va_start(va, format);
|
|
vsnprintf(errbuf, msgsize, format, va);
|
|
va_end(va);
|
|
|
|
ret = client->warn_handler(errbuf, client, client->warn_handler_data);
|
|
free(errbuf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_trace(const pkgconf_client_t *client, const char *filename, size_t len, const char *funcname, const char *format, ...)
|
|
*
|
|
* Report a message to a client-registered trace handler.
|
|
*
|
|
* :param pkgconf_client_t* client: The pkgconf client object to report the trace message to.
|
|
* :param char* filename: The file the function is in.
|
|
* :param size_t lineno: The line number currently being executed.
|
|
* :param char* funcname: The function name to use.
|
|
* :param char* format: A printf-style format string to use for formatting the trace message.
|
|
* :return: true if the trace handler processed the message, else false.
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_trace(const pkgconf_client_t *client, const char *filename, size_t lineno, const char *funcname, const char *format, ...)
|
|
{
|
|
char prefix[PKGCONF_ITEM_SIZE];
|
|
char *errbuf = NULL;
|
|
ssize_t errlen;
|
|
char *finalbuf = NULL;
|
|
ssize_t finallen;
|
|
bool ret;
|
|
va_list va;
|
|
|
|
if (client == NULL || client->trace_handler == NULL)
|
|
return false;
|
|
|
|
snprintf(prefix, sizeof prefix, "%s:" SIZE_FMT_SPECIFIER " [%s]:", filename, lineno, funcname);
|
|
|
|
va_start(va, format);
|
|
errlen = vsnprintf(NULL, 0, format, va);
|
|
va_end(va);
|
|
|
|
if (errlen < 0)
|
|
return false;
|
|
|
|
errlen++;
|
|
errbuf = calloc(1, errlen);
|
|
if (errbuf == NULL)
|
|
return false;
|
|
|
|
va_start(va, format);
|
|
vsnprintf(errbuf, errlen, format, va);
|
|
va_end(va);
|
|
|
|
finallen = snprintf(NULL, 0, "%s %s\n", prefix, errbuf);
|
|
if (finallen < 0)
|
|
{
|
|
free(errbuf);
|
|
return false;
|
|
}
|
|
|
|
finallen++;
|
|
finalbuf = calloc(1, finallen);
|
|
if (finalbuf == NULL)
|
|
{
|
|
free(errbuf);
|
|
return false;
|
|
}
|
|
|
|
snprintf(finalbuf, finallen, "%s %s\n", prefix, errbuf);
|
|
ret = client->trace_handler(finalbuf, client, client->trace_handler_data);
|
|
free(errbuf);
|
|
free(finalbuf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_default_error_handler(const char *msg, const pkgconf_client_t *client, const void *data)
|
|
*
|
|
* The default pkgconf error handler.
|
|
*
|
|
* :param char* msg: The error message to handle.
|
|
* :param pkgconf_client_t* client: The client object the error originated from.
|
|
* :param void* data: An opaque pointer to extra data associated with the client for error handling.
|
|
* :return: true (the function does nothing to process the message)
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_default_error_handler(const char *msg, const pkgconf_client_t *client, void *data)
|
|
{
|
|
(void) msg;
|
|
(void) client;
|
|
(void) data;
|
|
|
|
return true;
|
|
}
|
|
|
|
static void
|
|
default_unveil_handler(const pkgconf_client_t *client, const char *path, const char *permissions)
|
|
{
|
|
(void) client;
|
|
(void) path;
|
|
(void) permissions;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: unsigned int pkgconf_client_get_flags(const pkgconf_client_t *client)
|
|
*
|
|
* Retrieves resolver-specific flags associated with a client object.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to retrieve the resolver-specific flags from.
|
|
* :return: a bitfield of resolver-specific flags
|
|
* :rtype: uint
|
|
*/
|
|
unsigned int
|
|
pkgconf_client_get_flags(const pkgconf_client_t *client)
|
|
{
|
|
return client->flags;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_set_flags(pkgconf_client_t *client, unsigned int flags)
|
|
*
|
|
* Sets resolver-specific flags associated with a client object.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the resolver-specific flags on.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_flags(pkgconf_client_t *client, unsigned int flags)
|
|
{
|
|
client->flags = flags;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: const char *pkgconf_client_get_prefix_varname(const pkgconf_client_t *client)
|
|
*
|
|
* Retrieves the name of the variable that should contain a module's prefix.
|
|
* In some cases, it is necessary to override this variable to allow proper path relocation.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to retrieve the prefix variable name from.
|
|
* :return: the prefix variable name as a string
|
|
* :rtype: const char *
|
|
*/
|
|
const char *
|
|
pkgconf_client_get_prefix_varname(const pkgconf_client_t *client)
|
|
{
|
|
return client->prefix_varname;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_set_prefix_varname(pkgconf_client_t *client, const char *prefix_varname)
|
|
*
|
|
* Sets the name of the variable that should contain a module's prefix.
|
|
* If the variable name is ``NULL``, then the default variable name (``prefix``) is used.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the prefix variable name on.
|
|
* :param char* prefix_varname: The prefix variable name to set.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_prefix_varname(pkgconf_client_t *client, const char *prefix_varname)
|
|
{
|
|
if (prefix_varname == NULL)
|
|
prefix_varname = "prefix";
|
|
|
|
if (client->prefix_varname != NULL)
|
|
free(client->prefix_varname);
|
|
|
|
client->prefix_varname = strdup(prefix_varname);
|
|
|
|
PKGCONF_TRACE(client, "set prefix_varname to: %s", client->prefix_varname);
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_get_warn_handler(const pkgconf_client_t *client)
|
|
*
|
|
* Returns the warning handler if one is set, else ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to get the warn handler from.
|
|
* :return: a function pointer to the warn handler or ``NULL``
|
|
*/
|
|
pkgconf_error_handler_func_t
|
|
pkgconf_client_get_warn_handler(const pkgconf_client_t *client)
|
|
{
|
|
return client->warn_handler;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_set_warn_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t warn_handler, void *warn_handler_data)
|
|
*
|
|
* Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the warn handler on.
|
|
* :param pkgconf_error_handler_func_t warn_handler: The warn handler to set.
|
|
* :param void* warn_handler_data: Optional data to associate with the warn handler.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_warn_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t warn_handler, void *warn_handler_data)
|
|
{
|
|
client->warn_handler = warn_handler;
|
|
client->warn_handler_data = warn_handler_data;
|
|
|
|
if (client->warn_handler == NULL)
|
|
{
|
|
PKGCONF_TRACE(client, "installing default warn handler");
|
|
client->warn_handler = pkgconf_default_error_handler;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_get_error_handler(const pkgconf_client_t *client)
|
|
*
|
|
* Returns the error handler if one is set, else ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to get the error handler from.
|
|
* :return: a function pointer to the error handler or ``NULL``
|
|
*/
|
|
pkgconf_error_handler_func_t
|
|
pkgconf_client_get_error_handler(const pkgconf_client_t *client)
|
|
{
|
|
return client->error_handler;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_set_error_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data)
|
|
*
|
|
* Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the error handler on.
|
|
* :param pkgconf_error_handler_func_t error_handler: The error handler to set.
|
|
* :param void* error_handler_data: Optional data to associate with the error handler.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_error_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data)
|
|
{
|
|
client->error_handler = error_handler;
|
|
client->error_handler_data = error_handler_data;
|
|
|
|
if (client->error_handler == NULL)
|
|
{
|
|
PKGCONF_TRACE(client, "installing default error handler");
|
|
client->error_handler = pkgconf_default_error_handler;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_get_unveil_handler(const pkgconf_client_t *client)
|
|
*
|
|
* Returns the unveil handler if one is set, else ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to get the unveil handler from.
|
|
* :return: a function pointer to the error handler or ``NULL``
|
|
*/
|
|
pkgconf_unveil_handler_func_t
|
|
pkgconf_client_get_unveil_handler(const pkgconf_client_t *client)
|
|
{
|
|
return client->unveil_handler;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_set_unveil_handler(pkgconf_client_t *client, pkgconf_unveil_handler_func_t unveil_handler)
|
|
*
|
|
* Sets an unveil handler on a client object or uninstalls one if set to ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the error handler on.
|
|
* :param pkgconf_unveil_handler_func_t unveil_handler: The unveil handler to set.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_unveil_handler(pkgconf_client_t *client, pkgconf_unveil_handler_func_t unveil_handler)
|
|
{
|
|
client->unveil_handler = unveil_handler;
|
|
|
|
if (client->unveil_handler == NULL)
|
|
{
|
|
PKGCONF_TRACE(client, "installing default unveil handler");
|
|
client->unveil_handler = default_unveil_handler;
|
|
}
|
|
}
|
|
|
|
#ifndef PKGCONF_LITE
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_get_trace_handler(const pkgconf_client_t *client)
|
|
*
|
|
* Returns the error handler if one is set, else ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to get the error handler from.
|
|
* :return: a function pointer to the error handler or ``NULL``
|
|
*/
|
|
pkgconf_error_handler_func_t
|
|
pkgconf_client_get_trace_handler(const pkgconf_client_t *client)
|
|
{
|
|
return client->trace_handler;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: pkgconf_client_set_trace_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t trace_handler, void *trace_handler_data)
|
|
*
|
|
* Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the error handler on.
|
|
* :param pkgconf_error_handler_func_t trace_handler: The error handler to set.
|
|
* :param void* trace_handler_data: Optional data to associate with the error handler.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_trace_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t trace_handler, void *trace_handler_data)
|
|
{
|
|
client->trace_handler = trace_handler;
|
|
client->trace_handler_data = trace_handler_data;
|
|
|
|
if (client->trace_handler == NULL)
|
|
{
|
|
client->trace_handler = pkgconf_default_error_handler;
|
|
PKGCONF_TRACE(client, "installing default trace handler");
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_client_preload_one(pkgconf_client_t *client, pkgconf_pkg_t *pkg)
|
|
*
|
|
* Adds a package to the preloaded packages set.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object for preloading.
|
|
* :param pkgconf_pkg_t* pkg: The package to preload.
|
|
* :return: true on success, false on error
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_client_preload_one(pkgconf_client_t *client, pkgconf_pkg_t *pkg)
|
|
{
|
|
PKGCONF_TRACE(client, "preloading pkg %s@%p", pkg->id, pkg);
|
|
|
|
pkg->flags |= PKGCONF_PKG_PROPF_PRELOADED;
|
|
|
|
pkgconf_pkg_ref(client, pkg);
|
|
pkgconf_node_insert_tail(&pkg->preload_node, pkg, &client->preloaded_pkgs);
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_client_preload_path(pkgconf_client_t *client, const char *path)
|
|
*
|
|
* Loads a pkg-config file into the preloaded packages set.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object for preloading.
|
|
* :param char* path: The path to the pkg-config file to preload.
|
|
* :return: true on success, false on error
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_client_preload_path(pkgconf_client_t *client, const char *path)
|
|
{
|
|
pkgconf_pkg_t *pkg = pkgconf_pkg_new_from_path(client, path, PKGCONF_PKG_PROPF_PRELOADED);
|
|
if (pkg == NULL)
|
|
return false;
|
|
|
|
return pkgconf_client_preload_one(client, pkg);
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: bool pkgconf_client_preload_from_environ(pkgconf_client_t *client, const char *env)
|
|
*
|
|
* Loads zero or more pkg-config files specified in the given environmental
|
|
* variable.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object for preloading.
|
|
* :param char* environ: The environment variable to use for preloading.
|
|
* :return: true on success, false on error
|
|
* :rtype: bool
|
|
*/
|
|
bool
|
|
pkgconf_client_preload_from_environ(pkgconf_client_t *client, const char *env)
|
|
{
|
|
const char *data;
|
|
pkgconf_list_t pathlist = PKGCONF_LIST_INITIALIZER;
|
|
pkgconf_node_t *n;
|
|
bool ret = true;
|
|
|
|
data = getenv(env);
|
|
if (data == NULL)
|
|
return true;
|
|
|
|
pkgconf_path_split(data, &pathlist, true);
|
|
|
|
PKGCONF_FOREACH_LIST_ENTRY(pathlist.head, n)
|
|
{
|
|
pkgconf_path_t *pn = n->data;
|
|
|
|
ret = pkgconf_client_preload_path(client, pn->path);
|
|
if (!ret)
|
|
break;
|
|
}
|
|
|
|
pkgconf_path_free(&pathlist);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: void pkgconf_client_set_output(pkgconf_client_t *client, pkgconf_output_t *output)
|
|
*
|
|
* Sets the client's output object. This is mainly a convenience function for clients
|
|
* to use.
|
|
*
|
|
* :param pkgconf_client_t* client: The client object to set the output object for.
|
|
* :param pkgconf_output_t* output: The output object to use.
|
|
* :return: nothing
|
|
*/
|
|
void
|
|
pkgconf_client_set_output(pkgconf_client_t *client, pkgconf_output_t *output)
|
|
{
|
|
client->output = output;
|
|
}
|
|
|
|
/*
|
|
* !doc
|
|
*
|
|
* .. c:function:: const char *pkgconf_client_getenv(const pkgconf_client_t *client, const char *key)
|
|
*
|
|
* Looks up an environmental variable which may be mocked, otherwise fetches
|
|
* from the main environment.
|
|
*
|
|
* :param pkgconf_client_t* client: yhe client object to use for looking up environmental variables.
|
|
* :param char* key: the environmental variable to look up.
|
|
* :return: the environmental variable contents else NULL
|
|
* :rtype: const char*
|
|
*/
|
|
const char *
|
|
pkgconf_client_getenv(const pkgconf_client_t *client, const char *key)
|
|
{
|
|
if (client != NULL && client->environ_lookup_handler != NULL)
|
|
return client->environ_lookup_handler(client, key);
|
|
|
|
return getenv(key);
|
|
}
|