mirror of
https://gcc.gnu.org/git/gcc.git
synced 2024-11-23 02:44:18 +08:00
663 lines
17 KiB
C
663 lines
17 KiB
C
/* OpenACC Profiling Interface
|
|
|
|
Copyright (C) 2019-2024 Free Software Foundation, Inc.
|
|
|
|
Contributed by Mentor, a Siemens Business.
|
|
|
|
This file is part of the GNU Offloading and Multi Processing Library
|
|
(libgomp).
|
|
|
|
Libgomp is free software; you can redistribute it and/or modify it
|
|
under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3, or (at your option)
|
|
any later version.
|
|
|
|
Libgomp is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
more details.
|
|
|
|
Under Section 7 of GPL version 3, you are granted additional
|
|
permissions described in the GCC Runtime Library Exception, version
|
|
3.1, as published by the Free Software Foundation.
|
|
|
|
You should have received a copy of the GNU General Public License and
|
|
a copy of the GCC Runtime Library Exception along with this program;
|
|
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#define _GNU_SOURCE
|
|
#include "libgomp.h"
|
|
#include "oacc-int.h"
|
|
#include "secure_getenv.h"
|
|
#include "acc_prof.h"
|
|
#include <assert.h>
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
#ifdef PLUGIN_SUPPORT
|
|
# include <dlfcn.h>
|
|
#endif
|
|
|
|
#define STATIC_ASSERT(expr) _Static_assert (expr, "!(" #expr ")")
|
|
|
|
/* Statically assert that the layout of the common fields in the
|
|
'acc_event_info' variants matches. */
|
|
/* 'event_type' */
|
|
STATIC_ASSERT (offsetof (acc_event_info, event_type)
|
|
== offsetof (acc_event_info, data_event.event_type));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.event_type)
|
|
== offsetof (acc_event_info, launch_event.event_type));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.event_type)
|
|
== offsetof (acc_event_info, other_event.event_type));
|
|
/* 'valid_bytes' */
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.valid_bytes)
|
|
== offsetof (acc_event_info, launch_event.valid_bytes));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.valid_bytes)
|
|
== offsetof (acc_event_info, other_event.valid_bytes));
|
|
/* 'parent_construct' */
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.parent_construct)
|
|
== offsetof (acc_event_info, launch_event.parent_construct));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.parent_construct)
|
|
== offsetof (acc_event_info, other_event.parent_construct));
|
|
/* 'implicit' */
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.implicit)
|
|
== offsetof (acc_event_info, launch_event.implicit));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.implicit)
|
|
== offsetof (acc_event_info, other_event.implicit));
|
|
/* 'tool_info' */
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.tool_info)
|
|
== offsetof (acc_event_info, launch_event.tool_info));
|
|
STATIC_ASSERT (offsetof (acc_event_info, data_event.tool_info)
|
|
== offsetof (acc_event_info, other_event.tool_info));
|
|
|
|
struct goacc_prof_callback_entry
|
|
{
|
|
acc_prof_callback cb;
|
|
int ref;
|
|
bool enabled;
|
|
struct goacc_prof_callback_entry *next;
|
|
};
|
|
|
|
/* Use a separate flag to minimize run-time performance impact for the (very
|
|
common) case that profiling is not enabled.
|
|
|
|
Once enabled, we're not going to disable this anymore, anywhere. We
|
|
probably could, by adding appropriate logic to 'acc_prof_register',
|
|
'acc_prof_unregister'. */
|
|
bool goacc_prof_enabled = false;
|
|
|
|
/* Global state for registered callbacks.
|
|
'goacc_prof_callbacks_enabled[acc_ev_none]' acts as a global toggle. */
|
|
static bool goacc_prof_callbacks_enabled[acc_ev_last];
|
|
static struct goacc_prof_callback_entry *goacc_prof_callback_entries[acc_ev_last];
|
|
/* Lock used to protect access to 'goacc_prof_callbacks_enabled', and
|
|
'goacc_prof_callback_entries'. */
|
|
static gomp_mutex_t goacc_prof_lock;
|
|
|
|
void
|
|
goacc_profiling_initialize (void)
|
|
{
|
|
gomp_mutex_init (&goacc_prof_lock);
|
|
|
|
/* Initially, all callbacks for all events are enabled. */
|
|
for (int i = 0; i < acc_ev_last; ++i)
|
|
goacc_prof_callbacks_enabled[i] = true;
|
|
|
|
|
|
#ifdef PLUGIN_SUPPORT
|
|
char *acc_proflibs = secure_getenv ("ACC_PROFLIB");
|
|
while (acc_proflibs != NULL && acc_proflibs[0] != '\0')
|
|
{
|
|
char *acc_proflibs_sep = strchr (acc_proflibs, ';');
|
|
char *acc_proflib;
|
|
if (acc_proflibs_sep == acc_proflibs)
|
|
{
|
|
/* Stray ';' separator: make sure we don't 'dlopen' the main
|
|
program. */
|
|
acc_proflib = NULL;
|
|
}
|
|
else
|
|
{
|
|
if (acc_proflibs_sep != NULL)
|
|
{
|
|
/* Single out the first library. */
|
|
acc_proflib = gomp_malloc (acc_proflibs_sep - acc_proflibs + 1);
|
|
memcpy (acc_proflib, acc_proflibs,
|
|
acc_proflibs_sep - acc_proflibs);
|
|
acc_proflib[acc_proflibs_sep - acc_proflibs] = '\0';
|
|
}
|
|
else
|
|
{
|
|
/* No ';' separator, so only one library. */
|
|
acc_proflib = acc_proflibs;
|
|
}
|
|
|
|
gomp_debug (0, "%s: dlopen (\"%s\")\n", __FUNCTION__, acc_proflib);
|
|
void *dl_handle = dlopen (acc_proflib, RTLD_LAZY);
|
|
if (dl_handle != NULL)
|
|
{
|
|
typeof (&acc_register_library) a_r_l
|
|
= dlsym (dl_handle, "acc_register_library");
|
|
if (a_r_l == NULL)
|
|
goto dl_fail;
|
|
gomp_debug (0, " %s: calling %s:acc_register_library\n",
|
|
__FUNCTION__, acc_proflib);
|
|
a_r_l (acc_prof_register, acc_prof_unregister,
|
|
acc_prof_lookup);
|
|
}
|
|
else
|
|
{
|
|
dl_fail:
|
|
gomp_error ("while loading ACC_PROFLIB \"%s\": %s",
|
|
acc_proflib, dlerror ());
|
|
if (dl_handle != NULL)
|
|
{
|
|
int err = dlclose (dl_handle);
|
|
dl_handle = NULL;
|
|
if (err != 0)
|
|
goto dl_fail;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (acc_proflib != acc_proflibs)
|
|
{
|
|
free (acc_proflib);
|
|
|
|
acc_proflibs = acc_proflibs_sep + 1;
|
|
}
|
|
else
|
|
acc_proflibs = NULL;
|
|
}
|
|
#endif /* PLUGIN_SUPPORT */
|
|
}
|
|
|
|
void
|
|
acc_prof_register (acc_event_t ev, acc_prof_callback cb, acc_register_t reg)
|
|
{
|
|
gomp_debug (0, "%s: ev=%d, cb=%p, reg=%d\n",
|
|
__FUNCTION__, (int) ev, (void *) cb, (int) reg);
|
|
|
|
|
|
/* For any events to be dispatched, the user first has to register a
|
|
callback, which makes this here a good place for enabling the whole
|
|
machinery. */
|
|
if (!GOACC_PROF_ENABLED)
|
|
__atomic_store_n (&goacc_prof_enabled, true, MEMMODEL_RELEASE);
|
|
|
|
|
|
enum
|
|
{
|
|
EVENT_KIND_BOGUS,
|
|
EVENT_KIND_NORMAL,
|
|
/* As end events invoke callbacks in the reverse order, we register these
|
|
in the reverse order here. */
|
|
EVENT_KIND_END,
|
|
} event_kind = EVENT_KIND_BOGUS;
|
|
switch (ev)
|
|
{
|
|
case acc_ev_none:
|
|
case acc_ev_device_init_start:
|
|
case acc_ev_device_shutdown_start:
|
|
case acc_ev_runtime_shutdown:
|
|
case acc_ev_create:
|
|
case acc_ev_delete:
|
|
case acc_ev_alloc:
|
|
case acc_ev_free:
|
|
case acc_ev_enter_data_start:
|
|
case acc_ev_exit_data_start:
|
|
case acc_ev_update_start:
|
|
case acc_ev_compute_construct_start:
|
|
case acc_ev_enqueue_launch_start:
|
|
case acc_ev_enqueue_upload_start:
|
|
case acc_ev_enqueue_download_start:
|
|
case acc_ev_wait_start:
|
|
event_kind = EVENT_KIND_NORMAL;
|
|
break;
|
|
case acc_ev_device_init_end:
|
|
case acc_ev_device_shutdown_end:
|
|
case acc_ev_enter_data_end:
|
|
case acc_ev_exit_data_end:
|
|
case acc_ev_update_end:
|
|
case acc_ev_compute_construct_end:
|
|
case acc_ev_enqueue_launch_end:
|
|
case acc_ev_enqueue_upload_end:
|
|
case acc_ev_enqueue_download_end:
|
|
case acc_ev_wait_end:
|
|
event_kind = EVENT_KIND_END;
|
|
break;
|
|
case acc_ev_last:
|
|
break;
|
|
}
|
|
if (event_kind == EVENT_KIND_BOGUS)
|
|
{
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring request for bogus 'acc_event_t'\n");
|
|
return;
|
|
}
|
|
|
|
bool bogus = true;
|
|
switch (reg)
|
|
{
|
|
case acc_reg:
|
|
case acc_toggle:
|
|
case acc_toggle_per_thread:
|
|
bogus = false;
|
|
break;
|
|
}
|
|
if (bogus)
|
|
{
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring request with bogus 'acc_register_t'\n");
|
|
return;
|
|
}
|
|
|
|
/* Special cases. */
|
|
if (reg == acc_toggle)
|
|
{
|
|
if (cb == NULL)
|
|
{
|
|
gomp_debug (0, " globally enabling callbacks\n");
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
/* For 'acc_ev_none', this acts as a global toggle. */
|
|
goacc_prof_callbacks_enabled[ev] = true;
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
return;
|
|
}
|
|
else if (ev == acc_ev_none && cb != NULL)
|
|
{
|
|
gomp_debug (0, " ignoring request\n");
|
|
return;
|
|
}
|
|
}
|
|
else if (reg == acc_toggle_per_thread)
|
|
{
|
|
if (ev == acc_ev_none && cb == NULL)
|
|
{
|
|
gomp_debug (0, " thread: enabling callbacks\n");
|
|
goacc_lazy_initialize ();
|
|
struct goacc_thread *thr = goacc_thread ();
|
|
thr->prof_callbacks_enabled = true;
|
|
return;
|
|
}
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring bogus request\n");
|
|
return;
|
|
}
|
|
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
|
|
struct goacc_prof_callback_entry *it, *it_p;
|
|
it = goacc_prof_callback_entries[ev];
|
|
it_p = NULL;
|
|
while (it)
|
|
{
|
|
if (it->cb == cb)
|
|
break;
|
|
it_p = it;
|
|
it = it->next;
|
|
}
|
|
|
|
switch (reg)
|
|
{
|
|
case acc_reg:
|
|
/* If we already have this callback registered, just increment its
|
|
reference count. */
|
|
if (it != NULL)
|
|
{
|
|
it->ref++;
|
|
gomp_debug (0, " already registered;"
|
|
" incrementing reference count to: %d\n", it->ref);
|
|
}
|
|
else
|
|
{
|
|
struct goacc_prof_callback_entry *e
|
|
= gomp_malloc (sizeof (struct goacc_prof_callback_entry));
|
|
e->cb = cb;
|
|
e->ref = 1;
|
|
e->enabled = true;
|
|
bool prepend = (event_kind == EVENT_KIND_END);
|
|
/* If we don't have any callback registered yet, also use the
|
|
'prepend' code path. */
|
|
if (it_p == NULL)
|
|
prepend = true;
|
|
if (prepend)
|
|
{
|
|
gomp_debug (0, " prepending\n");
|
|
e->next = goacc_prof_callback_entries[ev];
|
|
goacc_prof_callback_entries[ev] = e;
|
|
}
|
|
else
|
|
{
|
|
gomp_debug (0, " appending\n");
|
|
e->next = NULL;
|
|
it_p->next = e;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case acc_toggle:
|
|
if (it == NULL)
|
|
{
|
|
gomp_debug (0, " ignoring request: is not registered\n");
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
gomp_debug (0, " enabling\n");
|
|
it->enabled = true;
|
|
}
|
|
break;
|
|
|
|
case acc_toggle_per_thread:
|
|
__builtin_unreachable ();
|
|
}
|
|
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
}
|
|
|
|
void
|
|
acc_prof_unregister (acc_event_t ev, acc_prof_callback cb, acc_register_t reg)
|
|
{
|
|
gomp_debug (0, "%s: ev=%d, cb=%p, reg=%d\n",
|
|
__FUNCTION__, (int) ev, (void *) cb, (int) reg);
|
|
|
|
/* If profiling is not enabled, there cannot be anything to unregister. */
|
|
if (!GOACC_PROF_ENABLED)
|
|
return;
|
|
|
|
if (ev < acc_ev_none
|
|
|| ev >= acc_ev_last)
|
|
{
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring request for bogus 'acc_event_t'\n");
|
|
return;
|
|
}
|
|
|
|
bool bogus = true;
|
|
switch (reg)
|
|
{
|
|
case acc_reg:
|
|
case acc_toggle:
|
|
case acc_toggle_per_thread:
|
|
bogus = false;
|
|
break;
|
|
}
|
|
if (bogus)
|
|
{
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring request with bogus 'acc_register_t'\n");
|
|
return;
|
|
}
|
|
|
|
/* Special cases. */
|
|
if (reg == acc_toggle)
|
|
{
|
|
if (cb == NULL)
|
|
{
|
|
gomp_debug (0, " globally disabling callbacks\n");
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
/* For 'acc_ev_none', this acts as a global toggle. */
|
|
goacc_prof_callbacks_enabled[ev] = false;
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
return;
|
|
}
|
|
else if (ev == acc_ev_none && cb != NULL)
|
|
{
|
|
gomp_debug (0, " ignoring request\n");
|
|
return;
|
|
}
|
|
}
|
|
else if (reg == acc_toggle_per_thread)
|
|
{
|
|
if (ev == acc_ev_none && cb == NULL)
|
|
{
|
|
gomp_debug (0, " thread: disabling callbacks\n");
|
|
goacc_lazy_initialize ();
|
|
struct goacc_thread *thr = goacc_thread ();
|
|
thr->prof_callbacks_enabled = false;
|
|
return;
|
|
}
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring bogus request\n");
|
|
return;
|
|
}
|
|
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
|
|
struct goacc_prof_callback_entry *it, *it_p;
|
|
it = goacc_prof_callback_entries[ev];
|
|
it_p = NULL;
|
|
while (it)
|
|
{
|
|
if (it->cb == cb)
|
|
break;
|
|
it_p = it;
|
|
it = it->next;
|
|
}
|
|
|
|
switch (reg)
|
|
{
|
|
case acc_reg:
|
|
if (it == NULL)
|
|
{
|
|
/* Silently ignore. */
|
|
gomp_debug (0, " ignoring bogus request: is not registered\n");
|
|
break;
|
|
}
|
|
it->ref--;
|
|
gomp_debug (0, " decrementing reference count to: %d\n", it->ref);
|
|
if (it->ref == 0)
|
|
{
|
|
if (it_p == NULL)
|
|
goacc_prof_callback_entries[ev] = it->next;
|
|
else
|
|
it_p->next = it->next;
|
|
free (it);
|
|
}
|
|
break;
|
|
|
|
case acc_toggle:
|
|
if (it == NULL)
|
|
{
|
|
gomp_debug (0, " ignoring request: is not registered\n");
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
gomp_debug (0, " disabling\n");
|
|
it->enabled = false;
|
|
}
|
|
break;
|
|
|
|
case acc_toggle_per_thread:
|
|
__builtin_unreachable ();
|
|
}
|
|
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
}
|
|
|
|
acc_query_fn
|
|
acc_prof_lookup (const char *name)
|
|
{
|
|
gomp_debug (0, "%s (%s)\n",
|
|
__FUNCTION__, name ?: "NULL");
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
acc_register_library (acc_prof_reg reg, acc_prof_reg unreg,
|
|
acc_prof_lookup_func lookup)
|
|
{
|
|
gomp_fatal ("TODO");
|
|
}
|
|
|
|
/* Prepare to dispatch events? */
|
|
|
|
bool
|
|
_goacc_profiling_dispatch_p (bool check_not_nested_p)
|
|
{
|
|
gomp_debug (0, "%s\n", __FUNCTION__);
|
|
|
|
bool ret;
|
|
|
|
struct goacc_thread *thr = goacc_thread ();
|
|
if (__builtin_expect (thr == NULL, false))
|
|
{
|
|
/* If we don't have any per-thread state yet, that means that per-thread
|
|
callback dispatch has not been explicitly disabled (which only a call
|
|
to 'acc_prof_unregister' with 'acc_toggle_per_thread' would do, and
|
|
that would have allocated per-thread state via
|
|
'goacc_lazy_initialize'); initially, all callbacks for all events are
|
|
enabled. */
|
|
gomp_debug (0, " %s: don't have any per-thread state yet\n", __FUNCTION__);
|
|
}
|
|
else
|
|
{
|
|
if (check_not_nested_p)
|
|
{
|
|
/* No nesting. */
|
|
assert (thr->prof_info == NULL);
|
|
assert (thr->api_info == NULL);
|
|
}
|
|
|
|
if (__builtin_expect (!thr->prof_callbacks_enabled, true))
|
|
{
|
|
gomp_debug (0, " %s: disabled for this thread\n", __FUNCTION__);
|
|
ret = false;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
|
|
/* 'goacc_prof_callbacks_enabled[acc_ev_none]' acts as a global toggle. */
|
|
if (__builtin_expect (!goacc_prof_callbacks_enabled[acc_ev_none], true))
|
|
{
|
|
gomp_debug (0, " %s: disabled globally\n", __FUNCTION__);
|
|
ret = false;
|
|
goto out_unlock;
|
|
}
|
|
else
|
|
ret = true;
|
|
|
|
out_unlock:
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
/* Set up to dispatch events? */
|
|
|
|
bool
|
|
_goacc_profiling_setup_p (struct goacc_thread *thr,
|
|
acc_prof_info *prof_info, acc_api_info *api_info)
|
|
{
|
|
gomp_debug (0, "%s (%p)\n", __FUNCTION__, thr);
|
|
|
|
/* If we don't have any per-thread state yet, we can't register 'prof_info'
|
|
and 'api_info'. */
|
|
if (__builtin_expect (thr == NULL, false))
|
|
{
|
|
gomp_debug (0, "Can't dispatch OpenACC Profiling Interface events for"
|
|
" the current call, construct, or directive\n");
|
|
return false;
|
|
}
|
|
|
|
if (thr->prof_info != NULL)
|
|
{
|
|
/* Profiling has already been set up for an outer construct. In this
|
|
case, we continue to use the existing information, and thus return
|
|
'false' here.
|
|
|
|
This can happen, for example, for an 'enter data' directive, which
|
|
sets up profiling, then calls into 'acc_copyin', which should not
|
|
again set up profiling, should not overwrite the existing
|
|
information. */
|
|
return false;
|
|
}
|
|
|
|
thr->prof_info = prof_info;
|
|
thr->api_info = api_info;
|
|
|
|
/* Fill in some defaults. */
|
|
|
|
prof_info->event_type = -1; /* Must be set later. */
|
|
prof_info->valid_bytes = _ACC_PROF_INFO_VALID_BYTES;
|
|
prof_info->version = _ACC_PROF_INFO_VERSION;
|
|
if (thr->dev)
|
|
{
|
|
prof_info->device_type = acc_device_type (thr->dev->type);
|
|
prof_info->device_number = thr->dev->target_id;
|
|
}
|
|
else
|
|
{
|
|
prof_info->device_type = -1;
|
|
prof_info->device_number = -1;
|
|
}
|
|
prof_info->thread_id = -1;
|
|
prof_info->async = acc_async_sync;
|
|
prof_info->async_queue = prof_info->async;
|
|
prof_info->src_file = NULL;
|
|
prof_info->func_name = NULL;
|
|
prof_info->line_no = -1;
|
|
prof_info->end_line_no = -1;
|
|
prof_info->func_line_no = -1;
|
|
prof_info->func_end_line_no = -1;
|
|
|
|
api_info->device_api = acc_device_api_none;
|
|
api_info->valid_bytes = _ACC_API_INFO_VALID_BYTES;
|
|
api_info->device_type = prof_info->device_type;
|
|
api_info->vendor = -1;
|
|
api_info->device_handle = NULL;
|
|
api_info->context_handle = NULL;
|
|
api_info->async_handle = NULL;
|
|
|
|
return true;
|
|
}
|
|
|
|
/* Dispatch events.
|
|
|
|
This must only be called if 'GOACC_PROFILING_DISPATCH_P' or
|
|
'GOACC_PROFILING_SETUP_P' returned a true result. */
|
|
|
|
void
|
|
goacc_profiling_dispatch (acc_prof_info *prof_info, acc_event_info *event_info,
|
|
acc_api_info *apt_info)
|
|
{
|
|
acc_event_t event_type = event_info->event_type;
|
|
gomp_debug (0, "%s: event_type=%d\n", __FUNCTION__, (int) event_type);
|
|
assert (event_type > acc_ev_none
|
|
&& event_type < acc_ev_last);
|
|
|
|
gomp_mutex_lock (&goacc_prof_lock);
|
|
|
|
if (!goacc_prof_callbacks_enabled[event_type])
|
|
{
|
|
gomp_debug (0, " disabled for this event type\n");
|
|
|
|
goto out_unlock;
|
|
}
|
|
|
|
for (struct goacc_prof_callback_entry *e
|
|
= goacc_prof_callback_entries[event_type];
|
|
e != NULL;
|
|
e = e->next)
|
|
{
|
|
if (!e->enabled)
|
|
{
|
|
gomp_debug (0, " disabled for callback %p\n", e->cb);
|
|
continue;
|
|
}
|
|
|
|
gomp_debug (0, " calling callback %p\n", e->cb);
|
|
e->cb (prof_info, event_info, apt_info);
|
|
}
|
|
|
|
out_unlock:
|
|
gomp_mutex_unlock (&goacc_prof_lock);
|
|
}
|