2003-12-22 19:29:21 +03:00
|
|
|
/*
|
2004-01-07 21:39:35 +03:00
|
|
|
* $HEADER$
|
2003-12-22 19:29:21 +03:00
|
|
|
*/
|
2004-01-07 10:57:00 +03:00
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
2004-02-11 01:15:55 +03:00
|
|
|
* @file:
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 02:51:07 +03:00
|
|
|
* A simple C-language object-oriented system with single inheritance
|
2004-02-10 17:04:27 +03:00
|
|
|
* and ownership-based memory management using a retain/release model.
|
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* A class consists of a struct and singly-instantiated class
|
2004-02-10 17:04:27 +03:00
|
|
|
* descriptor. The first element of the struct must be the parent
|
2004-02-10 17:34:18 +03:00
|
|
|
* class's struct. The class descriptor must be given a well-known
|
2004-02-13 02:51:07 +03:00
|
|
|
* name based upon the class struct name (if the struct is sally_t,
|
|
|
|
* the class descriptor should be sally_t_class) and must be
|
|
|
|
* statically initialized as discussed below.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
|
|
|
* (a) To define a class
|
|
|
|
*
|
|
|
|
* In a interface (.h) file, define the class. The first element
|
2004-02-13 22:20:57 +03:00
|
|
|
* should always be the parent class, for example
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-02-10 17:04:27 +03:00
|
|
|
* typedef struct sally_t sally_t;
|
|
|
|
* struct sally_t
|
|
|
|
* {
|
2004-02-13 02:51:07 +03:00
|
|
|
* parent_t parent;
|
2004-02-10 17:04:27 +03:00
|
|
|
* void *first_member;
|
|
|
|
* ...
|
|
|
|
* };
|
|
|
|
*
|
2004-06-22 01:20:39 +04:00
|
|
|
* OBJ_CLASS_DECLARATION(sally_t);
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-02-13 02:51:07 +03:00
|
|
|
* All classes must have a parent which is also class.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* In an implementation (.c) file, instantiate a class descriptor for
|
2004-02-13 22:20:57 +03:00
|
|
|
* the class like this:
|
|
|
|
* @code
|
|
|
|
* OBJ_CLASS_INSTANCE(sally_t, parent_t, sally_construct, sally_destruct);
|
|
|
|
* @endcode
|
|
|
|
* This macro actually expands to
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-06-07 19:33:53 +04:00
|
|
|
* ompi_class_t sally_t_class = {
|
2004-02-10 17:04:27 +03:00
|
|
|
* "sally_t",
|
2004-02-13 01:42:39 +03:00
|
|
|
* OBJ_CLASS(parent_t), // pointer to parent_t_class
|
2004-02-10 17:04:27 +03:00
|
|
|
* sally_construct,
|
2004-02-13 02:51:07 +03:00
|
|
|
* sally_destruct,
|
|
|
|
* 0, 0, NULL, NULL
|
2004-02-10 17:04:27 +03:00
|
|
|
* };
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-02-13 22:20:57 +03:00
|
|
|
* This variable should be declared in the interface (.h) file using
|
|
|
|
* the OBJ_CLASS_DECLARATION macro as shown above.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
|
|
|
* sally_construct, and sally_destruct are function pointers to the
|
|
|
|
* constructor and destructor for the class and are best defined as
|
2004-02-13 22:20:57 +03:00
|
|
|
* static functions in the implementation file. NULL pointers maybe
|
|
|
|
* supplied instead.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
|
|
|
* Other class methods may be added to the struct.
|
|
|
|
*
|
|
|
|
* (b) Class instantiation: dynamic
|
|
|
|
*
|
|
|
|
* To create a instance of a class (an object) use OBJ_NEW:
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-02-10 17:04:27 +03:00
|
|
|
* sally_t *sally = OBJ_NEW(sally_t);
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-02-10 17:04:27 +03:00
|
|
|
* which allocates memory of sizeof(sally_t) and runs the class's
|
2004-02-13 02:51:07 +03:00
|
|
|
* constructors.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
|
|
|
* Use OBJ_RETAIN, OBJ_RELEASE to do reference-count-based
|
|
|
|
* memory management:
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-02-10 17:04:27 +03:00
|
|
|
* OBJ_RETAIN(sally);
|
|
|
|
* OBJ_RELEASE(sally);
|
|
|
|
* OBJ_RELEASE(sally);
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-03-17 23:36:14 +03:00
|
|
|
* When the reference count reaches zero, the class's destructor, and
|
|
|
|
* those of its parents, are run and the memory is freed.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
|
|
|
* N.B. There is no explicit free/delete method for dynamic objects in
|
|
|
|
* this model.
|
|
|
|
*
|
|
|
|
* (c) Class instantiation: static
|
|
|
|
*
|
|
|
|
* For an object with static (or stack) allocation, it is only
|
|
|
|
* necessary to initialize the memory, which is done using
|
|
|
|
* OBJ_CONSTRUCT:
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-02-10 17:04:27 +03:00
|
|
|
* sally_t sally;
|
|
|
|
*
|
|
|
|
* OBJ_CONSTRUCT(&sally, sally_t);
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-02-10 17:04:27 +03:00
|
|
|
* The retain/release model is not necessary here, but before the
|
|
|
|
* object goes out of scope, OBJ_DESTRUCT should be run to release
|
|
|
|
* initialized resources:
|
2004-02-13 00:30:06 +03:00
|
|
|
* @code
|
2004-02-10 17:04:27 +03:00
|
|
|
* OBJ_DESTRUCT(&sally);
|
2004-02-13 00:30:06 +03:00
|
|
|
* @endcode
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
|
|
|
|
2004-06-07 19:33:53 +04:00
|
|
|
#ifndef OMPI_OBJECT_H
|
|
|
|
#define OMPI_OBJECT_H
|
2003-12-22 19:29:21 +03:00
|
|
|
|
2004-02-13 00:30:06 +03:00
|
|
|
#include <assert.h>
|
2003-12-22 19:29:21 +03:00
|
|
|
#include <stdlib.h>
|
2004-02-10 17:04:27 +03:00
|
|
|
|
2004-10-22 20:06:05 +04:00
|
|
|
#ifdef WIN32
|
2004-08-19 03:24:27 +04:00
|
|
|
#include <windows.h>
|
|
|
|
#else
|
2004-08-16 05:13:25 +04:00
|
|
|
#include "include/sys/atomic.h"
|
2004-08-19 03:24:27 +04:00
|
|
|
#endif
|
2004-06-16 20:27:32 +04:00
|
|
|
|
2003-12-22 19:29:21 +03:00
|
|
|
/*
|
2004-02-13 00:30:06 +03:00
|
|
|
* BEGIN_C_DECLS should be used at the beginning of your declarations,
|
|
|
|
* so that C++ compilers don't mangle their names. Use END_C_DECLS at
|
|
|
|
* the end of C declarations.
|
2003-12-22 19:29:21 +03:00
|
|
|
*/
|
2004-02-13 00:30:06 +03:00
|
|
|
#undef BEGIN_C_DECLS
|
|
|
|
#undef END_C_DECLS
|
2004-10-21 02:31:03 +04:00
|
|
|
#if defined(c_plusplus) || defined(__cplusplus)
|
2004-02-13 00:30:06 +03:00
|
|
|
# define BEGIN_C_DECLS extern "C" {
|
|
|
|
# define END_C_DECLS }
|
|
|
|
#else
|
2004-10-22 20:06:05 +04:00
|
|
|
#define BEGIN_C_DECLS /* empty */
|
|
|
|
#define END_C_DECLS /* empty */
|
2004-02-13 00:30:06 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/* macros ************************************************************/
|
2003-12-22 19:29:21 +03:00
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Return a pointer to the class descriptor associated with a
|
|
|
|
* class type.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 22:20:57 +03:00
|
|
|
* @param NAME Name of class
|
|
|
|
* @return Pointer to class descriptor
|
2004-02-13 00:30:06 +03:00
|
|
|
*/
|
2004-02-13 22:20:57 +03:00
|
|
|
#define OBJ_CLASS(NAME) (&(NAME ## _class))
|
2004-02-13 00:30:06 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Static initializer for a class descriptor
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 20:10:16 +03:00
|
|
|
* @param NAME Name of class
|
|
|
|
* @param PARENT Name of parent class
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param CONSTRUCTOR Pointer to constructor
|
|
|
|
* @param DESTRUCTOR Pointer to destructor
|
2004-02-13 22:20:57 +03:00
|
|
|
*
|
|
|
|
* Put this in NAME.c
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-02-13 22:20:57 +03:00
|
|
|
#define OBJ_CLASS_INSTANCE(NAME, PARENT, CONSTRUCTOR, DESTRUCTOR) \
|
2004-08-19 03:24:27 +04:00
|
|
|
ompi_class_t NAME ## _class = { \
|
2004-02-13 22:20:57 +03:00
|
|
|
# NAME, \
|
|
|
|
OBJ_CLASS(PARENT), \
|
2004-08-19 03:24:27 +04:00
|
|
|
(ompi_construct_t) CONSTRUCTOR, \
|
|
|
|
(ompi_destruct_t) DESTRUCTOR, \
|
2004-02-13 22:20:57 +03:00
|
|
|
0, 0, NULL, NULL \
|
2004-02-13 20:10:16 +03:00
|
|
|
}
|
2004-01-16 03:32:41 +03:00
|
|
|
|
2004-02-13 00:30:06 +03:00
|
|
|
|
2004-02-13 22:20:57 +03:00
|
|
|
/**
|
|
|
|
* Declaration for class descriptor
|
|
|
|
*
|
|
|
|
* @param NAME Name of class
|
|
|
|
*
|
|
|
|
* Put this in NAME.h
|
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
#define OBJ_CLASS_DECLARATION(NAME) \
|
2004-06-07 19:33:53 +04:00
|
|
|
extern ompi_class_t NAME ## _class
|
2004-02-13 22:20:57 +03:00
|
|
|
|
|
|
|
|
2004-02-13 00:30:06 +03:00
|
|
|
/**
|
|
|
|
* Create an object: dynamically allocate storage and run the class
|
|
|
|
* constructor.
|
|
|
|
*
|
|
|
|
* @param type Type (class) of the object
|
|
|
|
* @return Pointer to the object
|
2004-01-10 01:09:51 +03:00
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
#define OBJ_NEW(type) \
|
2004-06-07 19:33:53 +04:00
|
|
|
((type *) ompi_obj_new(sizeof(type), OBJ_CLASS(type)))
|
2004-01-10 01:09:51 +03:00
|
|
|
|
2003-12-22 19:29:21 +03:00
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
|
|
|
* Retain an object (by incrementing its reference count)
|
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param object Pointer to the object
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-02-18 09:09:58 +03:00
|
|
|
#define OBJ_RETAIN(object) \
|
|
|
|
do { \
|
|
|
|
assert(NULL != object); \
|
2004-08-19 03:24:27 +04:00
|
|
|
assert(NULL != ((ompi_object_t *) (object))->obj_class); \
|
2004-02-18 09:09:58 +03:00
|
|
|
if (object) { \
|
2004-08-19 03:24:27 +04:00
|
|
|
ompi_obj_update((ompi_object_t *) (object), 1); \
|
2004-02-18 09:09:58 +03:00
|
|
|
} \
|
2004-08-19 03:24:27 +04:00
|
|
|
assert(((ompi_object_t *) (object))->obj_reference_count >= 0); \
|
2004-02-13 00:30:06 +03:00
|
|
|
} while (0)
|
2004-02-10 17:04:27 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Release an object (by decrementing its reference count). If the
|
|
|
|
* reference count reaches zero, destruct (finalize) the object and
|
|
|
|
* free its storage.
|
|
|
|
*
|
2004-02-18 09:09:58 +03:00
|
|
|
* Note: If the object is freed, then the value of the pointer is set
|
|
|
|
* to NULL.
|
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param object Pointer to the object
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-02-18 09:09:58 +03:00
|
|
|
#define OBJ_RELEASE(object) \
|
|
|
|
do { \
|
|
|
|
assert(NULL != object); \
|
2004-08-19 03:24:27 +04:00
|
|
|
assert(NULL != ((ompi_object_t *) (object))->obj_class); \
|
|
|
|
if (0 == ompi_obj_update((ompi_object_t *) (object), -1)) { \
|
|
|
|
ompi_obj_run_destructors((ompi_object_t *) (object)); \
|
2004-02-18 09:09:58 +03:00
|
|
|
free(object); \
|
|
|
|
object = NULL; \
|
|
|
|
} \
|
2004-02-13 00:30:06 +03:00
|
|
|
} while (0)
|
2003-12-22 19:29:21 +03:00
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
|
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Construct (initialize) objects that are not dynamically allocated.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param object Pointer to the object
|
|
|
|
* @param type The object type
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-03-31 20:58:40 +04:00
|
|
|
|
2004-08-19 03:24:27 +04:00
|
|
|
#define OBJ_CONSTRUCT(object, type) \
|
2004-03-31 20:58:40 +04:00
|
|
|
OBJ_CONSTRUCT_INTERNAL(object, OBJ_CLASS(type))
|
|
|
|
|
|
|
|
#define OBJ_CONSTRUCT_INTERNAL(object, type) \
|
2004-02-13 00:30:06 +03:00
|
|
|
do { \
|
2004-03-31 20:58:40 +04:00
|
|
|
if (0 == (type)->cls_initialized) { \
|
2004-08-19 03:24:27 +04:00
|
|
|
ompi_class_initialize((type)); \
|
2004-02-13 00:30:06 +03:00
|
|
|
} \
|
|
|
|
if (object) { \
|
2004-08-19 03:24:27 +04:00
|
|
|
((ompi_object_t *) (object))->obj_class = (type); \
|
|
|
|
((ompi_object_t *) (object))->obj_reference_count = 1; \
|
|
|
|
ompi_obj_run_constructors((ompi_object_t *) (object)); \
|
2004-02-13 00:30:06 +03:00
|
|
|
} \
|
|
|
|
} while (0)
|
2004-02-10 17:04:27 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Destruct (finalize) an object that is not dynamically allocated.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param object Pointer to the object
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
#define OBJ_DESTRUCT(object) \
|
|
|
|
do { \
|
|
|
|
if (object) { \
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_obj_run_destructors((ompi_object_t *) (object)); \
|
2004-08-19 03:24:27 +04:00
|
|
|
} \
|
2004-02-13 00:30:06 +03:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
2004-02-13 22:20:57 +03:00
|
|
|
/* typedefs ***********************************************************/
|
|
|
|
|
2004-06-07 19:33:53 +04:00
|
|
|
typedef struct ompi_object_t ompi_object_t;
|
|
|
|
typedef struct ompi_class_t ompi_class_t;
|
|
|
|
typedef void (*ompi_construct_t) (ompi_object_t *);
|
|
|
|
typedef void (*ompi_destruct_t) (ompi_object_t *);
|
2004-02-13 22:20:57 +03:00
|
|
|
|
|
|
|
|
|
|
|
/* types **************************************************************/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class descriptor.
|
|
|
|
*
|
|
|
|
* There should be a single instance of this descriptor for each class
|
|
|
|
* definition.
|
|
|
|
*/
|
2004-06-07 19:33:53 +04:00
|
|
|
struct ompi_class_t {
|
2004-02-13 22:20:57 +03:00
|
|
|
const char *cls_name; /**< symbolic name for class */
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_class_t *cls_parent; /**< parent class descriptor */
|
|
|
|
ompi_construct_t cls_construct; /**< class constructor */
|
|
|
|
ompi_destruct_t cls_destruct; /**< class destructor */
|
2004-02-13 22:20:57 +03:00
|
|
|
int cls_initialized; /**< is class initialized */
|
|
|
|
int cls_depth; /**< depth of class hierarchy tree */
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_construct_t *cls_construct_array;
|
2004-02-13 22:20:57 +03:00
|
|
|
/**< array of parent class constructors */
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_destruct_t *cls_destruct_array;
|
2004-02-13 22:20:57 +03:00
|
|
|
/**< array of parent class destructors */
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base object.
|
|
|
|
*
|
|
|
|
* This is special and does not follow the pattern for other classes.
|
|
|
|
*/
|
2004-06-07 19:33:53 +04:00
|
|
|
struct ompi_object_t {
|
2004-08-19 03:24:27 +04:00
|
|
|
ompi_class_t *obj_class; /**< class descriptor */
|
|
|
|
volatile int obj_reference_count; /**< reference count */
|
2004-02-13 22:20:57 +03:00
|
|
|
};
|
|
|
|
|
2004-10-22 20:06:05 +04:00
|
|
|
BEGIN_C_DECLS
|
|
|
|
OMPI_DECLSPEC OBJ_CLASS_DECLARATION(ompi_object_t);
|
2004-02-13 22:20:57 +03:00
|
|
|
|
|
|
|
|
2004-02-13 00:30:06 +03:00
|
|
|
/* declarations *******************************************************/
|
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Lazy initialization of class descriptor.
|
|
|
|
*
|
|
|
|
* Specifically cache arrays of function pointers for the constructor
|
|
|
|
* and destructor hierarchies for this class.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param class Pointer to class descriptor
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-10-22 20:06:05 +04:00
|
|
|
OMPI_DECLSPEC void ompi_class_initialize(ompi_class_t *);
|
2004-02-10 17:04:27 +03:00
|
|
|
|
2004-02-13 00:30:06 +03:00
|
|
|
END_C_DECLS
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Run the hierarchy of class constructors for this object, in a
|
|
|
|
* parent-first order.
|
|
|
|
*
|
|
|
|
* Do not use this function directly: use OBJ_CONSTRUCT() instead.
|
|
|
|
*
|
|
|
|
* WARNING: This implementation relies on a hardwired maximum depth of
|
|
|
|
* the inheritance tree!!!
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* Hardwired for fairly shallow inheritance trees
|
|
|
|
* @param size Pointer to the object.
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
static inline void ompi_obj_run_constructors(ompi_object_t * object)
|
2004-02-13 00:30:06 +03:00
|
|
|
{
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_class_t *cls;
|
2004-02-13 00:30:06 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
assert(NULL != object);
|
2004-02-13 01:42:39 +03:00
|
|
|
assert(NULL != object->obj_class);
|
2004-02-13 00:30:06 +03:00
|
|
|
|
2004-02-13 01:42:39 +03:00
|
|
|
cls = object->obj_class;
|
2004-02-13 00:30:06 +03:00
|
|
|
for (i = cls->cls_depth - 1; i >= 0; i--) {
|
|
|
|
if (cls->cls_construct_array[i]) {
|
2004-08-19 03:24:27 +04:00
|
|
|
(cls->cls_construct_array[i]) (object);
|
2004-02-13 00:30:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-02-10 17:04:27 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Run the hierarchy of class destructors for this object, in a
|
|
|
|
* parent-last order.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* Do not use this function directly: use OBJ_DESTRUCT() instead.
|
|
|
|
*
|
|
|
|
* @param size Pointer to the object.
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
static inline void ompi_obj_run_destructors(ompi_object_t * object)
|
2004-02-13 00:30:06 +03:00
|
|
|
{
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_class_t *cls;
|
2004-02-13 00:30:06 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
assert(NULL != object);
|
2004-02-13 01:42:39 +03:00
|
|
|
assert(NULL != object->obj_class);
|
2004-02-13 00:30:06 +03:00
|
|
|
|
2004-02-13 01:42:39 +03:00
|
|
|
cls = object->obj_class;
|
2004-02-13 00:30:06 +03:00
|
|
|
for (i = 0; i < cls->cls_depth; i++) {
|
|
|
|
if (cls->cls_destruct_array[i]) {
|
2004-08-19 03:24:27 +04:00
|
|
|
(cls->cls_destruct_array[i]) (object);
|
2004-02-13 00:30:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-02-10 17:04:27 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2004-02-13 00:30:06 +03:00
|
|
|
* Create new object: dynamically allocate storage and run the class
|
|
|
|
* constructor.
|
|
|
|
*
|
|
|
|
* Do not use this function directly: use OBJ_NEW() instead.
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param size Size of the object
|
|
|
|
* @param cls Pointer to the class descriptor of this object
|
|
|
|
* @return Pointer to the object
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-08-19 03:24:27 +04:00
|
|
|
static inline ompi_object_t *ompi_obj_new(size_t size, ompi_class_t * cls)
|
2004-02-13 00:30:06 +03:00
|
|
|
{
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_object_t *object;
|
2004-02-13 00:30:06 +03:00
|
|
|
|
2004-06-07 19:33:53 +04:00
|
|
|
assert(size >= sizeof(ompi_object_t));
|
2004-02-13 00:30:06 +03:00
|
|
|
|
2004-06-07 19:33:53 +04:00
|
|
|
object = (ompi_object_t *) malloc(size);
|
2004-02-13 16:56:55 +03:00
|
|
|
if (0 == cls->cls_initialized) {
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_class_initialize(cls);
|
2004-02-13 16:56:55 +03:00
|
|
|
}
|
2004-02-13 00:30:06 +03:00
|
|
|
if (NULL != object) {
|
2004-08-19 03:24:27 +04:00
|
|
|
object->obj_class = cls;
|
2004-02-13 00:30:06 +03:00
|
|
|
object->obj_reference_count = 1;
|
2004-06-07 19:33:53 +04:00
|
|
|
ompi_obj_run_constructors(object);
|
2004-02-13 00:30:06 +03:00
|
|
|
}
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-10 17:04:27 +03:00
|
|
|
/**
|
2004-02-18 09:09:58 +03:00
|
|
|
* Atomically update the object's reference count by some increment.
|
2004-02-13 00:30:06 +03:00
|
|
|
*
|
2004-02-18 09:09:58 +03:00
|
|
|
* This function should not be used directly: it is called via the
|
|
|
|
* macros OBJ_RETAIN and OBJ_RELEASE
|
2004-02-10 17:04:27 +03:00
|
|
|
*
|
2004-02-13 00:30:06 +03:00
|
|
|
* @param object Pointer to the object
|
2004-02-18 09:09:58 +03:00
|
|
|
* @param inc Increment by which to update reference count
|
|
|
|
* @return New value of the reference count
|
2004-02-10 17:04:27 +03:00
|
|
|
*/
|
2004-06-07 19:33:53 +04:00
|
|
|
static inline int ompi_obj_update(ompi_object_t *object, int inc)
|
2004-02-13 00:30:06 +03:00
|
|
|
{
|
2004-10-22 20:06:05 +04:00
|
|
|
#ifdef WIN32
|
2004-08-19 03:24:27 +04:00
|
|
|
|
|
|
|
LONG volatile *addr;
|
|
|
|
|
|
|
|
addr = (LONG volatile *) &(object->obj_reference_count);
|
2004-10-29 02:21:20 +04:00
|
|
|
InterlockedExchangeAdd(addr, (LONG) inc);
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-08-20 03:34:46 +04:00
|
|
|
#elif OMPI_HAVE_ATOMIC
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-10-28 21:41:10 +04:00
|
|
|
ompi_atomic_add(&(object->obj_reference_count), 1 );
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-06-17 22:13:31 +04:00
|
|
|
#else
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-06-17 22:13:31 +04:00
|
|
|
object->obj_reference_count += inc;
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-06-17 22:13:31 +04:00
|
|
|
#endif
|
2004-08-19 03:24:27 +04:00
|
|
|
|
2004-10-28 21:41:10 +04:00
|
|
|
return object->obj_reference_count;
|
2004-02-13 00:30:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**********************************************************************/
|
2004-02-10 17:04:27 +03:00
|
|
|
|
2004-08-19 03:24:27 +04:00
|
|
|
#endif /* OMPI_OBJECT_H */
|