You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
232 lines
7.2 KiB
232 lines
7.2 KiB
2 years ago
|
/*
|
||
|
* Copyright(C) 1999-2020, 2022 National Technology & Engineering Solutions
|
||
|
* of Sandia, LLC (NTESS). Under the terms of Contract DE-NA0003525 with
|
||
|
* NTESS, the U.S. Government retains certain rights in this software.
|
||
|
*
|
||
|
* See packages/seacas/LICENSE for details
|
||
|
*/
|
||
|
#include "fmt/ostream.h"
|
||
|
#include <cstdarg> // for va_arg, va_end, va_list, etc
|
||
|
#include <cstddef> // for size_t
|
||
|
#include <cstdio> // for nullptr, stderr
|
||
|
#include <cstdlib> // for exit, free, malloc
|
||
|
|
||
|
/*#include "rf_allo.h"*/
|
||
|
|
||
|
#ifdef DEBUG
|
||
|
extern int Proc;
|
||
|
#endif
|
||
|
|
||
|
static void *smalloc(size_t n, char *filename, int lineno);
|
||
|
|
||
|
/******************************************************************************
|
||
|
*
|
||
|
* Dynamic Allocation of Multidimensional Arrays
|
||
|
*-----------------------------------------------------------------------------
|
||
|
*
|
||
|
* Example Usage:
|
||
|
*
|
||
|
* typedef struct
|
||
|
* { int bus1;
|
||
|
* int bus2;
|
||
|
* int dest;
|
||
|
* } POINT;
|
||
|
*
|
||
|
* POINT **points, corner;
|
||
|
*
|
||
|
* points = (POINT **) array_alloc (2, x, y, sizeof(POINT));
|
||
|
* ^ ^ ^
|
||
|
* | | |
|
||
|
* number of dimensions--+ | |
|
||
|
* | |
|
||
|
* first dimension max----+ |
|
||
|
* |
|
||
|
* second dimension max------+
|
||
|
*
|
||
|
* (points may be now be used as if it were declared
|
||
|
* POINT points[x][y])
|
||
|
*
|
||
|
* This particular version is limited to dimensions of 3 or less.
|
||
|
*
|
||
|
* corner = points[2][3]; (refer to the structure as you would any array)
|
||
|
*
|
||
|
* free (points); (frees the entire structure in one fell swoop)
|
||
|
*
|
||
|
*****************************************************************************/
|
||
|
/******************************************************************************
|
||
|
* The following section is a commented section containing
|
||
|
* an example main code:
|
||
|
*******************************************************************************
|
||
|
*double *array_alloc();
|
||
|
*main()
|
||
|
*{
|
||
|
* int ***temp;
|
||
|
* int *temp2;
|
||
|
* int i, j, k;
|
||
|
* int il, jl, kl;
|
||
|
*
|
||
|
* malloc_debug(2);
|
||
|
* il = 2;
|
||
|
* jl = 3;
|
||
|
* kl = 3;
|
||
|
* temp = (int ***) array_alloc(3,il,jl,kl,sizeof(int));
|
||
|
* for (i=0; i<il; i++) {
|
||
|
* for (j=0; j<jl; j++) {
|
||
|
* for (k=0; k<kl; k++) temp[i][j][k] = 1;
|
||
|
* }
|
||
|
* }
|
||
|
*
|
||
|
* temp2 = (int *) malloc(10*sizeof(int));
|
||
|
* for (i=0; i<10; i++) temp2[i] = 0;
|
||
|
*
|
||
|
* for (i=0; i<il; i++) {
|
||
|
* for (j=0; j<jl; j++) {
|
||
|
* for (k=0; k<kl; k++) fmt::print(" {}\n", temp[i][j][k]);
|
||
|
* }
|
||
|
* }
|
||
|
* malloc_verify();
|
||
|
*}
|
||
|
******************************************************************************/
|
||
|
|
||
|
/*****************************************************************************/
|
||
|
/*****************************************************************************/
|
||
|
|
||
|
void *array_alloc(const char *file, int lineno, int numdim, ...)
|
||
|
{
|
||
|
struct dim
|
||
|
{
|
||
|
size_t index; /* Number of elements in the dimension */
|
||
|
size_t total; /* Total number of elements */
|
||
|
size_t size; /* Size of a single element in bytes */
|
||
|
size_t off; /* offset from beginning of array */
|
||
|
} dim[3]; /* Info about each dimension */
|
||
|
size_t total; /* Total size of the array */
|
||
|
void *dfield; /* ptr to avoid lint complaints */
|
||
|
char *field; /* The multi-dimensional array */
|
||
|
char **ptr; /* Pointer offset */
|
||
|
char *data; /* Data offset */
|
||
|
va_list va; /* Current pointer in the argument list */
|
||
|
|
||
|
va_start(va, numdim);
|
||
|
|
||
|
if (numdim <= 0) {
|
||
|
fmt::print(stderr, "{} ({}: {}) ERROR: number of dimensions, {}, is <=0\n", __func__, file,
|
||
|
lineno, numdim);
|
||
|
exit(1);
|
||
|
}
|
||
|
else if (numdim > 3) {
|
||
|
fmt::print(stderr, "{} ({}: {}) ERROR: number of dimensions, {}, is > 3\n", __func__, file,
|
||
|
lineno, numdim);
|
||
|
exit(1);
|
||
|
}
|
||
|
|
||
|
dim[0].index = va_arg(va, size_t);
|
||
|
|
||
|
if (dim[0].index == 0) {
|
||
|
#ifdef DEBUG
|
||
|
fmt::print(stderr,
|
||
|
"WARNING, {} ({}: {}) called with first "
|
||
|
"dimension == 0; will return nullptr\n",
|
||
|
__func__, file, lineno);
|
||
|
#endif
|
||
|
va_end(va);
|
||
|
return (nullptr);
|
||
|
}
|
||
|
|
||
|
dim[0].total = dim[0].index;
|
||
|
dim[0].size = sizeof(void *);
|
||
|
dim[0].off = 0;
|
||
|
for (int i = 1; i < numdim; i++) {
|
||
|
dim[i].index = va_arg(va, size_t);
|
||
|
if (dim[i].index == 0) {
|
||
|
fmt::print(stderr,
|
||
|
"WARNING: {} ({}: {}) called with dimension {} == 0, "
|
||
|
" will return nullptr\n",
|
||
|
__func__, file, lineno, i + 1);
|
||
|
va_end(va);
|
||
|
return (nullptr);
|
||
|
}
|
||
|
dim[i].total = dim[i - 1].total * dim[i].index;
|
||
|
dim[i].size = sizeof(void *);
|
||
|
dim[i].off = dim[i - 1].off + dim[i - 1].total * dim[i - 1].size;
|
||
|
}
|
||
|
dim[numdim - 1].size = va_arg(va, size_t);
|
||
|
va_end(va);
|
||
|
|
||
|
/* Round up the last offset value so data is properly aligned. */
|
||
|
dim[numdim - 1].off = dim[numdim - 1].size *
|
||
|
((dim[numdim - 1].off + dim[numdim - 1].size - 1) / dim[numdim - 1].size);
|
||
|
|
||
|
total = dim[numdim - 1].off + dim[numdim - 1].total * dim[numdim - 1].size;
|
||
|
|
||
|
dfield = smalloc(total, const_cast<char *>(file), lineno);
|
||
|
field = reinterpret_cast<char *>(dfield);
|
||
|
|
||
|
for (int i = 0; i < numdim - 1; i++) {
|
||
|
ptr = reinterpret_cast<char **>(field + dim[i].off);
|
||
|
data = (field + dim[i + 1].off);
|
||
|
for (size_t j = 0; j < dim[i].total; j++) {
|
||
|
ptr[j] = data + j * dim[i + 1].size * dim[i + 1].index;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (dfield);
|
||
|
}
|
||
|
|
||
|
/* Safe version of malloc. Does not initialize memory .*/
|
||
|
|
||
|
/* Modified by Scott Hutchinson (1421) 20 January 1993 */
|
||
|
|
||
|
/*****************************************************************************/
|
||
|
/*****************************************************************************/
|
||
|
/*****************************************************************************/
|
||
|
|
||
|
static void *smalloc(size_t n, char *filename, int lineno)
|
||
|
|
||
|
{
|
||
|
void *pntr = nullptr; /* return value */
|
||
|
|
||
|
if (n != 0) {
|
||
|
pntr = malloc(n);
|
||
|
}
|
||
|
|
||
|
if (pntr == nullptr && n != 0) {
|
||
|
fmt::print(stderr,
|
||
|
"{} (from {},{}) Out of space - number of bytes "
|
||
|
"requested = {}\n",
|
||
|
__func__, filename, lineno, fmt::group_digits(n));
|
||
|
exit(0);
|
||
|
}
|
||
|
|
||
|
return pntr;
|
||
|
}
|
||
|
|
||
|
/*****************************************************************************/
|
||
|
/*****************************************************************************/
|
||
|
/*****************************************************************************/
|
||
|
|
||
|
void safe_free(void **ptr)
|
||
|
{
|
||
|
/*
|
||
|
* This version of free calls the system's free function
|
||
|
* with maximum error checking. It also doesn't call free if ptr is
|
||
|
* the nullptr pointer.
|
||
|
*/
|
||
|
|
||
|
if (*ptr != nullptr) {
|
||
|
|
||
|
free(*ptr);
|
||
|
|
||
|
/*
|
||
|
* Set the value of ptr to nullptr, so that further references
|
||
|
* to it will be flagged.
|
||
|
*/
|
||
|
|
||
|
*ptr = nullptr;
|
||
|
}
|
||
|
} /* safe_free */
|
||
|
|
||
|
/*****************************************************************************/
|
||
|
/* END of rf_allo.c */
|
||
|
/*****************************************************************************/
|