Blame shared/nm-glib-aux/nm-shared-utils.h

Packit Service b23acc
// SPDX-License-Identifier: LGPL-2.1+
Packit Service b23acc
/*
Packit Service b23acc
 * Copyright (C) 2016 Red Hat, Inc.
Packit Service b23acc
 */
Packit Service b23acc
Packit Service b23acc
#ifndef __NM_SHARED_UTILS_H__
Packit Service b23acc
#define __NM_SHARED_UTILS_H__
Packit Service b23acc
Packit Service b23acc
#include <netinet/in.h>
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
pid_t nm_utils_gettid (void);
Packit Service b23acc
Packit Service b23acc
gboolean _nm_assert_on_main_thread (void);
Packit Service b23acc
Packit Service b23acc
#if NM_MORE_ASSERTS > 5
Packit Service b23acc
#define NM_ASSERT_ON_MAIN_THREAD() G_STMT_START { nm_assert (_nm_assert_on_main_thread ()); } G_STMT_END
Packit Service b23acc
#else
Packit Service b23acc
#define NM_ASSERT_ON_MAIN_THREAD() G_STMT_START {                                         ; } G_STMT_END
Packit Service b23acc
#endif
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
_NM_INT_NOT_NEGATIVE (gssize val)
Packit Service b23acc
{
Packit Service b23acc
	/* whether an enum (without negative values) is a signed int, depends on compiler options
Packit Service b23acc
	 * and compiler implementation.
Packit Service b23acc
	 *
Packit Service b23acc
	 * When using such an enum for accessing an array, one naturally wants to check
Packit Service b23acc
	 * that the enum is not negative. However, the compiler doesn't like a plain
Packit Service b23acc
	 * comparison "enum_val >= 0", because (if the enum is unsigned), it will warn
Packit Service b23acc
	 * that the expression is always true *duh*. Not even a cast to a signed
Packit Service b23acc
	 * type helps to avoid the compiler warning in any case.
Packit Service b23acc
	 *
Packit Service b23acc
	 * The sole purpose of this function is to avoid a compiler warning, when checking
Packit Service b23acc
	 * that an enum is not negative. */
Packit Service b23acc
	return val >= 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/* check whether the integer value is smaller than G_MAXINT32. This macro exists
Packit Service b23acc
 * for the sole purpose, that a plain "((int) value <= G_MAXINT32)" comparison
Packit Service b23acc
 * may cause the compiler or coverity that this check is always TRUE. But the
Packit Service b23acc
 * check depends on compile time and the size of C type "int".  Of course, most
Packit Service b23acc
 * of the time in is gint32 and an int value is always <= G_MAXINT32.  The check
Packit Service b23acc
 * exists to catch cases where that is not true.
Packit Service b23acc
 *
Packit Service b23acc
 * Together with the G_STATIC_ASSERT(), we make sure that this is always satisfied. */
Packit Service b23acc
G_STATIC_ASSERT (sizeof (int) == sizeof (gint32));
Packit Service b23acc
#if _NM_CC_SUPPORT_GENERIC
Packit Service b23acc
#define _NM_INT_LE_MAXINT32(value) \
Packit Service b23acc
	({ \
Packit Service b23acc
		_nm_unused typeof (value) _value = (value); \
Packit Service b23acc
		\
Packit Service b23acc
		_Generic((value), \
Packit Service b23acc
		         int: TRUE \
Packit Service b23acc
		); \
Packit Service b23acc
	})
Packit Service b23acc
#else
Packit Service b23acc
#define _NM_INT_LE_MAXINT32(value) ({ \
Packit Service b23acc
		_nm_unused typeof (value) _value = (value); \
Packit Service b23acc
		_nm_unused const int *_p_value = &_value; \
Packit Service b23acc
		\
Packit Service b23acc
		TRUE; \
Packit Service b23acc
	})
Packit Service b23acc
#endif
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline char
Packit Service b23acc
nm_utils_addr_family_to_char (int addr_family)
Packit Service b23acc
{
Packit Service b23acc
	switch (addr_family) {
Packit Service b23acc
	case AF_UNSPEC: return 'X';
Packit Service b23acc
	case AF_INET:   return '4';
Packit Service b23acc
	case AF_INET6:  return '6';
Packit Service b23acc
	}
Packit Service b23acc
	g_return_val_if_reached ('?');
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gsize
Packit Service b23acc
nm_utils_addr_family_to_size (int addr_family)
Packit Service b23acc
{
Packit Service b23acc
	switch (addr_family) {
Packit Service b23acc
	case AF_INET:  return sizeof (in_addr_t);
Packit Service b23acc
	case AF_INET6: return sizeof (struct in6_addr);
Packit Service b23acc
	}
Packit Service b23acc
	g_return_val_if_reached (0);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline int
Packit Service b23acc
nm_utils_addr_family_from_size (gsize len)
Packit Service b23acc
{
Packit Service b23acc
	switch (len) {
Packit Service b23acc
	case sizeof (in_addr_t):       return AF_INET;
Packit Service b23acc
	case sizeof (struct in6_addr): return AF_INET6;
Packit Service b23acc
	}
Packit Service b23acc
	return AF_UNSPEC;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_assert_addr_family(addr_family) \
Packit Service b23acc
	nm_assert (NM_IN_SET ((addr_family), AF_INET, AF_INET6))
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	union {
Packit Service b23acc
		guint8 addr_ptr[1];
Packit Service b23acc
		in_addr_t addr4;
Packit Service b23acc
		struct in_addr addr4_struct;
Packit Service b23acc
		struct in6_addr addr6;
Packit Service b23acc
Packit Service b23acc
		/* NMIPAddr is really a union for IP addresses.
Packit Service b23acc
		 * However, as ethernet addresses fit in here nicely, use
Packit Service b23acc
		 * it also for an ethernet MAC address. */
Packit Service b23acc
		guint8 addr_eth[6 /*ETH_ALEN*/];
Packit Service b23acc
Packit Service b23acc
		guint8 array[sizeof (struct in6_addr)];
Packit Service b23acc
	};
Packit Service b23acc
} NMIPAddr;
Packit Service b23acc
Packit Service b23acc
#define NM_IP_ADDR_INIT { .array = { 0 } }
Packit Service b23acc
Packit Service b23acc
extern const NMIPAddr nm_ip_addr_zero;
Packit Service b23acc
Packit Service b23acc
static inline int
Packit Service b23acc
nm_ip_addr_cmp (int addr_family, gconstpointer a, gconstpointer b)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert_addr_family (addr_family);
Packit Service b23acc
	nm_assert (a);
Packit Service b23acc
	nm_assert (b);
Packit Service b23acc
Packit Service b23acc
	return memcmp (a, b, nm_utils_addr_family_to_size (addr_family));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_ip_addr_equal (int addr_family, gconstpointer a, gconstpointer b)
Packit Service b23acc
{
Packit Service b23acc
	return nm_ip_addr_cmp (addr_family, a, b) == 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_ip_addr_is_null (int addr_family, gconstpointer addr)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (addr);
Packit Service b23acc
	if (addr_family == AF_INET6)
Packit Service b23acc
		return IN6_IS_ADDR_UNSPECIFIED ((const struct in6_addr *) addr);
Packit Service b23acc
	nm_assert (addr_family == AF_INET);
Packit Service b23acc
	return ((const struct in_addr *) addr)->s_addr == 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_ip_addr_set (int addr_family, gpointer dst, gconstpointer src)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert_addr_family (addr_family);
Packit Service b23acc
	nm_assert (dst);
Packit Service b23acc
	nm_assert (src);
Packit Service b23acc
Packit Service b23acc
	memcpy (dst,
Packit Service b23acc
	        src,
Packit Service b23acc
	        (addr_family != AF_INET6)
Packit Service b23acc
	          ? sizeof (in_addr_t)
Packit Service b23acc
	          : sizeof (struct in6_addr));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_ip_addr_set_from_untrusted (int addr_family,
Packit Service b23acc
                                        gpointer dst,
Packit Service b23acc
                                        gconstpointer src,
Packit Service b23acc
                                        gsize src_len,
Packit Service b23acc
                                        int *out_addr_family);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_ip4_addr_is_localhost (in_addr_t addr4)
Packit Service b23acc
{
Packit Service b23acc
	return (addr4 & htonl (0xFF000000u)) == htonl (0x7F000000u);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
struct ether_addr;
Packit Service b23acc
Packit Service b23acc
static inline int
Packit Service b23acc
nm_utils_ether_addr_cmp (const struct ether_addr *a1, const struct ether_addr *a2)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (a1);
Packit Service b23acc
	nm_assert (a2);
Packit Service b23acc
	return memcmp (a1, a2, 6 /*ETH_ALEN*/);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_ether_addr_equal (const struct ether_addr *a1, const struct ether_addr *a2)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_ether_addr_cmp (a1, a2) == 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_INET_ADDRSTRLEN INET6_ADDRSTRLEN
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
nm_utils_inet_ntop (int addr_family, gconstpointer addr, char *dst)
Packit Service b23acc
{
Packit Service b23acc
	const char *s;
Packit Service b23acc
Packit Service b23acc
	const char *inet_ntop (int af,
Packit Service b23acc
	                       const void *src,
Packit Service b23acc
	                       char *dst,
Packit Service b23acc
	                       socklen_t size);
Packit Service b23acc
Packit Service b23acc
	nm_assert_addr_family (addr_family);
Packit Service b23acc
	nm_assert (addr);
Packit Service b23acc
	nm_assert (dst);
Packit Service b23acc
Packit Service b23acc
	s = inet_ntop (addr_family,
Packit Service b23acc
	               addr,
Packit Service b23acc
	               dst,
Packit Service b23acc
	               addr_family == AF_INET6 ? INET6_ADDRSTRLEN : INET_ADDRSTRLEN);
Packit Service b23acc
	nm_assert (s);
Packit Service b23acc
	return s;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
_nm_utils_inet4_ntop (in_addr_t addr, char dst[static INET_ADDRSTRLEN])
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_inet_ntop (AF_INET, &addr, dst);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
_nm_utils_inet6_ntop (const struct in6_addr *addr, char dst[static INET6_ADDRSTRLEN])
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_inet_ntop (AF_INET6, addr, dst);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline char *
Packit Service b23acc
nm_utils_inet_ntop_dup (int addr_family, gconstpointer addr)
Packit Service b23acc
{
Packit Service b23acc
	char buf[NM_UTILS_INET_ADDRSTRLEN];
Packit Service b23acc
Packit Service b23acc
	return g_strdup (nm_utils_inet_ntop (addr_family, addr, buf));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline char *
Packit Service b23acc
nm_utils_inet4_ntop_dup (in_addr_t addr)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_inet_ntop_dup (AF_INET, &addr);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline char *
Packit Service b23acc
nm_utils_inet6_ntop_dup (const struct in6_addr *addr)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_inet_ntop_dup (AF_INET6, addr);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_ipaddr_is_valid (int addr_family,
Packit Service b23acc
                                   const char *str_addr);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_ipaddr_is_normalized (int addr_family,
Packit Service b23acc
                                        const char *str_addr);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_RETURN(c) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        const int _cc = (c); \
Packit Service b23acc
        if (_cc) \
Packit Service b23acc
            return _cc < 0 ? -1 : 1; \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_RETURN_DIRECT(c) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        const int _cc = (c); \
Packit Service b23acc
        if (_cc) \
Packit Service b23acc
            return _cc; \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_SELF(a, b) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        typeof (a) _a = (a); \
Packit Service b23acc
        typeof (b) _b = (b); \
Packit Service b23acc
        \
Packit Service b23acc
        if (_a == _b) \
Packit Service b23acc
            return 0; \
Packit Service b23acc
        if (!_a) \
Packit Service b23acc
            return -1; \
Packit Service b23acc
        if (!_b) \
Packit Service b23acc
            return 1; \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_DIRECT(a, b) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        typeof (a) _a = (a); \
Packit Service b23acc
        typeof (b) _b = (b); \
Packit Service b23acc
        \
Packit Service b23acc
        if (_a != _b) \
Packit Service b23acc
            return (_a < _b) ? -1 : 1; \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
/* In the general case, direct pointer comparison is undefined behavior in C.
Packit Service b23acc
 * Avoid that by casting pointers to void* and then to uintptr_t. This comparison
Packit Service b23acc
 * is not really meaningful, except that it provides some kind of stable sort order
Packit Service b23acc
 * between pointers (that can otherwise not be compared). */
Packit Service b23acc
#define NM_CMP_DIRECT_PTR(a, b) \
Packit Service b23acc
	NM_CMP_DIRECT ((uintptr_t) ((void *) (a)), \
Packit Service b23acc
	               (uintptr_t) ((void *) (b)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_DIRECT_MEMCMP(a, b, size) \
Packit Service b23acc
    NM_CMP_RETURN (memcmp ((a), (b), (size)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_DIRECT_STRCMP(a, b) \
Packit Service b23acc
    NM_CMP_RETURN_DIRECT (strcmp ((a), (b)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_DIRECT_STRCMP0(a, b) \
Packit Service b23acc
    NM_CMP_RETURN_DIRECT (nm_strcmp0 ((a), (b)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_DIRECT_IN6ADDR(a, b) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        const struct in6_addr *const _a = (a); \
Packit Service b23acc
        const struct in6_addr *const _b = (b); \
Packit Service b23acc
        NM_CMP_RETURN (memcmp (_a, _b, sizeof (struct in6_addr))); \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD(a, b, field) \
Packit Service b23acc
    NM_CMP_DIRECT (((a)->field), ((b)->field))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_UNSAFE(a, b, field) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        /* it's unsafe, because it evaluates the arguments more then once.
Packit Service b23acc
         * This is necessary for bitfields, for which typeof() doesn't work. */ \
Packit Service b23acc
        if (((a)->field) != ((b)->field)) \
Packit Service b23acc
            return ((a)->field < ((b)->field)) ? -1 : 1; \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_BOOL(a, b, field) \
Packit Service b23acc
    NM_CMP_DIRECT (!!((a)->field), !!((b)->field))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_STR(a, b, field) \
Packit Service b23acc
    NM_CMP_RETURN (strcmp (((a)->field), ((b)->field)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_STR_INTERNED(a, b, field) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        const char *_a = ((a)->field); \
Packit Service b23acc
        const char *_b = ((b)->field); \
Packit Service b23acc
        \
Packit Service b23acc
        if (_a != _b) { \
Packit Service b23acc
            NM_CMP_RETURN_DIRECT (nm_strcmp0 (_a, _b)); \
Packit Service b23acc
        } \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_STR0(a, b, field) \
Packit Service b23acc
    NM_CMP_RETURN_DIRECT (nm_strcmp0 (((a)->field), ((b)->field)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_MEMCMP_LEN(a, b, field, len) \
Packit Service b23acc
    NM_CMP_RETURN (memcmp (&((a)->field), &((b)->field), \
Packit Service b23acc
                           NM_MIN (len, sizeof ((a)->field))))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_MEMCMP(a, b, field) \
Packit Service b23acc
    NM_CMP_RETURN (memcmp (&((a)->field), \
Packit Service b23acc
                           &((b)->field), \
Packit Service b23acc
                           sizeof ((a)->field)))
Packit Service b23acc
Packit Service b23acc
#define NM_CMP_FIELD_IN6ADDR(a, b, field) \
Packit Service b23acc
    G_STMT_START { \
Packit Service b23acc
        const struct in6_addr *const _a = &((a)->field); \
Packit Service b23acc
        const struct in6_addr *const _b = &((b)->field); \
Packit Service b23acc
        NM_CMP_RETURN (memcmp (_a, _b, sizeof (struct in6_addr))); \
Packit Service b23acc
    } G_STMT_END
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_memeqzero (gconstpointer data, gsize length);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
extern const void *const _NM_PTRARRAY_EMPTY[1];
Packit Service b23acc
Packit Service b23acc
#define NM_PTRARRAY_EMPTY(type) ((type const*) _NM_PTRARRAY_EMPTY)
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
_nm_utils_strbuf_init (char *buf, gsize len, char **p_buf_ptr, gsize *p_buf_len)
Packit Service b23acc
{
Packit Service b23acc
	NM_SET_OUT (p_buf_len, len);
Packit Service b23acc
	NM_SET_OUT (p_buf_ptr, buf);
Packit Service b23acc
	buf[0] = '\0';
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_utils_strbuf_init(buf, p_buf_ptr, p_buf_len) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		G_STATIC_ASSERT (G_N_ELEMENTS (buf) == sizeof (buf) && sizeof (buf) > sizeof (char *)); \
Packit Service b23acc
		_nm_utils_strbuf_init ((buf), sizeof (buf), (p_buf_ptr), (p_buf_len)); \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
void nm_utils_strbuf_append (char **buf, gsize *len, const char *format, ...) _nm_printf (3, 4);
Packit Service b23acc
void nm_utils_strbuf_append_c (char **buf, gsize *len, char c);
Packit Service b23acc
void nm_utils_strbuf_append_str (char **buf, gsize *len, const char *str);
Packit Service b23acc
void nm_utils_strbuf_append_bin (char **buf, gsize *len, gconstpointer str, gsize str_len);
Packit Service b23acc
void nm_utils_strbuf_seek_end (char **buf, gsize *len);
Packit Service b23acc
Packit Service b23acc
const char *nm_strquote (char *buf, gsize buf_len, const char *str);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_is_separator (const char c)
Packit Service b23acc
{
Packit Service b23acc
	return NM_IN_SET (c, ' ', '\t');
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
GBytes *nm_gbytes_get_empty (void);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_gbytes_equal0 (GBytes *a, GBytes *b)
Packit Service b23acc
{
Packit Service b23acc
	return a == b || (a && b && g_bytes_equal (a, b));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_gbytes_equal_mem (GBytes *bytes,
Packit Service b23acc
                                    gconstpointer mem_data,
Packit Service b23acc
                                    gsize mem_len);
Packit Service b23acc
Packit Service b23acc
GVariant *nm_utils_gbytes_to_variant_ay (GBytes *bytes);
Packit Service b23acc
Packit Service b23acc
GVariant *nm_utils_strdict_to_variant_ass (GHashTable *strdict);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
GVariant *nm_utils_gvariant_vardict_filter (GVariant *src,
Packit Service b23acc
                                            gboolean (*filter_fcn) (const char *key,
Packit Service b23acc
                                                                    GVariant *val,
Packit Service b23acc
                                                                    char **out_key,
Packit Service b23acc
                                                                    GVariant **out_val,
Packit Service b23acc
                                                                    gpointer user_data),
Packit Service b23acc
                                            gpointer user_data);
Packit Service b23acc
Packit Service b23acc
GVariant *
Packit Service b23acc
nm_utils_gvariant_vardict_filter_drop_one (GVariant *src,
Packit Service b23acc
                                           const char *key);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline int
Packit Service b23acc
nm_utils_hexchar_to_int (char ch)
Packit Service b23acc
{
Packit Service b23acc
	G_STATIC_ASSERT_EXPR ('0' < 'A');
Packit Service b23acc
	G_STATIC_ASSERT_EXPR ('A' < 'a');
Packit Service b23acc
Packit Service b23acc
	if (ch >= '0') {
Packit Service b23acc
		if (ch <= '9')
Packit Service b23acc
			return ch - '0';
Packit Service b23acc
		if (ch >= 'A') {
Packit Service b23acc
			if (ch <= 'F')
Packit Service b23acc
				return ((int) ch) + (10 - (int) 'A');
Packit Service b23acc
			if (ch >= 'a' && ch <= 'f')
Packit Service b23acc
				return ((int) ch) + (10 - (int) 'a');
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
	return -1;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
const char *nm_utils_dbus_path_get_last_component (const char *dbus_path);
Packit Service b23acc
Packit Service b23acc
int nm_utils_dbus_path_cmp (const char *dbus_path_a, const char *dbus_path_b);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef enum {
Packit Service b23acc
	NM_UTILS_STRSPLIT_SET_FLAGS_NONE           = 0,
Packit Service b23acc
Packit Service b23acc
	/* by default, strsplit will coalesce consecutive delimiters and remove
Packit Service b23acc
	 * them from the result. If this flag is present, empty values are preserved
Packit Service b23acc
	 * and returned.
Packit Service b23acc
	 *
Packit Service b23acc
	 * When combined with %NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP, if a value gets
Packit Service b23acc
	 * empty after strstrip(), it also gets removed. */
Packit Service b23acc
	NM_UTILS_STRSPLIT_SET_FLAGS_PRESERVE_EMPTY = (1u << 0),
Packit Service b23acc
Packit Service b23acc
	/* %NM_UTILS_STRSPLIT_SET_FLAGS_ALLOW_ESCAPING means that delimiters prefixed
Packit Service b23acc
	 * by a backslash are not treated as a separator. Such delimiters and their escape
Packit Service b23acc
	 * character are copied to the current word without unescaping them. In general,
Packit Service b23acc
	 * nm_utils_strsplit_set_full() does not remove any backslash escape characters
Packit Service b23acc
	 * and does no unescaping. It only considers them for skipping to split at
Packit Service b23acc
	 * an escaped delimiter.
Packit Service b23acc
	 *
Packit Service b23acc
	 * If this is combined with (or implied by %NM_UTILS_STRSPLIT_SET_FLAGS_ESCAPED), then
Packit Service b23acc
	 * the backslash escapes are removed from the result.
Packit Service b23acc
	 */
Packit Service b23acc
	NM_UTILS_STRSPLIT_SET_FLAGS_ALLOW_ESCAPING = (1u << 1),
Packit Service b23acc
Packit Service b23acc
	/* If flag is set, does the same as g_strstrip() on the returned tokens.
Packit Service b23acc
	 * This will remove leading and trailing ascii whitespaces (g_ascii_isspace()
Packit Service b23acc
	 * and NM_ASCII_SPACES).
Packit Service b23acc
	 *
Packit Service b23acc
	 * - when combined with !%NM_UTILS_STRSPLIT_SET_FLAGS_PRESERVE_EMPTY,
Packit Service b23acc
	 *   empty tokens will be removed (and %NULL will be returned if that
Packit Service b23acc
	 *   results in an empty string array).
Packit Service b23acc
	 * - when combined with %NM_UTILS_STRSPLIT_SET_FLAGS_ALLOW_ESCAPING,
Packit Service b23acc
	 *   trailing whitespace escaped by backslash are not stripped. */
Packit Service b23acc
	NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP       = (1u << 2),
Packit Service b23acc
Packit Service b23acc
	/* This implies %NM_UTILS_STRSPLIT_SET_FLAGS_ALLOW_ESCAPING.
Packit Service b23acc
	 *
Packit Service b23acc
	 * This will do a final run over all tokens and remove all backslash
Packit Service b23acc
	 * escape characters that
Packit Service b23acc
	 *   - precede a delimiter.
Packit Service b23acc
	 *   - precede a backslash.
Packit Service b23acc
	 *   - preceed a whitespace (with %NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP).
Packit Service b23acc
	 *
Packit Service b23acc
	 *  Note that with %NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP, it is only
Packit Service b23acc
	 *  necessary to escape the very last whitespace (if the delimiters
Packit Service b23acc
	 *  are not whitespace themself). So, technically, it would be sufficient
Packit Service b23acc
	 *  to only unescape a backslash before the last whitespace and the user
Packit Service b23acc
	 *  still could express everything. However, such a rule would be complicated
Packit Service b23acc
	 *  to understand, so when using backslash escaping with nm_utils_strsplit_set_full(),
Packit Service b23acc
	 *  then all characters (including backslash) are treated verbatim, except:
Packit Service b23acc
	 *
Packit Service b23acc
	 *    - "\\$DELIMITER" (escaped delimiter)
Packit Service b23acc
	 *    - "\\\\" (escaped backslash)
Packit Service b23acc
	 *    - "\\$SPACE" (escaped space) (with %NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP).
Packit Service b23acc
	 *
Packit Service b23acc
	 * Note that all other escapes like "\\n" or "\\001" are left alone.
Packit Service b23acc
	 * That makes the escaping/unescaping rules simple. Also, for the most part
Packit Service b23acc
	 * a text is just taken as-is, with little additional rules. Only backslashes
Packit Service b23acc
	 * need extra care, and then only if they proceed one of the relevant characters.
Packit Service b23acc
	 */
Packit Service b23acc
	NM_UTILS_STRSPLIT_SET_FLAGS_ESCAPED        = (1u << 3),
Packit Service b23acc
Packit Service b23acc
} NMUtilsStrsplitSetFlags;
Packit Service b23acc
Packit Service b23acc
const char **nm_utils_strsplit_set_full (const char *str,
Packit Service b23acc
                                         const char *delimiter,
Packit Service b23acc
                                         NMUtilsStrsplitSetFlags flags);
Packit Service b23acc
Packit Service b23acc
static inline const char **
Packit Service b23acc
nm_utils_strsplit_set_with_empty (const char *str,
Packit Service b23acc
                                  const char *delimiters)
Packit Service b23acc
{
Packit Service b23acc
	/* this returns the same result as g_strsplit_set(str, delimiters, -1), except
Packit Service b23acc
	 * it does not deep-clone the strv array.
Packit Service b23acc
	 * Also, for @str == "", this returns %NULL while g_strsplit_set() would return
Packit Service b23acc
	 * an empty strv array. */
Packit Service b23acc
	return nm_utils_strsplit_set_full (str, delimiters, NM_UTILS_STRSPLIT_SET_FLAGS_PRESERVE_EMPTY);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char **
Packit Service b23acc
nm_utils_strsplit_set (const char *str,
Packit Service b23acc
                       const char *delimiters)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_strsplit_set_full (str, delimiters, NM_UTILS_STRSPLIT_SET_FLAGS_NONE);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gssize nm_utils_strv_find_first (char **list, gssize len, const char *needle);
Packit Service b23acc
Packit Service b23acc
char **_nm_utils_strv_cleanup (char **strv,
Packit Service b23acc
                               gboolean strip_whitespace,
Packit Service b23acc
                               gboolean skip_empty,
Packit Service b23acc
                               gboolean skip_repeated);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline const char **
Packit Service b23acc
nm_utils_escaped_tokens_split (const char *str,
Packit Service b23acc
                               const char *delimiters)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_strsplit_set_full (str,
Packit Service b23acc
	                                   delimiters,
Packit Service b23acc
	                                     NM_UTILS_STRSPLIT_SET_FLAGS_ESCAPED
Packit Service b23acc
	                                   | NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
typedef enum {
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_NONE                       = 0,
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_SPACES              = (1ull << 0),
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_LEADING_SPACE       = (1ull << 1),
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_TRAILING_SPACE      = (1ull << 2),
Packit Service b23acc
Packit Service b23acc
	/* Backslash characters will be escaped as "\\\\" if they precede another
Packit Service b23acc
	 * character that makes it necessary. Such characters are:
Packit Service b23acc
	 *
Packit Service b23acc
	 *  1) before another '\\' backslash.
Packit Service b23acc
	 *  2) before any delimiter in @delimiters.
Packit Service b23acc
	 *  3) before any delimiter in @delimiters_as_needed.
Packit Service b23acc
	 *  4) before a white space, if ESCAPE_LEADING_SPACE or ESCAPE_TRAILING_SPACE is set.
Packit Service b23acc
	 *  5) before the end of the word
Packit Service b23acc
	 *
Packit Service b23acc
	 * Rule 4) is an extension. It's not immediately clear why with ESCAPE_LEADING_SPACE
Packit Service b23acc
	 * and ESCAPE_TRAILING_SPACE we want *all* backslashes before a white space escaped.
Packit Service b23acc
	 * The reason is, that we obviously want to use ESCAPE_LEADING_SPACE and ESCAPE_TRAILING_SPACE
Packit Service b23acc
	 * in cases, where we later parse the backslash escaped strings back, but allowing to strip
Packit Service b23acc
	 * unescaped white spaces. That means, we want that " a " gets escaped as "\\ a\\ ".
Packit Service b23acc
	 * On the other hand, we also want that " a\\ b " gets escaped as "\\ a\\\\ b\\ ",
Packit Service b23acc
	 * and not "\\ a\\ b\\ ". Because otherwise, the parser would need to treat "\\ "
Packit Service b23acc
	 * differently depending on whether the sequence is at the beginning, end or middle
Packit Service b23acc
	 * of the word.
Packit Service b23acc
	 *
Packit Service b23acc
	 * Rule 5) is also not immediately obvious. When used with ESCAPE_TRAILING_SPACE,
Packit Service b23acc
	 * we clearly want to allow that an escaped word can have arbitrary
Packit Service b23acc
	 * whitespace suffixes. That's why this mode exists. So we must escape "a\\" as
Packit Service b23acc
	 * "a\\\\", so that appending " " does not change the meaning.
Packit Service b23acc
	 * Also without ESCAPE_TRAILING_SPACE, we want in general that we can concatenate
Packit Service b23acc
	 * two escaped words without changing their meaning. If the words would be "a\\"
Packit Service b23acc
	 * and "," (with ',' being a delimiter), then the result must be "a\\\\" and "\\,"
Packit Service b23acc
	 * so that the concatenated word ("a\\\\\\,") is still the same. If we would escape
Packit Service b23acc
	 * them instead as "a\\" + "\\,", then the concatenated word would be "a\\\\," and
Packit Service b23acc
	 * different.
Packit Service b23acc
	 * */
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_BACKSLASH_AS_NEEDED = (1ull << 3),
Packit Service b23acc
Packit Service b23acc
	NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_BACKSLASH_ALWAYS    = (1ull << 4),
Packit Service b23acc
} NMUtilsEscapedTokensEscapeFlags;
Packit Service b23acc
Packit Service b23acc
const char *nm_utils_escaped_tokens_escape_full (const char *str,
Packit Service b23acc
                                                 const char *delimiters,
Packit Service b23acc
                                                 const char *delimiters_as_needed,
Packit Service b23acc
                                                 NMUtilsEscapedTokensEscapeFlags flags,
Packit Service b23acc
                                                 char **out_to_free);
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
nm_utils_escaped_tokens_escape (const char *str,
Packit Service b23acc
                                const char *delimiters,
Packit Service b23acc
                                char **out_to_free)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_escaped_tokens_escape_full (str,
Packit Service b23acc
	                                            delimiters,
Packit Service b23acc
	                                            NULL,
Packit Service b23acc
	                                              NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_BACKSLASH_ALWAYS
Packit Service b23acc
	                                            | NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_TRAILING_SPACE,
Packit Service b23acc
	                                            out_to_free);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline GString *
Packit Service b23acc
nm_utils_escaped_tokens_escape_gstr_assert (const char *str,
Packit Service b23acc
                                            const char *delimiters,
Packit Service b23acc
                                            GString *gstring)
Packit Service b23acc
{
Packit Service b23acc
#if NM_MORE_ASSERTS > 0
Packit Service b23acc
Packit Service b23acc
	/* Just appends @str to @gstring, but also assert that
Packit Service b23acc
	 * no escaping is necessary.
Packit Service b23acc
	 *
Packit Service b23acc
	 * Use nm_utils_escaped_tokens_escape_gstr_assert() instead
Packit Service b23acc
	 * of nm_utils_escaped_tokens_escape_gstr(), if you *know* that
Packit Service b23acc
	 * @str contains no delimiters, no backslashes, and no trailing
Packit Service b23acc
	 * whitespace that requires escaping. */
Packit Service b23acc
Packit Service b23acc
	nm_assert (str);
Packit Service b23acc
	nm_assert (gstring);
Packit Service b23acc
	nm_assert (delimiters);
Packit Service b23acc
Packit Service b23acc
	{
Packit Service b23acc
		gs_free char *str_to_free = NULL;
Packit Service b23acc
		const char *str0;
Packit Service b23acc
Packit Service b23acc
		str0 = nm_utils_escaped_tokens_escape (str, delimiters, &str_to_free);
Packit Service b23acc
		nm_assert (str0 == str);
Packit Service b23acc
		nm_assert (!str_to_free);
Packit Service b23acc
	}
Packit Service b23acc
#endif
Packit Service b23acc
Packit Service b23acc
	g_string_append (gstring, str);
Packit Service b23acc
	return gstring;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline GString *
Packit Service b23acc
nm_utils_escaped_tokens_escape_gstr (const char *str,
Packit Service b23acc
                                     const char *delimiters,
Packit Service b23acc
                                     GString *gstring)
Packit Service b23acc
{
Packit Service b23acc
	gs_free char *str_to_free = NULL;
Packit Service b23acc
Packit Service b23acc
	nm_assert (str);
Packit Service b23acc
	nm_assert (gstring);
Packit Service b23acc
Packit Service b23acc
	g_string_append (gstring,
Packit Service b23acc
	                 nm_utils_escaped_tokens_escape (str, delimiters, &str_to_free));
Packit Service b23acc
	return gstring;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
char **nm_utils_strsplit_quoted (const char *str);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline const char **
Packit Service b23acc
nm_utils_escaped_tokens_options_split_list (const char *str)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_strsplit_set_full (str,
Packit Service b23acc
	                                   ",",
Packit Service b23acc
	                                     NM_UTILS_STRSPLIT_SET_FLAGS_STRSTRIP
Packit Service b23acc
	                                   | NM_UTILS_STRSPLIT_SET_FLAGS_ALLOW_ESCAPING);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void nm_utils_escaped_tokens_options_split (char *str,
Packit Service b23acc
                                            const char **out_key,
Packit Service b23acc
                                            const char **out_val);
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
nm_utils_escaped_tokens_options_escape_key (const char *key,
Packit Service b23acc
                                            char **out_to_free)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_escaped_tokens_escape_full (key,
Packit Service b23acc
	                                            ",=",
Packit Service b23acc
	                                            NULL,
Packit Service b23acc
	                                              NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_BACKSLASH_AS_NEEDED
Packit Service b23acc
	                                            | NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_LEADING_SPACE
Packit Service b23acc
	                                            | NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_TRAILING_SPACE,
Packit Service b23acc
	                                            out_to_free);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char *
Packit Service b23acc
nm_utils_escaped_tokens_options_escape_val (const char *val,
Packit Service b23acc
                                            char **out_to_free)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_escaped_tokens_escape_full (val,
Packit Service b23acc
	                                            ",",
Packit Service b23acc
	                                            "=",
Packit Service b23acc
	                                              NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_BACKSLASH_AS_NEEDED
Packit Service b23acc
	                                            | NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_LEADING_SPACE
Packit Service b23acc
	                                            | NM_UTILS_ESCAPED_TOKENS_ESCAPE_FLAGS_ESCAPE_TRAILING_SPACE,
Packit Service b23acc
	                                            out_to_free);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_CHECKSUM_LENGTH_MD5          16
Packit Service b23acc
#define NM_UTILS_CHECKSUM_LENGTH_SHA1         20
Packit Service b23acc
#define NM_UTILS_CHECKSUM_LENGTH_SHA256       32
Packit Service b23acc
Packit Service b23acc
#define nm_utils_checksum_get_digest(sum, arr) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		GChecksum *const _sum = (sum); \
Packit Service b23acc
		gsize _len; \
Packit Service b23acc
		\
Packit Service b23acc
		G_STATIC_ASSERT_EXPR (   sizeof (arr) == NM_UTILS_CHECKSUM_LENGTH_MD5 \
Packit Service b23acc
		                      || sizeof (arr) == NM_UTILS_CHECKSUM_LENGTH_SHA1 \
Packit Service b23acc
		                      || sizeof (arr) == NM_UTILS_CHECKSUM_LENGTH_SHA256); \
Packit Service b23acc
		G_STATIC_ASSERT_EXPR (sizeof (arr) == G_N_ELEMENTS (arr)); \
Packit Service b23acc
		\
Packit Service b23acc
		nm_assert (_sum); \
Packit Service b23acc
		\
Packit Service b23acc
		_len = G_N_ELEMENTS (arr); \
Packit Service b23acc
		\
Packit Service b23acc
		g_checksum_get_digest (_sum, (arr), &_len); \
Packit Service b23acc
		nm_assert (_len == G_N_ELEMENTS (arr)); \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define nm_utils_checksum_get_digest_len(sum, buf, len) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		GChecksum *const _sum = (sum); \
Packit Service b23acc
		const gsize _len0 = (len); \
Packit Service b23acc
		gsize _len; \
Packit Service b23acc
		\
Packit Service b23acc
		nm_assert (NM_IN_SET (_len0, NM_UTILS_CHECKSUM_LENGTH_MD5, \
Packit Service b23acc
		                             NM_UTILS_CHECKSUM_LENGTH_SHA1, \
Packit Service b23acc
		                             NM_UTILS_CHECKSUM_LENGTH_SHA256)); \
Packit Service b23acc
		nm_assert (_sum); \
Packit Service b23acc
		\
Packit Service b23acc
		_len = _len0; \
Packit Service b23acc
		g_checksum_get_digest (_sum, (buf), &_len); \
Packit Service b23acc
		nm_assert (_len == _len0); \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
guint32 _nm_utils_ip4_prefix_to_netmask (guint32 prefix);
Packit Service b23acc
guint32 _nm_utils_ip4_get_default_prefix (guint32 ip);
Packit Service b23acc
Packit Service b23acc
gconstpointer          nm_utils_ipx_address_clear_host_address (int family, gpointer dst, gconstpointer src, guint8 plen);
Packit Service b23acc
in_addr_t              nm_utils_ip4_address_clear_host_address (in_addr_t addr, guint8 plen);
Packit Service b23acc
const struct in6_addr *nm_utils_ip6_address_clear_host_address (struct in6_addr *dst, const struct in6_addr *src, guint8 plen);
Packit Service b23acc
int nm_utils_ip6_address_same_prefix_cmp (const struct in6_addr *addr_a, const struct in6_addr *addr_b, guint8 plen);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_ip_is_site_local (int addr_family,
Packit Service b23acc
                                    const void *address);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_parse_inaddr_bin_full (int addr_family,
Packit Service b23acc
                                         gboolean accept_legacy,
Packit Service b23acc
                                         const char *text,
Packit Service b23acc
                                         int *out_addr_family,
Packit Service b23acc
                                         gpointer out_addr);
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_parse_inaddr_bin (int addr_family,
Packit Service b23acc
                           const char *text,
Packit Service b23acc
                           int *out_addr_family,
Packit Service b23acc
                           gpointer out_addr)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_parse_inaddr_bin_full (addr_family, FALSE, text, out_addr_family, out_addr);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_parse_inaddr (int addr_family,
Packit Service b23acc
                                const char *text,
Packit Service b23acc
                                char **out_addr);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_parse_inaddr_prefix_bin (int addr_family,
Packit Service b23acc
                                           const char *text,
Packit Service b23acc
                                           int *out_addr_family,
Packit Service b23acc
                                           gpointer out_addr,
Packit Service b23acc
                                           int *out_prefix);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_parse_inaddr_prefix (int addr_family,
Packit Service b23acc
                                       const char *text,
Packit Service b23acc
                                       char **out_addr,
Packit Service b23acc
                                       int *out_prefix);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_parse_next_line (const char **inout_ptr,
Packit Service b23acc
                                   gsize *inout_len,
Packit Service b23acc
                                   const char **out_line,
Packit Service b23acc
                                   gsize *out_line_len);
Packit Service b23acc
Packit Service b23acc
gint64 nm_g_ascii_strtoll (const char *nptr,
Packit Service b23acc
                           char **endptr,
Packit Service b23acc
                           guint base);
Packit Service b23acc
Packit Service b23acc
guint64 nm_g_ascii_strtoull (const char *nptr,
Packit Service b23acc
                             char **endptr,
Packit Service b23acc
                             guint base);
Packit Service b23acc
Packit Service b23acc
double nm_g_ascii_strtod (const char *nptr,
Packit Service b23acc
                          char **endptr);
Packit Service b23acc
Packit Service b23acc
gint64  _nm_utils_ascii_str_to_int64  (const char *str, guint base, gint64  min, gint64  max, gint64  fallback);
Packit Service b23acc
guint64 _nm_utils_ascii_str_to_uint64 (const char *str, guint base, guint64 min, guint64 max, guint64 fallback);
Packit Service b23acc
Packit Service b23acc
int _nm_utils_ascii_str_to_bool (const char *str,
Packit Service b23acc
                                  int default_value);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
extern char _nm_utils_to_string_buffer[2096];
Packit Service b23acc
Packit Service b23acc
void     nm_utils_to_string_buffer_init (char **buf, gsize *len);
Packit Service b23acc
gboolean nm_utils_to_string_buffer_init_null (gconstpointer obj, char **buf, gsize *len);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	unsigned flag;
Packit Service b23acc
	const char *name;
Packit Service b23acc
} NMUtilsFlags2StrDesc;
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_FLAGS2STR(f, n) { .flag = f, .name = ""n, }
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_FLAGS2STR_DEFINE(fcn_name, flags_type, ...) \
Packit Service b23acc
const char * \
Packit Service b23acc
fcn_name (flags_type flags, char *buf, gsize len) \
Packit Service b23acc
{ \
Packit Service b23acc
	static const NMUtilsFlags2StrDesc descs[] = { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	}; \
Packit Service b23acc
	G_STATIC_ASSERT (sizeof (flags_type) <= sizeof (unsigned)); \
Packit Service b23acc
	return nm_utils_flags2str (descs, G_N_ELEMENTS (descs), flags, buf, len); \
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
const char *nm_utils_flags2str (const NMUtilsFlags2StrDesc *descs,
Packit Service b23acc
                                gsize n_descs,
Packit Service b23acc
                                unsigned flags,
Packit Service b23acc
                                char *buf,
Packit Service b23acc
                                gsize len);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_ENUM2STR(v, n)     (void) 0; case v: s = ""n""; break; (void) 0
Packit Service b23acc
#define NM_UTILS_ENUM2STR_IGNORE(v) (void) 0; case v: break; (void) 0
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_ENUM2STR_DEFINE_FULL(fcn_name, lookup_type, int_fmt, ...) \
Packit Service b23acc
const char * \
Packit Service b23acc
fcn_name (lookup_type val, char *buf, gsize len) \
Packit Service b23acc
{ \
Packit Service b23acc
	nm_utils_to_string_buffer_init (&buf, &len;; \
Packit Service b23acc
	if (len) { \
Packit Service b23acc
		const char *s = NULL; \
Packit Service b23acc
		switch (val) { \
Packit Service b23acc
			(void) 0, \
Packit Service b23acc
			__VA_ARGS__ \
Packit Service b23acc
			(void) 0; \
Packit Service b23acc
		}; \
Packit Service b23acc
		if (s) \
Packit Service b23acc
			g_strlcpy (buf, s, len); \
Packit Service b23acc
		else \
Packit Service b23acc
			g_snprintf (buf, len, "(%"int_fmt")", val); \
Packit Service b23acc
	} \
Packit Service b23acc
	return buf; \
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_ENUM2STR_DEFINE(fcn_name, lookup_type, ...) \
Packit Service b23acc
	NM_UTILS_ENUM2STR_DEFINE_FULL (fcn_name, lookup_type, "d", __VA_ARGS__)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define _nm_g_slice_free_fcn_define(mem_size) \
Packit Service b23acc
static inline void \
Packit Service b23acc
_nm_g_slice_free_fcn_##mem_size (gpointer mem_block) \
Packit Service b23acc
{ \
Packit Service b23acc
	g_slice_free1 (mem_size, mem_block); \
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
_nm_g_slice_free_fcn_define (1)
Packit Service b23acc
_nm_g_slice_free_fcn_define (2)
Packit Service b23acc
_nm_g_slice_free_fcn_define (4)
Packit Service b23acc
_nm_g_slice_free_fcn_define (8)
Packit Service b23acc
_nm_g_slice_free_fcn_define (10)
Packit Service b23acc
_nm_g_slice_free_fcn_define (12)
Packit Service b23acc
_nm_g_slice_free_fcn_define (16)
Packit Service b23acc
_nm_g_slice_free_fcn_define (32)
Packit Service b23acc
Packit Service b23acc
#define nm_g_slice_free_fcn1(mem_size) \
Packit Service b23acc
	({ \
Packit Service b23acc
		void (*_fcn) (gpointer); \
Packit Service b23acc
		\
Packit Service b23acc
		/* If mem_size is a compile time constant, the compiler
Packit Service b23acc
		 * will be able to optimize this. Hence, you don't want
Packit Service b23acc
		 * to call this with a non-constant size argument. */ \
Packit Service b23acc
		G_STATIC_ASSERT_EXPR (   ((mem_size) ==  1) \
Packit Service b23acc
		                      || ((mem_size) ==  2) \
Packit Service b23acc
		                      || ((mem_size) ==  4) \
Packit Service b23acc
		                      || ((mem_size) ==  8) \
Packit Service b23acc
		                      || ((mem_size) == 10) \
Packit Service b23acc
		                      || ((mem_size) == 12) \
Packit Service b23acc
		                      || ((mem_size) == 16) \
Packit Service b23acc
		                      || ((mem_size) == 32)); \
Packit Service b23acc
		switch ((mem_size)) { \
Packit Service b23acc
		case  1: _fcn = _nm_g_slice_free_fcn_1;  break; \
Packit Service b23acc
		case  2: _fcn = _nm_g_slice_free_fcn_2;  break; \
Packit Service b23acc
		case  4: _fcn = _nm_g_slice_free_fcn_4;  break; \
Packit Service b23acc
		case  8: _fcn = _nm_g_slice_free_fcn_8;  break; \
Packit Service b23acc
		case 10: _fcn = _nm_g_slice_free_fcn_10; break; \
Packit Service b23acc
		case 12: _fcn = _nm_g_slice_free_fcn_12; break; \
Packit Service b23acc
		case 16: _fcn = _nm_g_slice_free_fcn_16; break; \
Packit Service b23acc
		case 32: _fcn = _nm_g_slice_free_fcn_32; break; \
Packit Service b23acc
		default: g_assert_not_reached (); _fcn = NULL; break; \
Packit Service b23acc
		} \
Packit Service b23acc
		_fcn; \
Packit Service b23acc
	})
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nm_g_slice_free_fcn:
Packit Service b23acc
 * @type: type argument for sizeof() operator that you would
Packit Service b23acc
 *   pass to g_slice_new().
Packit Service b23acc
 *
Packit Service b23acc
 * Returns: a function pointer with GDestroyNotify signature
Packit Service b23acc
 *   for g_slice_free(type,*).
Packit Service b23acc
 *
Packit Service b23acc
 * Only certain types are implemented. You'll get a compile time
Packit Service b23acc
 * error for the wrong types. */
Packit Service b23acc
#define nm_g_slice_free_fcn(type) (nm_g_slice_free_fcn1 (sizeof (type)))
Packit Service b23acc
Packit Service b23acc
#define nm_g_slice_free_fcn_gint64 (nm_g_slice_free_fcn (gint64))
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
/* Like g_error_matches() however:
Packit Service b23acc
 * - as macro it is always inlined.
Packit Service b23acc
 * - the @domain is usually a error quark getter function that cannot
Packit Service b23acc
 *   be inlined. This macro calls the getter only if there is an error (lazy).
Packit Service b23acc
 * - accept a list of allowed codes, instead of only one.
Packit Service b23acc
 */
Packit Service b23acc
#define nm_g_error_matches(error, err_domain, ...) \
Packit Service b23acc
	({ \
Packit Service b23acc
		const GError *const _error = (error); \
Packit Service b23acc
		\
Packit Service b23acc
		   _error \
Packit Service b23acc
		&& _error->domain == (err_domain) \
Packit Service b23acc
		&& NM_IN_SET (_error->code, __VA_ARGS__); \
Packit Service b23acc
	})
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_set_error_take (GError **error, GError *error_take)
Packit Service b23acc
{
Packit Service b23acc
	if (!error_take)
Packit Service b23acc
		g_return_if_reached ();
Packit Service b23acc
	if (!error) {
Packit Service b23acc
		g_error_free (error_take);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
	if (*error) {
Packit Service b23acc
		g_error_free (error_take);
Packit Service b23acc
		g_return_if_reached ();
Packit Service b23acc
	}
Packit Service b23acc
	*error = error_take;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_g_set_error_take_lazy(error, error_take_lazy) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		GError **_error = (error); \
Packit Service b23acc
		\
Packit Service b23acc
		if (_error) \
Packit Service b23acc
			nm_g_set_error_take (_error, (error_take_lazy)); \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * NMUtilsError:
Packit Service b23acc
 * @NM_UTILS_ERROR_UNKNOWN: unknown or unclassified error
Packit Service b23acc
 * @NM_UTILS_ERROR_CANCELLED_DISPOSING: when disposing an object that has
Packit Service b23acc
 *   pending aynchronous operations, the operation is cancelled with this
Packit Service b23acc
 *   error reason. Depending on the usage, this might indicate a bug because
Packit Service b23acc
 *   usually the target object should stay alive as long as there are pending
Packit Service b23acc
 *   operations.
Packit Service b23acc
 *
Packit Service b23acc
 * @NM_UTILS_ERROR_CONNECTION_AVAILABLE_INCOMPATIBLE: used for a very particular
Packit Service b23acc
 *   purpose during nm_device_check_connection_compatible() to indicate that
Packit Service b23acc
 *   the profile does not match the device already because their type differs.
Packit Service b23acc
 *   That is, there is a fundamental reason of trying to check a profile that
Packit Service b23acc
 *   cannot possibly match on this device.
Packit Service b23acc
 * @NM_UTILS_ERROR_CONNECTION_AVAILABLE_UNMANAGED_DEVICE: used for a very particular
Packit Service b23acc
 *   purpose during nm_device_check_connection_available(), to indicate that the
Packit Service b23acc
 *   device is not available because it is unmanaged.
Packit Service b23acc
 * @NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY: the profile is currently not
Packit Service b23acc
 *   available/compatible with the device, but this may be only temporary.
Packit Service b23acc
 *
Packit Service b23acc
 * @NM_UTILS_ERROR_SETTING_MISSING: the setting is missing
Packit Service b23acc
 *
Packit Service b23acc
 * @NM_UTILS_ERROR_INVALID_ARGUMENT: invalid argument.
Packit Service b23acc
 */
Packit Service b23acc
typedef enum {
Packit Service b23acc
	NM_UTILS_ERROR_UNKNOWN = 0,                 /*< nick=Unknown >*/
Packit Service b23acc
	NM_UTILS_ERROR_CANCELLED_DISPOSING,         /*< nick=CancelledDisposing >*/
Packit Service b23acc
	NM_UTILS_ERROR_INVALID_ARGUMENT,            /*< nick=InvalidArgument >*/
Packit Service b23acc
Packit Service b23acc
	/* the following codes have a special meaning and are exactly used for
Packit Service b23acc
	 * nm_device_check_connection_compatible() and nm_device_check_connection_available().
Packit Service b23acc
	 *
Packit Service b23acc
	 * Actually, their meaning is not very important (so, don't think too
Packit Service b23acc
	 * hard about the name of these error codes). What is important, is their
Packit Service b23acc
	 * relative order (i.e. the integer value of the codes). When manager
Packit Service b23acc
	 * searches for a suitable device, it will check all devices whether
Packit Service b23acc
	 * a profile can be activated. If they all fail, it will pick the error
Packit Service b23acc
	 * message from the device that returned the *highest* error code,
Packit Service b23acc
	 * in the hope that this message makes the most sense for the caller.
Packit Service b23acc
	 * */
Packit Service b23acc
	NM_UTILS_ERROR_CONNECTION_AVAILABLE_INCOMPATIBLE,
Packit Service b23acc
	NM_UTILS_ERROR_CONNECTION_AVAILABLE_UNMANAGED_DEVICE,
Packit Service b23acc
	NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
Packit Service b23acc
	NM_UTILS_ERROR_SETTING_MISSING,
Packit Service b23acc
Packit Service b23acc
} NMUtilsError;
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_ERROR (nm_utils_error_quark ())
Packit Service b23acc
GQuark nm_utils_error_quark (void);
Packit Service b23acc
Packit Service b23acc
void nm_utils_error_set_cancelled (GError **error,
Packit Service b23acc
                                   gboolean is_disposing,
Packit Service b23acc
                                   const char *instance_name);
Packit Service b23acc
Packit Service b23acc
static inline GError *
Packit Service b23acc
nm_utils_error_new_cancelled (gboolean is_disposing,
Packit Service b23acc
                              const char *instance_name)
Packit Service b23acc
{
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	nm_utils_error_set_cancelled (&error, is_disposing, instance_name);
Packit Service b23acc
	return error;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_error_is_cancelled_or_disposing (GError *error);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_error_is_cancelled (GError *error)
Packit Service b23acc
{
Packit Service b23acc
	return    error
Packit Service b23acc
	       && error->code == G_IO_ERROR_CANCELLED
Packit Service b23acc
	       && error->domain == G_IO_ERROR;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_error_is_notfound (GError *error);
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_utils_error_set_literal (GError **error, int error_code, const char *literal)
Packit Service b23acc
{
Packit Service b23acc
	g_set_error_literal (error, NM_UTILS_ERROR, error_code, literal);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_utils_error_set(error, error_code, ...) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		if (NM_NARG (__VA_ARGS__) == 1) { \
Packit Service b23acc
			g_set_error_literal ((error), NM_UTILS_ERROR, (error_code), _NM_UTILS_MACRO_FIRST (__VA_ARGS__)); \
Packit Service b23acc
		} else { \
Packit Service b23acc
			g_set_error ((error), NM_UTILS_ERROR, (error_code), __VA_ARGS__); \
Packit Service b23acc
		} \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define nm_utils_error_set_errno(error, errsv, fmt, ...) \
Packit Service b23acc
	G_STMT_START { \
Packit Service b23acc
		char _bstrerr[NM_STRERROR_BUFSIZE]; \
Packit Service b23acc
		\
Packit Service b23acc
		g_set_error ((error), \
Packit Service b23acc
		             NM_UTILS_ERROR, \
Packit Service b23acc
		             NM_UTILS_ERROR_UNKNOWN, \
Packit Service b23acc
		             fmt, \
Packit Service b23acc
		             ##__VA_ARGS__, \
Packit Service b23acc
		             nm_strerror_native_r (({ \
Packit Service b23acc
		                                      const int _errsv = (errsv); \
Packit Service b23acc
		                                      \
Packit Service b23acc
		                                      (  _errsv >= 0 \
Packit Service b23acc
		                                       ? _errsv \
Packit Service b23acc
		                                       : (  G_UNLIKELY (_errsv == G_MININT) \
Packit Service b23acc
		                                          ? G_MAXINT \
Packit Service b23acc
		                                          : -errsv)); \
Packit Service b23acc
		                                   }), \
Packit Service b23acc
		                                   _bstrerr, \
Packit Service b23acc
		                                   sizeof (_bstrerr))); \
Packit Service b23acc
	} G_STMT_END
Packit Service b23acc
Packit Service b23acc
#define nm_utils_error_new(error_code, ...) \
Packit Service b23acc
	(   (NM_NARG (__VA_ARGS__) == 1) \
Packit Service b23acc
	 ? g_error_new_literal (NM_UTILS_ERROR, (error_code), _NM_UTILS_MACRO_FIRST (__VA_ARGS__)) \
Packit Service b23acc
	 : g_error_new         (NM_UTILS_ERROR, (error_code), __VA_ARGS__))
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property (GObject *object,
Packit Service b23acc
                                   const char *property_name,
Packit Service b23acc
                                   const GValue *value,
Packit Service b23acc
                                   GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_string (GObject *object,
Packit Service b23acc
                                          const char *property_name,
Packit Service b23acc
                                          const char *value,
Packit Service b23acc
                                          GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_string_static (GObject *object,
Packit Service b23acc
                                                 const char *property_name,
Packit Service b23acc
                                                 const char *value,
Packit Service b23acc
                                                 GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_string_take (GObject *object,
Packit Service b23acc
                                               const char *property_name,
Packit Service b23acc
                                               char *value,
Packit Service b23acc
                                               GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_boolean (GObject *object,
Packit Service b23acc
                                           const char *property_name,
Packit Service b23acc
                                           gboolean value,
Packit Service b23acc
                                           GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_char (GObject *object,
Packit Service b23acc
                                        const char *property_name,
Packit Service b23acc
                                        gint8 value,
Packit Service b23acc
                                        GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_uchar (GObject *object,
Packit Service b23acc
                                         const char *property_name,
Packit Service b23acc
                                         guint8 value,
Packit Service b23acc
                                         GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_int (GObject *object,
Packit Service b23acc
                                       const char *property_name,
Packit Service b23acc
                                       int value,
Packit Service b23acc
                                       GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_int64 (GObject *object,
Packit Service b23acc
                                         const char *property_name,
Packit Service b23acc
                                         gint64 value,
Packit Service b23acc
                                         GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_uint (GObject *object,
Packit Service b23acc
                                        const char *property_name,
Packit Service b23acc
                                        guint value,
Packit Service b23acc
                                        GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_uint64 (GObject *object,
Packit Service b23acc
                                          const char *property_name,
Packit Service b23acc
                                          guint64 value,
Packit Service b23acc
                                          GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_flags (GObject *object,
Packit Service b23acc
                                         const char *property_name,
Packit Service b23acc
                                         GType gtype,
Packit Service b23acc
                                         guint value,
Packit Service b23acc
                                         GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_g_object_set_property_enum (GObject *object,
Packit Service b23acc
                                        const char *property_name,
Packit Service b23acc
                                        GType gtype,
Packit Service b23acc
                                        int value,
Packit Service b23acc
                                        GError **error);
Packit Service b23acc
Packit Service b23acc
GParamSpec *nm_g_object_class_find_property_from_gtype (GType gtype,
Packit Service b23acc
                                                        const char *property_name);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define _NM_G_PARAM_SPEC_CAST(param_spec, _value_type, _c_type) \
Packit Service b23acc
	({ \
Packit Service b23acc
		const GParamSpec *const _param_spec = (param_spec); \
Packit Service b23acc
		\
Packit Service b23acc
		nm_assert (   !_param_spec \
Packit Service b23acc
		           || _param_spec->value_type == (_value_type)); \
Packit Service b23acc
		((const _c_type *) _param_spec); \
Packit Service b23acc
	})
Packit Service b23acc
Packit Service b23acc
#define NM_G_PARAM_SPEC_CAST_BOOLEAN(param_spec) _NM_G_PARAM_SPEC_CAST (param_spec, G_TYPE_BOOLEAN, GParamSpecBoolean)
Packit Service b23acc
#define NM_G_PARAM_SPEC_CAST_UINT(param_spec)    _NM_G_PARAM_SPEC_CAST (param_spec, G_TYPE_UINT,    GParamSpecUInt)
Packit Service b23acc
#define NM_G_PARAM_SPEC_CAST_UINT64(param_spec)  _NM_G_PARAM_SPEC_CAST (param_spec, G_TYPE_UINT64,  GParamSpecUInt64)
Packit Service b23acc
Packit Service b23acc
#define NM_G_PARAM_SPEC_GET_DEFAULT_BOOLEAN(param_spec) (NM_G_PARAM_SPEC_CAST_BOOLEAN (NM_ENSURE_NOT_NULL (param_spec))->default_value)
Packit Service b23acc
#define NM_G_PARAM_SPEC_GET_DEFAULT_UINT(param_spec)    (NM_G_PARAM_SPEC_CAST_UINT    (NM_ENSURE_NOT_NULL (param_spec))->default_value)
Packit Service b23acc
#define NM_G_PARAM_SPEC_GET_DEFAULT_UINT64(param_spec)  (NM_G_PARAM_SPEC_CAST_UINT64  (NM_ENSURE_NOT_NULL (param_spec))->default_value)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
GType nm_g_type_find_implementing_class_for_property (GType gtype,
Packit Service b23acc
                                                      const char *pname);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef enum {
Packit Service b23acc
	NM_UTILS_STR_UTF8_SAFE_FLAG_NONE                = 0,
Packit Service b23acc
Packit Service b23acc
	/* This flag only has an effect during escaping. */
Packit Service b23acc
	NM_UTILS_STR_UTF8_SAFE_FLAG_ESCAPE_CTRL         = 0x0001,
Packit Service b23acc
Packit Service b23acc
	/* This flag only has an effect during escaping. */
Packit Service b23acc
	NM_UTILS_STR_UTF8_SAFE_FLAG_ESCAPE_NON_ASCII    = 0x0002,
Packit Service b23acc
Packit Service b23acc
	/* This flag only has an effect during escaping to ensure we
Packit Service b23acc
	 * don't leak secrets in memory. Note that during unescape we
Packit Service b23acc
	 * know the maximum result size from the beginning, and no
Packit Service b23acc
	 * reallocation happens. Thus, unescape always avoids leaking
Packit Service b23acc
	 * secrets already. */
Packit Service b23acc
	NM_UTILS_STR_UTF8_SAFE_FLAG_SECRET              = 0x0004,
Packit Service b23acc
Packit Service b23acc
	/* This flag only has an effect during unescaping. It means
Packit Service b23acc
	 * that non-escaped whitespaces (g_ascii_isspace()) will be
Packit Service b23acc
	 * stripped from the front and end of the string. Note that
Packit Service b23acc
	 * this flag is only useful for gracefully accepting user input
Packit Service b23acc
	 * with spaces. With this flag, escape and unescape may no longer
Packit Service b23acc
	 * yield the original input. */
Packit Service b23acc
	NM_UTILS_STR_UTF8_SAFE_UNESCAPE_STRIP_SPACES    = 0x0008,
Packit Service b23acc
} NMUtilsStrUtf8SafeFlags;
Packit Service b23acc
Packit Service b23acc
const char *nm_utils_buf_utf8safe_escape (gconstpointer buf, gssize buflen, NMUtilsStrUtf8SafeFlags flags, char **to_free);
Packit Service b23acc
char *nm_utils_buf_utf8safe_escape_cp (gconstpointer buf, gssize buflen, NMUtilsStrUtf8SafeFlags flags);
Packit Service b23acc
const char *nm_utils_buf_utf8safe_escape_bytes (GBytes *bytes, NMUtilsStrUtf8SafeFlags flags, char **to_free);
Packit Service b23acc
gconstpointer nm_utils_buf_utf8safe_unescape (const char *str, NMUtilsStrUtf8SafeFlags flags, gsize *out_len, gpointer *to_free);
Packit Service b23acc
Packit Service b23acc
const char *nm_utils_str_utf8safe_escape   (const char *str, NMUtilsStrUtf8SafeFlags flags, char **to_free);
Packit Service b23acc
const char *nm_utils_str_utf8safe_unescape (const char *str, NMUtilsStrUtf8SafeFlags flags, char **to_free);
Packit Service b23acc
Packit Service b23acc
char *nm_utils_str_utf8safe_escape_cp   (const char *str, NMUtilsStrUtf8SafeFlags flags);
Packit Service b23acc
char *nm_utils_str_utf8safe_unescape_cp (const char *str, NMUtilsStrUtf8SafeFlags flags);
Packit Service b23acc
Packit Service b23acc
char *nm_utils_str_utf8safe_escape_take (char *str, NMUtilsStrUtf8SafeFlags flags);
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_variant_unref_floating (GVariant *var)
Packit Service b23acc
{
Packit Service b23acc
	/* often a function wants to keep a reference to an input variant.
Packit Service b23acc
	 * It uses g_variant_ref_sink() to either increase the ref-count,
Packit Service b23acc
	 * or take ownership of a possibly floating reference.
Packit Service b23acc
	 *
Packit Service b23acc
	 * If the function doesn't actually want to do anything with the
Packit Service b23acc
	 * input variant, it still must make sure that a passed in floating
Packit Service b23acc
	 * reference is consumed. Hence, this helper which:
Packit Service b23acc
	 *
Packit Service b23acc
	 *   - does nothing if @var is not floating
Packit Service b23acc
	 *   - unrefs (consumes) @var if it is floating. */
Packit Service b23acc
	if (g_variant_is_floating (var))
Packit Service b23acc
		g_variant_unref (var);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_g_variant_lookup(dictionary, ...) \
Packit Service b23acc
	({ \
Packit Service b23acc
		GVariant *const _dictionary = (dictionary); \
Packit Service b23acc
		\
Packit Service b23acc
		(   _dictionary \
Packit Service b23acc
		 && g_variant_lookup (_dictionary, __VA_ARGS__)); \
Packit Service b23acc
	})
Packit Service b23acc
Packit Service b23acc
static inline GVariant *
Packit Service b23acc
nm_g_variant_lookup_value (GVariant *dictionary,
Packit Service b23acc
                           const char *key,
Packit Service b23acc
                           const GVariantType *expected_type)
Packit Service b23acc
{
Packit Service b23acc
	return   dictionary
Packit Service b23acc
	       ? g_variant_lookup_value (dictionary, key, expected_type)
Packit Service b23acc
	       : NULL;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_g_variant_is_of_type (GVariant *value,
Packit Service b23acc
                         const GVariantType *type)
Packit Service b23acc
{
Packit Service b23acc
	return    value
Packit Service b23acc
	       && g_variant_is_of_type (value, type);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_variant_builder_add_sv (GVariantBuilder *builder, const char *key, GVariant *val)
Packit Service b23acc
{
Packit Service b23acc
	g_variant_builder_add (builder, "{sv}", key, val);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_variant_builder_add_sv_bytearray (GVariantBuilder *builder, const char *key, const guint8 *arr, gsize len)
Packit Service b23acc
{
Packit Service b23acc
	g_variant_builder_add (builder, "{sv}", key, g_variant_new_fixed_array (G_VARIANT_TYPE_BYTE, arr, len, 1));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_variant_builder_add_sv_uint32 (GVariantBuilder *builder, const char *key, guint32 val)
Packit Service b23acc
{
Packit Service b23acc
	nm_g_variant_builder_add_sv (builder, key, g_variant_new_uint32 (val));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_variant_builder_add_sv_str (GVariantBuilder *builder, const char *key, const char *str)
Packit Service b23acc
{
Packit Service b23acc
	nm_g_variant_builder_add_sv (builder, key, g_variant_new_string (str));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_g_source_destroy_and_unref (GSource *source)
Packit Service b23acc
{
Packit Service b23acc
	g_source_destroy (source);
Packit Service b23acc
	g_source_unref (source);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_clear_g_source_inst(ptr) (nm_clear_pointer ((ptr), nm_g_source_destroy_and_unref))
Packit Service b23acc
Packit Service b23acc
NM_AUTO_DEFINE_FCN0 (GSource *, _nm_auto_destroy_and_unref_gsource, nm_g_source_destroy_and_unref);
Packit Service b23acc
#define nm_auto_destroy_and_unref_gsource nm_auto(_nm_auto_destroy_and_unref_gsource)
Packit Service b23acc
Packit Service b23acc
NM_AUTO_DEFINE_FCN0 (GMainContext *, _nm_auto_pop_gmaincontext, g_main_context_pop_thread_default)
Packit Service b23acc
#define nm_auto_pop_gmaincontext nm_auto (_nm_auto_pop_gmaincontext)
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_source_func_unref_gobject (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (G_IS_OBJECT (user_data));
Packit Service b23acc
	g_object_unref (user_data);
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
GSource *nm_g_idle_source_new (int priority,
Packit Service b23acc
                               GSourceFunc func,
Packit Service b23acc
                               gpointer user_data,
Packit Service b23acc
                               GDestroyNotify destroy_notify);
Packit Service b23acc
Packit Service b23acc
GSource *nm_g_timeout_source_new (guint timeout_msec,
Packit Service b23acc
                                  int priority,
Packit Service b23acc
                                  GSourceFunc func,
Packit Service b23acc
                                  gpointer user_data,
Packit Service b23acc
                                  GDestroyNotify destroy_notify);
Packit Service b23acc
GSource *nm_g_unix_fd_source_new (int fd,
Packit Service b23acc
                                  GIOCondition io_condition,
Packit Service b23acc
                                  int priority,
Packit Service b23acc
                                  gboolean (*source_func) (int fd,
Packit Service b23acc
                                                           GIOCondition condition,
Packit Service b23acc
                                                           gpointer user_data),
Packit Service b23acc
                                  gpointer user_data,
Packit Service b23acc
                                  GDestroyNotify destroy_notify);
Packit Service b23acc
GSource *nm_g_unix_signal_source_new (int signum,
Packit Service b23acc
                                      int priority,
Packit Service b23acc
                                      GSourceFunc handler,
Packit Service b23acc
                                      gpointer user_data,
Packit Service b23acc
                                      GDestroyNotify notify);
Packit Service b23acc
Packit Service b23acc
static inline GSource *
Packit Service b23acc
nm_g_source_attach (GSource *source,
Packit Service b23acc
                    GMainContext *context)
Packit Service b23acc
{
Packit Service b23acc
	g_source_attach (source, context);
Packit Service b23acc
	return source;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
NM_AUTO_DEFINE_FCN0 (GMainContext *, _nm_auto_unref_gmaincontext, g_main_context_unref)
Packit Service b23acc
#define nm_auto_unref_gmaincontext nm_auto (_nm_auto_unref_gmaincontext)
Packit Service b23acc
Packit Service b23acc
static inline GMainContext *
Packit Service b23acc
nm_g_main_context_push_thread_default (GMainContext *context)
Packit Service b23acc
{
Packit Service b23acc
	/* This function is to work together with nm_auto_pop_gmaincontext. */
Packit Service b23acc
	if (G_UNLIKELY (!context))
Packit Service b23acc
		context = g_main_context_default ();
Packit Service b23acc
	g_main_context_push_thread_default (context);
Packit Service b23acc
	return context;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_g_main_context_is_thread_default (GMainContext *context)
Packit Service b23acc
{
Packit Service b23acc
	GMainContext *cur_context;
Packit Service b23acc
Packit Service b23acc
	cur_context = g_main_context_get_thread_default ();
Packit Service b23acc
	if (cur_context == context)
Packit Service b23acc
		return TRUE;
Packit Service b23acc
Packit Service b23acc
	if (G_UNLIKELY (!cur_context))
Packit Service b23acc
		cur_context = g_main_context_default ();
Packit Service b23acc
	else if (G_UNLIKELY (!context))
Packit Service b23acc
		context = g_main_context_default ();
Packit Service b23acc
	else
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	return (cur_context == context);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline GMainContext *
Packit Service b23acc
nm_g_main_context_push_thread_default_if_necessary (GMainContext *context)
Packit Service b23acc
{
Packit Service b23acc
	GMainContext *cur_context;
Packit Service b23acc
Packit Service b23acc
	cur_context = g_main_context_get_thread_default ();
Packit Service b23acc
	if (cur_context == context)
Packit Service b23acc
		return NULL;
Packit Service b23acc
Packit Service b23acc
	if (G_UNLIKELY (!cur_context)) {
Packit Service b23acc
		cur_context = g_main_context_default ();
Packit Service b23acc
		if (cur_context == context)
Packit Service b23acc
			return NULL;
Packit Service b23acc
	} else if (G_UNLIKELY (!context)) {
Packit Service b23acc
		context = g_main_context_default ();
Packit Service b23acc
		if (cur_context == context)
Packit Service b23acc
			return NULL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_main_context_push_thread_default (context);
Packit Service b23acc
	return context;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline int
Packit Service b23acc
nm_utf8_collate0 (const char *a, const char *b)
Packit Service b23acc
{
Packit Service b23acc
	if (!a)
Packit Service b23acc
		return !b ? 0 : -1;
Packit Service b23acc
	if (!b)
Packit Service b23acc
		return 1;
Packit Service b23acc
	return g_utf8_collate (a, b);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
int nm_strcmp_with_data (gconstpointer a, gconstpointer b, gpointer user_data);
Packit Service b23acc
int nm_strcmp_p_with_data (gconstpointer a, gconstpointer b, gpointer user_data);
Packit Service b23acc
int nm_strcmp0_p_with_data (gconstpointer a, gconstpointer b, gpointer user_data);
Packit Service b23acc
int nm_cmp_uint32_p_with_data (gconstpointer p_a, gconstpointer p_b, gpointer user_data);
Packit Service b23acc
int nm_cmp_int2ptr_p_with_data (gconstpointer p_a, gconstpointer p_b, gpointer user_data);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	const char *name;
Packit Service b23acc
} NMUtilsNamedEntry;
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	union {
Packit Service b23acc
		NMUtilsNamedEntry named_entry;
Packit Service b23acc
		const char *name;
Packit Service b23acc
	};
Packit Service b23acc
	union {
Packit Service b23acc
		const char *value_str;
Packit Service b23acc
		gconstpointer value_ptr;
Packit Service b23acc
	};
Packit Service b23acc
} NMUtilsNamedValue;
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_NAMED_VALUE_INIT(n, v)     { .name = (n), .value_ptr = (v) }
Packit Service b23acc
Packit Service b23acc
NMUtilsNamedValue *nm_utils_named_values_from_str_dict_with_sort (GHashTable *hash,
Packit Service b23acc
                                                                  guint *out_len,
Packit Service b23acc
                                                                  GCompareDataFunc compare_func,
Packit Service b23acc
                                                                  gpointer user_data);
Packit Service b23acc
Packit Service b23acc
static inline NMUtilsNamedValue *
Packit Service b23acc
nm_utils_named_values_from_str_dict (GHashTable *hash, guint *out_len)
Packit Service b23acc
{
Packit Service b23acc
	G_STATIC_ASSERT (G_STRUCT_OFFSET (NMUtilsNamedValue, name) == 0);
Packit Service b23acc
Packit Service b23acc
	return nm_utils_named_values_from_str_dict_with_sort (hash, out_len, nm_strcmp_p_with_data, NULL);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gssize nm_utils_named_value_list_find (const NMUtilsNamedValue *arr,
Packit Service b23acc
                                       gsize len,
Packit Service b23acc
                                       const char *name,
Packit Service b23acc
                                       gboolean sorted);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_named_value_list_is_sorted (const NMUtilsNamedValue *arr,
Packit Service b23acc
                                              gsize len,
Packit Service b23acc
                                              gboolean accept_duplicates,
Packit Service b23acc
                                              GCompareDataFunc compare_func,
Packit Service b23acc
                                              gpointer user_data);
Packit Service b23acc
Packit Service b23acc
void nm_utils_named_value_list_sort (NMUtilsNamedValue *arr,
Packit Service b23acc
                                     gsize len,
Packit Service b23acc
                                     GCompareDataFunc compare_func,
Packit Service b23acc
                                     gpointer user_data);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gpointer *nm_utils_hash_keys_to_array (GHashTable *hash,
Packit Service b23acc
                                       GCompareDataFunc compare_func,
Packit Service b23acc
                                       gpointer user_data,
Packit Service b23acc
                                       guint *out_len);
Packit Service b23acc
Packit Service b23acc
gpointer *nm_utils_hash_values_to_array (GHashTable *hash,
Packit Service b23acc
                                         GCompareDataFunc compare_func,
Packit Service b23acc
                                         gpointer user_data,
Packit Service b23acc
                                         guint *out_len);
Packit Service b23acc
Packit Service b23acc
static inline const char **
Packit Service b23acc
nm_utils_strdict_get_keys (const GHashTable *hash,
Packit Service b23acc
                           gboolean sorted,
Packit Service b23acc
                           guint *out_length)
Packit Service b23acc
{
Packit Service b23acc
	return (const char **) nm_utils_hash_keys_to_array ((GHashTable *) hash,
Packit Service b23acc
	                                                    sorted ? nm_strcmp_p_with_data : NULL,
Packit Service b23acc
	                                                    NULL,
Packit Service b23acc
	                                                    out_length);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_hashtable_same_keys (const GHashTable *a,
Packit Service b23acc
                                       const GHashTable *b);
Packit Service b23acc
Packit Service b23acc
char **nm_utils_strv_make_deep_copied (const char **strv);
Packit Service b23acc
Packit Service b23acc
char **nm_utils_strv_make_deep_copied_n (const char **strv, gsize len);
Packit Service b23acc
Packit Service b23acc
static inline char **
Packit Service b23acc
nm_utils_strv_make_deep_copied_nonnull (const char **strv)
Packit Service b23acc
{
Packit Service b23acc
	return nm_utils_strv_make_deep_copied (strv) ?: g_new0 (char *, 1);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
char **nm_utils_strv_dup (gpointer strv,
Packit Service b23acc
                          gssize len,
Packit Service b23acc
                          gboolean deep_copied);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
GSList *nm_utils_g_slist_find_str (const GSList *list,
Packit Service b23acc
                                   const char *needle);
Packit Service b23acc
Packit Service b23acc
int nm_utils_g_slist_strlist_cmp (const GSList *a, const GSList *b);
Packit Service b23acc
Packit Service b23acc
char *nm_utils_g_slist_strlist_join (const GSList *a, const char *separator);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline guint
Packit Service b23acc
nm_g_array_len (const GArray *arr)
Packit Service b23acc
{
Packit Service b23acc
	return arr ? arr->len : 0u;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline guint
Packit Service b23acc
nm_g_ptr_array_len (const GPtrArray *arr)
Packit Service b23acc
{
Packit Service b23acc
	return arr ? arr->len : 0u;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline guint
Packit Service b23acc
nm_g_hash_table_size (GHashTable *hash)
Packit Service b23acc
{
Packit Service b23acc
	return hash ? g_hash_table_size (hash) : 0u;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gpointer
Packit Service b23acc
nm_g_hash_table_lookup (GHashTable *hash, gconstpointer key)
Packit Service b23acc
{
Packit Service b23acc
	return hash ? g_hash_table_lookup (hash, key) : NULL;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_g_hash_table_remove (GHashTable *hash, gconstpointer key)
Packit Service b23acc
{
Packit Service b23acc
	return hash ? g_hash_table_remove (hash, key) : FALSE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gssize nm_utils_ptrarray_find_binary_search (gconstpointer *list,
Packit Service b23acc
                                             gsize len,
Packit Service b23acc
                                             gconstpointer needle,
Packit Service b23acc
                                             GCompareDataFunc cmpfcn,
Packit Service b23acc
                                             gpointer user_data,
Packit Service b23acc
                                             gssize *out_idx_first,
Packit Service b23acc
                                             gssize *out_idx_last);
Packit Service b23acc
Packit Service b23acc
gssize nm_utils_array_find_binary_search (gconstpointer list,
Packit Service b23acc
                                          gsize elem_size,
Packit Service b23acc
                                          gsize len,
Packit Service b23acc
                                          gconstpointer needle,
Packit Service b23acc
                                          GCompareDataFunc cmpfcn,
Packit Service b23acc
                                          gpointer user_data);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef gboolean (*NMUtilsHashTableEqualFunc) (gconstpointer a,
Packit Service b23acc
                                               gconstpointer b);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_hash_table_equal (const GHashTable *a,
Packit Service b23acc
                                    const GHashTable *b,
Packit Service b23acc
                                    gboolean treat_null_as_empty,
Packit Service b23acc
                                    NMUtilsHashTableEqualFunc equal_func);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
void _nm_utils_strv_sort (const char **strv, gssize len);
Packit Service b23acc
#define nm_utils_strv_sort(strv, len) _nm_utils_strv_sort (NM_CAST_STRV_MC (strv), len)
Packit Service b23acc
Packit Service b23acc
int _nm_utils_strv_cmp_n (const char *const*strv1,
Packit Service b23acc
                          gssize len1,
Packit Service b23acc
                          const char *const*strv2,
Packit Service b23acc
                          gssize len2);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
_nm_utils_strv_equal (char **strv1, char **strv2)
Packit Service b23acc
{
Packit Service b23acc
	return _nm_utils_strv_cmp_n ((const char *const*) strv1, -1,
Packit Service b23acc
	                             (const char *const*) strv2, -1) == 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_NSEC_PER_SEC  ((gint64) 1000000000)
Packit Service b23acc
#define NM_UTILS_USEC_PER_SEC  ((gint64) 1000000)
Packit Service b23acc
#define NM_UTILS_MSEC_PER_SEC  ((gint64) 1000)
Packit Service b23acc
#define NM_UTILS_NSEC_PER_MSEC ((gint64) 1000000)
Packit Service b23acc
Packit Service b23acc
static inline gint64
Packit Service b23acc
NM_UTILS_NSEC_TO_MSEC_CEIL (gint64 nsec)
Packit Service b23acc
{
Packit Service b23acc
	return (nsec + (NM_UTILS_NSEC_PER_MSEC - 1)) / NM_UTILS_NSEC_PER_MSEC;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
int nm_utils_fd_wait_for_event (int fd, int event, gint64 timeout_nsec);
Packit Service b23acc
ssize_t nm_utils_fd_read_loop (int fd, void *buf, size_t nbytes, bool do_poll);
Packit Service b23acc
int nm_utils_fd_read_loop_exact (int fd, void *buf, size_t nbytes, bool do_poll);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_ARG_INFO_FULL(name_, ...) \
Packit Service b23acc
	((GDBusArgInfo *) (&((const GDBusArgInfo) { \
Packit Service b23acc
		.ref_count = -1, \
Packit Service b23acc
		.name = name_, \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	})))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_ARG_INFO(name_, a_signature) \
Packit Service b23acc
	NM_DEFINE_GDBUS_ARG_INFO_FULL ( \
Packit Service b23acc
		name_, \
Packit Service b23acc
		.signature = a_signature, \
Packit Service b23acc
	)
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_ARG_INFOS(...) \
Packit Service b23acc
	((GDBusArgInfo **) ((const GDBusArgInfo *[]) { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
		NULL, \
Packit Service b23acc
	}))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_PROPERTY_INFO(name_, ...) \
Packit Service b23acc
	((GDBusPropertyInfo *) (&((const GDBusPropertyInfo) { \
Packit Service b23acc
		.ref_count = -1, \
Packit Service b23acc
		.name = name_, \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	})))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_PROPERTY_INFO_READABLE(name_, m_signature) \
Packit Service b23acc
	NM_DEFINE_GDBUS_PROPERTY_INFO ( \
Packit Service b23acc
		name_, \
Packit Service b23acc
		.signature = m_signature, \
Packit Service b23acc
		.flags = G_DBUS_PROPERTY_INFO_FLAGS_READABLE, \
Packit Service b23acc
	)
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_PROPERTY_INFOS(...) \
Packit Service b23acc
	((GDBusPropertyInfo **) ((const GDBusPropertyInfo *[]) { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
		NULL, \
Packit Service b23acc
	}))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_SIGNAL_INFO_INIT(name_, ...) \
Packit Service b23acc
	{ \
Packit Service b23acc
		.ref_count = -1, \
Packit Service b23acc
		.name = name_, \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_SIGNAL_INFO(name_, ...) \
Packit Service b23acc
	((GDBusSignalInfo *) (&((const GDBusSignalInfo) NM_DEFINE_GDBUS_SIGNAL_INFO_INIT (name_, __VA_ARGS__))))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_SIGNAL_INFOS(...) \
Packit Service b23acc
	((GDBusSignalInfo **) ((const GDBusSignalInfo *[]) { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
		NULL, \
Packit Service b23acc
	}))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_METHOD_INFO_INIT(name_, ...) \
Packit Service b23acc
	{ \
Packit Service b23acc
		.ref_count = -1, \
Packit Service b23acc
		.name = name_, \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_METHOD_INFO(name_, ...) \
Packit Service b23acc
	((GDBusMethodInfo *) (&((const GDBusMethodInfo) NM_DEFINE_GDBUS_METHOD_INFO_INIT (name_, __VA_ARGS__))))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_METHOD_INFOS(...) \
Packit Service b23acc
	((GDBusMethodInfo **) ((const GDBusMethodInfo *[]) { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
		NULL, \
Packit Service b23acc
	}))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_INTERFACE_INFO_INIT(name_, ...) \
Packit Service b23acc
	{ \
Packit Service b23acc
		.ref_count = -1, \
Packit Service b23acc
		.name = name_, \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_INTERFACE_INFO(name_, ...) \
Packit Service b23acc
	((GDBusInterfaceInfo *) (&((const GDBusInterfaceInfo) NM_DEFINE_GDBUS_INTERFACE_INFO_INIT (name_, __VA_ARGS__))))
Packit Service b23acc
Packit Service b23acc
#define NM_DEFINE_GDBUS_INTERFACE_VTABLE(...) \
Packit Service b23acc
	((GDBusInterfaceVTable *) (&((const GDBusInterfaceVTable) { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	})))
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
guint64 nm_utils_get_start_time_for_pid (pid_t pid, char *out_state, pid_t *out_ppid);
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_process_state_is_dead (char pstate)
Packit Service b23acc
{
Packit Service b23acc
	/* "/proc/[pid]/stat" returns a state as the 3rd fields (see `man 5 proc`).
Packit Service b23acc
	 * Some of these states indicate the the process is effectively dead (or a zombie).
Packit Service b23acc
	 */
Packit Service b23acc
	return NM_IN_SET (pstate, 'Z', 'x', 'X');
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gpointer _nm_utils_user_data_pack (int nargs, gconstpointer *args);
Packit Service b23acc
Packit Service b23acc
#define nm_utils_user_data_pack(...) \
Packit Service b23acc
	_nm_utils_user_data_pack(NM_NARG (__VA_ARGS__), (gconstpointer[]) { __VA_ARGS__ })
Packit Service b23acc
Packit Service b23acc
void _nm_utils_user_data_unpack (gpointer user_data, int nargs, ...);
Packit Service b23acc
Packit Service b23acc
#define nm_utils_user_data_unpack(user_data, ...) \
Packit Service b23acc
	_nm_utils_user_data_unpack(user_data, NM_NARG (__VA_ARGS__), __VA_ARGS__)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef void (*NMUtilsInvokeOnIdleCallback) (gpointer user_data,
Packit Service b23acc
                                             GCancellable *cancellable);
Packit Service b23acc
Packit Service b23acc
void nm_utils_invoke_on_idle (GCancellable *cancellable,
Packit Service b23acc
                              NMUtilsInvokeOnIdleCallback callback,
Packit Service b23acc
                              gpointer callback_user_data);
Packit Service b23acc
Packit Service b23acc
void nm_utils_invoke_on_timeout (guint timeout_msec,
Packit Service b23acc
                                 GCancellable *cancellable,
Packit Service b23acc
                                 NMUtilsInvokeOnIdleCallback callback,
Packit Service b23acc
                                 gpointer callback_user_data);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
GSource *nm_utils_g_main_context_create_integrate_source (GMainContext *internal);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_strv_ptrarray_add_string_take (GPtrArray *cmd,
Packit Service b23acc
                                  char *str)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (cmd);
Packit Service b23acc
	nm_assert (str);
Packit Service b23acc
Packit Service b23acc
	g_ptr_array_add (cmd, str);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_strv_ptrarray_add_string_dup (GPtrArray *cmd,
Packit Service b23acc
                                 const char *str)
Packit Service b23acc
{
Packit Service b23acc
	nm_strv_ptrarray_add_string_take (cmd,
Packit Service b23acc
	                                  g_strdup (str));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define nm_strv_ptrarray_add_string_concat(cmd, ...) \
Packit Service b23acc
	nm_strv_ptrarray_add_string_take ((cmd), g_strconcat (__VA_ARGS__, NULL))
Packit Service b23acc
Packit Service b23acc
#define nm_strv_ptrarray_add_string_printf(cmd, ...) \
Packit Service b23acc
	nm_strv_ptrarray_add_string_take ((cmd), g_strdup_printf (__VA_ARGS__))
Packit Service b23acc
Packit Service b23acc
#define nm_strv_ptrarray_add_int(cmd, val) \
Packit Service b23acc
	nm_strv_ptrarray_add_string_take ((cmd), nm_strdup_int (val))
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_strv_ptrarray_take_gstring (GPtrArray *cmd,
Packit Service b23acc
                               GString **gstr)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (gstr && *gstr);
Packit Service b23acc
Packit Service b23acc
	nm_strv_ptrarray_add_string_take (cmd,
Packit Service b23acc
	                                  g_string_free (g_steal_pointer (gstr),
Packit Service b23acc
	                                                 FALSE));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
int nm_utils_getpagesize (void);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
char *nm_utils_bin2hexstr_full (gconstpointer addr,
Packit Service b23acc
                                gsize length,
Packit Service b23acc
                                char delimiter,
Packit Service b23acc
                                gboolean upper_case,
Packit Service b23acc
                                char *out);
Packit Service b23acc
Packit Service b23acc
guint8 *nm_utils_hexstr2bin_full (const char *hexstr,
Packit Service b23acc
                                  gboolean allow_0x_prefix,
Packit Service b23acc
                                  gboolean delimiter_required,
Packit Service b23acc
                                  const char *delimiter_candidates,
Packit Service b23acc
                                  gsize required_len,
Packit Service b23acc
                                  guint8 *buffer,
Packit Service b23acc
                                  gsize buffer_len,
Packit Service b23acc
                                  gsize *out_len);
Packit Service b23acc
Packit Service b23acc
#define nm_utils_hexstr2bin_buf(hexstr, allow_0x_prefix, delimiter_required, delimiter_candidates, buffer) \
Packit Service b23acc
    nm_utils_hexstr2bin_full ((hexstr), (allow_0x_prefix), (delimiter_required), (delimiter_candidates), G_N_ELEMENTS (buffer), (buffer), G_N_ELEMENTS (buffer), NULL)
Packit Service b23acc
Packit Service b23acc
guint8 *nm_utils_hexstr2bin_alloc (const char *hexstr,
Packit Service b23acc
                                   gboolean allow_0x_prefix,
Packit Service b23acc
                                   gboolean delimiter_required,
Packit Service b23acc
                                   const char *delimiter_candidates,
Packit Service b23acc
                                   gsize required_len,
Packit Service b23acc
                                   gsize *out_len);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define _NM_UTILS_STRING_TABLE_LOOKUP_DEFINE(fcn_name, \
Packit Service b23acc
                                             value_type, \
Packit Service b23acc
                                             value_type_result, \
Packit Service b23acc
                                             entry_cmd, \
Packit Service b23acc
                                             unknown_val_cmd, \
Packit Service b23acc
                                             get_operator, \
Packit Service b23acc
                                             ...) \
Packit Service b23acc
value_type_result \
Packit Service b23acc
fcn_name (const char *name) \
Packit Service b23acc
{ \
Packit Service b23acc
	static const struct { \
Packit Service b23acc
		const char *name; \
Packit Service b23acc
		value_type value; \
Packit Service b23acc
	} LIST[] = { \
Packit Service b23acc
		__VA_ARGS__ \
Packit Service b23acc
	}; \
Packit Service b23acc
	\
Packit Service b23acc
	if (NM_MORE_ASSERT_ONCE (5)) { \
Packit Service b23acc
		int i; \
Packit Service b23acc
		\
Packit Service b23acc
		for (i = 0; i < G_N_ELEMENTS (LIST); i++) { \
Packit Service b23acc
			nm_assert (LIST[i].name); \
Packit Service b23acc
			if (i > 0) \
Packit Service b23acc
				nm_assert (strcmp (LIST[i - 1].name, LIST[i].name) < 0); \
Packit Service b23acc
		} \
Packit Service b23acc
	} \
Packit Service b23acc
	\
Packit Service b23acc
	{ entry_cmd; } \
Packit Service b23acc
	\
Packit Service b23acc
	if (G_LIKELY (name)) { \
Packit Service b23acc
		G_STATIC_ASSERT (G_N_ELEMENTS (LIST) > 1); \
Packit Service b23acc
		G_STATIC_ASSERT (G_N_ELEMENTS (LIST) < G_MAXINT / 2 - 10); \
Packit Service b23acc
		int imin = 0; \
Packit Service b23acc
		int imax = (G_N_ELEMENTS (LIST) - 1); \
Packit Service b23acc
		int imid = (G_N_ELEMENTS (LIST) - 1) / 2; \
Packit Service b23acc
		\
Packit Service b23acc
		for (;;) { \
Packit Service b23acc
			const int cmp = strcmp (LIST[imid].name, name); \
Packit Service b23acc
			\
Packit Service b23acc
			if (G_UNLIKELY (cmp == 0)) \
Packit Service b23acc
				return get_operator (LIST[imid].value); \
Packit Service b23acc
			\
Packit Service b23acc
			if (cmp < 0) \
Packit Service b23acc
				imin = imid + 1; \
Packit Service b23acc
			else \
Packit Service b23acc
				imax = imid - 1; \
Packit Service b23acc
			\
Packit Service b23acc
			if (G_UNLIKELY (imin > imax)) \
Packit Service b23acc
				break; \
Packit Service b23acc
			\
Packit Service b23acc
			/* integer overflow cannot happen, because LIST is shorter than G_MAXINT/2. */ \
Packit Service b23acc
			imid = (imin + imax) / 2;\
Packit Service b23acc
		} \
Packit Service b23acc
	} \
Packit Service b23acc
	\
Packit Service b23acc
	{ unknown_val_cmd; } \
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_STRING_TABLE_LOOKUP_STRUCT_DEFINE(fcn_name, \
Packit Service b23acc
                                                   result_type, \
Packit Service b23acc
                                                   entry_cmd, \
Packit Service b23acc
                                                   unknown_val_cmd, \
Packit Service b23acc
                                                   ...) \
Packit Service b23acc
	_NM_UTILS_STRING_TABLE_LOOKUP_DEFINE (fcn_name, \
Packit Service b23acc
	                                      result_type, \
Packit Service b23acc
	                                      const result_type *, \
Packit Service b23acc
	                                      entry_cmd, \
Packit Service b23acc
	                                      unknown_val_cmd, \
Packit Service b23acc
	                                      &, \
Packit Service b23acc
	                                      __VA_ARGS__)
Packit Service b23acc
Packit Service b23acc
#define NM_UTILS_STRING_TABLE_LOOKUP_DEFINE(fcn_name, \
Packit Service b23acc
                                            result_type, \
Packit Service b23acc
                                            entry_cmd, \
Packit Service b23acc
                                            unknown_val_cmd, \
Packit Service b23acc
                                            ...) \
Packit Service b23acc
	_NM_UTILS_STRING_TABLE_LOOKUP_DEFINE (fcn_name, \
Packit Service b23acc
	                                      result_type, \
Packit Service b23acc
	                                      result_type, \
Packit Service b23acc
	                                      entry_cmd, \
Packit Service b23acc
	                                      unknown_val_cmd, \
Packit Service b23acc
	                                      , \
Packit Service b23acc
	                                      __VA_ARGS__)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline GTask *
Packit Service b23acc
nm_g_task_new (gpointer source_object,
Packit Service b23acc
               GCancellable *cancellable,
Packit Service b23acc
               gpointer source_tag,
Packit Service b23acc
               GAsyncReadyCallback callback,
Packit Service b23acc
               gpointer callback_data)
Packit Service b23acc
{
Packit Service b23acc
	GTask *task;
Packit Service b23acc
Packit Service b23acc
	task = g_task_new (source_object, cancellable, callback, callback_data);
Packit Service b23acc
	if (source_tag)
Packit Service b23acc
		g_task_set_source_tag (task, source_tag);
Packit Service b23acc
	return task;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_g_task_is_valid (gpointer task,
Packit Service b23acc
                    gpointer source_object,
Packit Service b23acc
                    gpointer source_tag)
Packit Service b23acc
{
Packit Service b23acc
	return    g_task_is_valid (task, source_object)
Packit Service b23acc
	       && g_task_get_source_tag (task) == source_tag;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
guint nm_utils_parse_debug_string (const char *string,
Packit Service b23acc
                                   const GDebugKey *keys,
Packit Service b23acc
                                   guint nkeys);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline gboolean
Packit Service b23acc
nm_utils_strdup_reset (char **dst, const char *src)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (dst);
Packit Service b23acc
Packit Service b23acc
	if (nm_streq0 (*dst, src))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	g_free (*dst);
Packit Service b23acc
	*dst = g_strdup (src);
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void nm_indirect_g_free (gpointer arg);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
/* nm_utils_get_next_realloc_size() is used to grow buffers exponentially, when
Packit Service b23acc
 * the final size is unknown. As such, it has borders for which it allocates
Packit Service b23acc
 * certain buffer sizes.
Packit Service b23acc
 *
Packit Service b23acc
 * The use of these defines is to get favorable allocation sequences.
Packit Service b23acc
 * For example, nm_str_buf_init() asks for an initial allocation size. Note that
Packit Service b23acc
 * it reserves the exactly requested amount, under the assumption that the
Packit Service b23acc
 * user may know how many bytes will be required. However, often the caller
Packit Service b23acc
 * doesn't know in advance, and NMStrBuf grows exponentially by calling
Packit Service b23acc
 * nm_utils_get_next_realloc_size().
Packit Service b23acc
 * Imagine you call nm_str_buf_init() with an initial buffer size 100, and you
Packit Service b23acc
 * add one character at a time. Then the first reallocation will increase the
Packit Service b23acc
 * buffer size only from 100 to 104.
Packit Service b23acc
 * If you however start with an initial buffer size of 104, then the next reallocation
Packit Service b23acc
 * via nm_utils_get_next_realloc_size() gives you 232, and so on. By using
Packit Service b23acc
 * these sizes, it results in one less allocation, if you anyway don't know the
Packit Service b23acc
 * exact size in advance. */
Packit Service b23acc
#define NM_UTILS_GET_NEXT_REALLOC_SIZE_104     ((gsize) 104)
Packit Service b23acc
#define NM_UTILS_GET_NEXT_REALLOC_SIZE_1000    ((gsize) 1000)
Packit Service b23acc
Packit Service b23acc
gsize nm_utils_get_next_realloc_size (gboolean true_realloc, gsize requested);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef enum {
Packit Service b23acc
	NMU_IFACE_ANY,
Packit Service b23acc
	NMU_IFACE_KERNEL,
Packit Service b23acc
	NMU_IFACE_OVS,
Packit Service b23acc
	NMU_IFACE_OVS_AND_KERNEL,
Packit Service b23acc
} NMUtilsIfaceType;
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_ifname_valid_kernel (const char *name, GError **error);
Packit Service b23acc
Packit Service b23acc
gboolean nm_utils_ifname_valid (const char* name,
Packit Service b23acc
                                NMUtilsIfaceType type,
Packit Service b23acc
                                GError **error);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static inline GArray *
Packit Service b23acc
nm_strvarray_ensure (GArray **p)
Packit Service b23acc
{
Packit Service b23acc
	if (!*p) {
Packit Service b23acc
		*p = g_array_new (TRUE, FALSE, sizeof (char *));
Packit Service b23acc
		g_array_set_clear_func (*p, nm_indirect_g_free);
Packit Service b23acc
	}
Packit Service b23acc
	return *p;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_strvarray_add (GArray *array, const char *str)
Packit Service b23acc
{
Packit Service b23acc
	char *s;
Packit Service b23acc
Packit Service b23acc
	s = g_strdup (str);
Packit Service b23acc
	g_array_append_val (array, s);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char *const*
Packit Service b23acc
nm_strvarray_get_strv_non_empty (GArray *arr, guint *length)
Packit Service b23acc
{
Packit Service b23acc
	if (!arr || arr->len == 0) {
Packit Service b23acc
		NM_SET_OUT (length, 0);
Packit Service b23acc
		return NULL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	NM_SET_OUT (length, arr->len);
Packit Service b23acc
	return &g_array_index (arr, const char *, 0);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline const char *const*
Packit Service b23acc
nm_strvarray_get_strv (GArray **arr, guint *length)
Packit Service b23acc
{
Packit Service b23acc
	if (!*arr) {
Packit Service b23acc
		NM_SET_OUT (length, 0);
Packit Service b23acc
		return (const char *const*) arr;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	NM_SET_OUT (length, (*arr)->len);
Packit Service b23acc
	return &g_array_index (*arr, const char *, 0);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static inline void
Packit Service b23acc
nm_strvarray_set_strv (GArray **array, const char *const*strv)
Packit Service b23acc
{
Packit Service b23acc
	gs_unref_array GArray *array_old = NULL;
Packit Service b23acc
Packit Service b23acc
	array_old = g_steal_pointer (array);
Packit Service b23acc
Packit Service b23acc
	if (!strv || !strv[0])
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	nm_strvarray_ensure (array);
Packit Service b23acc
	for (; strv[0]; strv++)
Packit Service b23acc
		nm_strvarray_add (*array, strv[0]);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
void _nm_utils_format_variant_attributes_full (GString *str,
Packit Service b23acc
                                               const NMUtilsNamedValue *values,
Packit Service b23acc
                                               guint num_values,
Packit Service b23acc
                                               char attr_separator,
Packit Service b23acc
                                               char key_value_separator);
Packit Service b23acc
Packit Service b23acc
char *_nm_utils_format_variant_attributes (GHashTable *attributes,
Packit Service b23acc
                                           char attr_separator,
Packit Service b23acc
                                           char key_value_separator);
Packit Service b23acc
Packit Service b23acc
#endif /* __NM_SHARED_UTILS_H__ */