-
Mathieu Faverge authoredMathieu Faverge authored
descriptor_ipiv.c 7.39 KiB
/**
*
* @file descriptor_ipiv.c
*
* @copyright 2022-2025 Bordeaux INP, CNRS (LaBRI UMR 5800), Inria,
* Univ. Bordeaux. All rights reserved.
*
***
*
* @brief Chameleon descriptors routines
*
* @version 1.3.0
* @author Mathieu Faverge
* @author Matthieu Kuhn
* @author Alycia Lisito
* @author Florent Pruvost
* @date 2024-08-29
*
***
*
* @defgroup Descriptor
* @brief Group descriptor routines exposed to users to manipulate IPIV data structures
*
*/
#define _GNU_SOURCE 1
#include <stdlib.h>
#include <stdio.h>
#include <assert.h>
#include <string.h>
#include "control/common.h"
#include "control/descriptor.h"
#include "chameleon/runtime.h"
/**
******************************************************************************
*
* @ingroup Descriptor
*
* @brief Internal function to create tiled descriptor associated to a pivot array.
*
******************************************************************************
*
* @param[in,out] ipiv
* The pointer to the ipiv descriptor to initialize.
*
* @param[in] desc
* The tile descriptor for which an associated ipiv descriptor must be generated.
*
* @param[in] data
* The pointer to the original vector where to store the pivot values.
*
******************************************************************************
*
* @return CHAMELEON_SUCCESS on success, CHAMELEON_ERR_NOT_INITIALIZED otherwise.
*
*/
int chameleon_ipiv_init( CHAM_ipiv_t *ipiv, const CHAM_desc_t *desc, void *data )
{
CHAM_context_t *chamctxt;
int rc = CHAMELEON_SUCCESS;
memset( ipiv, 0, sizeof(CHAM_ipiv_t) );
chamctxt = chameleon_context_self();
if (chamctxt == NULL) {
chameleon_error("CHAMELEON_Desc_Create", "CHAMELEON not initialized");
return CHAMELEON_ERR_NOT_INITIALIZED;
}
ipiv->desc = desc;
ipiv->data = data;
ipiv->i = 0;
ipiv->m = chameleon_min( desc->m, desc->n );
ipiv->mb = desc->mb;
ipiv->mt = chameleon_ceil( ipiv->m, ipiv->mb );
/* Create runtime specific structure like registering data */
RUNTIME_ipiv_create( ipiv, desc );
return rc;
}
/**
******************************************************************************
*
* @ingroup Descriptor
*
* @brief Internal function to destroy a tiled descriptor associated to a pivot array.
*
******************************************************************************
*
* @param[in,out] ipiv
* The pointer to the ipiv descriptor to destroy.
*
*/
void chameleon_ipiv_destroy( CHAM_ipiv_t *ipiv,
const CHAM_desc_t *desc )
{
RUNTIME_ipiv_destroy( ipiv, desc );
}
/**
*****************************************************************************
*
* @ingroup Descriptor
*
* @brief Create a tiled ipiv descriptor associated to a given matrix.
*
******************************************************************************
*
* @param[in,out] ipiv
* The pointer to the ipiv descriptor to initialize.
*
* @param[in] desc
* The tile descriptor for which an associated ipiv descriptor must be generated.
*
* @param[in] data
* The pointer to the original vector where to store the pivot values.
*
******************************************************************************
*
* @retval CHAMELEON_SUCCESS on successful exit
* @retval CHAMELEON_ERR_NOT_INITIALIZED if failed to initialize the descriptor.
* @retval CHAMELEON_ERR_OUT_OF_RESOURCES if failed to allocated some ressources.
*
*/
int CHAMELEON_Ipiv_Create( CHAM_ipiv_t **ipivptr, const CHAM_desc_t *desc, void *data )
{
CHAM_context_t *chamctxt;
CHAM_ipiv_t *ipiv;
chamctxt = chameleon_context_self();
if (chamctxt == NULL) {
chameleon_error("CHAMELEON_Ipiv_Create", "CHAMELEON not initialized");
return CHAMELEON_ERR_NOT_INITIALIZED;
}
/* Allocate memory and initialize the ipivriptor */
ipiv = (CHAM_ipiv_t*)malloc(sizeof(CHAM_ipiv_t));
if (ipiv == NULL) {
chameleon_error("CHAMELEON_Ipiv_Create", "malloc() failed");
return CHAMELEON_ERR_OUT_OF_RESOURCES;
}
chameleon_ipiv_init( ipiv, desc, data );
*ipivptr = ipiv;
return CHAMELEON_SUCCESS;
}
/**
*****************************************************************************
*
* @ingroup Descriptor
*
* @brief Destroys an ipiv tile descriptor.
*
******************************************************************************
*
* @param[in] ipivptr
* The Ipiv tile descriptor to destroy.
*
******************************************************************************
*
* @retval CHAMELEON_SUCCESS successful exit
*
*/
int CHAMELEON_Ipiv_Destroy( CHAM_ipiv_t **ipivptr,
const CHAM_desc_t *desc )
{
CHAM_context_t *chamctxt;
CHAM_ipiv_t *ipiv;
chamctxt = chameleon_context_self();
if (chamctxt == NULL) {
chameleon_error("CHAMELEON_Ipiv_Destroy", "CHAMELEON not initialized");
return CHAMELEON_ERR_NOT_INITIALIZED;
}
if ((ipivptr == NULL) || (*ipivptr == NULL)) {
chameleon_error("CHAMELEON_Ipiv_Destroy", "attempting to destroy a NULL descriptor");
return CHAMELEON_ERR_UNALLOCATED;
}
ipiv = *ipivptr;
chameleon_ipiv_destroy( ipiv, desc );
free(ipiv);
*ipivptr = NULL;
return CHAMELEON_SUCCESS;
}
/**
*****************************************************************************
*
* @ingroup Descriptor
*
* @brief Flushes the data in the sequence when they won't be reused. This calls
* cleans up the distributed communication caches, and transfer the data back to
* the CPU.
*
******************************************************************************
*
* @param[in] ipiv
* ipiv vector descriptor.
*
* @param[in] sequence
* The seqeunce in which to submit the calls to flush the data.
*
******************************************************************************
*
* @retval CHAMELEON_SUCCESS successful exit
*
*/
int CHAMELEON_Ipiv_Flush( const CHAM_ipiv_t *ipiv,
const RUNTIME_sequence_t *sequence )
{
RUNTIME_ipiv_flush( sequence, ipiv );
return CHAMELEON_SUCCESS;
}
/**
*****************************************************************************
*
* @ingroup Descriptor
*
* @brief Gathers an IPIV tile descriptor in a single vector on the given root node.
*
******************************************************************************
*
* @param[in] ipivdesc
* the ipiv vector descriptor to gather.
*
* @param[in] ipiv
* The ipiv vector where to store the result. Allocated vector of size
* ipivdesc->m on root, not referenced on other nodes.
*
* @param[in] root
* root node on which to gather the data.
*
******************************************************************************
*
* @retval CHAMELEON_SUCCESS successful exit
*
*/
int CHAMELEON_Ipiv_Gather( CHAM_ipiv_t *ipivdesc, int *ipiv, int root )
{
CHAM_context_t *chamctxt = chameleon_context_self();
RUNTIME_sequence_t *sequence = NULL;
int status;
chameleon_sequence_create( chamctxt, &sequence );
/* Submit the tasks to collect the ipiv array on root */
RUNTIME_ipiv_gather( sequence, ipivdesc, ipiv, root );
/* Wait for the end */
chameleon_sequence_wait( chamctxt, sequence );
status = sequence->status;
chameleon_sequence_destroy( chamctxt, sequence );
return status;
}