2007-07-27 05:35:13 -04:00
|
|
|
/** These routines represent handling of struct memory_list.
|
|
|
|
* @file */
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "elinks.h"
|
|
|
|
|
|
|
|
#include "util/error.h"
|
|
|
|
#include "util/memlist.h"
|
|
|
|
#include "util/memory.h"
|
|
|
|
|
|
|
|
|
2007-07-27 05:35:13 -04:00
|
|
|
/**
|
2007-07-31 06:38:00 -04:00
|
|
|
* @struct memory_list
|
2005-09-15 09:58:31 -04:00
|
|
|
* memory_list is used to track information about all allocated memory
|
|
|
|
* belonging to something. Then we can free it when we won't need it
|
|
|
|
* anymore, but the one who allocated it won't be able to get control
|
|
|
|
* back in order to free it himself.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define ML_SIZE(n) (sizeof(struct memory_list) + (n) * sizeof(void *))
|
|
|
|
|
2007-07-27 05:35:13 -04:00
|
|
|
/** Create a memory list. If @a p is NULL or allocation fails, it will
|
|
|
|
* return NULL.
|
2007-07-27 07:14:00 -04:00
|
|
|
* It always stops at first NULL element.
|
|
|
|
* @relates memory_list */
|
2007-03-11 06:22:02 -04:00
|
|
|
#if defined(DEBUG_MEMLIST) && defined(HAVE_VARIADIC_MACROS)
|
2005-09-15 09:58:31 -04:00
|
|
|
struct memory_list *
|
2021-01-02 10:20:27 -05:00
|
|
|
debug_getml(char *file, int line, void *p, ...)
|
2005-09-15 09:58:31 -04:00
|
|
|
#else
|
|
|
|
struct memory_list *
|
|
|
|
getml(void *p, ...)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
struct memory_list *ml;
|
|
|
|
va_list ap;
|
|
|
|
void *q;
|
|
|
|
int n = 1;
|
|
|
|
|
|
|
|
/* If first element is NULL, there's no need to allocate memory, so
|
|
|
|
* just return. */
|
|
|
|
if (!p) return NULL;
|
|
|
|
|
|
|
|
/* How many elements ? */
|
|
|
|
va_start(ap, p);
|
|
|
|
while ((q = va_arg(ap, void *))) n++;
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
/* Allocate space for memory list. */
|
2022-01-16 13:09:27 -05:00
|
|
|
ml = (struct memory_list *)mem_alloc(ML_SIZE(n));
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!ml) return NULL;
|
|
|
|
|
|
|
|
/* First element. */
|
|
|
|
ml->n = 1;
|
|
|
|
ml->p[0] = p;
|
|
|
|
|
|
|
|
/* Next ones. */
|
|
|
|
va_start(ap, p);
|
|
|
|
while ((q = va_arg(ap, void *))) ml->p[ml->n++] = q;
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
/* The end. */
|
|
|
|
return ml;
|
|
|
|
}
|
|
|
|
|
2007-07-27 05:35:13 -04:00
|
|
|
/** Add elements to a memory list.
|
2005-09-15 09:58:31 -04:00
|
|
|
* If memory list exists, it enlarges it, else it creates it.
|
|
|
|
* if there's no elements or first element is NULL, it does nothing.
|
2007-07-27 07:14:00 -04:00
|
|
|
* It always stops at first NULL element.
|
|
|
|
* @relates memory_list */
|
2007-03-11 06:22:02 -04:00
|
|
|
#if defined(DEBUG_MEMLIST) && defined(HAVE_VARIADIC_MACROS)
|
2005-09-15 09:58:31 -04:00
|
|
|
void
|
2021-01-02 10:20:27 -05:00
|
|
|
debug_add_to_ml(char *file, int line, struct memory_list **ml, ...)
|
2005-09-15 09:58:31 -04:00
|
|
|
#else
|
|
|
|
void
|
|
|
|
add_to_ml(struct memory_list **ml, ...)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
void *q;
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
/* How many new elements ? */
|
|
|
|
va_start(ap, ml);
|
|
|
|
while ((q = va_arg(ap, void *))) n++;
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
/* None, so just return. */
|
|
|
|
if (!n) {
|
|
|
|
#ifdef DEBUG_MEMLIST
|
2007-03-11 06:22:02 -04:00
|
|
|
#ifdef HAVE_VARIADIC_MACROS
|
2005-09-15 09:58:31 -04:00
|
|
|
errline = line, errfile = file;
|
2007-03-11 06:22:02 -04:00
|
|
|
#else
|
|
|
|
errline = 0, errfile = "?";
|
2007-04-26 09:02:04 -04:00
|
|
|
#endif
|
2005-09-15 09:58:31 -04:00
|
|
|
elinks_error("add_to_ml(%p, NULL, ...)", ml);
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!*ml) {
|
|
|
|
/* If getml() wasn't called before or returned NULL,
|
|
|
|
* then we create it. */
|
2022-01-16 13:09:27 -05:00
|
|
|
*ml = (struct memory_list *)mem_alloc(ML_SIZE(n));
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!*ml) return;
|
|
|
|
|
|
|
|
(*ml)->n = 0;
|
|
|
|
} else {
|
|
|
|
/* Enlarge existing ml. */
|
|
|
|
struct memory_list *nml;
|
|
|
|
|
2022-01-16 13:38:30 -05:00
|
|
|
nml = (struct memory_list *)mem_realloc(*ml, ML_SIZE(n + (*ml)->n));
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!nml) return;
|
|
|
|
|
|
|
|
*ml = nml;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set ml with new elements and update count. */
|
|
|
|
va_start(ap, ml);
|
|
|
|
while ((q = va_arg(ap, void *))) (*ml)->p[(*ml)->n++] = q;
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
|
2007-07-27 07:14:00 -04:00
|
|
|
/** @relates memory_list */
|
2005-09-15 09:58:31 -04:00
|
|
|
#ifdef DEBUG_MEMLIST
|
|
|
|
void
|
2021-01-02 10:20:27 -05:00
|
|
|
debug_add_one_to_ml(char *file, int line, struct memory_list **ml, void *p)
|
2005-09-15 09:58:31 -04:00
|
|
|
#else
|
|
|
|
void
|
|
|
|
add_one_to_ml(struct memory_list **ml, void *p)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
/* None, so just return. */
|
|
|
|
if (!p) {
|
|
|
|
#ifdef DEBUG_MEMLIST
|
|
|
|
errline = line, errfile = file;
|
|
|
|
elinks_error("add_one_to_ml(%p, NULL)", ml);
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!*ml) {
|
|
|
|
/* If getml() wasn't called before or returned NULL,
|
|
|
|
* then we create it. */
|
2022-01-16 13:09:27 -05:00
|
|
|
*ml = (struct memory_list *)mem_alloc(ML_SIZE(1));
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!*ml) return;
|
|
|
|
|
|
|
|
(*ml)->n = 0;
|
|
|
|
} else {
|
|
|
|
/* Enlarge existing ml. */
|
|
|
|
struct memory_list *nml;
|
|
|
|
|
2022-01-16 13:38:30 -05:00
|
|
|
nml = (struct memory_list *)mem_realloc(*ml, ML_SIZE(1 + (*ml)->n));
|
2005-09-15 09:58:31 -04:00
|
|
|
if (!nml) return;
|
|
|
|
|
|
|
|
*ml = nml;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set ml with new element and update count. */
|
|
|
|
(*ml)->p[(*ml)->n++] = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-27 05:35:13 -04:00
|
|
|
/** Free elements and memory list.
|
2007-07-27 07:14:00 -04:00
|
|
|
* It returns safely if passed a NULL pointer.
|
|
|
|
* @relates memory_list */
|
2005-09-15 09:58:31 -04:00
|
|
|
void
|
|
|
|
freeml(struct memory_list *ml)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!ml) return;
|
|
|
|
|
|
|
|
for (i = 0; i < ml->n; i++)
|
|
|
|
mem_free(ml->p[i]);
|
|
|
|
|
|
|
|
mem_free(ml);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef ML_SIZE
|