Home | History | Annotate | Download | only in netlink-private
      1 /*
      2  * netlink-private/netlink.h	Local Netlink Interface
      3  *
      4  *	This library is free software; you can redistribute it and/or
      5  *	modify it under the terms of the GNU Lesser General Public
      6  *	License as published by the Free Software Foundation version 2.1
      7  *	of the License.
      8  *
      9  * Copyright (c) 2003-2013 Thomas Graf <tgraf (at) suug.ch>
     10  */
     11 
     12 #ifndef NETLINK_LOCAL_H_
     13 #define NETLINK_LOCAL_H_
     14 
     15 #include <stdio.h>
     16 #include <errno.h>
     17 #include <stdlib.h>
     18 #include <string.h>
     19 #include <unistd.h>
     20 #include <fcntl.h>
     21 #include <math.h>
     22 #include <time.h>
     23 #include <stdarg.h>
     24 #include <ctype.h>
     25 #include <sys/types.h>
     26 #include <sys/stat.h>
     27 #include <sys/socket.h>
     28 #include <inttypes.h>
     29 #include <assert.h>
     30 #include <limits.h>
     31 #include <search.h>
     32 
     33 #include <arpa/inet.h>
     34 #include <netdb.h>
     35 
     36 #include <defs.h>
     37 
     38 #ifndef SOL_NETLINK
     39 #define SOL_NETLINK 270
     40 #endif
     41 
     42 #include <linux/types.h>
     43 
     44 /* local header copies */
     45 #include <linux/if.h>
     46 #include <linux/if_arp.h>
     47 #include <linux/if_ether.h>
     48 #include <linux/ethtool.h>
     49 #include <linux/pkt_sched.h>
     50 #include <linux/pkt_cls.h>
     51 #include <linux/gen_stats.h>
     52 #include <linux/ip_mp_alg.h>
     53 #include <linux/atm.h>
     54 #include <linux/ip.h>
     55 #include <linux/ipv6.h>
     56 #include <linux/snmp.h>
     57 
     58 #ifndef DISABLE_PTHREADS
     59 #include <pthread.h>
     60 #endif
     61 
     62 #include <netlink/netlink.h>
     63 #include <netlink/handlers.h>
     64 #include <netlink/cache.h>
     65 #include <netlink/route/tc.h>
     66 #include <netlink-private/object-api.h>
     67 #include <netlink-private/cache-api.h>
     68 #include <netlink-private/types.h>
     69 
     70 #define NSEC_PER_SEC	1000000000L
     71 
     72 struct trans_tbl {
     73 	int i;
     74 	const char *a;
     75 };
     76 
     77 #define __ADD(id, name) { .i = id, .a = #name },
     78 
     79 struct trans_list {
     80 	int i;
     81 	char *a;
     82 	struct nl_list_head list;
     83 };
     84 
     85 #ifdef NL_DEBUG
     86 #define NL_DBG(LVL,FMT,ARG...)						\
     87 	do {								\
     88 		if (LVL <= nl_debug)					\
     89 			fprintf(stderr,					\
     90 				"DBG<" #LVL ">%20s:%-4u %s: " FMT,	\
     91 				__FILE__, __LINE__,			\
     92 				__PRETTY_FUNCTION__, ##ARG);		\
     93 	} while (0)
     94 #else /* NL_DEBUG */
     95 #define NL_DBG(LVL,FMT,ARG...) do { } while(0)
     96 #endif /* NL_DEBUG */
     97 
     98 #define BUG()                            				\
     99 	do {                                 				\
    100 		fprintf(stderr, "BUG at file position %s:%d:%s\n",  	\
    101 			__FILE__, __LINE__, __PRETTY_FUNCTION__); 	\
    102 		assert(0);						\
    103 	} while (0)
    104 
    105 #define BUG_ON(condition)						\
    106 	do {								\
    107 		if (condition)						\
    108 			BUG();						\
    109 	} while (0)
    110 
    111 
    112 #define APPBUG(msg)							\
    113 	do {								\
    114 		fprintf(stderr, "APPLICATION BUG: %s:%d:%s: %s\n",	\
    115 			__FILE__, __LINE__, __PRETTY_FUNCTION__, msg);	\
    116 		assert(0);						\
    117 	} while(0)
    118 
    119 extern int __nl_read_num_str_file(const char *path,
    120 				  int (*cb)(long, const char *));
    121 
    122 extern int __trans_list_add(int, const char *, struct nl_list_head *);
    123 extern void __trans_list_clear(struct nl_list_head *);
    124 
    125 extern char *__type2str(int, char *, size_t, const struct trans_tbl *, size_t);
    126 extern int __str2type(const char *, const struct trans_tbl *, size_t);
    127 
    128 extern char *__list_type2str(int, char *, size_t, struct nl_list_head *);
    129 extern int __list_str2type(const char *, struct nl_list_head *);
    130 
    131 extern char *__flags2str(int, char *, size_t, const struct trans_tbl *, size_t);
    132 extern int __str2flags(const char *, const struct trans_tbl *, size_t);
    133 
    134 extern void dump_from_ops(struct nl_object *, struct nl_dump_params *);
    135 
    136 static inline int nl_cb_call(struct nl_cb *cb, int type, struct nl_msg *msg)
    137 {
    138 	int ret;
    139 
    140 	cb->cb_active = type;
    141 	ret = cb->cb_set[type](msg, cb->cb_args[type]);
    142 	cb->cb_active = __NL_CB_TYPE_MAX;
    143 	return ret;
    144 }
    145 
    146 #define ARRAY_SIZE(X) (sizeof(X) / sizeof((X)[0]))
    147 
    148 /* This is also defined in stddef.h */
    149 #ifndef offsetof
    150 #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
    151 #endif
    152 
    153 #define __init __attribute__ ((constructor))
    154 #define __exit __attribute__ ((destructor))
    155 #undef __deprecated
    156 #define __deprecated __attribute__ ((deprecated))
    157 
    158 #define min(x,y) ({ \
    159 	typeof(x) _x = (x);	\
    160 	typeof(y) _y = (y);	\
    161 	(void) (&_x == &_y);		\
    162 	_x < _y ? _x : _y; })
    163 
    164 #define max(x,y) ({ \
    165 	typeof(x) _x = (x);	\
    166 	typeof(y) _y = (y);	\
    167 	(void) (&_x == &_y);		\
    168 	_x > _y ? _x : _y; })
    169 
    170 #define min_t(type,x,y) \
    171 	({ type __x = (x); type __y = (y); __x < __y ? __x: __y; })
    172 #define max_t(type,x,y) \
    173 	({ type __x = (x); type __y = (y); __x > __y ? __x: __y; })
    174 
    175 extern int nl_cache_parse(struct nl_cache_ops *, struct sockaddr_nl *,
    176 			  struct nlmsghdr *, struct nl_parser_param *);
    177 
    178 
    179 static inline void rtnl_copy_ratespec(struct rtnl_ratespec *dst,
    180 				      struct tc_ratespec *src)
    181 {
    182 	dst->rs_cell_log = src->cell_log;
    183 	dst->rs_overhead = src->overhead;
    184 	dst->rs_cell_align = src->cell_align;
    185 	dst->rs_mpu = src->mpu;
    186 	dst->rs_rate = src->rate;
    187 }
    188 
    189 static inline void rtnl_rcopy_ratespec(struct tc_ratespec *dst,
    190 				       struct rtnl_ratespec *src)
    191 {
    192 	dst->cell_log = src->rs_cell_log;
    193 	dst->overhead = src->rs_overhead;
    194 	dst->cell_align = src->rs_cell_align;
    195 	dst->mpu = src->rs_mpu;
    196 	dst->rate = src->rs_rate;
    197 }
    198 
    199 static inline char *nl_cache_name(struct nl_cache *cache)
    200 {
    201 	return cache->c_ops ? cache->c_ops->co_name : "unknown";
    202 }
    203 
    204 #define GENL_FAMILY(id, name) \
    205 	{ \
    206 		{ id, NL_ACT_UNSPEC, name }, \
    207 		END_OF_MSGTYPES_LIST, \
    208 	}
    209 
    210 static inline int wait_for_ack(struct nl_sock *sk)
    211 {
    212 	if (sk->s_flags & NL_NO_AUTO_ACK)
    213 		return 0;
    214 	else
    215 		return nl_wait_for_ack(sk);
    216 }
    217 
    218 static inline int build_sysconf_path(char **strp, const char *filename)
    219 {
    220 	char *sysconfdir;
    221 
    222 	sysconfdir = getenv("NLSYSCONFDIR");
    223 
    224 	if (!sysconfdir)
    225 		sysconfdir = SYSCONFDIR;
    226 
    227 	return asprintf(strp, "%s/%s", sysconfdir, filename);
    228 }
    229 
    230 #ifndef DISABLE_PTHREADS
    231 #define NL_LOCK(NAME) pthread_mutex_t (NAME) = PTHREAD_MUTEX_INITIALIZER
    232 #define NL_RW_LOCK(NAME) pthread_rwlock_t (NAME) = PTHREAD_RWLOCK_INITIALIZER
    233 
    234 static inline void nl_lock(pthread_mutex_t *lock)
    235 {
    236 	pthread_mutex_lock(lock);
    237 }
    238 
    239 static inline void nl_unlock(pthread_mutex_t *lock)
    240 {
    241 	pthread_mutex_unlock(lock);
    242 }
    243 
    244 static inline void nl_read_lock(pthread_rwlock_t *lock)
    245 {
    246 	pthread_rwlock_rdlock(lock);
    247 }
    248 
    249 static inline void nl_read_unlock(pthread_rwlock_t *lock)
    250 {
    251 	pthread_rwlock_unlock(lock);
    252 }
    253 
    254 static inline void nl_write_lock(pthread_rwlock_t *lock)
    255 {
    256 	pthread_rwlock_wrlock(lock);
    257 }
    258 
    259 static inline void nl_write_unlock(pthread_rwlock_t *lock)
    260 {
    261 	pthread_rwlock_unlock(lock);
    262 }
    263 
    264 #else
    265 #define NL_LOCK(NAME) int __unused_lock_ ##NAME __attribute__((unused))
    266 #define NL_RW_LOCK(NAME) int __unused_lock_ ##NAME __attribute__((unused))
    267 
    268 #define nl_lock(LOCK) do { } while(0)
    269 #define nl_unlock(LOCK) do { } while(0)
    270 #define nl_read_lock(LOCK) do { } while(0)
    271 #define nl_read_unlock(LOCK) do { } while(0)
    272 #define nl_write_lock(LOCK) do { } while(0)
    273 #define nl_write_unlock(LOCK) do { } while(0)
    274 #endif
    275 
    276 #endif
    277