Skip to content

Commit

Permalink
Merge pull request #1672 from CEED/jeremy/work-vector
Browse files Browse the repository at this point in the history
ceed - add Ceed[Get,Restore]WorkVector
  • Loading branch information
jeremylt authored Sep 24, 2024
2 parents 3156c35 + 73501bf commit f009e52
Show file tree
Hide file tree
Showing 4 changed files with 175 additions and 7 deletions.
23 changes: 16 additions & 7 deletions include/ceed-impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,14 @@ typedef struct {
Ceed delegate;
} ObjDelegate;

// Work vector tracking
typedef struct CeedWorkVectors_private *CeedWorkVectors;
struct CeedWorkVectors_private {
CeedInt num_vecs, max_vecs;
bool *is_in_use;
CeedVector *vecs;
};

struct Ceed_private {
const char *resource;
Ceed delegate;
Expand Down Expand Up @@ -113,13 +121,14 @@ struct Ceed_private {
int (*OperatorCreate)(CeedOperator);
int (*OperatorCreateAtPoints)(CeedOperator);
int (*CompositeOperatorCreate)(CeedOperator);
int ref_count;
void *data;
bool is_debug;
bool has_valid_op_fallback_resource;
bool is_deterministic;
char err_msg[CEED_MAX_RESOURCE_LEN];
FOffset *f_offsets;
int ref_count;
void *data;
bool is_debug;
bool has_valid_op_fallback_resource;
bool is_deterministic;
char err_msg[CEED_MAX_RESOURCE_LEN];
FOffset *f_offsets;
CeedWorkVectors work_vectors;
};

struct CeedVector_private {
Expand Down
2 changes: 2 additions & 0 deletions include/ceed/backend.h
Original file line number Diff line number Diff line change
Expand Up @@ -252,6 +252,8 @@ CEED_EXTERN int CeedSetBackendFunctionImpl(Ceed ceed, const char *type, void *ob
CEED_EXTERN int CeedGetData(Ceed ceed, void *data);
CEED_EXTERN int CeedSetData(Ceed ceed, void *data);
CEED_EXTERN int CeedReference(Ceed ceed);
CEED_EXTERN int CeedGetWorkVector(Ceed ceed, CeedSize len, CeedVector *vec);
CEED_EXTERN int CeedRestoreWorkVector(Ceed ceed, CeedVector *vec);

CEED_EXTERN int CeedVectorHasValidArray(CeedVector vec, bool *has_valid_array);
CEED_EXTERN int CeedVectorHasBorrowedArrayOfType(CeedVector vec, CeedMemType mem_type, bool *has_borrowed_array_of_type);
Expand Down
113 changes: 113 additions & 0 deletions interface/ceed.c
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,43 @@ int CeedRegisterImpl(const char *prefix, int (*init)(const char *, Ceed), unsign
return CEED_ERROR_SUCCESS;
}

/**
@brief Create a work vector space for a `ceed`
@param[in,out] ceed `Ceed` to create work vector space for
@return An error code: 0 - success, otherwise - failure
@ref Developer
**/
static int CeedWorkVectorsCreate(Ceed ceed) {
CeedCall(CeedCalloc(1, &ceed->work_vectors));
return CEED_ERROR_SUCCESS;
}

/**
@brief Destroy a work vector space for a `ceed`
@param[in,out] ceed `Ceed` to destroy work vector space for
@return An error code: 0 - success, otherwise - failure
@ref Developer
**/
static int CeedWorkVectorsDestroy(Ceed ceed) {
if (!ceed->work_vectors) return CEED_ERROR_SUCCESS;
for (CeedSize i = 0; i < ceed->work_vectors->num_vecs; i++) {
CeedCheck(!ceed->work_vectors->is_in_use[i], ceed, CEED_ERROR_ACCESS, "Work vector %" CeedSize_FMT " checked out but not returned");
ceed->ref_count += 2; // Note: increase ref_count to prevent Ceed destructor from triggering again
CeedCall(CeedVectorDestroy(&ceed->work_vectors->vecs[i]));
ceed->ref_count -= 1; // Note: restore ref_count
}
CeedCall(CeedFree(&ceed->work_vectors->is_in_use));
CeedCall(CeedFree(&ceed->work_vectors->vecs));
CeedCall(CeedFree(&ceed->work_vectors));
return CEED_ERROR_SUCCESS;
}

/// @}

/// ----------------------------------------------------------------------------
Expand Down Expand Up @@ -751,6 +788,81 @@ int CeedReference(Ceed ceed) {
return CEED_ERROR_SUCCESS;
}

/**
@brief Get a `CeedVector` for scratch work from a `Ceed` context.
Note: This vector must be restored with @ref CeedRestoreWorkVector().
@param[in] ceed `Ceed` context
@param[in] len Minimum length of work vector
@param[out] vec Address of the variable where `CeedVector` will be stored
@return An error code: 0 - success, otherwise - failure
@ref Backend
**/
int CeedGetWorkVector(Ceed ceed, CeedSize len, CeedVector *vec) {
CeedInt i = 0;

if (!ceed->work_vectors) CeedCall(CeedWorkVectorsCreate(ceed));

// Search for big enough work vector
for (i = 0; i < ceed->work_vectors->num_vecs; i++) {
if (!ceed->work_vectors->is_in_use[i]) {
CeedSize work_len;

CeedCall(CeedVectorGetLength(ceed->work_vectors->vecs[i], &work_len));
if (work_len >= len) break;
}
}
// Long enough vector was not found
if (i == ceed->work_vectors->num_vecs) {
if (ceed->work_vectors->max_vecs == 0) {
ceed->work_vectors->max_vecs = 1;
CeedCall(CeedCalloc(ceed->work_vectors->max_vecs, &ceed->work_vectors->vecs));
CeedCall(CeedCalloc(ceed->work_vectors->max_vecs, &ceed->work_vectors->is_in_use));
} else if (ceed->work_vectors->max_vecs == i) {
ceed->work_vectors->max_vecs *= 2;
CeedCall(CeedRealloc(ceed->work_vectors->max_vecs, &ceed->work_vectors->vecs));
CeedCall(CeedRealloc(ceed->work_vectors->max_vecs, &ceed->work_vectors->is_in_use));
}
ceed->work_vectors->num_vecs++;
CeedCallBackend(CeedVectorCreate(ceed, len, &ceed->work_vectors->vecs[i]));
ceed->ref_count--; // Note: ref_count manipulation to prevent a ref-loop
}
// Return pointer to work vector
ceed->work_vectors->is_in_use[i] = true;
*vec = NULL;
CeedCall(CeedVectorReferenceCopy(ceed->work_vectors->vecs[i], vec));
ceed->ref_count++; // Note: bump ref_count to account for external access
return CEED_ERROR_SUCCESS;
}

/**
@brief Restore a `CeedVector` for scratch work from a `Ceed` context from @ref CeedGetWorkVector()
@param[in] ceed `Ceed` context
@param[out] vec `CeedVector` to restore
@return An error code: 0 - success, otherwise - failure
@ref Backend
**/
int CeedRestoreWorkVector(Ceed ceed, CeedVector *vec) {
for (CeedInt i = 0; i < ceed->work_vectors->num_vecs; i++) {
if (*vec == ceed->work_vectors->vecs[i]) {
CeedCheck(ceed->work_vectors->is_in_use[i], ceed, CEED_ERROR_ACCESS, "Work vector %" CeedSize_FMT " was not checked out but is being returned");
CeedCall(CeedVectorDestroy(vec));
ceed->work_vectors->is_in_use[i] = false;
ceed->ref_count--; // Note: reduce ref_count again to prevent a ref-loop
return CEED_ERROR_SUCCESS;
}
}
// LCOV_EXCL_START
return CeedError(ceed, CEED_ERROR_MAJOR, "vec was not checked out via CeedGetWorkVector()");
// LCOV_EXCL_STOP
}

/// @}

/// ----------------------------------------------------------------------------
Expand Down Expand Up @@ -1200,6 +1312,7 @@ int CeedDestroy(Ceed *ceed) {
CeedCall(CeedFree(&(*ceed)->resource));
CeedCall(CeedDestroy(&(*ceed)->op_fallback_ceed));
CeedCall(CeedFree(&(*ceed)->op_fallback_resource));
CeedCall(CeedWorkVectorsDestroy(*ceed));
CeedCall(CeedFree(ceed));
return CEED_ERROR_SUCCESS;
}
Expand Down
44 changes: 44 additions & 0 deletions tests/t130-vector.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
/// @file
/// Test getting and restoring work vectors
/// \test Test getting and restoring work vectors

#include <ceed.h>
#include <ceed/backend.h>
#include <stdio.h>

int main(int argc, char **argv) {
Ceed ceed;

CeedInit(argv[1], &ceed);

// Check for getting the same work vector back
{
CeedVector x, y;

CeedGetWorkVector(ceed, 20, &x);
// Do not do this!
CeedVector x_copy = x;

CeedRestoreWorkVector(ceed, &x);
CeedGetWorkVector(ceed, 20, &y);
if (y != x_copy) printf("failed to return same work vector");
CeedRestoreWorkVector(ceed, &y);
}

// Check for getting a new work vector back
{
CeedVector x, y;

CeedGetWorkVector(ceed, 20, &x);
// Do not do this!
CeedVector x_copy = x;

CeedRestoreWorkVector(ceed, &x);
CeedGetWorkVector(ceed, 30, &y);
if (y == x_copy) printf("failed to return new work vector");
CeedRestoreWorkVector(ceed, &y);
}

CeedDestroy(&ceed);
return 0;
}

0 comments on commit f009e52

Please sign in to comment.