1
1
openmpi/opal/class/opal_list.h
Gleb Natapov 140dce7614 Fix ABA problem in atomic_lifo code. This is temporary solution for now. We
are looking for a better one.

This commit was SVN r16091.
2007-09-11 15:40:30 +00:00

807 строки
26 KiB
C++

/*
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
* University Research and Technology
* Corporation. All rights reserved.
* Copyright (c) 2004-2006 The University of Tennessee and The University
* of Tennessee Research Foundation. All rights
* reserved.
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
* University of Stuttgart. All rights reserved.
* Copyright (c) 2004-2005 The Regents of the University of California.
* All rights reserved.
* Copyright (c) 2007 Voltaire All rights reserved.
* $COPYRIGHT$
*
* Additional copyrights may follow
*
* $HEADER$
*/
/**
* @file
*
* The opal_list_t interface is used to provide a generic
* doubly-linked list container for Open MPI. It was inspired by (but
* is slightly different than) the Stantard Template Library (STL)
* std::list class. One notable difference from std::list is that
* when an opal_list_t is destroyed, all of the opal_list_item_t
* objects that it contains are orphaned -- they are \em not
* destroyed.
*
* The general idea is that opal_list_item_t objects can be put on an
* opal_list_t. Hence, you create a new type that derives from
* opal_list_item_t; this new type can then be used with opal_list_t
* containers.
*
* NOTE: opal_list_item_t instances can only be on \em one list at a
* time. Specifically, if you add an opal_list_item_t to one list,
* and then add it to another list (without first removing it from the
* first list), you will effectively be hosing the first list. You
* have been warned.
*
* If OMPI_ENABLE_DEBUG is true, a bunch of checks occur, including
* some spot checks for a debugging reference count in an attempt to
* ensure that an opal_list_item_t is only one *one* list at a time.
* Given the highly concurrent nature of this class, these spot checks
* cannot guarantee that an item is only one list at a time.
* Specifically, since it is a desirable attribute of this class to
* not use locks for normal operations, it is possible that two
* threads may [erroneously] modify an opal_list_item_t concurrently.
*
* The only way to guarantee that a debugging reference count is valid
* for the duration of an operation is to lock the item_t during the
* operation. But this fundamentally changes the desirable attribute
* of this class (i.e., no locks). So all we can do is spot-check the
* reference count in a bunch of places and check that it is still the
* value that we think it should be. But this doesn't mean that you
* can run into "unlucky" cases where two threads are concurrently
* modifying an item_t, but all the spot checks still return the
* "right" values. All we can do is hope that we have enough spot
* checks to statistically drive down the possibility of the unlucky
* cases happening.
*/
#ifndef OPAL_LIST_H
#define OPAL_LIST_H
#include <stdio.h>
#include <stdlib.h>
#include "opal/class/opal_object.h"
#if OMPI_ENABLE_DEBUG
/* Need atomics for debugging (reference counting) */
#include "opal/sys/atomic.h"
#include "opal/threads/mutex.h"
#endif
#if defined(c_plusplus) || defined(__cplusplus)
extern "C" {
#endif
/**
* \internal
*
* The class for the list container.
*/
OPAL_DECLSPEC OBJ_CLASS_DECLARATION(opal_list_t);
/**
* \internal
*
* Base class for items that are put in list (opal_list_t) containers.
*/
OPAL_DECLSPEC OBJ_CLASS_DECLARATION(opal_list_item_t);
/**
* \internal
*
* Struct of an opal_list_item_t
*/
struct opal_list_item_t
{
opal_object_t super;
/**< Generic parent class for all Open MPI objects */
volatile struct opal_list_item_t *opal_list_next;
/**< Pointer to next list item */
volatile struct opal_list_item_t *opal_list_prev;
/**< Pointer to previous list item */
int32_t item_free;
#if OMPI_ENABLE_DEBUG
/** Atomic reference count for debugging */
volatile int32_t opal_list_item_refcount;
/** The list this item belong to */
volatile struct opal_list_t* opal_list_item_belong_to;
#endif
};
/**
* Base type for items that are put in a list (opal_list_t) containers.
*/
typedef struct opal_list_item_t opal_list_item_t;
/**
* Get the next item in a list.
*
* @param item A list item.
*
* @returns The next item in the list
*/
#define opal_list_get_next(item) \
((item) ? ((opal_list_item_t*) ((opal_list_item_t*)(item))->opal_list_next) : NULL)
/**
* Get the next item in a list.
*
* @param item A list item.
*
* @returns The next item in the list
*/
#define opal_list_get_prev(item) \
((item) ? ((opal_list_item_t*) ((opal_list_item_t*)(item))->opal_list_prev) : NULL)
/**
* \internal
*
* Struct of an opal_list_t
*/
struct opal_list_t
{
opal_object_t super;
/**< Generic parent class for all Open MPI objects */
opal_list_item_t opal_list_sentinel;
/**< Head and tail item of the list */
volatile size_t opal_list_length;
/**< Quick reference to the number of items in the list */
};
/**
* List container type.
*/
typedef struct opal_list_t opal_list_t;
/**
* Check for empty list
*
* @param list The list container
*
* @returns true if list's size is 0, false otherwise
*
* This is an O(1) operation.
*
* This is an inlined function in compilers that support inlining,
* so it's usually a cheap operation.
*/
static inline bool opal_list_is_empty(opal_list_t* list)
{
return (list->opal_list_sentinel.opal_list_next ==
&(list->opal_list_sentinel) ? true : false);
}
/**
* Return the first item on the list (does not remove it).
*
* @param list The list container
*
* @returns A pointer to the first item on the list
*
* This is an O(1) operation to return the first item on the list. It
* should be compared against the returned value from
* opal_list_get_end() to ensure that the list is not empty.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t* opal_list_get_first(opal_list_t* list)
{
opal_list_item_t* item = (opal_list_item_t*)list->opal_list_sentinel.opal_list_next;
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that the first item is only on one list */
assert(1 == item->opal_list_item_refcount);
assert( list == item->opal_list_item_belong_to );
#endif
return item;
}
/**
* Return the last item on the list (does not remove it).
*
* @param list The list container
*
* @returns A pointer to the last item on the list
*
* This is an O(1) operation to return the last item on the list. It
* should be compared against the returned value from
* opal_list_get_begin() to ensure that the list is not empty.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t* opal_list_get_last(opal_list_t* list)
{
opal_list_item_t* item = (opal_list_item_t *)list->opal_list_sentinel.opal_list_prev;
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that the last item is only on one list */
assert( 1 == item->opal_list_item_refcount );
assert( list == item->opal_list_item_belong_to );
#endif
return item;
}
/**
* Return the beginning of the list; an invalid list entry suitable
* for comparison only.
*
* @param list The list container
*
* @returns A pointer to the beginning of the list.
*
* This is an O(1) operation to return the beginning of the list.
* Similar to the STL, this is a special invalid list item -- it
* should \em not be used for storage. It is only suitable for
* comparison to other items in the list to see if they are valid or
* not; it's ususally used when iterating through the items in a list.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t* opal_list_get_begin(opal_list_t* list)
{
return &(list->opal_list_sentinel);
}
/**
* Return the end of the list; an invalid list entry suitable for
* comparison only.
*
* @param list The list container
*
* @returns A pointer to the end of the list.
*
* This is an O(1) operation to return the end of the list.
* Similar to the STL, this is a special invalid list item -- it
* should \em not be used for storage. It is only suitable for
* comparison to other items in the list to see if they are valid or
* not; it's ususally used when iterating through the items in a list.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t* opal_list_get_end(opal_list_t* list)
{
return &(list->opal_list_sentinel);
}
/**
* Return the number of items in a list
*
* @param list The list container
*
* @returns The size of the list (size_t)
*
* This is an O(1) lookup to return the size of the list.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*
* \warning The size of the list is cached as part of the list. In
* the future, calling \c opal_list_splice or \c opal_list_join may
* result in this function recomputing the list size, which would be
* an O(N) operation. If \c opal_list_splice or \c opal_list_join is
* never called on the specified list, this function will always be
* O(1).
*/
static inline size_t opal_list_get_size(opal_list_t* list)
{
#if OMPI_ENABLE_DEBUG && 0
/* not sure if we really want this running in devel, as it does
* slow things down. Wanted for development of splice / join to
* make sure length was reset properly
*/
size_t check_len = 0;
opal_list_item_t *item;
for (item = opal_list_get_first(list) ;
item != opal_list_get_end(list) ;
item = opal_list_get_next(item)) {
check_len++;
}
if (check_len != list->opal_list_length) {
fprintf(stderr," Error :: opal_list_get_size - opal_list_length does not match actual list length\n");
fflush(stderr);
abort();
}
#endif
return list->opal_list_length;
}
/**
* Remove an item from a list.
*
* @param list The list container
* @param item The item to remove
*
* @returns A pointer to the item on the list previous to the one
* that was removed.
*
* This is an O(1) operation to remove an item from the list. The
* forward / reverse pointers in the list are updated and the item is
* removed. The list item that is returned is now "owned" by the
* caller -- they are responsible for OBJ_RELEASE()'ing it.
*
* If debugging is enabled (specifically, if --enable-debug was used
* to configure Open MPI), this is an O(N) operation because it checks
* to see if the item is actually in the list first.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t *opal_list_remove_item
(opal_list_t *list, opal_list_item_t *item)
{
#if OMPI_ENABLE_DEBUG
opal_list_item_t *item_ptr;
bool found = false;
/* check to see that the item is in the list */
for (item_ptr = opal_list_get_first(list);
item_ptr != opal_list_get_end(list);
item_ptr = (opal_list_item_t *)(item_ptr->opal_list_next)) {
if (item_ptr == (opal_list_item_t *) item) {
found = true;
break;
}
}
if (!found) {
fprintf(stderr," Warning :: opal_list_remove_item - the item %p is not on the list %p \n",(void*) item, (void*) list);
fflush(stderr);
return (opal_list_item_t *)NULL;
}
assert( list == item->opal_list_item_belong_to );
#endif
/* reset next pointer of previous element */
item->opal_list_prev->opal_list_next=item->opal_list_next;
/* reset previous pointer of next element */
item->opal_list_next->opal_list_prev=item->opal_list_prev;
list->opal_list_length--;
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that this item is still only on one list */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), -1 );
assert(0 == item->opal_list_item_refcount);
item->opal_list_item_belong_to = NULL;
#endif
return (opal_list_item_t *)item->opal_list_prev;
}
/**
* Append an item to the end of the list.
*
* @param list The list container
* @param item The item to append
*
* This is an O(1) operation to append an item to the end of a list.
* The opal_list_item_t is not OBJ_RETAIN()'ed; it is assumed that
* "ownership" of the item is passed from the caller to the list.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
#if OMPI_ENABLE_DEBUG
#define opal_list_append(l,i) \
_opal_list_append(l,i,__FILE__,__LINE__)
#else
#define opal_list_append(l,i) \
_opal_list_append(l,i)
#endif /* OMPI_ENABLE_DEBUG */
static inline void _opal_list_append(opal_list_t *list, opal_list_item_t *item
#if OMPI_ENABLE_DEBUG
, char* FILE_NAME, int LINENO
#endif /* OMPI_ENABLE_DEBUG */
)
{
opal_list_item_t* sentinel = &(list->opal_list_sentinel);
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that this item is previously on no lists */
assert(0 == item->opal_list_item_refcount);
assert( NULL == item->opal_list_item_belong_to );
item->super.cls_init_file_name = FILE_NAME;
item->super.cls_init_lineno = LINENO;
#endif
/* set new element's previous pointer */
item->opal_list_prev = sentinel->opal_list_prev;
/* reset previous pointer on current last element */
sentinel->opal_list_prev->opal_list_next = item;
/* reset new element's next pointer */
item->opal_list_next = sentinel;
/* reset the list's tail element previous pointer */
sentinel->opal_list_prev = item;
/* increment list element counter */
list->opal_list_length++;
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure this item is only on the list that we just
appended it to */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), 1 );
assert(1 == item->opal_list_item_refcount);
item->opal_list_item_belong_to = list;
#endif
}
/**
* Prepend an item to the beginning of the list.
*
* @param list The list container
* @param item The item to prepend
*
* This is an O(1) operation to prepend an item to the beginning of a
* list. The opal_list_item_t is not OBJ_RETAIN()'ed; it is assumed
* that "ownership" of the item is passed from the caller to the list.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline void opal_list_prepend(opal_list_t *list,
opal_list_item_t *item)
{
opal_list_item_t* sentinel = &(list->opal_list_sentinel);
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that this item is previously on no lists */
assert(0 == item->opal_list_item_refcount);
assert( NULL == item->opal_list_item_belong_to );
#endif
/* reset item's next pointer */
item->opal_list_next = sentinel->opal_list_next;
/* reset item's previous pointer */
item->opal_list_prev = sentinel;
/* reset previous first element's previous poiner */
sentinel->opal_list_next->opal_list_prev = item;
/* reset head's next pointer */
sentinel->opal_list_next = item;
/* increment list element counter */
list->opal_list_length++;
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure this item is only on the list that we just
prepended it to */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), 1 );
assert(1 == item->opal_list_item_refcount);
item->opal_list_item_belong_to = list;
#endif
}
/**
* Remove the first item from the list and return it.
*
* @param list The list container
*
* @returns The first item on the list. If the list is empty,
* NULL will be returned
*
* This is an O(1) operation to return the first item on the list. If
* the list is not empty, a pointer to the first item in the list will
* be returned. Ownership of the item is transferred from the list to
* the caller; no calls to OBJ_RETAIN() or OBJ_RELEASE() are invoked.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t *opal_list_remove_first(opal_list_t *list)
{
/* Removes and returns first item on list.
Caller now owns the item and should release the item
when caller is done with it.
*/
volatile opal_list_item_t *item;
if ( 0 == list->opal_list_length ) {
return (opal_list_item_t *)NULL;
}
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that the first item is only on this list */
assert(1 == list->opal_list_sentinel.opal_list_next->opal_list_item_refcount);
#endif
/* reset list length counter */
list->opal_list_length--;
/* get pointer to first element on the list */
item = list->opal_list_sentinel.opal_list_next;
/* reset previous pointer of next item on the list */
item->opal_list_next->opal_list_prev = item->opal_list_prev;
/* reset the head next pointer */
list->opal_list_sentinel.opal_list_next = item->opal_list_next;
#if OMPI_ENABLE_DEBUG
assert( list == item->opal_list_item_belong_to );
item->opal_list_item_belong_to = NULL;
item->opal_list_prev=(opal_list_item_t *)NULL;
item->opal_list_next=(opal_list_item_t *)NULL;
/* Spot check: ensure that the item we're returning is now on no
lists */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), -1 );
assert(0 == item->opal_list_item_refcount);
#endif
return (opal_list_item_t *) item;
}
/**
* Remove the last item from the list and return it.
*
* @param list The list container
*
* @returns The last item on the list. If the list is empty,
* NULL will be returned
*
* This is an O(1) operation to return the last item on the list. If
* the list is not empty, a pointer to the last item in the list will
* be returned. Ownership of the item is transferred from the list to
* the caller; no calls to OBJ_RETAIN() or OBJ_RELEASE() are invoked.
*
* This is an inlined function in compilers that support inlining, so
* it's usually a cheap operation.
*/
static inline opal_list_item_t *opal_list_remove_last(opal_list_t *list)
{
/* Removes, releases and returns last item on list.
Caller now owns the item and should release the item
when caller is done with it.
*/
volatile opal_list_item_t *item;
if ( 0 == list->opal_list_length ) {
return (opal_list_item_t *)NULL;
}
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that the first item is only on this list */
assert(1 == list->opal_list_sentinel.opal_list_prev->opal_list_item_refcount);
#endif
/* reset list length counter */
list->opal_list_length--;
/* get item */
item = list->opal_list_sentinel.opal_list_prev;
/* reset previous pointer on next to last pointer */
item->opal_list_prev->opal_list_next = item->opal_list_next;
/* reset tail's previous pointer */
list->opal_list_sentinel.opal_list_prev = item->opal_list_prev;
#if OMPI_ENABLE_DEBUG
assert( list == item->opal_list_item_belong_to );
item->opal_list_next = item->opal_list_prev = (opal_list_item_t *)NULL;
/* Spot check: ensure that the item we're returning is now on no
lists */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), -1 );
assert(0 == item->opal_list_item_refcount);
item->opal_list_item_belong_to = NULL;
#endif
return (opal_list_item_t *) item;
}
/**
* Add an item to the list before a given element
*
* @param list The list container
* @param pos List element to insert \c item before
* @param item The item to insert
*
* Inserts \c item before \c pos. This is an O(1) operation.
*/
static inline void opal_list_insert_pos(opal_list_t *list, opal_list_item_t *pos,
opal_list_item_t *item)
{
#if OMPI_ENABLE_DEBUG
/* Spot check: ensure that the item we're insertting is currently
not on any list */
assert(0 == item->opal_list_item_refcount);
assert( NULL == item->opal_list_item_belong_to );
#endif
/* point item at the existing elements */
item->opal_list_next = pos;
item->opal_list_prev = pos->opal_list_prev;
/* splice into the list */
pos->opal_list_prev->opal_list_next = item;
pos->opal_list_prev = item;
/* reset list length counter */
list->opal_list_length++;
#if OMPI_ENABLE_DEBUG
/* Spot check: double check that this item is only on the list
that we just added it to */
OPAL_THREAD_ADD32( &(item->opal_list_item_refcount), 1 );
assert(1 == item->opal_list_item_refcount);
item->opal_list_item_belong_to = list;
#endif
}
/**
* Add an item to the list at a specific index location in the list.
*
* @param list The list container
* @param item The item to insert
* @param index Location to add the item
*
* @returns true if insertion succeeded; otherwise false
*
* This is potentially an O(N) operation to traverse down to the
* correct location in the list and add an item.
*
* Example: if idx = 2 and list = item1->item2->item3->item4, then
* after insert, list = item1->item2->item->item3->item4.
*
* If index is greater than the length of the list, no action is
* performed and false is returned.
*/
OPAL_DECLSPEC bool opal_list_insert(opal_list_t *list, opal_list_item_t *item,
long long idx);
/**
* Join a list into another list
*
* @param thislist List container for list being operated on
* @param pos List item in \c thislist marking the position before
* which items are inserted
* @param xlist List container for list being spliced from
*
* Join a list into another list. All of the elements of \c xlist
* are inserted before \c pos and removed from \c xlist.
*
* This operation is an O(1) operation. Both \c thislist and \c
* xlist must be valid list containsers. \c xlist will be empty
* but valid after the call. All pointers to \c opal_list_item_t
* containers remain valid, including those that point to elements
* in \c xlist.
*/
OPAL_DECLSPEC void opal_list_join(opal_list_t *thislist, opal_list_item_t *pos,
opal_list_t *xlist);
/**
* Splice a list into another list
*
* @param thislist List container for list being operated on
* @param pos List item in \c thislist marking the position before
* which items are inserted
* @param xlist List container for list being spliced from
* @param first List item in \c xlist marking the start of elements
* to be copied into \c thislist
* @param last List item in \c xlist marking the end of elements
* to be copied into \c thislist
*
* Splice a subset of a list into another list. The \c [first,
* last) elements of \c xlist are moved into \c thislist,
* inserting them before \c pos. \c pos must be a valid iterator
* in \c thislist and \c [first, last) must be a valid range in \c
* xlist. \c postition must not be in the range \c [first, last).
* It is, however, valid for \c xlist and \c thislist to be the
* same list.
*
* This is an O(N) operation because the length of both lists must
* be recomputed.
*/
OPAL_DECLSPEC void opal_list_splice(opal_list_t *thislist, opal_list_item_t *pos,
opal_list_t *xlist, opal_list_item_t *first,
opal_list_item_t *last);
/**
* Comparison function for opal_list_sort(), below.
*
* @param a Pointer to a pointer to an opal_list_item_t.
* Explanation below.
* @param b Pointer to a pointer to an opal_list_item_t.
* Explanation below.
* @retval 1 if \em a is greater than \em b
* @retval 0 if \em a is equal to \em b
* @retval 11 if \em a is less than \em b
*
* This function is invoked by qsort(3) from within
* opal_list_sort(). It is important to understand what
* opal_list_sort() does before invoking qsort, so go read that
* documentation first.
*
* The important thing to realize here is that a and b will be \em
* double pointers to the items that you need to compare. Here's
* a sample compare function to illustrate this point:
*
* \verb
* static int compare(opal_list_item_t **a, opal_list_item_t **b)
* {
* orte_pls_base_cmp_t *aa = *((orte_pls_base_cmp_t **) a);
* orte_pls_base_cmp_t *bb = *((orte_pls_base_cmp_t **) b);
*
* if (bb->priority > aa->priority) {
* return 1;
* } else if (bb->priority == aa->priority) {
* return 0;
* } else {
* return -1;
* }
* }
* \endverb
*/
typedef int (*opal_list_item_compare_fn_t)(opal_list_item_t **a,
opal_list_item_t **b);
/**
* Sort a list with a provided compare function.
*
* @param list The list to sort
* @param compare Compare function
*
* Put crassly, this function's complexity is O(N) + O(log(N)).
* Its algorithm is:
*
* - remove every item from the list and put the corresponding
* (opal_list_item_t*)'s in an array
* - call qsort(3) with that array and your compare function
* - re-add every element of the now-sorted array to the list
*
* The resulting list is now ordered. Note, however, that since
* an array of pointers is sorted, the comparison function must do
* a double de-reference to get to the actual opal_list_item_t (or
* whatever the underlying type is). See the documentation of
* opal_list_item_compare_fn_t for an example).
*/
OPAL_DECLSPEC int opal_list_sort(opal_list_t* list, opal_list_item_compare_fn_t compare);
#if defined(c_plusplus) || defined(__cplusplus)
}
#endif
#endif /* OPAL_LIST_H */