eccodes/src/action.c

221 lines
5.2 KiB
C
Raw Normal View History

2013-03-25 12:04:10 +00:00
/*
2020-01-28 14:32:34 +00:00
* (C) Copyright 2005- ECMWF.
2013-03-25 12:04:10 +00:00
*
* This software is licensed under the terms of the Apache Licence Version 2.0
* which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
*
* In applying this licence, ECMWF does not waive the privileges and immunities granted to it by
* virtue of its status as an intergovernmental organisation nor does it submit to any jurisdiction.
*/
/***************************************************************************
* Jean Baptiste Filippi - 01.11.2005 *
* *
***************************************************************************/
#include "grib_api_internal.h"
#if GRIB_PTHREADS
2020-01-22 13:10:59 +00:00
static pthread_once_t once = PTHREAD_ONCE_INIT;
2013-03-25 12:04:10 +00:00
static pthread_mutex_t mutex1 = PTHREAD_MUTEX_INITIALIZER;
2020-01-22 13:10:59 +00:00
static void init_mutex()
{
2013-03-25 12:04:10 +00:00
pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
2020-01-22 13:10:59 +00:00
pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
pthread_mutex_init(&mutex1, &attr);
2013-03-25 12:04:10 +00:00
pthread_mutexattr_destroy(&attr);
}
2015-12-30 14:39:02 +00:00
#elif GRIB_OMP_THREADS
static int once = 0;
static omp_nest_lock_t mutex1;
static void init_mutex()
{
GRIB_OMP_CRITICAL(lock_action_c)
{
2020-01-22 13:10:59 +00:00
if (once == 0) {
2015-12-30 14:39:02 +00:00
omp_init_nest_lock(&mutex1);
once = 1;
}
}
}
2013-03-25 12:04:10 +00:00
#endif
2020-01-22 13:10:59 +00:00
static void init(grib_action_class* c)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
if (!c)
return;
2020-01-22 13:10:59 +00:00
GRIB_MUTEX_INIT_ONCE(&once, &init_mutex);
2016-02-09 18:00:49 +00:00
GRIB_MUTEX_LOCK(&mutex1);
2020-01-22 13:10:59 +00:00
if (!c->inited) {
if (c->super) {
2020-01-22 13:10:59 +00:00
init(*(c->super));
}
2013-03-25 12:04:10 +00:00
c->init_class(c);
c->inited = 1;
}
2016-02-09 18:00:49 +00:00
GRIB_MUTEX_UNLOCK(&mutex1);
2013-03-25 12:04:10 +00:00
}
2019-07-11 16:53:16 +00:00
#if 0
/* A non-recursive version */
static void init(grib_action_class *c)
{
if (!c) return;
GRIB_MUTEX_INIT_ONCE(&once,&init_mutex);
GRIB_MUTEX_LOCK(&mutex1);
if(!c->inited)
{
if(c->super) {
grib_action_class *g = *(c->super);
if (g && !g->inited) {
Assert(g->super == NULL);
g->init_class(g);
g->inited = 1;
}
}
c->init_class(c);
c->inited = 1;
}
GRIB_MUTEX_UNLOCK(&mutex1);
}
#endif
2013-03-25 12:04:10 +00:00
void grib_dump(grib_action* a, FILE* f, int l)
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->dump) {
2013-03-25 12:04:10 +00:00
c->dump(a, f, l);
return;
}
c = c->super ? *(c->super) : NULL;
}
Assert(0);
}
2020-01-22 13:10:59 +00:00
void grib_xref(grib_action* a, FILE* f, const char* path)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->xref) {
c->xref(a, f, path);
2013-03-25 12:04:10 +00:00
return;
}
c = c->super ? *(c->super) : NULL;
}
2020-01-22 13:10:59 +00:00
printf("xref not implemented for %s\n", a->cclass->name);
2013-03-25 12:04:10 +00:00
Assert(0);
}
2020-01-22 13:10:59 +00:00
void grib_action_delete(grib_context* context, grib_action* a)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->destroy)
2013-03-25 12:04:10 +00:00
c->destroy(context, a);
c = c->super ? *(c->super) : NULL;
}
grib_context_free_persistent(context, a);
}
2020-01-22 13:10:59 +00:00
int grib_create_accessor(grib_section* p, grib_action* a, grib_loader* h)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->create_accessor) {
2016-10-03 15:39:45 +00:00
int ret;
/* ECC-604: Do not lock excessively */
/*GRIB_MUTEX_INIT_ONCE(&once,&init_mutex);*/
/*GRIB_MUTEX_LOCK(&mutex1);*/
2020-01-22 13:10:59 +00:00
ret = c->create_accessor(p, a, h);
/*GRIB_MUTEX_UNLOCK(&mutex1);*/
2013-03-25 12:04:10 +00:00
return ret;
2016-10-03 15:39:45 +00:00
}
2013-03-25 12:04:10 +00:00
c = c->super ? *(c->super) : NULL;
}
2020-01-22 13:10:59 +00:00
fprintf(stderr, "Cannot create accessor %s %s\n", a->name, a->cclass->name);
2013-03-25 12:04:10 +00:00
Assert(0);
return 0;
}
2020-01-22 13:10:59 +00:00
int grib_action_notify_change(grib_action* a, grib_accessor* observer, grib_accessor* observed)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2016-02-09 18:00:49 +00:00
/*GRIB_MUTEX_INIT_ONCE(&once,&init_mutex);*/
/*GRIB_MUTEX_LOCK(&mutex1);*/
2016-02-09 18:00:49 +00:00
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->notify_change) {
int result = c->notify_change(a, observer, observed);
/*GRIB_MUTEX_UNLOCK(&mutex1);*/
2016-02-09 18:00:49 +00:00
return result;
}
2013-03-25 12:04:10 +00:00
c = c->super ? *(c->super) : NULL;
}
/*GRIB_MUTEX_UNLOCK(&mutex1);*/
2013-03-25 12:04:10 +00:00
Assert(0);
return 0;
}
2020-01-22 13:10:59 +00:00
grib_action* grib_action_reparse(grib_action* a, grib_accessor* acc, int* doit)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->reparse)
return c->reparse(a, acc, doit);
2013-03-25 12:04:10 +00:00
c = c->super ? *(c->super) : NULL;
}
Assert(0);
return 0;
}
2020-01-22 13:10:59 +00:00
int grib_action_execute(grib_action* a, grib_handle* h)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_action_class* c = a->cclass;
2013-03-25 12:04:10 +00:00
init(c);
2020-01-22 13:10:59 +00:00
while (c) {
if (c->execute)
return c->execute(a, h);
2013-03-25 12:04:10 +00:00
c = c->super ? *(c->super) : NULL;
}
Assert(0);
return 0;
}
2020-01-22 13:10:59 +00:00
void grib_dump_action_branch(FILE* out, grib_action* a, int decay)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
while (a) {
grib_dump(a, out, decay);
a = a->next;
2013-03-25 12:04:10 +00:00
}
}
2020-01-22 13:10:59 +00:00
void grib_dump_action_tree(grib_context* ctx, FILE* out)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
grib_dump_action_branch(out, ctx->grib_reader->first->root, 0);
2013-03-25 12:04:10 +00:00
}
2020-01-22 13:10:59 +00:00
void grib_xref_action_branch(FILE* out, grib_action* a, const char* path)
2013-03-25 12:04:10 +00:00
{
2020-01-22 13:10:59 +00:00
while (a) {
grib_xref(a, out, path);
a = a->next;
2013-03-25 12:04:10 +00:00
}
}