Blob Blame History Raw
// SPDX-License-Identifier: LGPL-2.1+
/*
 * Copyright (C) 2019 Red Hat, Inc.
 */

#ifndef __NM_VALUE_TYPE_H__
#define __NM_VALUE_TYPE_H__

typedef enum {
	NM_VALUE_TYPE_UNSPEC = 1,
	NM_VALUE_TYPE_BOOL   = 2,
	NM_VALUE_TYPE_INT32  = 3,
	NM_VALUE_TYPE_INT    = 4,
	NM_VALUE_TYPE_STRING = 5,
} NMValueType;

/*****************************************************************************/

#ifdef NM_VALUE_TYPE_DEFINE_FUNCTIONS

typedef union {
	bool             v_bool;
	gint32           v_int32;
	int              v_int;
	const char      *v_string;

	/* for convenience, also let the union contain other pointer types. These are
	 * for NM_VALUE_TYPE_UNSPEC. */
	gconstpointer   *v_ptr;
	const GPtrArray *v_ptrarray;

} NMValueTypUnion;

/* Set the NMValueTypUnion. You can also assign the member directly.
 * The only purpose of this is that it also returns a pointer to the
 * union. So, you can do
 *
 *   ptr = NM_VALUE_TYP_UNION_SET (&value_typ_union_storage, v_bool, TRUE);
 */
#define NM_VALUE_TYP_UNION_SET(_arg, _type, _val) \
	({ \
		NMValueTypUnion *const _arg2 = (_arg); \
		\
		*_arg2 = (NMValueTypUnion) { \
			._type = (_val), \
		}; \
		_arg2; \
	})

typedef struct {
	bool has;
	NMValueTypUnion val;
} NMValueTypUnioMaybe;

#define NM_VALUE_TYP_UNIO_MAYBE_SET(_arg, _type, _val) \
	({ \
		NMValueTypUnioMaybe *const _arg2 = (_arg); \
		\
		*_arg2 = (NMValueTypUnioMaybe) { \
			.has       = TRUE, \
			.val._type = (_val), \
		}; \
		_arg2; \
	})

/*****************************************************************************/

static inline int
nm_value_type_cmp (NMValueType value_type,
                   gconstpointer p_a,
                   gconstpointer p_b)
{
	switch (value_type) {
	case NM_VALUE_TYPE_BOOL:   NM_CMP_DIRECT (*((const bool   *) p_a), *((const bool   *) p_b)); return 0;
	case NM_VALUE_TYPE_INT32:  NM_CMP_DIRECT (*((const gint32 *) p_a), *((const gint32 *) p_b)); return 0;
	case NM_VALUE_TYPE_INT:    NM_CMP_DIRECT (*((const int    *) p_a), *((const int    *) p_b)); return 0;
	case NM_VALUE_TYPE_STRING: return nm_strcmp0 (*((const char *const*) p_a), *((const char *const*) p_b));
	case NM_VALUE_TYPE_UNSPEC:
		break;
	}
	nm_assert_not_reached ();
	return 0;
}

static inline gboolean
nm_value_type_equal (NMValueType value_type,
                     gconstpointer p_a,
                     gconstpointer p_b)
{
	return nm_value_type_cmp (value_type, p_a, p_b) == 0;
}

static inline void
nm_value_type_copy (NMValueType value_type,
                    gpointer dst,
                    gconstpointer src)
{
	switch (value_type) {
	case NM_VALUE_TYPE_BOOL:   (*((bool   *) dst) = *((const bool   *) src)); return;
	case NM_VALUE_TYPE_INT32:  (*((gint32 *) dst) = *((const gint32 *) src)); return;
	case NM_VALUE_TYPE_INT:    (*((int    *) dst) = *((const int    *) src)); return;
	case NM_VALUE_TYPE_STRING:
		/* self assignment safe! */
		if (*((char **) dst) != *((const char *const*) src)) {
			g_free (*((char **) dst));
			*((char **) dst) = g_strdup (*((const char *const*) src));
		}
		return;
	case NM_VALUE_TYPE_UNSPEC:
		break;
	}
	nm_assert_not_reached ();
}

static inline void
nm_value_type_get_from_variant (NMValueType value_type,
                                gpointer dst,
                                GVariant *variant,
                                gboolean clone)
{
	switch (value_type) {
	case NM_VALUE_TYPE_BOOL:   *((bool   *) dst) = g_variant_get_boolean (variant); return;
	case NM_VALUE_TYPE_INT32:  *((gint32 *) dst) = g_variant_get_int32 (variant);   return;
	case NM_VALUE_TYPE_STRING:
		if (clone) {
			g_free (*((char **) dst));
			*((char **) dst) = g_variant_dup_string (variant, NULL);
		} else {
			/* we don't clone the string, nor free the previous value. */
			*((const char **) dst) = g_variant_get_string (variant, NULL);
		}
		return;

	case NM_VALUE_TYPE_INT:
		/* "int" also does not have a define variant type, because it's not
		 * clear how many bits we would need. */

		/* fall-through */
	case NM_VALUE_TYPE_UNSPEC:
		break;
	}
	nm_assert_not_reached ();
}

static inline GVariant *
nm_value_type_to_variant (NMValueType value_type,
                          gconstpointer src)
{
	const char *v_string;

	switch (value_type) {
	case NM_VALUE_TYPE_BOOL:   return g_variant_new_boolean (*((const bool   *) src));
	case NM_VALUE_TYPE_INT32:  return g_variant_new_int32   (*((const gint32 *) src));;
	case NM_VALUE_TYPE_STRING:
		v_string = *((const char *const*) src);
		return v_string ? g_variant_new_string (v_string) : NULL;

	case NM_VALUE_TYPE_INT:
		/* "int" also does not have a define variant type, because it's not
		 * clear how many bits we would need. */

		/* fall-through */
	case NM_VALUE_TYPE_UNSPEC:
		break;
	}
	nm_assert_not_reached ();
	return NULL;
}

static inline const GVariantType *
nm_value_type_get_variant_type (NMValueType value_type)
{
	switch (value_type) {
	case NM_VALUE_TYPE_BOOL:   return G_VARIANT_TYPE_BOOLEAN;
	case NM_VALUE_TYPE_INT32:  return G_VARIANT_TYPE_INT32;
	case NM_VALUE_TYPE_STRING: return G_VARIANT_TYPE_STRING;

	case NM_VALUE_TYPE_INT:
		/* "int" also does not have a define variant type, because it's not
		 * clear how many bits we would need. */

		/* fall-through */
	case NM_VALUE_TYPE_UNSPEC:
		break;
	}
	nm_assert_not_reached ();
	return NULL;
}

/*****************************************************************************/

#endif /* NM_VALUE_TYPE_DEFINE_FUNCTIONS */

#endif  /* __NM_VALUE_TYPE_H__ */