199 строки
7.5 KiB
C
199 строки
7.5 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) 2010 Cisco Systems, Inc. All rights reserved.
|
||
|
* $COPYRIGHT$
|
||
|
*
|
||
|
* Additional copyrights may follow
|
||
|
*
|
||
|
* $HEADER$
|
||
|
*/
|
||
|
|
||
|
#include "ompi_config.h"
|
||
|
|
||
|
#include "ompi/class/ompi_seq_tracker.h"
|
||
|
|
||
|
|
||
|
|
||
|
OBJ_CLASS_INSTANCE(ompi_seq_tracker_range_t,
|
||
|
opal_list_item_t,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
|
||
|
|
||
|
static void ompi_seq_tracker_construct(ompi_seq_tracker_t* seq_tracker) {
|
||
|
OBJ_CONSTRUCT(&seq_tracker->seq_ids, opal_list_t);
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*)opal_list_get_end(&seq_tracker->seq_ids);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void ompi_seq_tracker_destruct(ompi_seq_tracker_t* seq_tracker)
|
||
|
{
|
||
|
opal_list_item_t* item;
|
||
|
while(NULL != (item = opal_list_remove_first(&seq_tracker->seq_ids)))
|
||
|
OBJ_RELEASE(item);
|
||
|
OBJ_DESTRUCT(&seq_tracker->seq_ids);
|
||
|
}
|
||
|
|
||
|
OBJ_CLASS_INSTANCE(
|
||
|
ompi_seq_tracker_t,
|
||
|
opal_object_t,
|
||
|
ompi_seq_tracker_construct,
|
||
|
ompi_seq_tracker_destruct);
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Look for duplicate sequence number in current range.
|
||
|
* Must be called w/ matching lock held.
|
||
|
*/
|
||
|
|
||
|
bool ompi_seq_tracker_check_duplicate(
|
||
|
ompi_seq_tracker_t* seq_tracker,
|
||
|
uint32_t seq_id)
|
||
|
{
|
||
|
ompi_seq_tracker_range_t* item;
|
||
|
const ompi_seq_tracker_range_t* sentinel = (ompi_seq_tracker_range_t*)opal_list_get_end(&seq_tracker->seq_ids);
|
||
|
int8_t direction = 0; /* 1 is next, -1 is previous */
|
||
|
|
||
|
item = seq_tracker->seq_ids_current;
|
||
|
while(true) {
|
||
|
if(sentinel == item) {
|
||
|
return false;
|
||
|
} else if(item->seq_id_high >= seq_id && item->seq_id_low <= seq_id) {
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) item;
|
||
|
return true;
|
||
|
} else if(seq_id > item->seq_id_high && direction != -1) {
|
||
|
direction = 1;
|
||
|
item = (ompi_seq_tracker_range_t*) opal_list_get_next(item);
|
||
|
} else if(seq_id < item->seq_id_low && direction != 1) {
|
||
|
direction = -1;
|
||
|
item = (ompi_seq_tracker_range_t*) opal_list_get_prev(item);
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* insert item into sequence tracking list,
|
||
|
* compacts continuous regions into a single entry
|
||
|
* GMS::: Use a free list for the items (don't do OBJ_NEW)!
|
||
|
*/
|
||
|
void ompi_seq_tracker_insert(ompi_seq_tracker_t* seq_tracker,
|
||
|
uint32_t seq_id)
|
||
|
{
|
||
|
opal_list_t* seq_ids = &seq_tracker->seq_ids;
|
||
|
ompi_seq_tracker_range_t* item = seq_tracker->seq_ids_current;
|
||
|
int8_t direction = 0; /* 1 is next, -1 is previous */
|
||
|
ompi_seq_tracker_range_t *new_item, *next_item, *prev_item;
|
||
|
const ompi_seq_tracker_range_t* sentinel = (ompi_seq_tracker_range_t*)opal_list_get_end(seq_ids);
|
||
|
|
||
|
while( true ) {
|
||
|
if( item == sentinel ) {
|
||
|
new_item = OBJ_NEW(ompi_seq_tracker_range_t);
|
||
|
new_item->seq_id_low = new_item->seq_id_high = seq_id;
|
||
|
if( -1 == direction ) {
|
||
|
opal_list_prepend(seq_ids, (opal_list_item_t*) new_item);
|
||
|
} else {
|
||
|
opal_list_append(seq_ids, (opal_list_item_t*) new_item);
|
||
|
}
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) new_item;
|
||
|
return;
|
||
|
|
||
|
} else if(item->seq_id_high >= seq_id && item->seq_id_low <= seq_id ) {
|
||
|
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) item;
|
||
|
return;
|
||
|
|
||
|
} else if((item->seq_id_high + 1) == seq_id) {
|
||
|
|
||
|
next_item = (ompi_seq_tracker_range_t*) opal_list_get_next(item);
|
||
|
/* try to consolidate */
|
||
|
if( (sentinel != next_item) && next_item->seq_id_low == (seq_id+1)) {
|
||
|
item->seq_id_high = next_item->seq_id_high;
|
||
|
opal_list_remove_item(seq_ids, (opal_list_item_t*) next_item);
|
||
|
OBJ_RELEASE(next_item);
|
||
|
} else {
|
||
|
item->seq_id_high = seq_id;
|
||
|
}
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) item;
|
||
|
return;
|
||
|
|
||
|
} else if((item->seq_id_low - 1) == seq_id) {
|
||
|
|
||
|
prev_item = (ompi_seq_tracker_range_t*) opal_list_get_prev(item);
|
||
|
/* try to consolidate */
|
||
|
if( (sentinel != prev_item) && prev_item->seq_id_high == (seq_id-1)) {
|
||
|
item->seq_id_low = prev_item->seq_id_low;
|
||
|
opal_list_remove_item(seq_ids, (opal_list_item_t*) prev_item);
|
||
|
OBJ_RELEASE(prev_item);
|
||
|
} else {
|
||
|
item->seq_id_low = seq_id;
|
||
|
}
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) item;
|
||
|
return;
|
||
|
|
||
|
} else if(seq_id > item->seq_id_high ) {
|
||
|
if(direction == -1) {
|
||
|
/* we have gone back in the list, and we went one item too far */
|
||
|
new_item = OBJ_NEW(ompi_seq_tracker_range_t);
|
||
|
new_item->seq_id_low = new_item->seq_id_high = seq_id;
|
||
|
next_item = (ompi_seq_tracker_range_t*) opal_list_get_next(item);
|
||
|
/* insert new_item directly before item */
|
||
|
opal_list_insert_pos(seq_ids,
|
||
|
(opal_list_item_t*) next_item,
|
||
|
(opal_list_item_t*) new_item);
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) new_item;
|
||
|
return;
|
||
|
} else {
|
||
|
direction = 1;
|
||
|
item = (ompi_seq_tracker_range_t*) opal_list_get_next(item);
|
||
|
}
|
||
|
} else if(seq_id < item->seq_id_low) {
|
||
|
if(direction == 1) {
|
||
|
/* we have gone forward in the list, and we went one item too far */
|
||
|
new_item = OBJ_NEW(ompi_seq_tracker_range_t);
|
||
|
new_item->seq_id_low = new_item->seq_id_high = seq_id;
|
||
|
opal_list_insert_pos(seq_ids,
|
||
|
(opal_list_item_t*) item,
|
||
|
(opal_list_item_t*) new_item);
|
||
|
|
||
|
seq_tracker->seq_ids_current = (ompi_seq_tracker_range_t*) new_item;
|
||
|
return;
|
||
|
} else {
|
||
|
direction = -1;
|
||
|
item = (ompi_seq_tracker_range_t*) opal_list_get_prev(item);
|
||
|
}
|
||
|
} else {
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void ompi_seq_tracker_copy(ompi_seq_tracker_t* dst, ompi_seq_tracker_t* src)
|
||
|
{
|
||
|
opal_list_item_t* item;
|
||
|
for( item = opal_list_get_first(&src->seq_ids);
|
||
|
item != opal_list_get_end(&src->seq_ids);
|
||
|
item = opal_list_get_next(item)) {
|
||
|
ompi_seq_tracker_range_t* src_item = (ompi_seq_tracker_range_t*)item;
|
||
|
ompi_seq_tracker_range_t* dst_item = OBJ_NEW(ompi_seq_tracker_range_t);
|
||
|
dst_item->seq_id_high = src_item->seq_id_high;
|
||
|
dst_item->seq_id_low = src_item->seq_id_low;
|
||
|
opal_list_append(&dst->seq_ids, &dst_item->super);
|
||
|
if(src->seq_ids_current == src_item) {
|
||
|
dst->seq_ids_current = dst_item;
|
||
|
}
|
||
|
}
|
||
|
}
|