#include <urcu-bp.h>
#include <tracepoint-types.h>
#include <urcu/compiler.h>
+#include <dlfcn.h> /* for dlopen */
#ifdef __cplusplus
extern "C" {
{ \
struct tracepoint_probe *__tp_probe; \
\
+ if (!rcu_read_lock_bp) \
+ return; \
rcu_read_lock_bp(); \
- __tp_probe = rcu_dereference(__tracepoint_##provider##___##name.probes); \
+ __tp_probe = rcu_dereference_bp(__tracepoint_##provider##___##name.probes); \
if (caa_unlikely(!__tp_probe)) \
goto end; \
do { \
__tracepoint_probe_unregister(name, func, data); \
}
+extern int __tracepoint_probe_register(const char *name, void *func, void *data);
+extern int __tracepoint_probe_unregister(const char *name, void *func, void *data);
+
+#ifdef TRACEPOINT_DEFINE
+
/*
* Note: to allow PIC code, we need to allow the linker to update the pointers
* in the __tracepoints_ptrs section.
__attribute__((used, section("__tracepoints_ptrs"))) = \
&__tracepoint_##provider##___##name;
-extern int __tracepoint_probe_register(const char *name, void *func, void *data);
-extern int __tracepoint_probe_unregister(const char *name, void *func, void *data);
-extern int tracepoint_register_lib(struct tracepoint * const *tracepoints_start,
+static int (*tracepoint_register_lib)(struct tracepoint * const *tracepoints_start,
int tracepoints_count);
-extern int tracepoint_unregister_lib(struct tracepoint * const *tracepoints_start);
+static int (*tracepoint_unregister_lib)(struct tracepoint * const *tracepoints_start);
+static void *liblttngust_handle;
/*
* These weak symbols, the constructor, and destructor take care of
{
if (__tracepoint_registered++)
return;
+
+ liblttngust_handle = dlopen("liblttng-ust.so", RTLD_NOW | RTLD_GLOBAL);
+ if (!liblttngust_handle)
+ return;
+ tracepoint_register_lib = dlsym(liblttngust_handle,
+ "tracepoint_register_lib");
+ tracepoint_unregister_lib = dlsym(liblttngust_handle,
+ "tracepoint_unregister_lib");
tracepoint_register_lib(__start___tracepoints_ptrs,
__stop___tracepoints_ptrs -
__start___tracepoints_ptrs);
static void __attribute__((destructor)) __tracepoints__destroy(void)
{
+ int ret;
if (--__tracepoint_registered)
return;
- tracepoint_unregister_lib(__start___tracepoints_ptrs);
+ if (tracepoint_unregister_lib)
+ tracepoint_unregister_lib(__start___tracepoints_ptrs);
+ if (liblttngust_handle) {
+ tracepoint_unregister_lib = NULL;
+ tracepoint_register_lib = NULL;
+ ret = dlclose(liblttngust_handle);
+ assert(!ret);
+ }
}
+#else /* TRACEPOINT_DEFINE */
+
+#define _DEFINE_TRACEPOINT(provider, name)
+
+#endif /* #else TRACEPOINT_DEFINE */
+
#ifdef __cplusplus
}
#endif
*/
#define TRACEPOINT_EVENT(provider, name, args, fields) \
- _DECLARE_TRACEPOINT(provider, name, _TP_PARAMS(args))
+ _DECLARE_TRACEPOINT(provider, name, _TP_PARAMS(args)) \
+ _DEFINE_TRACEPOINT(provider, name)
#define TRACEPOINT_EVENT_CLASS(provider, name, args, fields)
#define TRACEPOINT_EVENT_INSTANCE(provider, _template, name, args) \
- _DECLARE_TRACEPOINT(provider, name, _TP_PARAMS(args))
+ _DECLARE_TRACEPOINT(provider, name, _TP_PARAMS(args)) \
+ _DEFINE_TRACEPOINT(provider, name)
#endif /* #ifndef TRACEPOINT_EVENT */
* Typical use of these loglevels:
*
* 1) Declare the mapping between loglevel names and an integer values
- * within TRACEPOINT_LOGLEVEL_ENUM, using TP_LOGLEVEL for each tuple.
- * Do _NOT_ add comma (,) nor semicolon (;) between the
- * TRACEPOINT_LOGLEVEL_ENUM entries. Do _NOT_ add comma (,) nor
- * semicolon (;) after the TRACEPOINT_LOGLEVEL_ENUM declaration. The
- * name should be a proper C99 identifier.
+ * within TRACEPOINT_LOGLEVEL_ENUM(), using tp_loglevel() for each
+ * tuple. Do _NOT_ add comma (,) nor semicolon (;) between the
+ * tp_loglevel entries contained within TRACEPOINT_LOGLEVEL_ENUM().
+ * Do _NOT_ add comma (,) nor semicolon (;) after the
+ * TRACEPOINT_LOGLEVEL_ENUM() declaration. The name should be a
+ * proper C99 identifier.
*
* TRACEPOINT_LOGLEVEL_ENUM(
- * TP_LOGLEVEL( < loglevel_name >, < value > )
- * TP_LOGLEVEL( < loglevel_name >, < value > )
+ * tp_loglevel( < loglevel_name >, < value > )
+ * tp_loglevel( < loglevel_name >, < value > )
* ...
* )
*
* e.g.:
*
* TRACEPOINT_LOGLEVEL_ENUM(
- * TP_LOGLEVEL(LOG_EMERG, 0)
- * TP_LOGLEVEL(LOG_ALERT, 1)
- * TP_LOGLEVEL(LOG_CRIT, 2)
- * TP_LOGLEVEL(LOG_ERR, 3)
- * TP_LOGLEVEL(LOG_WARNING, 4)
- * TP_LOGLEVEL(LOG_NOTICE, 5)
- * TP_LOGLEVEL(LOG_INFO, 6)
- * TP_LOGLEVEL(LOG_DEBUG, 7)
+ * tp_loglevel(LOG_EMERG, 0)
+ * tp_loglevel(LOG_ALERT, 1)
+ * tp_loglevel(LOG_CRIT, 2)
+ * tp_loglevel(LOG_ERR, 3)
+ * tp_loglevel(LOG_WARNING, 4)
+ * tp_loglevel(LOG_NOTICE, 5)
+ * tp_loglevel(LOG_INFO, 6)
+ * tp_loglevel(LOG_DEBUG, 7)
* )
*
* 2) Then, declare tracepoint loglevels for tracepoints. A
* provider.
*/
-#define TRACEPOINT_LOGLEVEL_ENUM()
-#define TRACEPOINT_LOGLEVEL(name, loglevel)
+#define TRACEPOINT_LOGLEVEL_ENUM(...)
+#define TRACEPOINT_LOGLEVEL(provider, name, loglevel)
#endif /* #ifndef TRACEPOINT_LOGLEVEL */