1
0
mirror of https://github.com/rkd77/elinks.git synced 2025-01-03 14:57:44 -05:00
elinks/src/util/memory.c
2022-01-16 21:13:17 +01:00

189 lines
3.3 KiB
C

/** Memory allocation manager
* @file */
#ifndef _GNU_SOURCE
#define _GNU_SOURCE /* MREMAP_MAYMOVE */
#endif
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <stdlib.h>
#include <sys/types.h>
#ifdef HAVE_MMAP
#include <sys/mman.h>
#endif
#include <unistd.h>
#include "elinks.h"
#include "util/error.h"
#include "util/memory.h"
#if !defined(DEBUG_MEMLEAK) && !defined(CONFIG_FASTMEM)
static int alloc_try = 0;
static int
patience(const char *of)
{
++alloc_try;
if (alloc_try < ALLOC_MAXTRIES) {
ERROR("Out of memory (%s returned NULL): retry #%d/%d, "
"I still exercise my patience and retry tirelessly.",
of, alloc_try, ALLOC_MAXTRIES);
sleep(ALLOC_DELAY);
return alloc_try;
}
#ifdef CRASH_IF_ALLOC_MAXTRIES
INTERNAL("Out of memory (%s returned NULL) after %d tries, "
"I give up. See ya on the other side.",
of, alloc_try);
#else
ERROR("Out of memory (%s returned NULL) after %d tries, "
"I give up and try to continue. Pray for me, please.",
of, alloc_try);
#endif
alloc_try = 0;
return 0;
}
void *
mem_alloc(size_t size)
{
if (size)
do {
void *p = malloc(size);
if (p) return p;
} while (patience("malloc"));
return NULL;
}
void *
mem_calloc(size_t count, size_t eltsize)
{
if (eltsize && count)
do {
void *p = calloc(count, eltsize);
if (p) return p;
} while (patience("calloc"));
return NULL;
}
void
mem_free(void *p)
{
if (!p) {
INTERNAL("mem_free(NULL)");
return;
}
free(p);
}
void *
mem_realloc(void *p, size_t size)
{
if (!p) return mem_alloc(size);
if (size)
do {
void *p2 = realloc(p, size);
if (p2) return p2;
} while (patience("realloc"));
else
mem_free(p);
return NULL;
}
#endif
/* TODO: Leak detector and the usual protection gear? patience()?
*
* We could just alias mem_mmap_* to mem_debug_* #if DEBUG_MEMLEAK, *WHEN* we are
* confident that the mmap() code is really bugless ;-). --pasky */
#ifdef HAVE_MMAP
static int page_size;
/** Round up to a full page.
* This tries to prevent useless reallocations, especially since they
* are quite expensive in the mremap()-less case. */
static size_t
round_size(size_t size)
{
#ifdef HAVE_SC_PAGE_SIZE
if (!page_size) page_size = sysconf(_SC_PAGE_SIZE);
#endif
if (page_size <= 0) page_size = 1;
return (size / page_size + 1) * page_size;
}
/** Some systems may not have MAP_ANON but MAP_ANONYMOUS instead. */
#if defined(MAP_ANONYMOUS) && !defined(MAP_ANON)
#define MAP_ANON MAP_ANONYMOUS
#endif
void *
mem_mmap_alloc(size_t size)
{
if (size) {
void *p = mmap(NULL, round_size(size), PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANON, -1, 0);
if (p != MAP_FAILED)
return p;
}
return NULL;
}
void
mem_mmap_free(void *p, size_t size)
{
if (!p) {
INTERNAL("mem_mmap_free(NULL)");
return;
}
munmap(p, round_size(size));
}
void *
mem_mmap_realloc(void *p, size_t old_size, size_t new_size)
{
if (!p) return mem_mmap_alloc(new_size);
if (round_size(old_size) == round_size(new_size))
return p;
if (new_size) {
#ifdef HAVE_MREMAP
void *p2 = mremap(p, round_size(old_size), round_size(new_size), MREMAP_MAYMOVE);
if (p2 != MAP_FAILED)
return p2;
#else
void *p2 = mem_mmap_alloc(new_size);
if (p2) {
memcpy(p2, p, MIN(old_size, new_size));
mem_mmap_free(p, old_size);
return p2;
}
#endif
} else {
mem_mmap_free(p, old_size);
}
return NULL;
}
#endif