OpenFAM Reference Implementation

openfam::fam::fam_compare_swap

Atomically conditionally replace a value in FAM with the given value and return the old value.

Synopsis

int32_t fam_compare_swap(Fam_Descriptor *descriptor, uint64_t offset, int32_t oldValue, int32_t newValue);

int64_t fam_compare_swap(Fam_Descriptor *descriptor, uint64_t offset, int64_t oldValue, int64_t newValue);

uint32_t fam_compare_swap(Fam_Descriptor *descriptor, uint64_t offset, uint32_t oldValue, uint32_t newValue);

uint64_t fam_compare_swap(Fam_Descriptor *descriptor, uint64_t offset, uint64_t oldValue, uint64_t newValue);

int128_t fam_compare_swap(Fam_Descriptor *descriptor, uint64_t offset, int128_t oldValue, int128_t newValue);

Description

These methods atomically perform a compare and swap of a value in FAM, and return the old value from FAM. All of these methods atomically implement the following algorithm (where TYPE is one of the data types listed in the synopsis):

TYPE cas(TYPE *p, TYPE oldValue, TYPE newValue){
  if(*p != oldValue){
    return *p;
  } else {
    *p = newValue;
    return oldValue;
  }
}

Input Arguments

Name Description
descriptor Descriptor associated with the data item.
offset Offset within the data item in FAM where value is located.
oldValue Expected old value at the FAM location.
newValue New value to use if successful.

Return Values

The old value from FAM. Throws an exception on error.

Exceptions

ExceptionDescription
Fam_InvalidOption_Exceptionincorrect parameters are passed.
Fam_Datapath_Exceptionerror occurred during atomic operation over fabric.
FAM_ERR_NOPERMCaller does not have access rights.
FAM_ERR_LIBFABRIClibfabric error occurred.
FAM_ERR_NOTFOUNDItem not found in the region.
FAM_ERR_GRPCif there is a communication error with memory server
FAM_ERR_RPC_CLIENT_NOTFOUNDMemory server initialization failure
Fam_Timeout_Exceptionnumber of libfabric retry count reached

Notes

These methods atomically compare a value in FAM with an expected value and if the value in FAM is equal to the expected value, swap a new value. The old value is returned. Note that the offset argument must point to the correct value for the data type. Availability of these methods is dependent on hardware support for the operations. GCC currently does not define int128_t; it is defined in fam.h.

Example

#include <string.h>
#include <fam/fam.h>
#include <fam/fam_exception.h>
using namespace std;
using namespace openfam;
	
int main(void) {
	fam *myFam = new fam();
	
	// ... Initialization code here
	try {
		// look up the descriptor to a previously allocated data item
		Fam_Descriptor *descriptor = myFam->fam_lookup("myItem", "myRegion");

		// Atomically compare and replaces the first integer element in FAM with 0
		// only if it contains 100; It returns the old value
		int32_t oldValue = myFam->fam_compare_swap(descriptor, 0, 100, 0);

		// The first integer in FAM is replaced with 0 only if it contains 100;
		// Old value is returned
		printf("Old value of first integer element in FAM: %d\n", oldValue);
		
		// ... subsequent code here
	} catch (Fam_Exception &e) {
		printf("fam API failed: %d: %s\n", e.fam_error(), e.fam_error_msg());
	}
	// ... Finalization code follows
}