2003-12-22 16:29:21 +00:00
|
|
|
/*
|
2005-11-05 19:57:48 +00:00
|
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
* University Research and Technology
|
|
|
|
* Corporation. All rights reserved.
|
|
|
|
* Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
|
|
* of Tennessee Research Foundation. All rights
|
|
|
|
* reserved.
|
2004-11-28 20:09:25 +00:00
|
|
|
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
|
|
* University of Stuttgart. All rights reserved.
|
2005-03-24 12:43:37 +00:00
|
|
|
* Copyright (c) 2004-2005 The Regents of the University of California.
|
|
|
|
* All rights reserved.
|
2004-11-22 01:38:40 +00:00
|
|
|
* $COPYRIGHT$
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
2004-01-07 07:57:00 +00:00
|
|
|
* $HEADER$
|
2003-12-22 16:29:21 +00:00
|
|
|
*/
|
2004-01-07 07:57:00 +00:00
|
|
|
|
2004-02-10 14:04:27 +00:00
|
|
|
/**
|
2004-02-12 21:30:06 +00:00
|
|
|
* @file
|
|
|
|
*
|
2005-07-04 00:16:24 +00:00
|
|
|
* Implementation of opal_object_t, the base opal foundation class
|
2004-02-10 14:04:27 +00:00
|
|
|
*/
|
2003-12-22 16:29:21 +00:00
|
|
|
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "opal_config.h"
|
2004-08-18 23:24:27 +00:00
|
|
|
|
2004-02-12 21:30:06 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "opal/sys/atomic.h"
|
2005-07-03 16:06:07 +00:00
|
|
|
#include "opal/class/opal_object.h"
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "opal/constants.h"
|
2003-12-22 16:29:21 +00:00
|
|
|
|
2004-02-12 21:30:06 +00:00
|
|
|
/*
|
|
|
|
* Instantiation of class descriptor for the base class. This is
|
|
|
|
* special, since be mark it as already initialized, with no parent
|
|
|
|
* and no constructor or desctructor
|
|
|
|
*/
|
2005-07-03 16:06:07 +00:00
|
|
|
opal_class_t opal_object_t_class = {
|
|
|
|
"opal_object_t", /* name */
|
2004-02-12 21:30:06 +00:00
|
|
|
NULL, /* parent class */
|
|
|
|
NULL, /* constructor */
|
|
|
|
NULL, /* destructor */
|
|
|
|
1, /* initialized -- this class is preinitialized */
|
|
|
|
0, /* class hierarchy depth */
|
|
|
|
NULL, /* array of constructors */
|
|
|
|
NULL /* array of destructors */
|
|
|
|
};
|
2003-12-22 16:29:21 +00:00
|
|
|
|
2005-03-24 22:13:05 +00:00
|
|
|
/*
|
|
|
|
* Local variables
|
|
|
|
*/
|
2005-07-03 21:38:51 +00:00
|
|
|
static opal_atomic_lock_t class_lock = { { OPAL_ATOMIC_UNLOCKED } };
|
2005-03-24 22:13:05 +00:00
|
|
|
static void** classes = NULL;
|
|
|
|
static int num_classes = 0;
|
|
|
|
static int max_classes = 0;
|
|
|
|
static const int increment = 10;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Local functions
|
|
|
|
*/
|
2005-07-03 16:06:07 +00:00
|
|
|
static void save_class(opal_class_t *cls);
|
2005-03-24 22:13:05 +00:00
|
|
|
static void expand_array(void);
|
|
|
|
|
2004-02-10 14:04:27 +00:00
|
|
|
|
2004-02-12 21:30:06 +00:00
|
|
|
/*
|
|
|
|
* Lazy initialization of class descriptor.
|
|
|
|
*/
|
2005-07-03 16:06:07 +00:00
|
|
|
void opal_class_initialize(opal_class_t *cls)
|
2003-12-22 16:29:21 +00:00
|
|
|
{
|
2005-07-03 16:06:07 +00:00
|
|
|
opal_class_t *c;
|
2006-02-10 21:04:26 +00:00
|
|
|
opal_construct_t* cls_construct_array;
|
|
|
|
opal_destruct_t* cls_destruct_array;
|
2004-02-12 21:30:06 +00:00
|
|
|
int i;
|
2003-12-22 16:29:21 +00:00
|
|
|
|
2004-10-18 16:55:20 +00:00
|
|
|
assert(cls);
|
2005-03-24 22:13:05 +00:00
|
|
|
|
|
|
|
/* Check to see if any other thread got in here and initialized
|
|
|
|
this class before we got a chance to */
|
|
|
|
|
|
|
|
if (1 == cls->cls_initialized) {
|
|
|
|
return;
|
|
|
|
}
|
2005-07-03 21:38:51 +00:00
|
|
|
opal_atomic_lock(&class_lock);
|
2005-03-24 22:13:05 +00:00
|
|
|
|
|
|
|
/* If another thread initializing this same class came in at
|
|
|
|
roughly the same time, it may have gotten the lock and
|
|
|
|
initialized. So check again. */
|
|
|
|
|
|
|
|
if (1 == cls->cls_initialized) {
|
2005-07-03 21:38:51 +00:00
|
|
|
opal_atomic_unlock(&class_lock);
|
2005-03-24 22:13:05 +00:00
|
|
|
return;
|
|
|
|
}
|
2004-02-10 14:04:27 +00:00
|
|
|
|
2004-02-12 21:30:06 +00:00
|
|
|
/*
|
|
|
|
* First calculate depth of class hierarchy
|
|
|
|
*/
|
|
|
|
|
2004-10-18 16:55:20 +00:00
|
|
|
cls->cls_depth = 0;
|
|
|
|
for (c = cls; c; c = c->cls_parent) {
|
|
|
|
cls->cls_depth += 1;
|
2004-02-12 21:30:06 +00:00
|
|
|
}
|
2006-02-10 21:04:26 +00:00
|
|
|
|
2004-02-12 21:30:06 +00:00
|
|
|
/*
|
|
|
|
* Allocate arrays for hierarchy of constructors and destructors
|
|
|
|
*/
|
2004-02-10 14:04:27 +00:00
|
|
|
|
2005-03-24 22:13:05 +00:00
|
|
|
cls->cls_construct_array =
|
2006-02-10 21:04:26 +00:00
|
|
|
(void (**)(opal_object_t*))malloc((cls->cls_depth + 1)*
|
|
|
|
sizeof(opal_construct_t) * 2 );
|
2004-10-18 16:55:20 +00:00
|
|
|
if (NULL == cls->cls_construct_array) {
|
2004-02-12 21:30:06 +00:00
|
|
|
perror("Out of memory");
|
2005-03-24 22:13:05 +00:00
|
|
|
exit(-1);
|
2004-02-12 21:30:06 +00:00
|
|
|
}
|
2006-05-18 16:21:29 +00:00
|
|
|
cls->cls_destruct_array = cls->cls_construct_array + cls->cls_depth + 1;
|
2006-02-10 21:04:26 +00:00
|
|
|
cls_construct_array = cls->cls_construct_array;
|
|
|
|
cls_destruct_array = cls->cls_destruct_array;
|
2004-02-12 21:30:06 +00:00
|
|
|
|
2004-10-18 16:55:20 +00:00
|
|
|
c = cls;
|
|
|
|
for (i = 0; i < cls->cls_depth; i++) {
|
2006-02-10 21:04:26 +00:00
|
|
|
if( NULL != c->cls_construct ) {
|
|
|
|
*cls_construct_array = c->cls_construct;
|
|
|
|
cls_construct_array++;
|
|
|
|
}
|
|
|
|
if( NULL != c->cls_destruct ) {
|
|
|
|
*cls_destruct_array = c->cls_destruct;
|
|
|
|
cls_destruct_array++;
|
|
|
|
}
|
2004-02-12 21:30:06 +00:00
|
|
|
c = c->cls_parent;
|
2006-02-10 21:04:26 +00:00
|
|
|
}
|
|
|
|
*cls_construct_array = NULL; /* end marker for the constructors */
|
|
|
|
*cls_destruct_array = NULL; /* end marker for the destructors */
|
|
|
|
/* Now we have to invert the constructors */
|
|
|
|
for( i = 0, --cls_construct_array;
|
|
|
|
cls_construct_array > (cls->cls_construct_array + i);
|
|
|
|
i++, cls_construct_array-- ) {
|
|
|
|
opal_construct_t temp_construct = *cls_construct_array;
|
|
|
|
*cls_construct_array = cls->cls_construct_array[i];
|
|
|
|
cls->cls_construct_array[i] = temp_construct;
|
2004-02-12 21:30:06 +00:00
|
|
|
}
|
2004-02-10 14:04:27 +00:00
|
|
|
|
2004-10-18 16:55:20 +00:00
|
|
|
cls->cls_initialized = 1;
|
2005-03-24 22:13:05 +00:00
|
|
|
save_class(cls);
|
|
|
|
|
|
|
|
/* All done */
|
|
|
|
|
2005-07-03 21:38:51 +00:00
|
|
|
opal_atomic_unlock(&class_lock);
|
2004-02-12 21:30:06 +00:00
|
|
|
}
|
2005-03-24 22:13:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note that this is finalize for *all* classes.
|
|
|
|
*/
|
2005-07-03 16:06:07 +00:00
|
|
|
int opal_class_finalize(void)
|
2005-03-24 22:13:05 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (NULL != classes) {
|
|
|
|
for (i = 0; i < num_classes; ++i) {
|
|
|
|
if (NULL != classes[i]) {
|
|
|
|
free(classes[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(classes);
|
|
|
|
classes = NULL;
|
|
|
|
num_classes = 0;
|
|
|
|
max_classes = 0;
|
|
|
|
}
|
|
|
|
|
2006-02-12 01:33:29 +00:00
|
|
|
return OPAL_SUCCESS;
|
2005-03-24 22:13:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-03 16:06:07 +00:00
|
|
|
static void save_class(opal_class_t *cls)
|
2005-03-24 22:13:05 +00:00
|
|
|
{
|
|
|
|
if (num_classes >= max_classes) {
|
|
|
|
expand_array();
|
|
|
|
}
|
|
|
|
|
|
|
|
classes[num_classes] = cls->cls_construct_array;
|
|
|
|
++num_classes;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void expand_array(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
max_classes += increment;
|
2006-08-22 20:07:42 +00:00
|
|
|
classes = (void**)realloc(classes, sizeof(void *) * max_classes);
|
2005-03-24 22:13:05 +00:00
|
|
|
if (NULL == classes) {
|
|
|
|
perror("class malloc failed");
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
for (i = num_classes; i < max_classes; ++i) {
|
|
|
|
classes[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|