xref: /freebsd/contrib/pkgconf/libpkgconf/client.c (revision a3cefe7f2b4df0f70ff92d4570ce18e517af43ec)
1*a3cefe7fSPierre Pronchery /*
2*a3cefe7fSPierre Pronchery  * client.c
3*a3cefe7fSPierre Pronchery  * libpkgconf consumer lifecycle management
4*a3cefe7fSPierre Pronchery  *
5*a3cefe7fSPierre Pronchery  * Copyright (c) 2016 pkgconf authors (see AUTHORS).
6*a3cefe7fSPierre Pronchery  *
7*a3cefe7fSPierre Pronchery  * Permission to use, copy, modify, and/or distribute this software for any
8*a3cefe7fSPierre Pronchery  * purpose with or without fee is hereby granted, provided that the above
9*a3cefe7fSPierre Pronchery  * copyright notice and this permission notice appear in all copies.
10*a3cefe7fSPierre Pronchery  *
11*a3cefe7fSPierre Pronchery  * This software is provided 'as is' and without any warranty, express or
12*a3cefe7fSPierre Pronchery  * implied.  In no event shall the authors be liable for any damages arising
13*a3cefe7fSPierre Pronchery  * from the use of this software.
14*a3cefe7fSPierre Pronchery  */
15*a3cefe7fSPierre Pronchery 
16*a3cefe7fSPierre Pronchery #include <libpkgconf/config.h>
17*a3cefe7fSPierre Pronchery #include <libpkgconf/stdinc.h>
18*a3cefe7fSPierre Pronchery #include <libpkgconf/libpkgconf.h>
19*a3cefe7fSPierre Pronchery 
20*a3cefe7fSPierre Pronchery /*
21*a3cefe7fSPierre Pronchery  * !doc
22*a3cefe7fSPierre Pronchery  *
23*a3cefe7fSPierre Pronchery  * libpkgconf `client` module
24*a3cefe7fSPierre Pronchery  * ==========================
25*a3cefe7fSPierre Pronchery  *
26*a3cefe7fSPierre Pronchery  * The libpkgconf `client` module implements the `pkgconf_client_t` "client" object.
27*a3cefe7fSPierre Pronchery  * Client objects store all necessary state for libpkgconf allowing for multiple instances to run
28*a3cefe7fSPierre Pronchery  * in parallel.
29*a3cefe7fSPierre Pronchery  *
30*a3cefe7fSPierre Pronchery  * Client objects are not thread safe, in other words, a client object should not be shared across
31*a3cefe7fSPierre Pronchery  * thread boundaries.
32*a3cefe7fSPierre Pronchery  */
33*a3cefe7fSPierre Pronchery 
34*a3cefe7fSPierre Pronchery static void
trace_path_list(const pkgconf_client_t * client,const char * desc,pkgconf_list_t * list)35*a3cefe7fSPierre Pronchery trace_path_list(const pkgconf_client_t *client, const char *desc, pkgconf_list_t *list)
36*a3cefe7fSPierre Pronchery {
37*a3cefe7fSPierre Pronchery 	const pkgconf_node_t *n;
38*a3cefe7fSPierre Pronchery 
39*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "%s:", desc);
40*a3cefe7fSPierre Pronchery 	PKGCONF_FOREACH_LIST_ENTRY(list->head, n)
41*a3cefe7fSPierre Pronchery 	{
42*a3cefe7fSPierre Pronchery 		const pkgconf_path_t *p = n->data;
43*a3cefe7fSPierre Pronchery 
44*a3cefe7fSPierre Pronchery 		PKGCONF_TRACE(client, "  - '%s'", p->path);
45*a3cefe7fSPierre Pronchery 	}
46*a3cefe7fSPierre Pronchery }
47*a3cefe7fSPierre Pronchery 
48*a3cefe7fSPierre Pronchery /*
49*a3cefe7fSPierre Pronchery  * !doc
50*a3cefe7fSPierre Pronchery  *
51*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_dir_list_build(pkgconf_client_t *client)
52*a3cefe7fSPierre Pronchery  *
53*a3cefe7fSPierre Pronchery  *    Bootstraps the package search paths.  If the ``PKGCONF_PKG_PKGF_ENV_ONLY`` `flag` is set on the client,
54*a3cefe7fSPierre Pronchery  *    then only the ``PKG_CONFIG_PATH`` environment variable will be used, otherwise both the
55*a3cefe7fSPierre Pronchery  *    ``PKG_CONFIG_PATH`` and ``PKG_CONFIG_LIBDIR`` environment variables will be used.
56*a3cefe7fSPierre Pronchery  *
57*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The pkgconf client object to bootstrap.
58*a3cefe7fSPierre Pronchery  *    :return: nothing
59*a3cefe7fSPierre Pronchery  */
60*a3cefe7fSPierre Pronchery void
pkgconf_client_dir_list_build(pkgconf_client_t * client,const pkgconf_cross_personality_t * personality)61*a3cefe7fSPierre Pronchery pkgconf_client_dir_list_build(pkgconf_client_t *client, const pkgconf_cross_personality_t *personality)
62*a3cefe7fSPierre Pronchery {
63*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("PKG_CONFIG_PATH", NULL, &client->dir_list, true);
64*a3cefe7fSPierre Pronchery 
65*a3cefe7fSPierre Pronchery 	if (!(client->flags & PKGCONF_PKG_PKGF_ENV_ONLY))
66*a3cefe7fSPierre Pronchery 	{
67*a3cefe7fSPierre Pronchery 		pkgconf_list_t dir_list = PKGCONF_LIST_INITIALIZER;
68*a3cefe7fSPierre Pronchery 		const pkgconf_list_t *prepend_list = &personality->dir_list;
69*a3cefe7fSPierre Pronchery 
70*a3cefe7fSPierre Pronchery #ifdef _WIN32
71*a3cefe7fSPierre Pronchery 		(void) pkgconf_path_build_from_registry(HKEY_CURRENT_USER, &client->dir_list, true);
72*a3cefe7fSPierre Pronchery 		(void) pkgconf_path_build_from_registry(HKEY_LOCAL_MACHINE, &client->dir_list, true);
73*a3cefe7fSPierre Pronchery #endif
74*a3cefe7fSPierre Pronchery 
75*a3cefe7fSPierre Pronchery 		if (getenv("PKG_CONFIG_LIBDIR") != NULL)
76*a3cefe7fSPierre Pronchery 		{
77*a3cefe7fSPierre Pronchery 			/* PKG_CONFIG_LIBDIR= should empty the search path entirely. */
78*a3cefe7fSPierre Pronchery 			(void) pkgconf_path_build_from_environ("PKG_CONFIG_LIBDIR", NULL, &dir_list, true);
79*a3cefe7fSPierre Pronchery 			prepend_list = &dir_list;
80*a3cefe7fSPierre Pronchery 		}
81*a3cefe7fSPierre Pronchery 
82*a3cefe7fSPierre Pronchery 		pkgconf_path_copy_list(&client->dir_list, prepend_list);
83*a3cefe7fSPierre Pronchery 		pkgconf_path_free(&dir_list);
84*a3cefe7fSPierre Pronchery 	}
85*a3cefe7fSPierre Pronchery }
86*a3cefe7fSPierre Pronchery 
87*a3cefe7fSPierre Pronchery /*
88*a3cefe7fSPierre Pronchery  * !doc
89*a3cefe7fSPierre Pronchery  *
90*a3cefe7fSPierre Pronchery  * .. 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)
91*a3cefe7fSPierre Pronchery  *
92*a3cefe7fSPierre Pronchery  *    Initialise a pkgconf client object.
93*a3cefe7fSPierre Pronchery  *
94*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client to initialise.
95*a3cefe7fSPierre Pronchery  *    :param pkgconf_error_handler_func_t error_handler: An optional error handler to use for logging errors.
96*a3cefe7fSPierre Pronchery  *    :param void* error_handler_data: user data passed to optional error handler
97*a3cefe7fSPierre Pronchery  *    :param pkgconf_cross_personality_t* personality: the cross-compile personality to use for defaults
98*a3cefe7fSPierre Pronchery  *    :return: nothing
99*a3cefe7fSPierre Pronchery  */
100*a3cefe7fSPierre Pronchery 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)101*a3cefe7fSPierre Pronchery pkgconf_client_init(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality)
102*a3cefe7fSPierre Pronchery {
103*a3cefe7fSPierre Pronchery 	client->error_handler_data = error_handler_data;
104*a3cefe7fSPierre Pronchery 	client->error_handler = error_handler;
105*a3cefe7fSPierre Pronchery 	client->auditf = NULL;
106*a3cefe7fSPierre Pronchery 	client->cache_table = NULL;
107*a3cefe7fSPierre Pronchery 	client->cache_count = 0;
108*a3cefe7fSPierre Pronchery 
109*a3cefe7fSPierre Pronchery #ifndef PKGCONF_LITE
110*a3cefe7fSPierre Pronchery 	if (client->trace_handler == NULL)
111*a3cefe7fSPierre Pronchery 		pkgconf_client_set_trace_handler(client, NULL, NULL);
112*a3cefe7fSPierre Pronchery #endif
113*a3cefe7fSPierre Pronchery 
114*a3cefe7fSPierre Pronchery 	if (client->unveil_handler == NULL)
115*a3cefe7fSPierre Pronchery 		pkgconf_client_set_unveil_handler(client, NULL);
116*a3cefe7fSPierre Pronchery 
117*a3cefe7fSPierre Pronchery 	pkgconf_client_set_error_handler(client, error_handler, error_handler_data);
118*a3cefe7fSPierre Pronchery 	pkgconf_client_set_warn_handler(client, NULL, NULL);
119*a3cefe7fSPierre Pronchery 
120*a3cefe7fSPierre Pronchery 	pkgconf_client_set_sysroot_dir(client, personality->sysroot_dir);
121*a3cefe7fSPierre Pronchery 	pkgconf_client_set_buildroot_dir(client, NULL);
122*a3cefe7fSPierre Pronchery 	pkgconf_client_set_prefix_varname(client, NULL);
123*a3cefe7fSPierre Pronchery 
124*a3cefe7fSPierre Pronchery 	if(getenv("PKG_CONFIG_SYSTEM_LIBRARY_PATH") == NULL)
125*a3cefe7fSPierre Pronchery 		pkgconf_path_copy_list(&client->filter_libdirs, &personality->filter_libdirs);
126*a3cefe7fSPierre Pronchery 	else
127*a3cefe7fSPierre Pronchery 		pkgconf_path_build_from_environ("PKG_CONFIG_SYSTEM_LIBRARY_PATH", NULL, &client->filter_libdirs, false);
128*a3cefe7fSPierre Pronchery 
129*a3cefe7fSPierre Pronchery 	if(getenv("PKG_CONFIG_SYSTEM_INCLUDE_PATH") == NULL)
130*a3cefe7fSPierre Pronchery 		pkgconf_path_copy_list(&client->filter_includedirs, &personality->filter_includedirs);
131*a3cefe7fSPierre Pronchery 	else
132*a3cefe7fSPierre Pronchery 		pkgconf_path_build_from_environ("PKG_CONFIG_SYSTEM_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
133*a3cefe7fSPierre Pronchery 
134*a3cefe7fSPierre Pronchery 	/* GCC uses these environment variables to define system include paths, so we should check them. */
135*a3cefe7fSPierre Pronchery #ifdef __HAIKU__
136*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("BELIBRARIES", NULL, &client->filter_libdirs, false);
137*a3cefe7fSPierre Pronchery #else
138*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("LIBRARY_PATH", NULL, &client->filter_libdirs, false);
139*a3cefe7fSPierre Pronchery #endif
140*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("CPATH", NULL, &client->filter_includedirs, false);
141*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("C_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
142*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("CPLUS_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
143*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("OBJC_INCLUDE_PATH", NULL, &client->filter_includedirs, false);
144*a3cefe7fSPierre Pronchery 
145*a3cefe7fSPierre Pronchery #ifdef _WIN32
146*a3cefe7fSPierre Pronchery 	/* also use the path lists that MSVC uses on windows */
147*a3cefe7fSPierre Pronchery 	pkgconf_path_build_from_environ("INCLUDE", NULL, &client->filter_includedirs, false);
148*a3cefe7fSPierre Pronchery #endif
149*a3cefe7fSPierre Pronchery 
150*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "initialized client @%p", client);
151*a3cefe7fSPierre Pronchery 
152*a3cefe7fSPierre Pronchery 	trace_path_list(client, "filtered library paths", &client->filter_libdirs);
153*a3cefe7fSPierre Pronchery 	trace_path_list(client, "filtered include paths", &client->filter_includedirs);
154*a3cefe7fSPierre Pronchery }
155*a3cefe7fSPierre Pronchery 
156*a3cefe7fSPierre Pronchery /*
157*a3cefe7fSPierre Pronchery  * !doc
158*a3cefe7fSPierre Pronchery  *
159*a3cefe7fSPierre Pronchery  * .. 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)
160*a3cefe7fSPierre Pronchery  *
161*a3cefe7fSPierre Pronchery  *    Allocate and initialise a pkgconf client object.
162*a3cefe7fSPierre Pronchery  *
163*a3cefe7fSPierre Pronchery  *    :param pkgconf_error_handler_func_t error_handler: An optional error handler to use for logging errors.
164*a3cefe7fSPierre Pronchery  *    :param void* error_handler_data: user data passed to optional error handler
165*a3cefe7fSPierre Pronchery  *    :param pkgconf_cross_personality_t* personality: cross-compile personality to use
166*a3cefe7fSPierre Pronchery  *    :return: A pkgconf client object.
167*a3cefe7fSPierre Pronchery  *    :rtype: pkgconf_client_t*
168*a3cefe7fSPierre Pronchery  */
169*a3cefe7fSPierre Pronchery pkgconf_client_t *
pkgconf_client_new(pkgconf_error_handler_func_t error_handler,void * error_handler_data,const pkgconf_cross_personality_t * personality)170*a3cefe7fSPierre Pronchery pkgconf_client_new(pkgconf_error_handler_func_t error_handler, void *error_handler_data, const pkgconf_cross_personality_t *personality)
171*a3cefe7fSPierre Pronchery {
172*a3cefe7fSPierre Pronchery 	pkgconf_client_t *out = calloc(1, sizeof(pkgconf_client_t));
173*a3cefe7fSPierre Pronchery 	if (out == NULL)
174*a3cefe7fSPierre Pronchery 		return NULL;
175*a3cefe7fSPierre Pronchery 
176*a3cefe7fSPierre Pronchery 	pkgconf_client_init(out, error_handler, error_handler_data, personality);
177*a3cefe7fSPierre Pronchery 	return out;
178*a3cefe7fSPierre Pronchery }
179*a3cefe7fSPierre Pronchery 
180*a3cefe7fSPierre Pronchery static void
unref_preload_list(pkgconf_client_t * client)181*a3cefe7fSPierre Pronchery unref_preload_list(pkgconf_client_t *client)
182*a3cefe7fSPierre Pronchery {
183*a3cefe7fSPierre Pronchery 	pkgconf_node_t *n, *tn;
184*a3cefe7fSPierre Pronchery 
185*a3cefe7fSPierre Pronchery 	PKGCONF_FOREACH_LIST_ENTRY_SAFE(client->preloaded_pkgs.head, n, tn)
186*a3cefe7fSPierre Pronchery 	{
187*a3cefe7fSPierre Pronchery 		pkgconf_pkg_t *pkg = n->data;
188*a3cefe7fSPierre Pronchery 		pkgconf_pkg_unref(client, pkg);
189*a3cefe7fSPierre Pronchery 	}
190*a3cefe7fSPierre Pronchery }
191*a3cefe7fSPierre Pronchery 
192*a3cefe7fSPierre Pronchery /*
193*a3cefe7fSPierre Pronchery  * !doc
194*a3cefe7fSPierre Pronchery  *
195*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_deinit(pkgconf_client_t *client)
196*a3cefe7fSPierre Pronchery  *
197*a3cefe7fSPierre Pronchery  *    Release resources belonging to a pkgconf client object.
198*a3cefe7fSPierre Pronchery  *
199*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client to deinitialise.
200*a3cefe7fSPierre Pronchery  *    :return: nothing
201*a3cefe7fSPierre Pronchery  */
202*a3cefe7fSPierre Pronchery void
pkgconf_client_deinit(pkgconf_client_t * client)203*a3cefe7fSPierre Pronchery pkgconf_client_deinit(pkgconf_client_t *client)
204*a3cefe7fSPierre Pronchery {
205*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "deinit @%p", client);
206*a3cefe7fSPierre Pronchery 
207*a3cefe7fSPierre Pronchery 	unref_preload_list(client);
208*a3cefe7fSPierre Pronchery 
209*a3cefe7fSPierre Pronchery 	if (client->prefix_varname != NULL)
210*a3cefe7fSPierre Pronchery 		free(client->prefix_varname);
211*a3cefe7fSPierre Pronchery 
212*a3cefe7fSPierre Pronchery 	if (client->sysroot_dir != NULL)
213*a3cefe7fSPierre Pronchery 		free(client->sysroot_dir);
214*a3cefe7fSPierre Pronchery 
215*a3cefe7fSPierre Pronchery 	if (client->buildroot_dir != NULL)
216*a3cefe7fSPierre Pronchery 		free(client->buildroot_dir);
217*a3cefe7fSPierre Pronchery 
218*a3cefe7fSPierre Pronchery 	pkgconf_path_free(&client->filter_libdirs);
219*a3cefe7fSPierre Pronchery 	pkgconf_path_free(&client->filter_includedirs);
220*a3cefe7fSPierre Pronchery 
221*a3cefe7fSPierre Pronchery 	pkgconf_tuple_free_global(client);
222*a3cefe7fSPierre Pronchery 	pkgconf_path_free(&client->dir_list);
223*a3cefe7fSPierre Pronchery 	pkgconf_cache_free(client);
224*a3cefe7fSPierre Pronchery }
225*a3cefe7fSPierre Pronchery 
226*a3cefe7fSPierre Pronchery /*
227*a3cefe7fSPierre Pronchery  * !doc
228*a3cefe7fSPierre Pronchery  *
229*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_free(pkgconf_client_t *client)
230*a3cefe7fSPierre Pronchery  *
231*a3cefe7fSPierre Pronchery  *    Release resources belonging to a pkgconf client object and then free the client object itself.
232*a3cefe7fSPierre Pronchery  *
233*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client to deinitialise and free.
234*a3cefe7fSPierre Pronchery  *    :return: nothing
235*a3cefe7fSPierre Pronchery  */
236*a3cefe7fSPierre Pronchery void
pkgconf_client_free(pkgconf_client_t * client)237*a3cefe7fSPierre Pronchery pkgconf_client_free(pkgconf_client_t *client)
238*a3cefe7fSPierre Pronchery {
239*a3cefe7fSPierre Pronchery 	pkgconf_client_deinit(client);
240*a3cefe7fSPierre Pronchery 	free(client);
241*a3cefe7fSPierre Pronchery }
242*a3cefe7fSPierre Pronchery 
243*a3cefe7fSPierre Pronchery /*
244*a3cefe7fSPierre Pronchery  * !doc
245*a3cefe7fSPierre Pronchery  *
246*a3cefe7fSPierre Pronchery  * .. c:function:: const char *pkgconf_client_get_sysroot_dir(const pkgconf_client_t *client)
247*a3cefe7fSPierre Pronchery  *
248*a3cefe7fSPierre Pronchery  *    Retrieves the client's sysroot directory (if any).
249*a3cefe7fSPierre Pronchery  *
250*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object being accessed.
251*a3cefe7fSPierre Pronchery  *    :return: A string containing the sysroot directory or NULL.
252*a3cefe7fSPierre Pronchery  *    :rtype: const char *
253*a3cefe7fSPierre Pronchery  */
254*a3cefe7fSPierre Pronchery const char *
pkgconf_client_get_sysroot_dir(const pkgconf_client_t * client)255*a3cefe7fSPierre Pronchery pkgconf_client_get_sysroot_dir(const pkgconf_client_t *client)
256*a3cefe7fSPierre Pronchery {
257*a3cefe7fSPierre Pronchery 	return client->sysroot_dir;
258*a3cefe7fSPierre Pronchery }
259*a3cefe7fSPierre Pronchery 
260*a3cefe7fSPierre Pronchery /*
261*a3cefe7fSPierre Pronchery  * !doc
262*a3cefe7fSPierre Pronchery  *
263*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_set_sysroot_dir(pkgconf_client_t *client, const char *sysroot_dir)
264*a3cefe7fSPierre Pronchery  *
265*a3cefe7fSPierre Pronchery  *    Sets or clears the sysroot directory on a client object.  Any previous sysroot directory setting is
266*a3cefe7fSPierre Pronchery  *    automatically released if one was previously set.
267*a3cefe7fSPierre Pronchery  *
268*a3cefe7fSPierre Pronchery  *    Additionally, the global tuple ``$(pc_sysrootdir)`` is set as appropriate based on the new setting.
269*a3cefe7fSPierre Pronchery  *
270*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object being modified.
271*a3cefe7fSPierre Pronchery  *    :param char* sysroot_dir: The sysroot directory to set or NULL to unset.
272*a3cefe7fSPierre Pronchery  *    :return: nothing
273*a3cefe7fSPierre Pronchery  */
274*a3cefe7fSPierre Pronchery void
pkgconf_client_set_sysroot_dir(pkgconf_client_t * client,const char * sysroot_dir)275*a3cefe7fSPierre Pronchery pkgconf_client_set_sysroot_dir(pkgconf_client_t *client, const char *sysroot_dir)
276*a3cefe7fSPierre Pronchery {
277*a3cefe7fSPierre Pronchery 	if (client->sysroot_dir != NULL)
278*a3cefe7fSPierre Pronchery 		free(client->sysroot_dir);
279*a3cefe7fSPierre Pronchery 
280*a3cefe7fSPierre Pronchery 	client->sysroot_dir = sysroot_dir != NULL ? strdup(sysroot_dir) : NULL;
281*a3cefe7fSPierre Pronchery 
282*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "set sysroot_dir to: %s", client->sysroot_dir != NULL ? client->sysroot_dir : "<default>");
283*a3cefe7fSPierre Pronchery 
284*a3cefe7fSPierre Pronchery 	pkgconf_tuple_add_global(client, "pc_sysrootdir", client->sysroot_dir != NULL ? client->sysroot_dir : "/");
285*a3cefe7fSPierre Pronchery }
286*a3cefe7fSPierre Pronchery 
287*a3cefe7fSPierre Pronchery /*
288*a3cefe7fSPierre Pronchery  * !doc
289*a3cefe7fSPierre Pronchery  *
290*a3cefe7fSPierre Pronchery  * .. c:function:: const char *pkgconf_client_get_buildroot_dir(const pkgconf_client_t *client)
291*a3cefe7fSPierre Pronchery  *
292*a3cefe7fSPierre Pronchery  *    Retrieves the client's buildroot directory (if any).
293*a3cefe7fSPierre Pronchery  *
294*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object being accessed.
295*a3cefe7fSPierre Pronchery  *    :return: A string containing the buildroot directory or NULL.
296*a3cefe7fSPierre Pronchery  *    :rtype: const char *
297*a3cefe7fSPierre Pronchery  */
298*a3cefe7fSPierre Pronchery const char *
pkgconf_client_get_buildroot_dir(const pkgconf_client_t * client)299*a3cefe7fSPierre Pronchery pkgconf_client_get_buildroot_dir(const pkgconf_client_t *client)
300*a3cefe7fSPierre Pronchery {
301*a3cefe7fSPierre Pronchery 	return client->buildroot_dir;
302*a3cefe7fSPierre Pronchery }
303*a3cefe7fSPierre Pronchery 
304*a3cefe7fSPierre Pronchery /*
305*a3cefe7fSPierre Pronchery  * !doc
306*a3cefe7fSPierre Pronchery  *
307*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_set_buildroot_dir(pkgconf_client_t *client, const char *buildroot_dir)
308*a3cefe7fSPierre Pronchery  *
309*a3cefe7fSPierre Pronchery  *    Sets or clears the buildroot directory on a client object.  Any previous buildroot directory setting is
310*a3cefe7fSPierre Pronchery  *    automatically released if one was previously set.
311*a3cefe7fSPierre Pronchery  *
312*a3cefe7fSPierre Pronchery  *    Additionally, the global tuple ``$(pc_top_builddir)`` is set as appropriate based on the new setting.
313*a3cefe7fSPierre Pronchery  *
314*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object being modified.
315*a3cefe7fSPierre Pronchery  *    :param char* buildroot_dir: The buildroot directory to set or NULL to unset.
316*a3cefe7fSPierre Pronchery  *    :return: nothing
317*a3cefe7fSPierre Pronchery  */
318*a3cefe7fSPierre Pronchery void
pkgconf_client_set_buildroot_dir(pkgconf_client_t * client,const char * buildroot_dir)319*a3cefe7fSPierre Pronchery pkgconf_client_set_buildroot_dir(pkgconf_client_t *client, const char *buildroot_dir)
320*a3cefe7fSPierre Pronchery {
321*a3cefe7fSPierre Pronchery 	if (client->buildroot_dir != NULL)
322*a3cefe7fSPierre Pronchery 		free(client->buildroot_dir);
323*a3cefe7fSPierre Pronchery 
324*a3cefe7fSPierre Pronchery 	client->buildroot_dir = buildroot_dir != NULL ? strdup(buildroot_dir) : NULL;
325*a3cefe7fSPierre Pronchery 
326*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "set buildroot_dir to: %s", client->buildroot_dir != NULL ? client->buildroot_dir : "<default>");
327*a3cefe7fSPierre Pronchery 
328*a3cefe7fSPierre Pronchery 	pkgconf_tuple_add_global(client, "pc_top_builddir", client->buildroot_dir != NULL ? client->buildroot_dir : "$(top_builddir)");
329*a3cefe7fSPierre Pronchery }
330*a3cefe7fSPierre Pronchery 
331*a3cefe7fSPierre Pronchery /*
332*a3cefe7fSPierre Pronchery  * !doc
333*a3cefe7fSPierre Pronchery  *
334*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_error(const pkgconf_client_t *client, const char *format, ...)
335*a3cefe7fSPierre Pronchery  *
336*a3cefe7fSPierre Pronchery  *    Report an error to a client-registered error handler.
337*a3cefe7fSPierre Pronchery  *
338*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The pkgconf client object to report the error to.
339*a3cefe7fSPierre Pronchery  *    :param char* format: A printf-style format string to use for formatting the error message.
340*a3cefe7fSPierre Pronchery  *    :return: true if the error handler processed the message, else false.
341*a3cefe7fSPierre Pronchery  *    :rtype: bool
342*a3cefe7fSPierre Pronchery  */
343*a3cefe7fSPierre Pronchery bool
pkgconf_error(const pkgconf_client_t * client,const char * format,...)344*a3cefe7fSPierre Pronchery pkgconf_error(const pkgconf_client_t *client, const char *format, ...)
345*a3cefe7fSPierre Pronchery {
346*a3cefe7fSPierre Pronchery 	char *errbuf;
347*a3cefe7fSPierre Pronchery 	ssize_t msgsize = 0;
348*a3cefe7fSPierre Pronchery 	bool ret;
349*a3cefe7fSPierre Pronchery 	va_list va;
350*a3cefe7fSPierre Pronchery 
351*a3cefe7fSPierre Pronchery 	va_start(va, format);
352*a3cefe7fSPierre Pronchery 	msgsize = vsnprintf(NULL, 0, format, va);
353*a3cefe7fSPierre Pronchery 	va_end(va);
354*a3cefe7fSPierre Pronchery 
355*a3cefe7fSPierre Pronchery 	if (msgsize < 0)
356*a3cefe7fSPierre Pronchery 		return false;
357*a3cefe7fSPierre Pronchery 
358*a3cefe7fSPierre Pronchery 	msgsize++;
359*a3cefe7fSPierre Pronchery 
360*a3cefe7fSPierre Pronchery 	errbuf = calloc(1, msgsize);
361*a3cefe7fSPierre Pronchery 	if (errbuf == NULL)
362*a3cefe7fSPierre Pronchery 		return false;
363*a3cefe7fSPierre Pronchery 
364*a3cefe7fSPierre Pronchery 	va_start(va, format);
365*a3cefe7fSPierre Pronchery 	vsnprintf(errbuf, msgsize, format, va);
366*a3cefe7fSPierre Pronchery 	va_end(va);
367*a3cefe7fSPierre Pronchery 
368*a3cefe7fSPierre Pronchery 	ret = client->error_handler(errbuf, client, client->error_handler_data);
369*a3cefe7fSPierre Pronchery 	free(errbuf);
370*a3cefe7fSPierre Pronchery 
371*a3cefe7fSPierre Pronchery 	return ret;
372*a3cefe7fSPierre Pronchery }
373*a3cefe7fSPierre Pronchery 
374*a3cefe7fSPierre Pronchery /*
375*a3cefe7fSPierre Pronchery  * !doc
376*a3cefe7fSPierre Pronchery  *
377*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_warn(const pkgconf_client_t *client, const char *format, ...)
378*a3cefe7fSPierre Pronchery  *
379*a3cefe7fSPierre Pronchery  *    Report an error to a client-registered warn handler.
380*a3cefe7fSPierre Pronchery  *
381*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The pkgconf client object to report the error to.
382*a3cefe7fSPierre Pronchery  *    :param char* format: A printf-style format string to use for formatting the warning message.
383*a3cefe7fSPierre Pronchery  *    :return: true if the warn handler processed the message, else false.
384*a3cefe7fSPierre Pronchery  *    :rtype: bool
385*a3cefe7fSPierre Pronchery  */
386*a3cefe7fSPierre Pronchery bool
pkgconf_warn(const pkgconf_client_t * client,const char * format,...)387*a3cefe7fSPierre Pronchery pkgconf_warn(const pkgconf_client_t *client, const char *format, ...)
388*a3cefe7fSPierre Pronchery {
389*a3cefe7fSPierre Pronchery 	char *errbuf;
390*a3cefe7fSPierre Pronchery 	ssize_t msgsize = 0;
391*a3cefe7fSPierre Pronchery 	bool ret;
392*a3cefe7fSPierre Pronchery 	va_list va;
393*a3cefe7fSPierre Pronchery 
394*a3cefe7fSPierre Pronchery 	va_start(va, format);
395*a3cefe7fSPierre Pronchery 	msgsize = vsnprintf(NULL, 0, format, va);
396*a3cefe7fSPierre Pronchery 	va_end(va);
397*a3cefe7fSPierre Pronchery 
398*a3cefe7fSPierre Pronchery 	if (msgsize < 0)
399*a3cefe7fSPierre Pronchery 		return false;
400*a3cefe7fSPierre Pronchery 
401*a3cefe7fSPierre Pronchery 	msgsize++;
402*a3cefe7fSPierre Pronchery 
403*a3cefe7fSPierre Pronchery 	errbuf = calloc(1, msgsize);
404*a3cefe7fSPierre Pronchery 	if (errbuf == NULL)
405*a3cefe7fSPierre Pronchery 		return false;
406*a3cefe7fSPierre Pronchery 
407*a3cefe7fSPierre Pronchery 	va_start(va, format);
408*a3cefe7fSPierre Pronchery 	vsnprintf(errbuf, msgsize, format, va);
409*a3cefe7fSPierre Pronchery 	va_end(va);
410*a3cefe7fSPierre Pronchery 
411*a3cefe7fSPierre Pronchery 	ret = client->warn_handler(errbuf, client, client->warn_handler_data);
412*a3cefe7fSPierre Pronchery 	free(errbuf);
413*a3cefe7fSPierre Pronchery 
414*a3cefe7fSPierre Pronchery 	return ret;
415*a3cefe7fSPierre Pronchery }
416*a3cefe7fSPierre Pronchery 
417*a3cefe7fSPierre Pronchery /*
418*a3cefe7fSPierre Pronchery  * !doc
419*a3cefe7fSPierre Pronchery  *
420*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_trace(const pkgconf_client_t *client, const char *filename, size_t len, const char *funcname, const char *format, ...)
421*a3cefe7fSPierre Pronchery  *
422*a3cefe7fSPierre Pronchery  *    Report a message to a client-registered trace handler.
423*a3cefe7fSPierre Pronchery  *
424*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The pkgconf client object to report the trace message to.
425*a3cefe7fSPierre Pronchery  *    :param char* filename: The file the function is in.
426*a3cefe7fSPierre Pronchery  *    :param size_t lineno: The line number currently being executed.
427*a3cefe7fSPierre Pronchery  *    :param char* funcname: The function name to use.
428*a3cefe7fSPierre Pronchery  *    :param char* format: A printf-style format string to use for formatting the trace message.
429*a3cefe7fSPierre Pronchery  *    :return: true if the trace handler processed the message, else false.
430*a3cefe7fSPierre Pronchery  *    :rtype: bool
431*a3cefe7fSPierre Pronchery  */
432*a3cefe7fSPierre Pronchery bool
pkgconf_trace(const pkgconf_client_t * client,const char * filename,size_t lineno,const char * funcname,const char * format,...)433*a3cefe7fSPierre Pronchery pkgconf_trace(const pkgconf_client_t *client, const char *filename, size_t lineno, const char *funcname, const char *format, ...)
434*a3cefe7fSPierre Pronchery {
435*a3cefe7fSPierre Pronchery 	char prefix[PKGCONF_ITEM_SIZE];
436*a3cefe7fSPierre Pronchery 	char *errbuf = NULL;
437*a3cefe7fSPierre Pronchery 	ssize_t errlen;
438*a3cefe7fSPierre Pronchery 	char *finalbuf = NULL;
439*a3cefe7fSPierre Pronchery 	ssize_t finallen;
440*a3cefe7fSPierre Pronchery 	bool ret;
441*a3cefe7fSPierre Pronchery 	va_list va;
442*a3cefe7fSPierre Pronchery 
443*a3cefe7fSPierre Pronchery 	if (client == NULL || client->trace_handler == NULL)
444*a3cefe7fSPierre Pronchery 		return false;
445*a3cefe7fSPierre Pronchery 
446*a3cefe7fSPierre Pronchery 	snprintf(prefix, sizeof prefix, "%s:" SIZE_FMT_SPECIFIER " [%s]:", filename, lineno, funcname);
447*a3cefe7fSPierre Pronchery 
448*a3cefe7fSPierre Pronchery 	va_start(va, format);
449*a3cefe7fSPierre Pronchery 	errlen = vsnprintf(NULL, 0, format, va);
450*a3cefe7fSPierre Pronchery 	va_end(va);
451*a3cefe7fSPierre Pronchery 
452*a3cefe7fSPierre Pronchery 	if (errlen < 0)
453*a3cefe7fSPierre Pronchery 		return false;
454*a3cefe7fSPierre Pronchery 
455*a3cefe7fSPierre Pronchery 	errlen++;
456*a3cefe7fSPierre Pronchery 	errbuf = calloc(1, errlen);
457*a3cefe7fSPierre Pronchery 	if (errbuf == NULL)
458*a3cefe7fSPierre Pronchery 		return false;
459*a3cefe7fSPierre Pronchery 
460*a3cefe7fSPierre Pronchery 	va_start(va, format);
461*a3cefe7fSPierre Pronchery 	vsnprintf(errbuf, errlen, format, va);
462*a3cefe7fSPierre Pronchery 	va_end(va);
463*a3cefe7fSPierre Pronchery 
464*a3cefe7fSPierre Pronchery 	finallen = snprintf(NULL, 0, "%s %s\n", prefix, errbuf);
465*a3cefe7fSPierre Pronchery 	if (finallen < 0)
466*a3cefe7fSPierre Pronchery 		return false;
467*a3cefe7fSPierre Pronchery 
468*a3cefe7fSPierre Pronchery 	finallen++;
469*a3cefe7fSPierre Pronchery 	finalbuf = calloc(1, finallen);
470*a3cefe7fSPierre Pronchery 	if (finalbuf == NULL)
471*a3cefe7fSPierre Pronchery 		return false;
472*a3cefe7fSPierre Pronchery 
473*a3cefe7fSPierre Pronchery 	snprintf(finalbuf, finallen, "%s %s\n", prefix, errbuf);
474*a3cefe7fSPierre Pronchery 	ret = client->trace_handler(finalbuf, client, client->trace_handler_data);
475*a3cefe7fSPierre Pronchery 	free(errbuf);
476*a3cefe7fSPierre Pronchery 	free(finalbuf);
477*a3cefe7fSPierre Pronchery 
478*a3cefe7fSPierre Pronchery 	return ret;
479*a3cefe7fSPierre Pronchery }
480*a3cefe7fSPierre Pronchery 
481*a3cefe7fSPierre Pronchery /*
482*a3cefe7fSPierre Pronchery  * !doc
483*a3cefe7fSPierre Pronchery  *
484*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_default_error_handler(const char *msg, const pkgconf_client_t *client, const void *data)
485*a3cefe7fSPierre Pronchery  *
486*a3cefe7fSPierre Pronchery  *    The default pkgconf error handler.
487*a3cefe7fSPierre Pronchery  *
488*a3cefe7fSPierre Pronchery  *    :param char* msg: The error message to handle.
489*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object the error originated from.
490*a3cefe7fSPierre Pronchery  *    :param void* data: An opaque pointer to extra data associated with the client for error handling.
491*a3cefe7fSPierre Pronchery  *    :return: true (the function does nothing to process the message)
492*a3cefe7fSPierre Pronchery  *    :rtype: bool
493*a3cefe7fSPierre Pronchery  */
494*a3cefe7fSPierre Pronchery bool
pkgconf_default_error_handler(const char * msg,const pkgconf_client_t * client,void * data)495*a3cefe7fSPierre Pronchery pkgconf_default_error_handler(const char *msg, const pkgconf_client_t *client, void *data)
496*a3cefe7fSPierre Pronchery {
497*a3cefe7fSPierre Pronchery 	(void) msg;
498*a3cefe7fSPierre Pronchery 	(void) client;
499*a3cefe7fSPierre Pronchery 	(void) data;
500*a3cefe7fSPierre Pronchery 
501*a3cefe7fSPierre Pronchery 	return true;
502*a3cefe7fSPierre Pronchery }
503*a3cefe7fSPierre Pronchery 
504*a3cefe7fSPierre Pronchery static void
default_unveil_handler(const pkgconf_client_t * client,const char * path,const char * permissions)505*a3cefe7fSPierre Pronchery default_unveil_handler(const pkgconf_client_t *client, const char *path, const char *permissions)
506*a3cefe7fSPierre Pronchery {
507*a3cefe7fSPierre Pronchery 	(void) client;
508*a3cefe7fSPierre Pronchery 	(void) path;
509*a3cefe7fSPierre Pronchery 	(void) permissions;
510*a3cefe7fSPierre Pronchery }
511*a3cefe7fSPierre Pronchery 
512*a3cefe7fSPierre Pronchery /*
513*a3cefe7fSPierre Pronchery  * !doc
514*a3cefe7fSPierre Pronchery  *
515*a3cefe7fSPierre Pronchery  * .. c:function:: unsigned int pkgconf_client_get_flags(const pkgconf_client_t *client)
516*a3cefe7fSPierre Pronchery  *
517*a3cefe7fSPierre Pronchery  *    Retrieves resolver-specific flags associated with a client object.
518*a3cefe7fSPierre Pronchery  *
519*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to retrieve the resolver-specific flags from.
520*a3cefe7fSPierre Pronchery  *    :return: a bitfield of resolver-specific flags
521*a3cefe7fSPierre Pronchery  *    :rtype: uint
522*a3cefe7fSPierre Pronchery  */
523*a3cefe7fSPierre Pronchery unsigned int
pkgconf_client_get_flags(const pkgconf_client_t * client)524*a3cefe7fSPierre Pronchery pkgconf_client_get_flags(const pkgconf_client_t *client)
525*a3cefe7fSPierre Pronchery {
526*a3cefe7fSPierre Pronchery 	return client->flags;
527*a3cefe7fSPierre Pronchery }
528*a3cefe7fSPierre Pronchery 
529*a3cefe7fSPierre Pronchery /*
530*a3cefe7fSPierre Pronchery  * !doc
531*a3cefe7fSPierre Pronchery  *
532*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_set_flags(pkgconf_client_t *client, unsigned int flags)
533*a3cefe7fSPierre Pronchery  *
534*a3cefe7fSPierre Pronchery  *    Sets resolver-specific flags associated with a client object.
535*a3cefe7fSPierre Pronchery  *
536*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the resolver-specific flags on.
537*a3cefe7fSPierre Pronchery  *    :return: nothing
538*a3cefe7fSPierre Pronchery  */
539*a3cefe7fSPierre Pronchery void
pkgconf_client_set_flags(pkgconf_client_t * client,unsigned int flags)540*a3cefe7fSPierre Pronchery pkgconf_client_set_flags(pkgconf_client_t *client, unsigned int flags)
541*a3cefe7fSPierre Pronchery {
542*a3cefe7fSPierre Pronchery 	client->flags = flags;
543*a3cefe7fSPierre Pronchery }
544*a3cefe7fSPierre Pronchery 
545*a3cefe7fSPierre Pronchery /*
546*a3cefe7fSPierre Pronchery  * !doc
547*a3cefe7fSPierre Pronchery  *
548*a3cefe7fSPierre Pronchery  * .. c:function:: const char *pkgconf_client_get_prefix_varname(const pkgconf_client_t *client)
549*a3cefe7fSPierre Pronchery  *
550*a3cefe7fSPierre Pronchery  *    Retrieves the name of the variable that should contain a module's prefix.
551*a3cefe7fSPierre Pronchery  *    In some cases, it is necessary to override this variable to allow proper path relocation.
552*a3cefe7fSPierre Pronchery  *
553*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to retrieve the prefix variable name from.
554*a3cefe7fSPierre Pronchery  *    :return: the prefix variable name as a string
555*a3cefe7fSPierre Pronchery  *    :rtype: const char *
556*a3cefe7fSPierre Pronchery  */
557*a3cefe7fSPierre Pronchery const char *
pkgconf_client_get_prefix_varname(const pkgconf_client_t * client)558*a3cefe7fSPierre Pronchery pkgconf_client_get_prefix_varname(const pkgconf_client_t *client)
559*a3cefe7fSPierre Pronchery {
560*a3cefe7fSPierre Pronchery 	return client->prefix_varname;
561*a3cefe7fSPierre Pronchery }
562*a3cefe7fSPierre Pronchery 
563*a3cefe7fSPierre Pronchery /*
564*a3cefe7fSPierre Pronchery  * !doc
565*a3cefe7fSPierre Pronchery  *
566*a3cefe7fSPierre Pronchery  * .. c:function:: void pkgconf_client_set_prefix_varname(pkgconf_client_t *client, const char *prefix_varname)
567*a3cefe7fSPierre Pronchery  *
568*a3cefe7fSPierre Pronchery  *    Sets the name of the variable that should contain a module's prefix.
569*a3cefe7fSPierre Pronchery  *    If the variable name is ``NULL``, then the default variable name (``prefix``) is used.
570*a3cefe7fSPierre Pronchery  *
571*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the prefix variable name on.
572*a3cefe7fSPierre Pronchery  *    :param char* prefix_varname: The prefix variable name to set.
573*a3cefe7fSPierre Pronchery  *    :return: nothing
574*a3cefe7fSPierre Pronchery  */
575*a3cefe7fSPierre Pronchery void
pkgconf_client_set_prefix_varname(pkgconf_client_t * client,const char * prefix_varname)576*a3cefe7fSPierre Pronchery pkgconf_client_set_prefix_varname(pkgconf_client_t *client, const char *prefix_varname)
577*a3cefe7fSPierre Pronchery {
578*a3cefe7fSPierre Pronchery 	if (prefix_varname == NULL)
579*a3cefe7fSPierre Pronchery 		prefix_varname = "prefix";
580*a3cefe7fSPierre Pronchery 
581*a3cefe7fSPierre Pronchery 	if (client->prefix_varname != NULL)
582*a3cefe7fSPierre Pronchery 		free(client->prefix_varname);
583*a3cefe7fSPierre Pronchery 
584*a3cefe7fSPierre Pronchery 	client->prefix_varname = strdup(prefix_varname);
585*a3cefe7fSPierre Pronchery 
586*a3cefe7fSPierre Pronchery 	PKGCONF_TRACE(client, "set prefix_varname to: %s", client->prefix_varname);
587*a3cefe7fSPierre Pronchery }
588*a3cefe7fSPierre Pronchery 
589*a3cefe7fSPierre Pronchery /*
590*a3cefe7fSPierre Pronchery  * !doc
591*a3cefe7fSPierre Pronchery  *
592*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_get_warn_handler(const pkgconf_client_t *client)
593*a3cefe7fSPierre Pronchery  *
594*a3cefe7fSPierre Pronchery  *    Returns the warning handler if one is set, else ``NULL``.
595*a3cefe7fSPierre Pronchery  *
596*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to get the warn handler from.
597*a3cefe7fSPierre Pronchery  *    :return: a function pointer to the warn handler or ``NULL``
598*a3cefe7fSPierre Pronchery  */
599*a3cefe7fSPierre Pronchery pkgconf_error_handler_func_t
pkgconf_client_get_warn_handler(const pkgconf_client_t * client)600*a3cefe7fSPierre Pronchery pkgconf_client_get_warn_handler(const pkgconf_client_t *client)
601*a3cefe7fSPierre Pronchery {
602*a3cefe7fSPierre Pronchery 	return client->warn_handler;
603*a3cefe7fSPierre Pronchery }
604*a3cefe7fSPierre Pronchery 
605*a3cefe7fSPierre Pronchery /*
606*a3cefe7fSPierre Pronchery  * !doc
607*a3cefe7fSPierre Pronchery  *
608*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_set_warn_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t warn_handler, void *warn_handler_data)
609*a3cefe7fSPierre Pronchery  *
610*a3cefe7fSPierre Pronchery  *    Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
611*a3cefe7fSPierre Pronchery  *
612*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the warn handler on.
613*a3cefe7fSPierre Pronchery  *    :param pkgconf_error_handler_func_t warn_handler: The warn handler to set.
614*a3cefe7fSPierre Pronchery  *    :param void* warn_handler_data: Optional data to associate with the warn handler.
615*a3cefe7fSPierre Pronchery  *    :return: nothing
616*a3cefe7fSPierre Pronchery  */
617*a3cefe7fSPierre Pronchery void
pkgconf_client_set_warn_handler(pkgconf_client_t * client,pkgconf_error_handler_func_t warn_handler,void * warn_handler_data)618*a3cefe7fSPierre Pronchery pkgconf_client_set_warn_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t warn_handler, void *warn_handler_data)
619*a3cefe7fSPierre Pronchery {
620*a3cefe7fSPierre Pronchery 	client->warn_handler = warn_handler;
621*a3cefe7fSPierre Pronchery 	client->warn_handler_data = warn_handler_data;
622*a3cefe7fSPierre Pronchery 
623*a3cefe7fSPierre Pronchery 	if (client->warn_handler == NULL)
624*a3cefe7fSPierre Pronchery 	{
625*a3cefe7fSPierre Pronchery 		PKGCONF_TRACE(client, "installing default warn handler");
626*a3cefe7fSPierre Pronchery 		client->warn_handler = pkgconf_default_error_handler;
627*a3cefe7fSPierre Pronchery 	}
628*a3cefe7fSPierre Pronchery }
629*a3cefe7fSPierre Pronchery 
630*a3cefe7fSPierre Pronchery /*
631*a3cefe7fSPierre Pronchery  * !doc
632*a3cefe7fSPierre Pronchery  *
633*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_get_error_handler(const pkgconf_client_t *client)
634*a3cefe7fSPierre Pronchery  *
635*a3cefe7fSPierre Pronchery  *    Returns the error handler if one is set, else ``NULL``.
636*a3cefe7fSPierre Pronchery  *
637*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to get the error handler from.
638*a3cefe7fSPierre Pronchery  *    :return: a function pointer to the error handler or ``NULL``
639*a3cefe7fSPierre Pronchery  */
640*a3cefe7fSPierre Pronchery pkgconf_error_handler_func_t
pkgconf_client_get_error_handler(const pkgconf_client_t * client)641*a3cefe7fSPierre Pronchery pkgconf_client_get_error_handler(const pkgconf_client_t *client)
642*a3cefe7fSPierre Pronchery {
643*a3cefe7fSPierre Pronchery 	return client->error_handler;
644*a3cefe7fSPierre Pronchery }
645*a3cefe7fSPierre Pronchery 
646*a3cefe7fSPierre Pronchery /*
647*a3cefe7fSPierre Pronchery  * !doc
648*a3cefe7fSPierre Pronchery  *
649*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_set_error_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data)
650*a3cefe7fSPierre Pronchery  *
651*a3cefe7fSPierre Pronchery  *    Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
652*a3cefe7fSPierre Pronchery  *
653*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the error handler on.
654*a3cefe7fSPierre Pronchery  *    :param pkgconf_error_handler_func_t error_handler: The error handler to set.
655*a3cefe7fSPierre Pronchery  *    :param void* error_handler_data: Optional data to associate with the error handler.
656*a3cefe7fSPierre Pronchery  *    :return: nothing
657*a3cefe7fSPierre Pronchery  */
658*a3cefe7fSPierre Pronchery void
pkgconf_client_set_error_handler(pkgconf_client_t * client,pkgconf_error_handler_func_t error_handler,void * error_handler_data)659*a3cefe7fSPierre Pronchery pkgconf_client_set_error_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t error_handler, void *error_handler_data)
660*a3cefe7fSPierre Pronchery {
661*a3cefe7fSPierre Pronchery 	client->error_handler = error_handler;
662*a3cefe7fSPierre Pronchery 	client->error_handler_data = error_handler_data;
663*a3cefe7fSPierre Pronchery 
664*a3cefe7fSPierre Pronchery 	if (client->error_handler == NULL)
665*a3cefe7fSPierre Pronchery 	{
666*a3cefe7fSPierre Pronchery 		PKGCONF_TRACE(client, "installing default error handler");
667*a3cefe7fSPierre Pronchery 		client->error_handler = pkgconf_default_error_handler;
668*a3cefe7fSPierre Pronchery 	}
669*a3cefe7fSPierre Pronchery }
670*a3cefe7fSPierre Pronchery 
671*a3cefe7fSPierre Pronchery /*
672*a3cefe7fSPierre Pronchery  * !doc
673*a3cefe7fSPierre Pronchery  *
674*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_get_unveil_handler(const pkgconf_client_t *client)
675*a3cefe7fSPierre Pronchery  *
676*a3cefe7fSPierre Pronchery  *    Returns the unveil handler if one is set, else ``NULL``.
677*a3cefe7fSPierre Pronchery  *
678*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to get the unveil handler from.
679*a3cefe7fSPierre Pronchery  *    :return: a function pointer to the error handler or ``NULL``
680*a3cefe7fSPierre Pronchery  */
681*a3cefe7fSPierre Pronchery pkgconf_unveil_handler_func_t
pkgconf_client_get_unveil_handler(const pkgconf_client_t * client)682*a3cefe7fSPierre Pronchery pkgconf_client_get_unveil_handler(const pkgconf_client_t *client)
683*a3cefe7fSPierre Pronchery {
684*a3cefe7fSPierre Pronchery 	return client->unveil_handler;
685*a3cefe7fSPierre Pronchery }
686*a3cefe7fSPierre Pronchery 
687*a3cefe7fSPierre Pronchery /*
688*a3cefe7fSPierre Pronchery  * !doc
689*a3cefe7fSPierre Pronchery  *
690*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_set_unveil_handler(pkgconf_client_t *client, pkgconf_unveil_handler_func_t unveil_handler)
691*a3cefe7fSPierre Pronchery  *
692*a3cefe7fSPierre Pronchery  *    Sets an unveil handler on a client object or uninstalls one if set to ``NULL``.
693*a3cefe7fSPierre Pronchery  *
694*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the error handler on.
695*a3cefe7fSPierre Pronchery  *    :param pkgconf_unveil_handler_func_t unveil_handler: The unveil handler to set.
696*a3cefe7fSPierre Pronchery  *    :return: nothing
697*a3cefe7fSPierre Pronchery  */
698*a3cefe7fSPierre Pronchery void
pkgconf_client_set_unveil_handler(pkgconf_client_t * client,pkgconf_unveil_handler_func_t unveil_handler)699*a3cefe7fSPierre Pronchery pkgconf_client_set_unveil_handler(pkgconf_client_t *client, pkgconf_unveil_handler_func_t unveil_handler)
700*a3cefe7fSPierre Pronchery {
701*a3cefe7fSPierre Pronchery 	client->unveil_handler = unveil_handler;
702*a3cefe7fSPierre Pronchery 
703*a3cefe7fSPierre Pronchery 	if (client->unveil_handler == NULL)
704*a3cefe7fSPierre Pronchery 	{
705*a3cefe7fSPierre Pronchery 		PKGCONF_TRACE(client, "installing default unveil handler");
706*a3cefe7fSPierre Pronchery 		client->unveil_handler = default_unveil_handler;
707*a3cefe7fSPierre Pronchery 	}
708*a3cefe7fSPierre Pronchery }
709*a3cefe7fSPierre Pronchery 
710*a3cefe7fSPierre Pronchery #ifndef PKGCONF_LITE
711*a3cefe7fSPierre Pronchery /*
712*a3cefe7fSPierre Pronchery  * !doc
713*a3cefe7fSPierre Pronchery  *
714*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_get_trace_handler(const pkgconf_client_t *client)
715*a3cefe7fSPierre Pronchery  *
716*a3cefe7fSPierre Pronchery  *    Returns the error handler if one is set, else ``NULL``.
717*a3cefe7fSPierre Pronchery  *
718*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to get the error handler from.
719*a3cefe7fSPierre Pronchery  *    :return: a function pointer to the error handler or ``NULL``
720*a3cefe7fSPierre Pronchery  */
721*a3cefe7fSPierre Pronchery pkgconf_error_handler_func_t
pkgconf_client_get_trace_handler(const pkgconf_client_t * client)722*a3cefe7fSPierre Pronchery pkgconf_client_get_trace_handler(const pkgconf_client_t *client)
723*a3cefe7fSPierre Pronchery {
724*a3cefe7fSPierre Pronchery 	return client->trace_handler;
725*a3cefe7fSPierre Pronchery }
726*a3cefe7fSPierre Pronchery 
727*a3cefe7fSPierre Pronchery /*
728*a3cefe7fSPierre Pronchery  * !doc
729*a3cefe7fSPierre Pronchery  *
730*a3cefe7fSPierre Pronchery  * .. c:function:: pkgconf_client_set_trace_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t trace_handler, void *trace_handler_data)
731*a3cefe7fSPierre Pronchery  *
732*a3cefe7fSPierre Pronchery  *    Sets a warn handler on a client object or uninstalls one if set to ``NULL``.
733*a3cefe7fSPierre Pronchery  *
734*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object to set the error handler on.
735*a3cefe7fSPierre Pronchery  *    :param pkgconf_error_handler_func_t trace_handler: The error handler to set.
736*a3cefe7fSPierre Pronchery  *    :param void* trace_handler_data: Optional data to associate with the error handler.
737*a3cefe7fSPierre Pronchery  *    :return: nothing
738*a3cefe7fSPierre Pronchery  */
739*a3cefe7fSPierre Pronchery void
pkgconf_client_set_trace_handler(pkgconf_client_t * client,pkgconf_error_handler_func_t trace_handler,void * trace_handler_data)740*a3cefe7fSPierre Pronchery pkgconf_client_set_trace_handler(pkgconf_client_t *client, pkgconf_error_handler_func_t trace_handler, void *trace_handler_data)
741*a3cefe7fSPierre Pronchery {
742*a3cefe7fSPierre Pronchery 	client->trace_handler = trace_handler;
743*a3cefe7fSPierre Pronchery 	client->trace_handler_data = trace_handler_data;
744*a3cefe7fSPierre Pronchery 
745*a3cefe7fSPierre Pronchery 	if (client->trace_handler == NULL)
746*a3cefe7fSPierre Pronchery 	{
747*a3cefe7fSPierre Pronchery 		client->trace_handler = pkgconf_default_error_handler;
748*a3cefe7fSPierre Pronchery 		PKGCONF_TRACE(client, "installing default trace handler");
749*a3cefe7fSPierre Pronchery 	}
750*a3cefe7fSPierre Pronchery }
751*a3cefe7fSPierre Pronchery #endif
752*a3cefe7fSPierre Pronchery 
753*a3cefe7fSPierre Pronchery /*
754*a3cefe7fSPierre Pronchery  * !doc
755*a3cefe7fSPierre Pronchery  *
756*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_client_preload_path(pkgconf_client_t *client, const char *path)
757*a3cefe7fSPierre Pronchery  *
758*a3cefe7fSPierre Pronchery  *    Loads a pkg-config file into the preloaded packages set.
759*a3cefe7fSPierre Pronchery  *
760*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object for preloading.
761*a3cefe7fSPierre Pronchery  *    :param char* path: The path to the pkg-config file to preload.
762*a3cefe7fSPierre Pronchery  *    :return: true on success, false on error
763*a3cefe7fSPierre Pronchery  *    :rtype: bool
764*a3cefe7fSPierre Pronchery  */
765*a3cefe7fSPierre Pronchery bool
pkgconf_client_preload_path(pkgconf_client_t * client,const char * path)766*a3cefe7fSPierre Pronchery pkgconf_client_preload_path(pkgconf_client_t *client, const char *path)
767*a3cefe7fSPierre Pronchery {
768*a3cefe7fSPierre Pronchery 	pkgconf_pkg_t *pkg = pkgconf_pkg_new_from_path(client, path, PKGCONF_PKG_PROPF_PRELOADED);
769*a3cefe7fSPierre Pronchery 	if (pkg == NULL)
770*a3cefe7fSPierre Pronchery 		return false;
771*a3cefe7fSPierre Pronchery 
772*a3cefe7fSPierre Pronchery 	pkgconf_pkg_ref(client, pkg);
773*a3cefe7fSPierre Pronchery 	pkgconf_node_insert_tail(&pkg->preload_node, pkg, &client->preloaded_pkgs);
774*a3cefe7fSPierre Pronchery 
775*a3cefe7fSPierre Pronchery 	return true;
776*a3cefe7fSPierre Pronchery }
777*a3cefe7fSPierre Pronchery 
778*a3cefe7fSPierre Pronchery /*
779*a3cefe7fSPierre Pronchery  * !doc
780*a3cefe7fSPierre Pronchery  *
781*a3cefe7fSPierre Pronchery  * .. c:function:: bool pkgconf_client_preload_from_environ(pkgconf_client_t *client, const char *env)
782*a3cefe7fSPierre Pronchery  *
783*a3cefe7fSPierre Pronchery  *    Loads zero or more pkg-config files specified in the given environmental
784*a3cefe7fSPierre Pronchery  *    variable.
785*a3cefe7fSPierre Pronchery  *
786*a3cefe7fSPierre Pronchery  *    :param pkgconf_client_t* client: The client object for preloading.
787*a3cefe7fSPierre Pronchery  *    :param char* environ: The environment variable to use for preloading.
788*a3cefe7fSPierre Pronchery  *    :return: true on success, false on error
789*a3cefe7fSPierre Pronchery  *    :rtype: bool
790*a3cefe7fSPierre Pronchery  */
791*a3cefe7fSPierre Pronchery bool
pkgconf_client_preload_from_environ(pkgconf_client_t * client,const char * env)792*a3cefe7fSPierre Pronchery pkgconf_client_preload_from_environ(pkgconf_client_t *client, const char *env)
793*a3cefe7fSPierre Pronchery {
794*a3cefe7fSPierre Pronchery 	const char *data;
795*a3cefe7fSPierre Pronchery 	pkgconf_list_t pathlist = PKGCONF_LIST_INITIALIZER;
796*a3cefe7fSPierre Pronchery 	pkgconf_node_t *n;
797*a3cefe7fSPierre Pronchery 	bool ret;
798*a3cefe7fSPierre Pronchery 
799*a3cefe7fSPierre Pronchery 	data = getenv(env);
800*a3cefe7fSPierre Pronchery 	if (data == NULL)
801*a3cefe7fSPierre Pronchery 		return true;
802*a3cefe7fSPierre Pronchery 
803*a3cefe7fSPierre Pronchery 	pkgconf_path_split(data, &pathlist, true);
804*a3cefe7fSPierre Pronchery 
805*a3cefe7fSPierre Pronchery 	PKGCONF_FOREACH_LIST_ENTRY(pathlist.head, n)
806*a3cefe7fSPierre Pronchery 	{
807*a3cefe7fSPierre Pronchery 		pkgconf_path_t *pn = n->data;
808*a3cefe7fSPierre Pronchery 
809*a3cefe7fSPierre Pronchery 		ret = pkgconf_client_preload_path(client, pn->path);
810*a3cefe7fSPierre Pronchery 		if (!ret)
811*a3cefe7fSPierre Pronchery 			break;
812*a3cefe7fSPierre Pronchery 	}
813*a3cefe7fSPierre Pronchery 
814*a3cefe7fSPierre Pronchery 	pkgconf_path_free(&pathlist);
815*a3cefe7fSPierre Pronchery 
816*a3cefe7fSPierre Pronchery 	return ret;
817*a3cefe7fSPierre Pronchery }
818