Home | History | Annotate | Line # | Download | only in libprop
prop_object_impl.h revision 1.38
      1  1.38   thorpej /*	$NetBSD: prop_object_impl.h,v 1.38 2025/04/26 17:13:23 thorpej Exp $	*/
      2   1.1   thorpej 
      3   1.1   thorpej /*-
      4  1.37   thorpej  * Copyright (c) 2006, 2020, 2025 The NetBSD Foundation, Inc.
      5   1.1   thorpej  * All rights reserved.
      6   1.1   thorpej  *
      7   1.1   thorpej  * This code is derived from software contributed to The NetBSD Foundation
      8   1.1   thorpej  * by Jason R. Thorpe.
      9   1.1   thorpej  *
     10   1.1   thorpej  * Redistribution and use in source and binary forms, with or without
     11   1.1   thorpej  * modification, are permitted provided that the following conditions
     12   1.1   thorpej  * are met:
     13   1.1   thorpej  * 1. Redistributions of source code must retain the above copyright
     14   1.1   thorpej  *    notice, this list of conditions and the following disclaimer.
     15   1.1   thorpej  * 2. Redistributions in binary form must reproduce the above copyright
     16   1.1   thorpej  *    notice, this list of conditions and the following disclaimer in the
     17   1.1   thorpej  *    documentation and/or other materials provided with the distribution.
     18   1.1   thorpej  *
     19   1.1   thorpej  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     20   1.1   thorpej  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     21   1.1   thorpej  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     22   1.1   thorpej  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     23   1.1   thorpej  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     24   1.1   thorpej  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     25   1.1   thorpej  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     26   1.1   thorpej  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     27   1.1   thorpej  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     28   1.1   thorpej  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     29   1.1   thorpej  * POSSIBILITY OF SUCH DAMAGE.
     30   1.1   thorpej  */
     31   1.1   thorpej 
     32   1.1   thorpej #ifndef _PROPLIB_PROP_OBJECT_IMPL_H_
     33   1.1   thorpej #define	_PROPLIB_PROP_OBJECT_IMPL_H_
     34   1.1   thorpej 
     35  1.33   thorpej #if defined(HAVE_NBTOOL_CONFIG_H)
     36  1.33   thorpej #include "nbtool_config.h"
     37  1.33   thorpej #endif
     38  1.33   thorpej 
     39   1.5   thorpej #if defined(_KERNEL) || defined(_STANDALONE)
     40   1.5   thorpej #include <lib/libkern/libkern.h>
     41   1.5   thorpej #else
     42   1.5   thorpej #include <inttypes.h>
     43   1.5   thorpej #endif
     44   1.5   thorpej 
     45  1.17     joerg #include "prop_stack.h"
     46  1.17     joerg 
     47  1.37   thorpej struct _prop_object;
     48  1.37   thorpej 
     49   1.1   thorpej struct _prop_object_externalize_context {
     50   1.1   thorpej 	char *		poec_buf;		/* string buffer */
     51   1.1   thorpej 	size_t		poec_capacity;		/* capacity of buffer */
     52   1.1   thorpej 	size_t		poec_len;		/* current length of string */
     53   1.1   thorpej 	unsigned int	poec_depth;		/* nesting depth */
     54  1.37   thorpej 	prop_format_t	poec_format;		/* output format */
     55  1.37   thorpej };
     56  1.37   thorpej 
     57  1.37   thorpej struct _prop_object_type_tags {
     58  1.37   thorpej 	const char	*xml_tag;
     59  1.37   thorpej 	const char	*json_open_tag;
     60  1.37   thorpej 	const char	*json_close_tag;
     61  1.37   thorpej 	const char	*json_empty_sep;
     62   1.1   thorpej };
     63   1.1   thorpej 
     64  1.37   thorpej bool		_prop_object_externalize_start_line(
     65  1.37   thorpej 				struct _prop_object_externalize_context *);
     66  1.37   thorpej bool		_prop_object_externalize_end_line(
     67  1.37   thorpej 				struct _prop_object_externalize_context *,
     68  1.37   thorpej 				const char *);
     69  1.27   thorpej bool		_prop_object_externalize_start_tag(
     70   1.1   thorpej 				struct _prop_object_externalize_context *,
     71  1.37   thorpej 				const struct _prop_object_type_tags *,
     72   1.1   thorpej 				const char *);
     73  1.27   thorpej bool		_prop_object_externalize_end_tag(
     74   1.1   thorpej 				struct _prop_object_externalize_context *,
     75  1.37   thorpej 				const struct _prop_object_type_tags *);
     76  1.27   thorpej bool		_prop_object_externalize_empty_tag(
     77   1.1   thorpej 				struct _prop_object_externalize_context *,
     78  1.37   thorpej 				const struct _prop_object_type_tags *);
     79  1.27   thorpej bool		_prop_object_externalize_append_cstring(
     80   1.1   thorpej 				struct _prop_object_externalize_context *,
     81   1.1   thorpej 				const char *);
     82  1.27   thorpej bool		_prop_object_externalize_append_encoded_cstring(
     83   1.1   thorpej 				struct _prop_object_externalize_context *,
     84   1.1   thorpej 				const char *);
     85  1.27   thorpej bool		_prop_object_externalize_append_char(
     86   1.1   thorpej 				struct _prop_object_externalize_context *,
     87   1.1   thorpej 				unsigned char);
     88  1.27   thorpej bool		_prop_object_externalize_header(
     89   1.4   thorpej 				struct _prop_object_externalize_context *);
     90  1.27   thorpej bool		_prop_object_externalize_footer(
     91   1.4   thorpej 				struct _prop_object_externalize_context *);
     92   1.1   thorpej 
     93  1.37   thorpej bool		_prop_object_externalize_to_file(struct _prop_object *,
     94  1.37   thorpej 				const char *, prop_format_t);
     95  1.37   thorpej 
     96  1.37   thorpej 
     97   1.1   thorpej struct _prop_object_externalize_context *
     98  1.37   thorpej 	_prop_object_externalize_context_alloc(prop_format_t);
     99  1.27   thorpej void	_prop_object_externalize_context_free(
    100   1.1   thorpej 				struct _prop_object_externalize_context *);
    101  1.37   thorpej char	*_prop_object_externalize(struct _prop_object *, prop_format_t fmt);
    102   1.1   thorpej 
    103   1.1   thorpej typedef enum {
    104   1.1   thorpej 	_PROP_TAG_TYPE_START,			/* e.g. <dict> */
    105   1.1   thorpej 	_PROP_TAG_TYPE_END,			/* e.g. </dict> */
    106   1.1   thorpej 	_PROP_TAG_TYPE_EITHER
    107   1.1   thorpej } _prop_tag_type_t;
    108   1.1   thorpej 
    109   1.1   thorpej struct _prop_object_internalize_context {
    110  1.37   thorpej 	prop_format_t poic_format;
    111  1.37   thorpej 
    112  1.37   thorpej 	const char *poic_data;
    113   1.1   thorpej 	const char *poic_cp;
    114   1.1   thorpej 
    115   1.1   thorpej 	const char *poic_tag_start;
    116   1.1   thorpej 
    117   1.1   thorpej 	const char *poic_tagname;
    118   1.1   thorpej 	size_t      poic_tagname_len;
    119   1.1   thorpej 	const char *poic_tagattr;
    120   1.1   thorpej 	size_t      poic_tagattr_len;
    121   1.1   thorpej 	const char *poic_tagattrval;
    122   1.1   thorpej 	size_t      poic_tagattrval_len;
    123   1.1   thorpej 
    124  1.16   thorpej 	bool   poic_is_empty_element;
    125   1.1   thorpej 	_prop_tag_type_t poic_tag_type;
    126   1.1   thorpej };
    127   1.1   thorpej 
    128  1.27   thorpej typedef enum {
    129  1.17     joerg 	_PROP_OBJECT_FREE_DONE,
    130  1.17     joerg 	_PROP_OBJECT_FREE_RECURSE,
    131  1.17     joerg 	_PROP_OBJECT_FREE_FAILED
    132  1.27   thorpej } _prop_object_free_rv_t;
    133  1.17     joerg 
    134  1.27   thorpej typedef enum {
    135  1.18     joerg 	_PROP_OBJECT_EQUALS_FALSE,
    136  1.18     joerg 	_PROP_OBJECT_EQUALS_TRUE,
    137  1.18     joerg 	_PROP_OBJECT_EQUALS_RECURSE
    138  1.27   thorpej } _prop_object_equals_rv_t;
    139  1.18     joerg 
    140   1.1   thorpej #define	_PROP_EOF(c)		((c) == '\0')
    141   1.1   thorpej #define	_PROP_ISSPACE(c)	\
    142  1.32  christos 	((c) == ' ' || (c) == '\t' || (c) == '\n' || (c) == '\r')
    143   1.1   thorpej 
    144   1.1   thorpej #define	_PROP_TAG_MATCH(ctx, t)					\
    145   1.1   thorpej 	_prop_object_internalize_match((ctx)->poic_tagname,	\
    146   1.1   thorpej 				       (ctx)->poic_tagname_len,	\
    147   1.1   thorpej 				       (t), strlen(t))
    148   1.1   thorpej 
    149   1.1   thorpej #define	_PROP_TAGATTR_MATCH(ctx, a)				\
    150   1.1   thorpej 	_prop_object_internalize_match((ctx)->poic_tagattr,	\
    151   1.1   thorpej 				       (ctx)->poic_tagattr_len,	\
    152   1.1   thorpej 				       (a), strlen(a))
    153   1.1   thorpej 
    154   1.1   thorpej #define	_PROP_TAGATTRVAL_MATCH(ctx, a)				  \
    155   1.1   thorpej 	_prop_object_internalize_match((ctx)->poic_tagattrval,	  \
    156   1.1   thorpej 				       (ctx)->poic_tagattrval_len,\
    157   1.1   thorpej 				       (a), strlen(a))
    158   1.1   thorpej 
    159  1.27   thorpej bool		_prop_object_internalize_find_tag(
    160   1.1   thorpej 				struct _prop_object_internalize_context *,
    161   1.1   thorpej 				const char *, _prop_tag_type_t);
    162  1.27   thorpej bool		_prop_object_internalize_match(const char *, size_t,
    163   1.1   thorpej 					       const char *, size_t);
    164  1.27   thorpej bool		_prop_object_internalize_decode_string(
    165   1.1   thorpej 				struct _prop_object_internalize_context *,
    166   1.1   thorpej 				char *, size_t, size_t *, const char **);
    167  1.37   thorpej const char *	_prop_object_internalize_skip_whitespace(const char *);
    168  1.37   thorpej prop_object_t	_prop_object_internalize(const char *,
    169  1.37   thorpej 				const struct _prop_object_type_tags *);
    170  1.37   thorpej prop_object_t	_prop_object_internalize_from_file(const char *,
    171  1.37   thorpej 				const struct _prop_object_type_tags *);
    172   1.1   thorpej 
    173   1.1   thorpej struct _prop_object_internalize_context *
    174  1.37   thorpej 		_prop_object_internalize_context_alloc(const char *,
    175  1.37   thorpej 				prop_format_t);
    176   1.1   thorpej void		_prop_object_internalize_context_free(
    177   1.1   thorpej 				struct _prop_object_internalize_context *);
    178   1.1   thorpej 
    179  1.27   thorpej typedef bool (*prop_object_internalizer_t)(prop_stack_t, prop_object_t *,
    180  1.27   thorpej 				struct _prop_object_internalize_context *);
    181  1.27   thorpej typedef bool (*prop_object_internalizer_continue_t)(prop_stack_t,
    182  1.27   thorpej 				prop_object_t *,
    183  1.27   thorpej 				struct _prop_object_internalize_context *,
    184  1.27   thorpej 				void *, prop_object_t);
    185  1.17     joerg 
    186   1.1   thorpej 	/* These are here because they're required by shared code. */
    187  1.27   thorpej bool		_prop_array_internalize(prop_stack_t, prop_object_t *,
    188  1.27   thorpej 				struct _prop_object_internalize_context *);
    189  1.27   thorpej bool		_prop_bool_internalize(prop_stack_t, prop_object_t *,
    190  1.27   thorpej 				struct _prop_object_internalize_context *);
    191  1.27   thorpej bool		_prop_data_internalize(prop_stack_t, prop_object_t *,
    192  1.27   thorpej 				struct _prop_object_internalize_context *);
    193  1.27   thorpej bool		_prop_dictionary_internalize(prop_stack_t, prop_object_t *,
    194  1.27   thorpej 				struct _prop_object_internalize_context *);
    195  1.27   thorpej bool		_prop_number_internalize(prop_stack_t, prop_object_t *,
    196  1.27   thorpej 				struct _prop_object_internalize_context *);
    197  1.27   thorpej bool		_prop_string_internalize(prop_stack_t, prop_object_t *,
    198  1.27   thorpej 				struct _prop_object_internalize_context *);
    199   1.1   thorpej 
    200  1.37   thorpej bool		_prop_string_externalize_internal(
    201  1.37   thorpej 				struct _prop_object_externalize_context *,
    202  1.37   thorpej 				const struct _prop_object_type_tags *,
    203  1.37   thorpej 				const char *);
    204  1.37   thorpej 
    205   1.2   thorpej struct _prop_object_type {
    206  1.17     joerg 	/* type indicator */
    207  1.17     joerg 	uint32_t	pot_type;
    208  1.17     joerg 	/* func to free object */
    209  1.27   thorpej 	_prop_object_free_rv_t
    210  1.27   thorpej 			(*pot_free)(prop_stack_t, prop_object_t *);
    211  1.17     joerg 	/*
    212  1.17     joerg 	 * func to free the child returned by pot_free with stack == NULL.
    213  1.17     joerg 	 *
    214  1.18     joerg 	 * Must be implemented if pot_free can return anything other than
    215  1.18     joerg 	 * _PROP_OBJECT_FREE_DONE.
    216  1.17     joerg 	 */
    217  1.17     joerg 	void	(*pot_emergency_free)(prop_object_t);
    218  1.17     joerg 	/* func to externalize object */
    219  1.17     joerg 	bool	(*pot_extern)(struct _prop_object_externalize_context *,
    220  1.17     joerg 			      void *);
    221  1.17     joerg 	/* func to test quality */
    222  1.27   thorpej 	_prop_object_equals_rv_t
    223  1.27   thorpej 		(*pot_equals)(prop_object_t, prop_object_t,
    224  1.18     joerg 			      void **, void **,
    225  1.18     joerg 			      prop_object_t *, prop_object_t *);
    226  1.18     joerg 	/*
    227  1.18     joerg 	 * func to finish equality iteration.
    228  1.18     joerg 	 *
    229  1.18     joerg 	 * Must be implemented if pot_equals can return
    230  1.18     joerg 	 * _PROP_OBJECT_EQUALS_RECURSE
    231  1.18     joerg 	 */
    232  1.18     joerg 	void	(*pot_equals_finish)(prop_object_t, prop_object_t);
    233  1.28      haad 	void    (*pot_lock)(void);
    234  1.28      haad 	void    (*pot_unlock)(void);
    235   1.2   thorpej };
    236   1.2   thorpej 
    237   1.1   thorpej struct _prop_object {
    238   1.2   thorpej 	const struct _prop_object_type *po_type;/* type descriptor */
    239   1.1   thorpej 	uint32_t	po_refcnt;		/* reference count */
    240   1.1   thorpej };
    241   1.1   thorpej 
    242  1.27   thorpej void		_prop_object_init(struct _prop_object *,
    243  1.27   thorpej 				  const struct _prop_object_type *);
    244  1.27   thorpej void		_prop_object_fini(struct _prop_object *);
    245   1.1   thorpej 
    246   1.1   thorpej struct _prop_object_iterator {
    247   1.1   thorpej 	prop_object_t	(*pi_next_object)(void *);
    248   1.1   thorpej 	void		(*pi_reset)(void *);
    249   1.1   thorpej 	prop_object_t	pi_obj;
    250   1.1   thorpej 	uint32_t	pi_version;
    251   1.1   thorpej };
    252   1.1   thorpej 
    253  1.29     pooka #define _PROP_NOTHREAD_ONCE_DECL(x)	static bool x = false;
    254  1.29     pooka #define _PROP_NOTHREAD_ONCE_RUN(x,f)					\
    255  1.29     pooka 	do {								\
    256  1.29     pooka 		if ((x) == false) {					\
    257  1.29     pooka 			f();						\
    258  1.29     pooka 			x = true;					\
    259  1.29     pooka 		}							\
    260  1.29     pooka 	} while (/*CONSTCOND*/0)
    261  1.29     pooka 
    262   1.1   thorpej #if defined(_KERNEL)
    263   1.1   thorpej 
    264   1.1   thorpej /*
    265   1.1   thorpej  * proplib in the kernel...
    266   1.1   thorpej  */
    267   1.1   thorpej 
    268   1.6   thorpej #include <sys/param.h>
    269   1.1   thorpej #include <sys/malloc.h>
    270   1.1   thorpej #include <sys/pool.h>
    271   1.1   thorpej #include <sys/systm.h>
    272  1.15        ad #include <sys/rwlock.h>
    273  1.29     pooka #include <sys/once.h>
    274   1.1   thorpej 
    275  1.27   thorpej #define	_PROP_ASSERT(x)			KASSERT(x)
    276   1.1   thorpej 
    277  1.27   thorpej #define	_PROP_MALLOC(s, t)		malloc((s), (t), M_WAITOK)
    278  1.27   thorpej #define	_PROP_CALLOC(s, t)		malloc((s), (t), M_WAITOK | M_ZERO)
    279  1.27   thorpej #define	_PROP_REALLOC(v, s, t)		realloc((v), (s), (t), M_WAITOK)
    280  1.27   thorpej #define	_PROP_FREE(v, t)		free((v), (t))
    281   1.1   thorpej 
    282  1.27   thorpej #define	_PROP_POOL_GET(p)		pool_get(&(p), PR_WAITOK)
    283  1.27   thorpej #define	_PROP_POOL_PUT(p, v)		pool_put(&(p), (v))
    284   1.1   thorpej 
    285  1.30     pooka struct prop_pool_init {
    286  1.30     pooka 	struct pool *pp;
    287  1.30     pooka 	size_t size;
    288  1.30     pooka 	const char *wchan;
    289  1.30     pooka };
    290  1.30     pooka #define	_PROP_POOL_INIT(pp, size, wchan)				\
    291  1.30     pooka struct pool pp;								\
    292  1.30     pooka static const struct prop_pool_init _link_ ## pp[1] = {			\
    293  1.30     pooka 	{ &pp, size, wchan }						\
    294  1.30     pooka };									\
    295  1.30     pooka __link_set_add_rodata(prop_linkpools, _link_ ## pp);
    296   1.1   thorpej 
    297   1.1   thorpej #define	_PROP_MALLOC_DEFINE(t, s, l)					\
    298   1.1   thorpej 		MALLOC_DEFINE(t, s, l);
    299   1.1   thorpej 
    300  1.29     pooka #define	_PROP_MUTEX_DECL_STATIC(x)	static kmutex_t x;
    301  1.29     pooka #define	_PROP_MUTEX_INIT(x)		mutex_init(&(x),MUTEX_DEFAULT,IPL_NONE)
    302  1.29     pooka #define	_PROP_MUTEX_LOCK(x)		mutex_enter(&(x))
    303  1.29     pooka #define	_PROP_MUTEX_UNLOCK(x)		mutex_exit(&(x))
    304   1.3   thorpej 
    305  1.27   thorpej #define	_PROP_RWLOCK_DECL(x)		krwlock_t x ;
    306  1.27   thorpej #define	_PROP_RWLOCK_INIT(x)		rw_init(&(x))
    307  1.27   thorpej #define	_PROP_RWLOCK_RDLOCK(x)		rw_enter(&(x), RW_READER)
    308  1.27   thorpej #define	_PROP_RWLOCK_WRLOCK(x)		rw_enter(&(x), RW_WRITER)
    309  1.27   thorpej #define	_PROP_RWLOCK_UNLOCK(x)		rw_exit(&(x))
    310  1.27   thorpej #define	_PROP_RWLOCK_DESTROY(x)		rw_destroy(&(x))
    311  1.22   xtraeme 
    312  1.29     pooka #define _PROP_ONCE_DECL(x)		static ONCE_DECL(x);
    313  1.29     pooka #define _PROP_ONCE_RUN(x,f)		RUN_ONCE(&(x), f)
    314  1.29     pooka 
    315  1.31     pooka #include <sys/atomic.h>
    316  1.31     pooka 
    317  1.34   thorpej #define	_PROP_ATOMIC_LOAD(x)		atomic_load_relaxed(x)
    318  1.31     pooka #define _PROP_ATOMIC_INC32(x)		atomic_inc_32(x)
    319  1.31     pooka #define _PROP_ATOMIC_DEC32(x)		atomic_dec_32(x)
    320  1.31     pooka #define _PROP_ATOMIC_INC32_NV(x, v)	v = atomic_inc_32_nv(x)
    321  1.31     pooka #define _PROP_ATOMIC_DEC32_NV(x, v)	v = atomic_dec_32_nv(x)
    322  1.31     pooka 
    323   1.1   thorpej #elif defined(_STANDALONE)
    324   1.1   thorpej 
    325   1.1   thorpej /*
    326   1.1   thorpej  * proplib in a standalone environment...
    327   1.1   thorpej  */
    328   1.1   thorpej 
    329   1.1   thorpej #include <lib/libsa/stand.h>
    330   1.1   thorpej 
    331   1.1   thorpej void *		_prop_standalone_calloc(size_t);
    332   1.2   thorpej void *		_prop_standalone_realloc(void *, size_t);
    333   1.1   thorpej 
    334  1.27   thorpej #define	_PROP_ASSERT(x)			/* nothing */
    335   1.1   thorpej 
    336  1.27   thorpej #define	_PROP_MALLOC(s, t)		alloc((s))
    337  1.27   thorpej #define	_PROP_CALLOC(s, t)		_prop_standalone_calloc((s))
    338  1.27   thorpej #define	_PROP_REALLOC(v, s, t)		_prop_standalone_realloc((v), (s))
    339  1.27   thorpej #define	_PROP_FREE(v, t)		dealloc((v), 0)		/* XXX */
    340   1.1   thorpej 
    341  1.27   thorpej #define	_PROP_POOL_GET(p)		alloc((p))
    342  1.27   thorpej #define	_PROP_POOL_PUT(p, v)		dealloc((v), (p))
    343   1.1   thorpej 
    344   1.1   thorpej #define	_PROP_POOL_INIT(p, s, d)	static const size_t p = s;
    345   1.1   thorpej 
    346   1.1   thorpej #define	_PROP_MALLOC_DEFINE(t, s, l)	/* nothing */
    347   1.1   thorpej 
    348   1.6   thorpej #define	_PROP_MUTEX_DECL_STATIC(x)	/* nothing */
    349  1.29     pooka #define	_PROP_MUTEX_INIT(x)		/* nothing */
    350   1.6   thorpej #define	_PROP_MUTEX_LOCK(x)		/* nothing */
    351   1.6   thorpej #define	_PROP_MUTEX_UNLOCK(x)		/* nothing */
    352   1.6   thorpej 
    353  1.27   thorpej #define	_PROP_RWLOCK_DECL(x)		/* nothing */
    354  1.27   thorpej #define	_PROP_RWLOCK_INIT(x)		/* nothing */
    355  1.27   thorpej #define	_PROP_RWLOCK_RDLOCK(x)		/* nothing */
    356  1.27   thorpej #define	_PROP_RWLOCK_WRLOCK(x)		/* nothing */
    357  1.27   thorpej #define	_PROP_RWLOCK_UNLOCK(x)		/* nothing */
    358  1.27   thorpej #define	_PROP_RWLOCK_DESTROY(x)		/* nothing */
    359  1.22   xtraeme 
    360  1.29     pooka #define _PROP_ONCE_DECL(x)		_PROP_NOTHREAD_ONCE_DECL(x)
    361  1.29     pooka #define _PROP_ONCE_RUN(x,f)		_PROP_NOTHREAD_ONCE_RUN(x,f)
    362  1.29     pooka 
    363  1.34   thorpej #define	_PROP_ATOMIC_LOAD(x)		*(x)
    364  1.31     pooka #define _PROP_ATOMIC_INC32(x)		++*(x)
    365  1.31     pooka #define _PROP_ATOMIC_DEC32(x)		--*(x)
    366  1.31     pooka #define _PROP_ATOMIC_INC32_NV(x, v)	v = ++*(x)
    367  1.31     pooka #define _PROP_ATOMIC_DEC32_NV(x, v)	v = --*(x)
    368  1.31     pooka 
    369   1.1   thorpej #else
    370   1.1   thorpej 
    371   1.1   thorpej /*
    372   1.1   thorpej  * proplib in user space...
    373   1.1   thorpej  */
    374   1.1   thorpej 
    375   1.1   thorpej #include <assert.h>
    376   1.1   thorpej #include <string.h>
    377   1.1   thorpej #include <stdio.h>
    378   1.1   thorpej #include <stdlib.h>
    379   1.5   thorpej #include <stddef.h>
    380   1.1   thorpej 
    381  1.27   thorpej #define	_PROP_ASSERT(x)			/*LINTED*/assert(x)
    382   1.1   thorpej 
    383  1.27   thorpej #define	_PROP_MALLOC(s, t)		malloc((s))
    384  1.27   thorpej #define	_PROP_CALLOC(s, t)		calloc(1, (s))
    385  1.27   thorpej #define	_PROP_REALLOC(v, s, t)		realloc((v), (s))
    386  1.27   thorpej #define	_PROP_FREE(v, t)		free((v))
    387   1.1   thorpej 
    388  1.27   thorpej #define	_PROP_POOL_GET(p)		malloc((p))
    389  1.27   thorpej #define	_PROP_POOL_PUT(p, v)		free((v))
    390   1.1   thorpej 
    391   1.1   thorpej #define	_PROP_POOL_INIT(p, s, d)	static const size_t p = s;
    392   1.1   thorpej 
    393   1.1   thorpej #define	_PROP_MALLOC_DEFINE(t, s, l)	/* nothing */
    394   1.1   thorpej 
    395   1.3   thorpej #if defined(__NetBSD__) && defined(_LIBPROP)
    396   1.3   thorpej /*
    397   1.3   thorpej  * Use the same mechanism as libc; we get pthread mutexes for threaded
    398   1.3   thorpej  * programs and do-nothing stubs for non-threaded programs.
    399   1.3   thorpej  */
    400  1.31     pooka #include <sys/atomic.h>
    401   1.3   thorpej #include "reentrant.h"
    402  1.29     pooka #define	_PROP_MUTEX_DECL_STATIC(x)	static mutex_t x;
    403  1.29     pooka #define	_PROP_MUTEX_INIT(x)		mutex_init(&(x), NULL)
    404   1.6   thorpej #define	_PROP_MUTEX_LOCK(x)		mutex_lock(&(x))
    405   1.6   thorpej #define	_PROP_MUTEX_UNLOCK(x)		mutex_unlock(&(x))
    406   1.6   thorpej 
    407  1.27   thorpej #define	_PROP_RWLOCK_DECL(x)		rwlock_t x ;
    408  1.27   thorpej #define	_PROP_RWLOCK_INIT(x)		rwlock_init(&(x), NULL)
    409  1.27   thorpej #define	_PROP_RWLOCK_RDLOCK(x)		rwlock_rdlock(&(x))
    410  1.27   thorpej #define	_PROP_RWLOCK_WRLOCK(x)		rwlock_wrlock(&(x))
    411  1.27   thorpej #define	_PROP_RWLOCK_UNLOCK(x)		rwlock_unlock(&(x))
    412  1.27   thorpej #define	_PROP_RWLOCK_DESTROY(x)		rwlock_destroy(&(x))
    413  1.29     pooka 
    414  1.29     pooka #define _PROP_ONCE_DECL(x)						\
    415  1.29     pooka 	static pthread_once_t x = PTHREAD_ONCE_INIT;
    416  1.29     pooka #define _PROP_ONCE_RUN(x,f)		thr_once(&(x), (void(*)(void))f);
    417  1.29     pooka 
    418  1.34   thorpej #define	_PROP_ATOMIC_LOAD(x)		*(x)
    419  1.31     pooka #define _PROP_ATOMIC_INC32(x)		atomic_inc_32(x)
    420  1.31     pooka #define _PROP_ATOMIC_DEC32(x)		atomic_dec_32(x)
    421  1.31     pooka #define _PROP_ATOMIC_INC32_NV(x, v)	v = atomic_inc_32_nv(x)
    422  1.31     pooka #define _PROP_ATOMIC_DEC32_NV(x, v)	v = atomic_dec_32_nv(x)
    423  1.31     pooka 
    424  1.38   thorpej #define	_PROP_EXPORT			__attribute__((visibility("default")))
    425  1.38   thorpej 
    426   1.3   thorpej #elif defined(HAVE_NBTOOL_CONFIG_H)
    427   1.3   thorpej /*
    428   1.3   thorpej  * None of NetBSD's build tools are multi-threaded.
    429   1.3   thorpej  */
    430   1.6   thorpej #define	_PROP_MUTEX_DECL_STATIC(x)	/* nothing */
    431  1.29     pooka #define	_PROP_MUTEX_INIT(x)		/* nothing */
    432   1.6   thorpej #define	_PROP_MUTEX_LOCK(x)		/* nothing */
    433   1.6   thorpej #define	_PROP_MUTEX_UNLOCK(x)		/* nothing */
    434   1.6   thorpej 
    435  1.27   thorpej #define	_PROP_RWLOCK_DECL(x)		/* nothing */
    436  1.27   thorpej #define	_PROP_RWLOCK_INIT(x)		/* nothing */
    437  1.27   thorpej #define	_PROP_RWLOCK_RDLOCK(x)		/* nothing */
    438  1.27   thorpej #define	_PROP_RWLOCK_WRLOCK(x)		/* nothing */
    439  1.27   thorpej #define	_PROP_RWLOCK_UNLOCK(x)		/* nothing */
    440  1.27   thorpej #define	_PROP_RWLOCK_DESTROY(x)		/* nothing */
    441  1.29     pooka 
    442  1.29     pooka #define _PROP_ONCE_DECL(x)		_PROP_NOTHREAD_ONCE_DECL(x)
    443  1.29     pooka #define _PROP_ONCE_RUN(x,f)		_PROP_NOTHREAD_ONCE_RUN(x,f)
    444  1.31     pooka 
    445  1.34   thorpej #define	_PROP_ATOMIC_LOAD(x)		*(x)
    446  1.31     pooka #define _PROP_ATOMIC_INC32(x)		++*(x)
    447  1.31     pooka #define _PROP_ATOMIC_DEC32(x)		--*(x)
    448  1.31     pooka #define _PROP_ATOMIC_INC32_NV(x, v)	v = ++*(x)
    449  1.31     pooka #define _PROP_ATOMIC_DEC32_NV(x, v)	v = --*(x)
    450  1.31     pooka 
    451   1.3   thorpej #else
    452   1.3   thorpej /*
    453   1.3   thorpej  * Use pthread mutexes everywhere else.
    454   1.3   thorpej  */
    455   1.3   thorpej #include <pthread.h>
    456  1.29     pooka #define	_PROP_MUTEX_DECL_STATIC(x)	static pthread_mutex_t x;
    457  1.29     pooka #define	_PROP_MUTEX_INIT(x)		pthread_mutex_init(&(x), NULL)
    458  1.27   thorpej #define	_PROP_MUTEX_LOCK(x)		pthread_mutex_lock(&(x))
    459  1.27   thorpej #define	_PROP_MUTEX_UNLOCK(x)		pthread_mutex_unlock(&(x))
    460   1.6   thorpej 
    461  1.27   thorpej #define	_PROP_RWLOCK_DECL(x)		pthread_rwlock_t x ;
    462  1.27   thorpej #define	_PROP_RWLOCK_INIT(x)		pthread_rwlock_init(&(x), NULL)
    463  1.27   thorpej #define	_PROP_RWLOCK_RDLOCK(x)		pthread_rwlock_rdlock(&(x))
    464  1.27   thorpej #define	_PROP_RWLOCK_WRLOCK(x)		pthread_rwlock_wrlock(&(x))
    465  1.27   thorpej #define	_PROP_RWLOCK_UNLOCK(x)		pthread_rwlock_unlock(&(x))
    466  1.27   thorpej #define	_PROP_RWLOCK_DESTROY(x)		pthread_rwlock_destroy(&(x))
    467  1.29     pooka 
    468  1.29     pooka #define _PROP_ONCE_DECL(x)						\
    469  1.29     pooka 	static pthread_once_t x = PTHREAD_ONCE_INIT;
    470  1.29     pooka #define _PROP_ONCE_RUN(x,f)		pthread_once(&(x),(void(*)(void))f)
    471  1.31     pooka 
    472  1.31     pooka #define _PROP_NEED_REFCNT_MTX
    473  1.31     pooka 
    474  1.34   thorpej #define	_PROP_ATOMIC_LOAD(x)		*(x)
    475  1.34   thorpej 
    476  1.31     pooka #define _PROP_ATOMIC_INC32(x)						\
    477  1.31     pooka do {									\
    478  1.31     pooka 	pthread_mutex_lock(&_prop_refcnt_mtx);				\
    479  1.31     pooka 	(*(x))++;							\
    480  1.31     pooka 	pthread_mutex_unlock(&_prop_refcnt_mtx);			\
    481  1.31     pooka } while (/*CONSTCOND*/0)
    482  1.31     pooka 
    483  1.31     pooka #define _PROP_ATOMIC_DEC32(x)						\
    484  1.31     pooka do {									\
    485  1.31     pooka 	pthread_mutex_lock(&_prop_refcnt_mtx);				\
    486  1.31     pooka 	(*(x))--;							\
    487  1.31     pooka 	pthread_mutex_unlock(&_prop_refcnt_mtx);			\
    488  1.31     pooka } while (/*CONSTCOND*/0)
    489  1.31     pooka 
    490  1.31     pooka #define _PROP_ATOMIC_INC32_NV(x, v)					\
    491  1.31     pooka do {									\
    492  1.31     pooka 	pthread_mutex_lock(&_prop_refcnt_mtx);				\
    493  1.31     pooka 	v = ++(*(x));							\
    494  1.31     pooka 	pthread_mutex_unlock(&_prop_refcnt_mtx);			\
    495  1.31     pooka } while (/*CONSTCOND*/0)
    496  1.31     pooka 
    497  1.31     pooka #define _PROP_ATOMIC_DEC32_NV(x, v)					\
    498  1.31     pooka do {									\
    499  1.31     pooka 	pthread_mutex_lock(&_prop_refcnt_mtx);				\
    500  1.31     pooka 	v = --(*(x));							\
    501  1.31     pooka 	pthread_mutex_unlock(&_prop_refcnt_mtx);			\
    502  1.31     pooka } while (/*CONSTCOND*/0)
    503  1.31     pooka 
    504   1.3   thorpej #endif
    505   1.1   thorpej #endif /* _KERNEL */
    506   1.1   thorpej 
    507  1.38   thorpej #ifndef _PROP_EXPORT
    508  1.38   thorpej #define	_PROP_EXPORT			/* nothing */
    509  1.38   thorpej #endif
    510  1.38   thorpej 
    511   1.9   thorpej /*
    512   1.9   thorpej  * Language features.
    513   1.9   thorpej  */
    514   1.9   thorpej #if defined(__NetBSD__)
    515   1.9   thorpej #include <sys/cdefs.h>
    516  1.27   thorpej #define	_PROP_ARG_UNUSED		__unused
    517  1.36   thorpej #if defined(__clang__)
    518  1.36   thorpej #define	_PROP_DEPRECATED(s, m)		/* delete */
    519  1.36   thorpej #else /* ! __clang__ */
    520  1.36   thorpej #define	_PROP_DEPRECATED(s, m)		__warn_references(s, m)
    521  1.36   thorpej #endif /* __clang__ */
    522  1.37   thorpej #define	_PROP_UNCONST(x)		__UNCONST(x)
    523   1.9   thorpej #else
    524  1.27   thorpej #define	_PROP_ARG_UNUSED		/* delete */
    525  1.34   thorpej #define	_PROP_DEPRECATED(s, m)		/* delete */
    526  1.37   thorpej #define	_PROP_UNCONST(x)	((void *)(unsigned long)(const void *)(x))
    527   1.9   thorpej #endif /* __NetBSD__ */
    528   1.9   thorpej 
    529   1.1   thorpej #endif /* _PROPLIB_PROP_OBJECT_IMPL_H_ */
    530