Blame liblttng-ust/lttng-context-provider.c

Packit c04fcb
/*
Packit c04fcb
 * lttng-context-provider.c
Packit c04fcb
 *
Packit c04fcb
 * LTTng UST application context provider.
Packit c04fcb
 *
Packit c04fcb
 * Copyright (C) 2016 Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
Packit c04fcb
 *
Packit c04fcb
 * This library is free software; you can redistribute it and/or
Packit c04fcb
 * modify it under the terms of the GNU Lesser General Public
Packit c04fcb
 * License as published by the Free Software Foundation; only
Packit c04fcb
 * version 2.1 of the License.
Packit c04fcb
 *
Packit c04fcb
 * This library is distributed in the hope that it will be useful,
Packit c04fcb
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit c04fcb
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Packit c04fcb
 * Lesser General Public License for more details.
Packit c04fcb
 *
Packit c04fcb
 * You should have received a copy of the GNU Lesser General Public
Packit c04fcb
 * License along with this library; if not, write to the Free Software
Packit c04fcb
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Packit c04fcb
 */
Packit c04fcb
Packit c04fcb
#include <sys/types.h>
Packit c04fcb
#include <unistd.h>
Packit c04fcb
#include <lttng/ust-context-provider.h>
Packit c04fcb
#include "lttng-tracer-core.h"
Packit c04fcb
#include "jhash.h"
Packit c04fcb
#include <helper.h>
Packit c04fcb
Packit c04fcb
#define CONTEXT_PROVIDER_HT_BITS	12
Packit c04fcb
#define CONTEXT_PROVIDER_HT_SIZE	(1U << CONTEXT_PROVIDER_HT_BITS)
Packit c04fcb
struct context_provider_ht {
Packit c04fcb
	struct cds_hlist_head table[CONTEXT_PROVIDER_HT_SIZE];
Packit c04fcb
};
Packit c04fcb
Packit c04fcb
static struct context_provider_ht context_provider_ht;
Packit c04fcb
Packit c04fcb
static struct lttng_ust_context_provider *
Packit c04fcb
		lookup_provider_by_name(const char *name)
Packit c04fcb
{
Packit c04fcb
	struct cds_hlist_head *head;
Packit c04fcb
	struct cds_hlist_node *node;
Packit c04fcb
	struct lttng_ust_context_provider *provider;
Packit c04fcb
	uint32_t hash;
Packit c04fcb
	const char *end;
Packit c04fcb
	size_t len;
Packit c04fcb
Packit c04fcb
	/* Lookup using everything before first ':' as key. */
Packit c04fcb
	end = strchr(name, ':');
Packit c04fcb
	if (end)
Packit c04fcb
		len = end - name;
Packit c04fcb
	else
Packit c04fcb
		len = strlen(name);
Packit c04fcb
	hash = jhash(name, len, 0);
Packit c04fcb
	head = &context_provider_ht.table[hash & (CONTEXT_PROVIDER_HT_SIZE - 1)];
Packit c04fcb
	cds_hlist_for_each_entry(provider, node, head, node) {
Packit c04fcb
		if (!strncmp(provider->name, name, len))
Packit c04fcb
			return provider;
Packit c04fcb
	}
Packit c04fcb
	return NULL;
Packit c04fcb
}
Packit c04fcb
Packit c04fcb
int lttng_ust_context_provider_register(struct lttng_ust_context_provider *provider)
Packit c04fcb
{
Packit c04fcb
	struct cds_hlist_head *head;
Packit c04fcb
	size_t name_len = strlen(provider->name);
Packit c04fcb
	uint32_t hash;
Packit c04fcb
	int ret = 0;
Packit c04fcb
Packit c04fcb
	/* Provider name starts with "$app.". */
Packit c04fcb
	if (strncmp("$app.", provider->name, strlen("$app.") != 0))
Packit c04fcb
		return -EINVAL;
Packit c04fcb
	/* Provider name cannot contain a column character. */
Packit c04fcb
	if (strchr(provider->name, ':'))
Packit c04fcb
		return -EINVAL;
Packit c04fcb
	if (ust_lock()) {
Packit c04fcb
		ret = -EBUSY;
Packit c04fcb
		goto end;
Packit c04fcb
	}
Packit c04fcb
	if (lookup_provider_by_name(provider->name)) {
Packit c04fcb
		ret = -EBUSY;
Packit c04fcb
		goto end;
Packit c04fcb
	}
Packit c04fcb
	hash = jhash(provider->name, name_len, 0);
Packit c04fcb
	head = &context_provider_ht.table[hash & (CONTEXT_PROVIDER_HT_SIZE - 1)];
Packit c04fcb
	cds_hlist_add_head(&provider->node, head);
Packit c04fcb
	lttng_ust_context_set_session_provider(provider->name,
Packit c04fcb
		provider->get_size, provider->record,
Packit c04fcb
		provider->get_value);
Packit c04fcb
end:
Packit c04fcb
	ust_unlock();
Packit c04fcb
	return ret;
Packit c04fcb
}
Packit c04fcb
Packit c04fcb
void lttng_ust_context_provider_unregister(struct lttng_ust_context_provider *provider)
Packit c04fcb
{
Packit c04fcb
	if (ust_lock())
Packit c04fcb
		goto end;
Packit c04fcb
	lttng_ust_context_set_session_provider(provider->name,
Packit c04fcb
		lttng_ust_dummy_get_size, lttng_ust_dummy_record,
Packit c04fcb
		lttng_ust_dummy_get_value);
Packit c04fcb
	cds_hlist_del(&provider->node);
Packit c04fcb
end:
Packit c04fcb
	ust_unlock();
Packit c04fcb
}
Packit c04fcb
Packit c04fcb
/*
Packit c04fcb
 * Called with ust mutex held.
Packit c04fcb
 * Add application context to array of context, even if the application
Packit c04fcb
 * context is not currently loaded by application. It will then use the
Packit c04fcb
 * dummy callbacks in that case.
Packit c04fcb
 * Always performed before tracing is started, since it modifies
Packit c04fcb
 * metadata describing the context.
Packit c04fcb
 */
Packit c04fcb
int lttng_ust_add_app_context_to_ctx_rcu(const char *name,
Packit c04fcb
		struct lttng_ctx **ctx)
Packit c04fcb
{
Packit c04fcb
	struct lttng_ust_context_provider *provider;
Packit c04fcb
	struct lttng_ctx_field new_field;
Packit c04fcb
	int ret;
Packit c04fcb
Packit c04fcb
	if (*ctx && lttng_find_context(*ctx, name))
Packit c04fcb
		return -EEXIST;
Packit c04fcb
	/*
Packit c04fcb
	 * For application context, add it by expanding
Packit c04fcb
	 * ctx array.
Packit c04fcb
	 */
Packit c04fcb
	memset(&new_field, 0, sizeof(new_field));
Packit c04fcb
	new_field.field_name = strdup(name);
Packit c04fcb
	if (!new_field.field_name)
Packit c04fcb
		return -ENOMEM;
Packit c04fcb
	new_field.event_field.name = new_field.field_name;
Packit c04fcb
	new_field.event_field.type.atype = atype_dynamic;
Packit c04fcb
	/*
Packit c04fcb
	 * If provider is not found, we add the context anyway, but
Packit c04fcb
	 * it will provide a dummy context.
Packit c04fcb
	 */
Packit c04fcb
	provider = lookup_provider_by_name(name);
Packit c04fcb
	if (provider) {
Packit c04fcb
		new_field.get_size = provider->get_size;
Packit c04fcb
		new_field.record = provider->record;
Packit c04fcb
		new_field.get_value = provider->get_value;
Packit c04fcb
	} else {
Packit c04fcb
		new_field.get_size = lttng_ust_dummy_get_size;
Packit c04fcb
		new_field.record = lttng_ust_dummy_record;
Packit c04fcb
		new_field.get_value = lttng_ust_dummy_get_value;
Packit c04fcb
	}
Packit c04fcb
	ret = lttng_context_add_rcu(ctx, &new_field);
Packit c04fcb
	if (ret) {
Packit c04fcb
		free(new_field.field_name);
Packit c04fcb
		return ret;
Packit c04fcb
	}
Packit c04fcb
	return 0;
Packit c04fcb
}