Blame tools/fpgad/api/opae_events_api.c

Packit Service 3975d1
// Copyright(c) 2018-2019, Intel Corporation
Packit Service 3975d1
//
Packit Service 3975d1
// Redistribution  and  use  in source  and  binary  forms,  with  or  without
Packit Service 3975d1
// modification, are permitted provided that the following conditions are met:
Packit Service 3975d1
//
Packit Service 3975d1
// * Redistributions of  source code  must retain the  above copyright notice,
Packit Service 3975d1
//   this list of conditions and the following disclaimer.
Packit Service 3975d1
// * Redistributions in binary form must reproduce the above copyright notice,
Packit Service 3975d1
//   this list of conditions and the following disclaimer in the documentation
Packit Service 3975d1
//   and/or other materials provided with the distribution.
Packit Service 3975d1
// * Neither the name  of Intel Corporation  nor the names of its contributors
Packit Service 3975d1
//   may be used to  endorse or promote  products derived  from this  software
Packit Service 3975d1
//   without specific prior written permission.
Packit Service 3975d1
//
Packit Service 3975d1
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
Packit Service 3975d1
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING,  BUT NOT LIMITED TO,  THE
Packit Service 3975d1
// IMPLIED WARRANTIES OF  MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit Service 3975d1
// ARE DISCLAIMED.  IN NO EVENT  SHALL THE COPYRIGHT OWNER  OR CONTRIBUTORS BE
Packit Service 3975d1
// LIABLE  FOR  ANY  DIRECT,  INDIRECT,  INCIDENTAL,  SPECIAL,  EXEMPLARY,  OR
Packit Service 3975d1
// CONSEQUENTIAL  DAMAGES  (INCLUDING,  BUT  NOT LIMITED  TO,  PROCUREMENT  OF
Packit Service 3975d1
// SUBSTITUTE GOODS OR SERVICES;  LOSS OF USE,  DATA, OR PROFITS;  OR BUSINESS
Packit Service 3975d1
// INTERRUPTION)  HOWEVER CAUSED  AND ON ANY THEORY  OF LIABILITY,  WHETHER IN
Packit Service 3975d1
// CONTRACT,  STRICT LIABILITY,  OR TORT  (INCLUDING NEGLIGENCE  OR OTHERWISE)
Packit Service 3975d1
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,  EVEN IF ADVISED OF THE
Packit Service 3975d1
// POSSIBILITY OF SUCH DAMAGE.
Packit Service 3975d1
Packit Service 3975d1
#ifdef HAVE_CONFIG_H
Packit Service 3975d1
#include <config.h>
Packit Service 3975d1
#endif // HAVE_CONFIG_H
Packit Service 3975d1
Packit Service 3975d1
#include <inttypes.h>
Packit Service 3975d1
Packit Service 3975d1
#include "opae_events_api.h"
Packit Service 3975d1
Packit Service 3975d1
#ifdef LOG
Packit Service 3975d1
#undef LOG
Packit Service 3975d1
#endif
Packit Service 3975d1
#define LOG(format, ...) \
Packit Service 3975d1
log_printf("opae_events_api: " format, ##__VA_ARGS__)
Packit Service 3975d1
Packit Service 3975d1
STATIC pthread_mutex_t list_lock = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
Packit Service 3975d1
STATIC api_client_event_registry *event_registry_list;
Packit Service 3975d1
Packit Service 3975d1
int opae_api_register_event(int conn_socket,
Packit Service 3975d1
			    int fd,
Packit Service 3975d1
			    fpga_event_type e,
Packit Service 3975d1
			    uint64_t object_id)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *r =
Packit Service 3975d1
		(api_client_event_registry *) malloc(sizeof(*r));
Packit Service 3975d1
	int err;
Packit Service 3975d1
Packit Service 3975d1
	if (!r)
Packit Service 3975d1
		return ENOMEM;
Packit Service 3975d1
Packit Service 3975d1
	r->conn_socket = conn_socket;
Packit Service 3975d1
	r->fd = fd;
Packit Service 3975d1
	r->data = 1;
Packit Service 3975d1
	r->event = e;
Packit Service 3975d1
	r->object_id = object_id;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_lock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	r->next = event_registry_list;
Packit Service 3975d1
	event_registry_list = r;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_unlock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	return 0;
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
STATIC void release_event_registry(api_client_event_registry *r)
Packit Service 3975d1
{
Packit Service 3975d1
	close(r->fd);
Packit Service 3975d1
	free(r);
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
int opae_api_unregister_event(int conn_socket,
Packit Service 3975d1
			      fpga_event_type e,
Packit Service 3975d1
			      uint64_t object_id)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *trash;
Packit Service 3975d1
	api_client_event_registry *save;
Packit Service 3975d1
	int err;
Packit Service 3975d1
	int res = 0;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_lock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	trash = event_registry_list;
Packit Service 3975d1
Packit Service 3975d1
	if (!trash) { // empty list
Packit Service 3975d1
		res = 1;
Packit Service 3975d1
		goto out_unlock;
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	if ((conn_socket == trash->conn_socket) &&
Packit Service 3975d1
		(e == trash->event) &&
Packit Service 3975d1
		(object_id == trash->object_id)) {
Packit Service 3975d1
Packit Service 3975d1
		// found at head of list
Packit Service 3975d1
Packit Service 3975d1
		event_registry_list = event_registry_list->next;
Packit Service 3975d1
		release_event_registry(trash);
Packit Service 3975d1
		goto out_unlock;
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	save = trash;
Packit Service 3975d1
	trash = trash->next;
Packit Service 3975d1
	while (trash) {
Packit Service 3975d1
Packit Service 3975d1
		if ((conn_socket == trash->conn_socket) &&
Packit Service 3975d1
			(e == trash->event) &&
Packit Service 3975d1
			(object_id == trash->object_id))
Packit Service 3975d1
			break;
Packit Service 3975d1
Packit Service 3975d1
		save = trash;
Packit Service 3975d1
		trash = trash->next;
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	if (!trash) { // not found
Packit Service 3975d1
		res = 1;
Packit Service 3975d1
		goto out_unlock;
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	// found at trash
Packit Service 3975d1
	save->next = trash->next;
Packit Service 3975d1
	release_event_registry(trash);
Packit Service 3975d1
Packit Service 3975d1
out_unlock:
Packit Service 3975d1
	fpgad_mutex_unlock(err, &list_lock);
Packit Service 3975d1
	return res;
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
STATIC api_client_event_registry *
Packit Service 3975d1
find_event_for(int conn_socket)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *r;
Packit Service 3975d1
Packit Service 3975d1
	for (r = event_registry_list ; r ; r = r->next)
Packit Service 3975d1
		if (conn_socket == r->conn_socket)
Packit Service 3975d1
			break;
Packit Service 3975d1
Packit Service 3975d1
	return r;
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
void opae_api_unregister_all_events_for(int conn_socket)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *r;
Packit Service 3975d1
	int err;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_lock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	r = find_event_for(conn_socket);
Packit Service 3975d1
	while (r) {
Packit Service 3975d1
		opae_api_unregister_event(conn_socket, r->event, r->object_id);
Packit Service 3975d1
		r = find_event_for(conn_socket);
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_unlock(err, &list_lock);
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
void opae_api_unregister_all_events(void)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *r;
Packit Service 3975d1
	int err;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_lock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	for (r = event_registry_list ; r != NULL ; ) {
Packit Service 3975d1
		api_client_event_registry *trash;
Packit Service 3975d1
		trash = r;
Packit Service 3975d1
		r = r->next;
Packit Service 3975d1
		release_event_registry(trash);
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	event_registry_list = NULL;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_unlock(err, &list_lock);
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
void opae_api_for_each_registered_event
Packit Service 3975d1
(void (*cb)(api_client_event_registry *r, void *context),
Packit Service 3975d1
void *context)
Packit Service 3975d1
{
Packit Service 3975d1
	api_client_event_registry *r;
Packit Service 3975d1
	int err;
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_lock(err, &list_lock);
Packit Service 3975d1
Packit Service 3975d1
	for (r = event_registry_list; r != NULL; r = r->next) {
Packit Service 3975d1
		cb(r, context);
Packit Service 3975d1
	}
Packit Service 3975d1
Packit Service 3975d1
	fpgad_mutex_unlock(err, &list_lock);
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
STATIC void check_and_send_EVENT_ERROR(api_client_event_registry *r,
Packit Service 3975d1
				       void *context)
Packit Service 3975d1
{
Packit Service 3975d1
	fpgad_monitored_device *d =
Packit Service 3975d1
		(fpgad_monitored_device *)context;
Packit Service 3975d1
Packit Service 3975d1
	if ((r->event == FPGA_EVENT_ERROR) &&
Packit Service 3975d1
	    (r->object_id == d->object_id)) {
Packit Service 3975d1
		LOG("object_id: 0x%" PRIx64 " event: FPGA_EVENT_ERROR\n",
Packit Service 3975d1
			d->object_id);
Packit Service 3975d1
		if (write(r->fd, &r->data, sizeof(r->data)) < 0)
Packit Service 3975d1
			LOG("write failed: %s\n", strerror(errno));
Packit Service 3975d1
		r->data++;
Packit Service 3975d1
	}
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
void opae_api_send_EVENT_ERROR(fpgad_monitored_device *d)
Packit Service 3975d1
{
Packit Service 3975d1
	opae_api_for_each_registered_event(check_and_send_EVENT_ERROR,
Packit Service 3975d1
					   d);
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
STATIC void check_and_send_EVENT_POWER_THERMAL(api_client_event_registry *r,
Packit Service 3975d1
					       void *context)
Packit Service 3975d1
{
Packit Service 3975d1
	fpgad_monitored_device *d =
Packit Service 3975d1
		(fpgad_monitored_device *)context;
Packit Service 3975d1
Packit Service 3975d1
	if ((r->event == FPGA_EVENT_POWER_THERMAL) &&
Packit Service 3975d1
	    (r->object_id == d->object_id)) {
Packit Service 3975d1
		LOG("object_id: 0x%" PRIx64 " event: FPGA_EVENT_POWER_THERMAL\n",
Packit Service 3975d1
			d->object_id);
Packit Service 3975d1
		if (write(r->fd, &r->data, sizeof(r->data)) < 0)
Packit Service 3975d1
			LOG("write failed: %s\n", strerror(errno));
Packit Service 3975d1
		r->data++;
Packit Service 3975d1
	}
Packit Service 3975d1
}
Packit Service 3975d1
Packit Service 3975d1
void opae_api_send_EVENT_POWER_THERMAL(fpgad_monitored_device *d)
Packit Service 3975d1
{
Packit Service 3975d1
	opae_api_for_each_registered_event(check_and_send_EVENT_POWER_THERMAL,
Packit Service 3975d1
					   d);
Packit Service 3975d1
}