// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2008 - 2013 Red Hat, Inc. */ #ifndef __NETWORKMANAGER_IP4_CONFIG_H__ #define __NETWORKMANAGER_IP4_CONFIG_H__ #include "nm-setting-connection.h" #include "nm-setting-ip4-config.h" #include "nm-glib-aux/nm-dedup-multi.h" #include "platform/nmp-object.h" /*****************************************************************************/ typedef enum _NMIPConfigFlags { NM_IP_CONFIG_FLAG_NONE = 0, /* if set, then the merge flag NM_IP_CONFIG_MERGE_NO_DEFAULT_ROUTES gets * ignored during merge. */ NM_IP_CONFIG_FLAGS_IGNORE_MERGE_NO_DEFAULT_ROUTES = (1ull << 0), } NMIPConfigFlags; typedef struct { NMDedupMultiIdxType parent; NMPObjectType obj_type; } NMIPConfigDedupMultiIdxType; void nm_ip_config_dedup_multi_idx_type_init (NMIPConfigDedupMultiIdxType *idx_type, NMPObjectType obj_type); /*****************************************************************************/ void nm_ip_config_iter_ip4_address_init (NMDedupMultiIter *iter, const NMIP4Config *self); void nm_ip_config_iter_ip4_route_init (NMDedupMultiIter *iter, const NMIP4Config *self); static inline gboolean nm_ip_config_iter_ip4_address_next (NMDedupMultiIter *ipconf_iter, const NMPlatformIP4Address **out_address) { gboolean has_next; has_next = nm_dedup_multi_iter_next (ipconf_iter); if (out_address) *out_address = has_next ? NMP_OBJECT_CAST_IP4_ADDRESS (ipconf_iter->current->obj) : NULL; return has_next; } static inline gboolean nm_ip_config_iter_ip4_route_next (NMDedupMultiIter *ipconf_iter, const NMPlatformIP4Route **out_route) { gboolean has_next; has_next = nm_dedup_multi_iter_next (ipconf_iter); if (out_route) *out_route = has_next ? NMP_OBJECT_CAST_IP4_ROUTE (ipconf_iter->current->obj) : NULL; return has_next; } #define nm_ip_config_iter_ip4_address_for_each(iter, self, address) \ for (nm_ip_config_iter_ip4_address_init ((iter), (self)); \ nm_ip_config_iter_ip4_address_next ((iter), (address)); \ ) #define nm_ip_config_iter_ip4_route_for_each(iter, self, route) \ for (nm_ip_config_iter_ip4_route_init ((iter), (self)); \ nm_ip_config_iter_ip4_route_next ((iter), (route)); \ ) /*****************************************************************************/ static inline gboolean nm_ip_config_best_default_route_is (const NMPObject *obj) { const NMPlatformIPRoute *r = NMP_OBJECT_CAST_IP_ROUTE (obj); /* return whether @obj is considered a default-route. * * NMIP4Config/NMIP6Config tracks the (best) default-route explicitly, because * at various places we act differently depending on whether there is a default-route * configured. * * Note that this only considers the main routing table. */ return r && NM_PLATFORM_IP_ROUTE_IS_DEFAULT (r) && nm_platform_route_table_is_main (r->table_coerced) && r->type_coerced == nm_platform_route_type_coerce (1 /*RTN_UNICAST*/); } const NMPObject *_nm_ip_config_best_default_route_find_better (const NMPObject *obj_cur, const NMPObject *obj_cmp); gboolean _nm_ip_config_best_default_route_set (const NMPObject **best_default_route, const NMPObject *new_candidate); gboolean _nm_ip_config_best_default_route_merge (const NMPObject **best_default_route, const NMPObject *new_candidate); /*****************************************************************************/ gboolean _nm_ip_config_add_obj (NMDedupMultiIndex *multi_idx, NMIPConfigDedupMultiIdxType *idx_type, int ifindex, const NMPObject *obj_new, const NMPlatformObject *pl_new, gboolean merge, gboolean append_force, const NMPObject **out_obj_old, const NMPObject **out_obj_new); const NMDedupMultiEntry *_nm_ip_config_lookup_ip_route (const NMDedupMultiIndex *multi_idx, const NMIPConfigDedupMultiIdxType *idx_type, const NMPObject *needle, NMPlatformIPRouteCmpType cmp_type); void _nm_ip_config_merge_route_attributes (int addr_family, NMIPRoute *s_route, NMPlatformIPRoute *r, guint32 route_table); /*****************************************************************************/ #define NM_TYPE_IP4_CONFIG (nm_ip4_config_get_type ()) #define NM_IP4_CONFIG(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), NM_TYPE_IP4_CONFIG, NMIP4Config)) #define NM_IP4_CONFIG_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), NM_TYPE_IP4_CONFIG, NMIP4ConfigClass)) #define NM_IS_IP4_CONFIG(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), NM_TYPE_IP4_CONFIG)) #define NM_IS_IP4_CONFIG_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), NM_TYPE_IP4_CONFIG)) #define NM_IP4_CONFIG_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), NM_TYPE_IP4_CONFIG, NMIP4ConfigClass)) typedef struct _NMIP4ConfigClass NMIP4ConfigClass; /* internal */ #define NM_IP4_CONFIG_MULTI_IDX "multi-idx" #define NM_IP4_CONFIG_IFINDEX "ifindex" /* public*/ #define NM_IP4_CONFIG_ADDRESS_DATA "address-data" #define NM_IP4_CONFIG_ROUTE_DATA "route-data" #define NM_IP4_CONFIG_GATEWAY "gateway" #define NM_IP4_CONFIG_NAMESERVER_DATA "nameserver-data" #define NM_IP4_CONFIG_DOMAINS "domains" #define NM_IP4_CONFIG_SEARCHES "searches" #define NM_IP4_CONFIG_DNS_OPTIONS "dns-options" #define NM_IP4_CONFIG_DNS_PRIORITY "dns-priority" #define NM_IP4_CONFIG_WINS_SERVER_DATA "wins-server-data" /* deprecated */ #define NM_IP4_CONFIG_ADDRESSES "addresses" #define NM_IP4_CONFIG_ROUTES "routes" #define NM_IP4_CONFIG_NAMESERVERS "nameservers" #define NM_IP4_CONFIG_WINS_SERVERS "wins-servers" GType nm_ip4_config_get_type (void); NMIP4Config * nm_ip4_config_new (NMDedupMultiIndex *multi_idx, int ifindex); NMIP4Config *nm_ip4_config_clone (const NMIP4Config *self); int nm_ip4_config_get_ifindex (const NMIP4Config *self); NMDedupMultiIndex *nm_ip4_config_get_multi_idx (const NMIP4Config *self); NMIP4Config *nm_ip4_config_capture (NMDedupMultiIndex *multi_idx, NMPlatform *platform, int ifindex); void nm_ip4_config_add_dependent_routes (NMIP4Config *self, guint32 route_table, guint32 route_metric, gboolean is_vrf, GPtrArray ** out_ip4_dev_route_blacklist); gboolean nm_ip4_config_commit (const NMIP4Config *self, NMPlatform *platform, NMIPRouteTableSyncMode route_table_sync); void nm_ip4_config_merge_setting (NMIP4Config *self, NMSettingIPConfig *setting, NMSettingConnectionMdns mdns, NMSettingConnectionLlmnr llmnr, guint32 route_table, guint32 route_metric); NMSetting *nm_ip4_config_create_setting (const NMIP4Config *self); void nm_ip4_config_merge (NMIP4Config *dst, const NMIP4Config *src, NMIPConfigMergeFlags merge_flags, guint32 default_route_metric_penalty); void nm_ip4_config_subtract (NMIP4Config *dst, const NMIP4Config *src, guint32 default_route_metric_penalty); void nm_ip4_config_intersect (NMIP4Config *dst, const NMIP4Config *src, gboolean intersect_addresses, gboolean intersect_routes, guint32 default_route_metric_penalty); NMIP4Config *nm_ip4_config_intersect_alloc (const NMIP4Config *a, const NMIP4Config *b, gboolean intersect_addresses, gboolean intersect_routes, guint32 default_route_metric_penalty); gboolean nm_ip4_config_replace (NMIP4Config *dst, const NMIP4Config *src, gboolean *relevant_changes); const NMPObject *nm_ip4_config_best_default_route_get (const NMIP4Config *self); const NMPObject *_nm_ip4_config_best_default_route_find (const NMIP4Config *self); in_addr_t nmtst_ip4_config_get_gateway (NMIP4Config *config); NMSettingConnectionMdns nm_ip4_config_mdns_get (const NMIP4Config *self); void nm_ip4_config_mdns_set (NMIP4Config *self, NMSettingConnectionMdns mdns); NMSettingConnectionLlmnr nm_ip4_config_llmnr_get (const NMIP4Config *self); void nm_ip4_config_llmnr_set (NMIP4Config *self, NMSettingConnectionLlmnr llmnr); void nm_ip4_config_set_config_flags (NMIP4Config *self, NMIPConfigFlags flags, NMIPConfigFlags mask); NMIPConfigFlags nm_ip4_config_get_config_flags (const NMIP4Config *self); const NMDedupMultiHeadEntry *nm_ip4_config_lookup_addresses (const NMIP4Config *self); void nm_ip4_config_reset_addresses (NMIP4Config *self); void nm_ip4_config_add_address (NMIP4Config *self, const NMPlatformIP4Address *address); void _nmtst_ip4_config_del_address (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_addresses (const NMIP4Config *self); const NMPlatformIP4Address *nm_ip4_config_get_first_address (const NMIP4Config *self); const NMPlatformIP4Address *_nmtst_ip4_config_get_address (const NMIP4Config *self, guint i); gboolean nm_ip4_config_address_exists (const NMIP4Config *self, const NMPlatformIP4Address *address); const NMDedupMultiHeadEntry *nm_ip4_config_lookup_routes (const NMIP4Config *self); void nm_ip4_config_reset_routes (NMIP4Config *self); void nm_ip4_config_add_route (NMIP4Config *self, const NMPlatformIP4Route *route, const NMPObject **out_obj_new); void _nmtst_ip4_config_del_route (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_routes (const NMIP4Config *self); const NMPlatformIP4Route *_nmtst_ip4_config_get_route (const NMIP4Config *self, guint i); const NMPlatformIP4Route *nm_ip4_config_get_direct_route_for_host (const NMIP4Config *self, in_addr_t host, guint32 route_table); void nm_ip4_config_update_routes_metric (NMIP4Config *self, gint64 metric); void nm_ip4_config_reset_nameservers (NMIP4Config *self); void nm_ip4_config_add_nameserver (NMIP4Config *self, guint32 nameserver); static inline void _nm_ip4_config_add_nameserver (NMIP4Config *self, const guint32 *nameserver) { nm_ip4_config_add_nameserver (self, *nameserver); } void nm_ip4_config_del_nameserver (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_nameservers (const NMIP4Config *self); guint32 nm_ip4_config_get_nameserver (const NMIP4Config *self, guint i); const in_addr_t *_nm_ip4_config_get_nameserver (const NMIP4Config *self, guint i); void nm_ip4_config_reset_domains (NMIP4Config *self); void nm_ip4_config_add_domain (NMIP4Config *self, const char *domain); void nm_ip4_config_del_domain (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_domains (const NMIP4Config *self); const char * nm_ip4_config_get_domain (const NMIP4Config *self, guint i); void nm_ip4_config_reset_searches (NMIP4Config *self); void nm_ip4_config_add_search (NMIP4Config *self, const char *search); void nm_ip4_config_del_search (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_searches (const NMIP4Config *self); const char * nm_ip4_config_get_search (const NMIP4Config *self, guint i); void nm_ip4_config_reset_dns_options (NMIP4Config *self); void nm_ip4_config_add_dns_option (NMIP4Config *self, const char *option); void nm_ip4_config_del_dns_option (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_dns_options (const NMIP4Config *self); const char * nm_ip4_config_get_dns_option (const NMIP4Config *self, guint i); void nm_ip4_config_set_dns_priority (NMIP4Config *self, int priority); int nm_ip4_config_get_dns_priority (const NMIP4Config *self); void nm_ip4_config_reset_nis_servers (NMIP4Config *self); void nm_ip4_config_add_nis_server (NMIP4Config *self, guint32 nis); void nm_ip4_config_del_nis_server (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_nis_servers (const NMIP4Config *self); guint32 nm_ip4_config_get_nis_server (const NMIP4Config *self, guint i); void nm_ip4_config_set_nis_domain (NMIP4Config *self, const char *domain); const char * nm_ip4_config_get_nis_domain (const NMIP4Config *self); void nm_ip4_config_reset_wins (NMIP4Config *self); void nm_ip4_config_add_wins (NMIP4Config *self, guint32 wins); void nm_ip4_config_del_wins (NMIP4Config *self, guint i); guint nm_ip4_config_get_num_wins (const NMIP4Config *self); guint32 nm_ip4_config_get_wins (const NMIP4Config *self, guint i); void nm_ip4_config_set_mtu (NMIP4Config *self, guint32 mtu, NMIPConfigSource source); guint32 nm_ip4_config_get_mtu (const NMIP4Config *self); NMIPConfigSource nm_ip4_config_get_mtu_source (const NMIP4Config *self); void nm_ip4_config_set_metered (NMIP4Config *self, gboolean metered); gboolean nm_ip4_config_get_metered (const NMIP4Config *self); const NMPObject *nm_ip4_config_nmpobj_lookup (const NMIP4Config *self, const NMPObject *needle); gboolean nm_ip4_config_nmpobj_remove (NMIP4Config *self, const NMPObject *needle); void nm_ip4_config_hash (const NMIP4Config *self, GChecksum *sum, gboolean dns_only); gboolean nm_ip4_config_equal (const NMIP4Config *a, const NMIP4Config *b); gboolean _nm_ip_config_check_and_add_domain (GPtrArray *array, const char *domain); void nm_ip_config_dump (const NMIPConfig *self, const char *detail, NMLogLevel level, NMLogDomain domain); /*****************************************************************************/ #include "nm-ip6-config.h" static inline gboolean NM_IS_IP_CONFIG (gconstpointer config, int addr_family) { if (addr_family == AF_UNSPEC) return NM_IS_IP4_CONFIG (config) || NM_IS_IP6_CONFIG (config); if (addr_family == AF_INET) return NM_IS_IP4_CONFIG (config); if (addr_family == AF_INET6) return NM_IS_IP6_CONFIG (config); g_return_val_if_reached (FALSE); } #if _NM_CC_SUPPORT_GENERIC /* _NM_IS_IP_CONFIG() is a bit unusual. If _Generic() is supported, * it checks whether @config is either NM_IS_IP4_CONFIG() or NM_IS_IP6_CONFIG(), * depending on the pointer type of @config. * * For example, with _Generic() support, the following assertions would fail: * NMIP6Config *ptr = (NMIP6Config *) nm_ip4_config_new(...); * g_assert (_NM_IS_IP_CONFIG (ptr, ptr)); * but the following would pass: * NMIP4Config *ptr = nm_ip4_config_new(...); * g_assert (_NM_IS_IP_CONFIG (ptr, ptr)); */ #define _NM_IS_IP_CONFIG(typeexpr, config) \ ({ \ const void *const _config = (config); \ _Generic ((typeexpr), \ const void *const: (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ const void * : (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ void *const: (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ void * : (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ const NMIPConfig *const: (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ const NMIPConfig * : (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ NMIPConfig *const: (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ NMIPConfig * : (NM_IS_IP4_CONFIG (_config) || NM_IS_IP6_CONFIG (_config)), \ const NMIP4Config *const: (NM_IS_IP4_CONFIG (_config)), \ const NMIP4Config * : (NM_IS_IP4_CONFIG (_config)), \ NMIP4Config *const: (NM_IS_IP4_CONFIG (_config)), \ NMIP4Config * : (NM_IS_IP4_CONFIG (_config)), \ const NMIP6Config *const: (NM_IS_IP6_CONFIG (_config)), \ const NMIP6Config * : (NM_IS_IP6_CONFIG (_config)), \ NMIP6Config *const: (NM_IS_IP6_CONFIG (_config)), \ NMIP6Config * : (NM_IS_IP6_CONFIG (_config))); \ }) #else #define _NM_IS_IP_CONFIG(typeexpr, config) NM_IS_IP_CONFIG(config, AF_UNSPEC) #endif #define NM_IP_CONFIG_CAST(config) \ ({ \ const void *const _configx = (config); \ \ nm_assert (!_configx || _NM_IS_IP_CONFIG ((config), _configx)); \ NM_CONSTCAST_FULL (NMIPConfig, (config), _configx, NMIP4Config, NMIP6Config); \ }) static inline int nm_ip_config_get_addr_family (const NMIPConfig *config) { if (NM_IS_IP4_CONFIG (config)) return AF_INET; if (NM_IS_IP6_CONFIG (config)) return AF_INET6; g_return_val_if_reached (AF_UNSPEC); } #define _NM_IP_CONFIG_DISPATCH(config, v4_func, v6_func, ...) \ G_STMT_START { \ gconstpointer _config = (config); \ \ if (NM_IS_IP4_CONFIG (_config)) { \ return v4_func ((NMIP4Config *) _config, ##__VA_ARGS__); \ } else { \ nm_assert (NM_IS_IP6_CONFIG (_config)); \ return v6_func ((NMIP6Config *) _config, ##__VA_ARGS__); \ } \ } G_STMT_END #define _NM_IP_CONFIG_DISPATCH_VOID(config, v4_func, v6_func, ...) \ G_STMT_START { \ gconstpointer _config = (config); \ \ if (NM_IS_IP4_CONFIG (_config)) { \ v4_func ((NMIP4Config *) _config, ##__VA_ARGS__); \ } else { \ nm_assert (NM_IS_IP6_CONFIG (_config)); \ v6_func ((NMIP6Config *) _config, ##__VA_ARGS__); \ } \ } G_STMT_END static inline int nm_ip_config_get_ifindex (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_ifindex, nm_ip6_config_get_ifindex); } static inline void nm_ip_config_hash (const NMIPConfig *self, GChecksum *sum, gboolean dns_only) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_hash, nm_ip6_config_hash, sum, dns_only); } static inline void nm_ip_config_add_address (NMIPConfig *self, const NMPlatformIPAddress *address) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_add_address, nm_ip6_config_add_address, (gconstpointer) address); } static inline void nm_ip_config_reset_addresses (NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_reset_addresses, nm_ip6_config_reset_addresses); } static inline void nm_ip_config_add_route (NMIPConfig *self, const NMPlatformIPRoute *new, const NMPObject **out_obj_new) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_add_route, nm_ip6_config_add_route, (gpointer) new, out_obj_new); } static inline void nm_ip_config_reset_routes (NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_reset_routes, nm_ip6_config_reset_routes); } static inline int nm_ip_config_get_dns_priority (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_dns_priority, nm_ip6_config_get_dns_priority); } static inline void nm_ip_config_set_dns_priority (NMIPConfig *self, int priority) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_set_dns_priority, nm_ip6_config_set_dns_priority, priority); } static inline void nm_ip_config_add_nameserver (NMIPConfig *self, const NMIPAddr *ns) { _NM_IP_CONFIG_DISPATCH_VOID (self, _nm_ip4_config_add_nameserver, nm_ip6_config_add_nameserver, (gconstpointer) ns); } static inline void nm_ip_config_reset_nameservers (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_reset_nameservers, nm_ip6_config_reset_nameservers); } static inline guint nm_ip_config_get_num_nameservers (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_num_nameservers, nm_ip6_config_get_num_nameservers); } static inline gconstpointer nm_ip_config_get_nameserver (const NMIPConfig *self, guint i) { _NM_IP_CONFIG_DISPATCH (self, _nm_ip4_config_get_nameserver, nm_ip6_config_get_nameserver, i); } static inline guint nm_ip_config_get_num_domains (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_num_domains, nm_ip6_config_get_num_domains); } static inline const char * nm_ip_config_get_domain (const NMIPConfig *self, guint i) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_domain, nm_ip6_config_get_domain, i); } static inline void nm_ip_config_reset_searches (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_reset_searches, nm_ip6_config_reset_searches); } static inline void nm_ip_config_add_search (const NMIPConfig *self, const char *new) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_add_search, nm_ip6_config_add_search, new); } static inline guint nm_ip_config_get_num_searches (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_num_searches, nm_ip6_config_get_num_searches); } static inline const char * nm_ip_config_get_search (const NMIPConfig *self, guint i) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_search, nm_ip6_config_get_search, i); } static inline guint nm_ip_config_get_num_dns_options (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_num_dns_options, nm_ip6_config_get_num_dns_options); } static inline const char * nm_ip_config_get_dns_option (const NMIPConfig *self, guint i) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_dns_option, nm_ip6_config_get_dns_option, i); } static inline const NMPObject * nm_ip_config_best_default_route_get (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_best_default_route_get, nm_ip6_config_best_default_route_get); } static inline NMIPConfigFlags nm_ip_config_get_config_flags (const NMIPConfig *self) { _NM_IP_CONFIG_DISPATCH (self, nm_ip4_config_get_config_flags, nm_ip6_config_get_config_flags); } static inline void nm_ip_config_set_config_flags (NMIPConfig *self, NMIPConfigFlags flags, NMIPConfigFlags mask) { _NM_IP_CONFIG_DISPATCH_VOID (self, nm_ip4_config_set_config_flags, nm_ip6_config_set_config_flags, flags, mask); } #define _NM_IP_CONFIG_DISPATCH_SET_OP(_return, dst, src, v4_func, v6_func, ...) \ G_STMT_START { \ gpointer _dst = (dst); \ gconstpointer _src = (src); \ \ if (NM_IS_IP4_CONFIG (_dst)) { \ nm_assert (NM_IS_IP4_CONFIG (_src)); \ _return v4_func ((NMIP4Config *) _dst, (const NMIP4Config *) _src, ##__VA_ARGS__); \ } else { \ nm_assert (NM_IS_IP6_CONFIG (_src)); \ _return v6_func ((NMIP6Config *) _dst, (const NMIP6Config *) _src, ##__VA_ARGS__); \ } \ } G_STMT_END static inline void nm_ip_config_intersect (NMIPConfig *dst, const NMIPConfig *src, gboolean intersect_addresses, gboolean intersect_routes, guint32 default_route_metric_penalty) { _NM_IP_CONFIG_DISPATCH_SET_OP (, dst, src, nm_ip4_config_intersect, nm_ip6_config_intersect, intersect_addresses, intersect_routes, default_route_metric_penalty); } static inline void nm_ip_config_subtract (NMIPConfig *dst, const NMIPConfig *src, guint32 default_route_metric_penalty) { _NM_IP_CONFIG_DISPATCH_SET_OP (, dst, src, nm_ip4_config_subtract, nm_ip6_config_subtract, default_route_metric_penalty); } static inline void nm_ip_config_merge (NMIPConfig *dst, const NMIPConfig *src, NMIPConfigMergeFlags merge_flags, guint32 default_route_metric_penalty) { _NM_IP_CONFIG_DISPATCH_SET_OP (, dst, src, nm_ip4_config_merge, nm_ip6_config_merge, merge_flags, default_route_metric_penalty); } static inline gboolean nm_ip_config_replace (NMIPConfig *dst, const NMIPConfig *src, gboolean *relevant_changes) { _NM_IP_CONFIG_DISPATCH_SET_OP (return, dst, src, nm_ip4_config_replace, nm_ip6_config_replace, relevant_changes); } static inline NMIPConfig * nm_ip_config_intersect_alloc (const NMIPConfig *a, const NMIPConfig *b, gboolean intersect_addresses, gboolean intersect_routes, guint32 default_route_metric_penalty) { if (NM_IS_IP4_CONFIG (a)) { nm_assert (NM_IS_IP4_CONFIG (b)); return (NMIPConfig *) nm_ip4_config_intersect_alloc ((const NMIP4Config *) a, (const NMIP4Config *) b, intersect_addresses, intersect_routes, default_route_metric_penalty); } else { nm_assert (NM_IS_IP6_CONFIG (a)); nm_assert (NM_IS_IP6_CONFIG (b)); return (NMIPConfig *) nm_ip6_config_intersect_alloc ((const NMIP6Config *) a, (const NMIP6Config *) b, intersect_addresses, intersect_routes, default_route_metric_penalty); } } #endif /* __NETWORKMANAGER_IP4_CONFIG_H__ */