Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
/*
|
2016-01-07 19:22:35 -08:00
|
|
|
* Copyright (c) 2012-2016 Cisco Systems, Inc. All rights reserved.
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* Copyright (c) 2012 Los Alamos National Security, LLC. All rights reserved
|
2015-04-29 16:17:55 -07:00
|
|
|
* Copyright (c) 2015 Intel, Inc. All rights reserved.
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* $COPYRIGHT$
|
2015-04-29 16:17:55 -07:00
|
|
|
*
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* Additional copyrights may follow
|
2015-04-29 16:17:55 -07:00
|
|
|
*
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* $HEADER$
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @file
|
|
|
|
*
|
|
|
|
* This file provides a "hotel" class:
|
|
|
|
*
|
|
|
|
* - A hotel has a fixed number of rooms (i.e., storage slots)
|
|
|
|
* - An arbitrary data pointer can check into an empty room at any time
|
|
|
|
* - The occupant of a room can check out at any time
|
|
|
|
* - Optionally, the occupant of a room can be forcibly evicted at a
|
|
|
|
* given time (i.e., when an opal timer event expires).
|
|
|
|
* - The hotel has finite occupancy; if you try to checkin a new
|
|
|
|
* occupant and the hotel is already full, it will gracefully fail
|
|
|
|
* to checkin.
|
|
|
|
*
|
|
|
|
* One use case for this class is for ACK-based network retransmission
|
|
|
|
* schemes (NACK-based retransmission schemes probably can use
|
2015-04-29 16:17:55 -07:00
|
|
|
* opal_ring_buffer).
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
*
|
|
|
|
* For ACK-based retransmission schemes, a hotel might be used
|
|
|
|
* something like this:
|
|
|
|
*
|
|
|
|
* - when a message is sent, check it in to a hotel with a timer
|
|
|
|
* - if an ACK is received, check it out of the hotel (which also cancels
|
|
|
|
* the timer)
|
|
|
|
* - if an ACK isn't received in time, the timer will expire and the
|
|
|
|
* upper layer will get a callback with the message
|
|
|
|
* - if an ACK is received late (i.e., after its timer has expired),
|
|
|
|
* then checkout will gracefully fail
|
|
|
|
*
|
|
|
|
* Note that this class intentionally provides pretty minimal
|
|
|
|
* functionality. It is intended to be used in performance-critical
|
|
|
|
* code paths -- extra functionality would simply add latency.
|
|
|
|
*
|
2012-11-02 14:00:54 +00:00
|
|
|
* There is an opal_hotel_init() function to create a hotel, but no
|
|
|
|
* corresponding finalize; the destructor will handle all finalization
|
|
|
|
* issues. Note that when a hotel is destroyed, it will delete all
|
|
|
|
* pending events from the event base (i.e., all pending eviction
|
|
|
|
* callbacks); no further eviction callbacks will be invoked.
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef OPAL_HOTEL_H
|
|
|
|
#define OPAL_HOTEL_H
|
|
|
|
|
|
|
|
#include "opal_config.h"
|
|
|
|
|
2013-04-29 17:02:37 +00:00
|
|
|
#include "opal/prefetch.h"
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
#include "opal/class/opal_object.h"
|
|
|
|
#include "opal/mca/event/event.h"
|
|
|
|
|
|
|
|
BEGIN_C_DECLS
|
|
|
|
|
|
|
|
struct opal_hotel_t;
|
|
|
|
|
|
|
|
/* User-supplied function to be invoked when an occupant is evicted. */
|
2015-04-29 16:17:55 -07:00
|
|
|
typedef void (*opal_hotel_eviction_callback_fn_t)(struct opal_hotel_t *hotel,
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
int room_num,
|
|
|
|
void *occupant);
|
|
|
|
|
2012-08-16 18:42:23 +00:00
|
|
|
/* Note that this is an internal data structure; it is not part of the
|
|
|
|
public opal_hotel interface. Public consumers of opal_hotel
|
|
|
|
shouldn't need to use this struct at all (we only have it here in
|
|
|
|
this .h file because some functions are inlined for speed, and need
|
|
|
|
to get to the internals of this struct).
|
|
|
|
|
|
|
|
The room struct should be as small as possible to be cache
|
|
|
|
friendly. Specifically: it would be great if multiple rooms could
|
|
|
|
fit in a single cache line because we'll always allocate a
|
|
|
|
contiguous set of rooms in an array. */
|
|
|
|
typedef struct {
|
|
|
|
void *occupant;
|
|
|
|
opal_event_t eviction_timer_event;
|
|
|
|
} opal_hotel_room_t;
|
|
|
|
|
|
|
|
/* Note that this is an internal data structure; it is not part of the
|
|
|
|
public opal_hotel interface. Public consumers of opal_hotel
|
|
|
|
shouldn't need to use this struct at all (we only have it here in
|
|
|
|
this .h file because some functions are inlined for speed, and need
|
|
|
|
to get to the internals of this struct).
|
|
|
|
|
|
|
|
Use a unique struct for holding the arguments for eviction
|
|
|
|
callbacks. We *could* make the to-be-evicted opal_hotel_room_t
|
|
|
|
instance as the argument, but we don't, for 2 reasons:
|
|
|
|
|
|
|
|
1. We want as many opal_hotel_room_t's to fit in a cache line as
|
|
|
|
possible (i.e., to be as cache-friendly as possible). The
|
|
|
|
common/fast code path only needs to access the data in the
|
|
|
|
opal_hotel_room_t (and not the callback argument data).
|
|
|
|
|
|
|
|
2. Evictions will be uncommon, so we don't mind penalizing them a
|
|
|
|
bit by making the data be in a separate cache line.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
struct opal_hotel_t *hotel;
|
|
|
|
int room_num;
|
|
|
|
} opal_hotel_room_eviction_callback_arg_t;
|
|
|
|
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
typedef struct opal_hotel_t {
|
|
|
|
/* make this an object */
|
|
|
|
opal_object_t super;
|
|
|
|
|
|
|
|
/* Max number of rooms in the hotel */
|
|
|
|
int num_rooms;
|
|
|
|
|
2015-07-11 06:42:23 -07:00
|
|
|
/* event base to be used for eviction timeout */
|
|
|
|
opal_event_base_t *evbase;
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
struct timeval eviction_timeout;
|
|
|
|
opal_hotel_eviction_callback_fn_t evict_callback_fn;
|
|
|
|
|
|
|
|
/* All rooms in this hotel */
|
2012-08-16 18:42:23 +00:00
|
|
|
opal_hotel_room_t *rooms;
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
|
|
|
|
/* Separate array for all the eviction callback arguments (see
|
|
|
|
rationale above for why this is a separate array) */
|
2012-08-16 18:42:23 +00:00
|
|
|
opal_hotel_room_eviction_callback_arg_t *eviction_args;
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
|
|
|
|
/* All currently unoccupied rooms in this hotel (not necessarily
|
|
|
|
in any particular order) */
|
|
|
|
int *unoccupied_rooms;
|
|
|
|
int last_unoccupied_room;
|
|
|
|
} opal_hotel_t;
|
|
|
|
OBJ_CLASS_DECLARATION(opal_hotel_t);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize the hotel.
|
|
|
|
*
|
|
|
|
* @param hotel Pointer to a hotel (IN)
|
|
|
|
* @param num_rooms The total number of rooms in the hotel (IN)
|
2015-07-11 06:42:23 -07:00
|
|
|
* @param evbase Pointer to event base used for eviction timeout
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* @param eviction_timeout Max length of a stay at the hotel before
|
|
|
|
* the eviction callback is invoked (in microseconds)
|
|
|
|
* @param eviction_event_priority Event lib priority for the eviction timeout
|
|
|
|
* @param evict_callback_fn Callback function invoked if an occupant
|
|
|
|
* does not check out before the eviction_timeout.
|
|
|
|
*
|
|
|
|
* NOTE: If the callback function is NULL, then no eviction timer
|
|
|
|
* will be set - occupants will remain checked into the hotel until
|
|
|
|
* explicitly checked out.
|
|
|
|
*
|
2016-01-07 19:22:35 -08:00
|
|
|
* Also note: the eviction_callback_fn should absolutely not call any
|
|
|
|
* of the hotel checkout functions. Specifically: the occupant has
|
|
|
|
* already been ("forcibly") checked out *before* the
|
|
|
|
* eviction_callback_fn is invoked.
|
|
|
|
*
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
* @return OPAL_SUCCESS if all initializations were succesful. Otherwise,
|
|
|
|
* the error indicate what went wrong in the function.
|
|
|
|
*/
|
|
|
|
OPAL_DECLSPEC int opal_hotel_init(opal_hotel_t *hotel, int num_rooms,
|
2015-07-11 06:42:23 -07:00
|
|
|
opal_event_base_t *evbase,
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
uint32_t eviction_timeout,
|
|
|
|
int eviction_event_priority,
|
|
|
|
opal_hotel_eviction_callback_fn_t evict_callback_fn);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check in an occupant to the hotel.
|
|
|
|
*
|
|
|
|
* @param hotel Pointer to hotel (IN)
|
|
|
|
* @param occupant Occupant to check in (opaque to the hotel) (IN)
|
|
|
|
* @param room The room number that identifies this occupant in the
|
|
|
|
* hotel (OUT).
|
|
|
|
*
|
|
|
|
* If there is room in the hotel, the occupant is checked in and the
|
|
|
|
* timer for that occupant is started. The occupant's room is
|
|
|
|
* returned in the "room" param.
|
|
|
|
*
|
|
|
|
* Note that once a room's checkout_expire timer expires, the occupant
|
|
|
|
* is forcibly checked out, and then the eviction callback is invoked.
|
|
|
|
*
|
|
|
|
* @return OPAL_SUCCESS if the occupant is successfully checked in,
|
|
|
|
* and the room parameter will contain a valid value.
|
|
|
|
* @return OPAL_ERR_TEMP_OUT_OF_RESOURCE is the hotel is full. Try
|
|
|
|
* again later.
|
|
|
|
*/
|
|
|
|
static inline int opal_hotel_checkin(opal_hotel_t *hotel,
|
|
|
|
void *occupant,
|
|
|
|
int *room_num)
|
|
|
|
{
|
|
|
|
opal_hotel_room_t *room;
|
|
|
|
|
|
|
|
/* Do we have any rooms available? */
|
|
|
|
if (OPAL_UNLIKELY(hotel->last_unoccupied_room < 0)) {
|
|
|
|
return OPAL_ERR_TEMP_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put this occupant into the first empty room that we have */
|
|
|
|
*room_num = hotel->unoccupied_rooms[hotel->last_unoccupied_room--];
|
|
|
|
room = &(hotel->rooms[*room_num]);
|
|
|
|
room->occupant = occupant;
|
|
|
|
|
|
|
|
/* Assign the event and make it pending */
|
2015-07-11 06:42:23 -07:00
|
|
|
if (NULL != hotel->evbase) {
|
|
|
|
opal_event_add(&(room->eviction_timer_event),
|
|
|
|
&(hotel->eviction_timeout));
|
|
|
|
}
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2013-07-11 20:00:37 +00:00
|
|
|
/**
|
|
|
|
* Same as opal_hotel_checkin(), but slightly optimized for when the
|
|
|
|
* caller *knows* that there is a room available.
|
|
|
|
*/
|
|
|
|
static inline void opal_hotel_checkin_with_res(opal_hotel_t *hotel,
|
|
|
|
void *occupant,
|
|
|
|
int *room_num)
|
|
|
|
{
|
|
|
|
opal_hotel_room_t *room;
|
|
|
|
|
|
|
|
/* Put this occupant into the first empty room that we have */
|
|
|
|
*room_num = hotel->unoccupied_rooms[hotel->last_unoccupied_room--];
|
|
|
|
room = &(hotel->rooms[*room_num]);
|
|
|
|
assert(room->occupant == NULL);
|
|
|
|
room->occupant = occupant;
|
|
|
|
|
|
|
|
/* Assign the event and make it pending */
|
2015-07-11 06:42:23 -07:00
|
|
|
if (NULL != hotel->evbase) {
|
|
|
|
opal_event_add(&(room->eviction_timer_event),
|
|
|
|
&(hotel->eviction_timeout));
|
|
|
|
}
|
2013-07-11 20:00:37 +00:00
|
|
|
}
|
|
|
|
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
/**
|
|
|
|
* Check the specified occupant out of the hotel.
|
|
|
|
*
|
|
|
|
* @param hotel Pointer to hotel (IN)
|
|
|
|
* @param room Room number to checkout (IN)
|
|
|
|
*
|
|
|
|
* If there is an occupant in the room, their timer is canceled and
|
|
|
|
* they are checked out.
|
|
|
|
*
|
|
|
|
* Nothing is returned (as a minor optimization).
|
|
|
|
*/
|
|
|
|
static inline void opal_hotel_checkout(opal_hotel_t *hotel, int room_num)
|
|
|
|
{
|
|
|
|
opal_hotel_room_t *room;
|
|
|
|
|
|
|
|
/* Bozo check */
|
2012-10-22 18:05:39 +00:00
|
|
|
assert(room_num < hotel->num_rooms);
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
|
|
|
|
/* If there's an occupant in the room, check them out */
|
|
|
|
room = &(hotel->rooms[room_num]);
|
|
|
|
if (OPAL_LIKELY(NULL != room->occupant)) {
|
2016-01-07 19:22:35 -08:00
|
|
|
/* Do not change this logic without also changing the same
|
|
|
|
logic in opal_hotel_checkout_and_return_occupant() and
|
|
|
|
opal_hotel.c:local_eviction_callback(). */
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
room->occupant = NULL;
|
2015-07-11 06:42:23 -07:00
|
|
|
if (NULL != hotel->evbase) {
|
|
|
|
opal_event_del(&(room->eviction_timer_event));
|
|
|
|
}
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
hotel->last_unoccupied_room++;
|
2012-10-22 18:05:39 +00:00
|
|
|
assert(hotel->last_unoccupied_room < hotel->num_rooms);
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
hotel->unoccupied_rooms[hotel->last_unoccupied_room] = room_num;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Don't bother returning whether we actually checked someone out
|
|
|
|
or not (because this is in the critical performance path) --
|
|
|
|
assume the upper layer knows what it's doing. */
|
|
|
|
}
|
|
|
|
|
2015-04-29 16:17:55 -07:00
|
|
|
/**
|
|
|
|
* Check the specified occupant out of the hotel and return the occupant.
|
|
|
|
*
|
|
|
|
* @param hotel Pointer to hotel (IN)
|
|
|
|
* @param room Room number to checkout (IN)
|
|
|
|
* @param void * occupant (OUT)
|
|
|
|
* If there is an occupant in the room, their timer is canceled and
|
|
|
|
* they are checked out.
|
|
|
|
*
|
|
|
|
* Use this checkout and when caller needs the occupant
|
|
|
|
*/
|
|
|
|
static inline void opal_hotel_checkout_and_return_occupant(opal_hotel_t *hotel, int room_num, void **occupant)
|
|
|
|
{
|
|
|
|
opal_hotel_room_t *room;
|
|
|
|
|
|
|
|
/* Bozo check */
|
|
|
|
assert(room_num < hotel->num_rooms);
|
|
|
|
|
|
|
|
/* If there's an occupant in the room, check them out */
|
|
|
|
room = &(hotel->rooms[room_num]);
|
|
|
|
if (OPAL_LIKELY(NULL != room->occupant)) {
|
|
|
|
opal_output (10, "checking out occupant %p from room num %d", room->occupant, room_num);
|
2016-01-07 19:22:35 -08:00
|
|
|
/* Do not change this logic without also changing the same
|
|
|
|
logic in opal_hotel_checkout() and
|
|
|
|
opal_hotel.c:local_eviction_callback(). */
|
2015-04-29 16:17:55 -07:00
|
|
|
*occupant = room->occupant;
|
|
|
|
room->occupant = NULL;
|
2015-07-11 06:42:23 -07:00
|
|
|
if (NULL != hotel->evbase) {
|
|
|
|
opal_event_del(&(room->eviction_timer_event));
|
|
|
|
}
|
2015-04-29 16:17:55 -07:00
|
|
|
hotel->last_unoccupied_room++;
|
|
|
|
assert(hotel->last_unoccupied_room < hotel->num_rooms);
|
|
|
|
hotel->unoccupied_rooms[hotel->last_unoccupied_room] = room_num;
|
|
|
|
}
|
|
|
|
else {
|
2015-06-18 09:53:20 -07:00
|
|
|
*occupant = NULL;
|
|
|
|
}
|
2015-04-29 16:17:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if the hotel is empty (no occupant)
|
|
|
|
* @param hotel Pointer to hotel (IN)
|
|
|
|
* @return bool true if empty false if there is a occupant(s)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static inline bool opal_hotel_is_empty (opal_hotel_t *hotel)
|
|
|
|
{
|
|
|
|
if (hotel->last_unoccupied_room == hotel->num_rooms - 1)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-18 09:53:20 -07:00
|
|
|
/**
|
|
|
|
* Access the occupant of a room, but leave them checked into their room.
|
|
|
|
*
|
|
|
|
* @param hotel Pointer to hotel (IN)
|
|
|
|
* @param room Room number to checkout (IN)
|
|
|
|
* @param void * occupant (OUT)
|
|
|
|
*
|
|
|
|
* This accessor function is typically used to cycle across the occupants
|
|
|
|
* to check for someone already present that matches a description.
|
|
|
|
*/
|
|
|
|
static inline void opal_hotel_knock(opal_hotel_t *hotel, int room_num, void **occupant)
|
|
|
|
{
|
|
|
|
opal_hotel_room_t *room;
|
|
|
|
|
|
|
|
/* Bozo check */
|
|
|
|
assert(room_num < hotel->num_rooms);
|
|
|
|
|
|
|
|
*occupant = NULL;
|
|
|
|
|
|
|
|
/* If there's an occupant in the room, have them come to the door */
|
|
|
|
room = &(hotel->rooms[room_num]);
|
|
|
|
if (OPAL_LIKELY(NULL != room->occupant)) {
|
|
|
|
opal_output (10, "occupant %p in room num %d responded to knock", room->occupant, room_num);
|
|
|
|
*occupant = room->occupant;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Add new "opal_hotel" class. Abstractly speaking, this class does the
following:
* Provides a fixed number of resource slots (i.e., "hotel rooms").
* Allows one thing to occupy a resource slot at a time (i.e., each
hotel room can have an occupant check in to that room).
* Resource slots can be vacated at any time (i.e., occupants can
voluntarily check out of their hotel room).
* Resource slots can be occupied for a specific maximum amount of
time. If that time expires, the occupant is forcibly evicted and
the upper layer is notified via (libevent) callback (i.e., the maid
will kick an occupant of out of their room when their reservation
is over).
This class can be to be used for things like retransmission schemes
for unreliable transports. For example, a message sent on an
unreliable transport can be checked in to a hotel room. If an ACK for
that message is received, the message can be checked out. But if the
ACK is never received, the message will eventually be evicted from its
room and the upper layer will be notified that the message failed to
check out in time (i.e., that an ACK for that message was not received
in time).
Code using this class is currently being developed off-trunk, but will
be coming to SVN soon.
This commit was SVN r27067.
2012-08-16 17:29:55 +00:00
|
|
|
END_C_DECLS
|
|
|
|
|
|
|
|
#endif /* OPAL_HOTEL_H */
|