1998-04-22 12:58:34 +00:00
|
|
|
/*
|
|
|
|
* BIRD Resource Manager
|
|
|
|
*
|
1999-03-29 19:35:47 +00:00
|
|
|
* (c) 1998--1999 Martin Mares <mj@ucw.cz>
|
2021-11-26 23:21:12 +00:00
|
|
|
* (c) 2021 Maria Matejka <mq@jmq.cz>
|
1998-04-22 12:58:34 +00:00
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _BIRD_RESOURCE_H_
|
|
|
|
#define _BIRD_RESOURCE_H_
|
|
|
|
|
2023-04-21 13:26:06 +00:00
|
|
|
#include "lib/locking.h"
|
|
|
|
#include "lib/tlists.h"
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2023-04-22 19:20:19 +00:00
|
|
|
#include <stdarg.h>
|
|
|
|
|
2021-11-26 23:21:12 +00:00
|
|
|
struct resmem {
|
|
|
|
size_t effective; /* Memory actually used for data storage */
|
|
|
|
size_t overhead; /* Overhead memory imposed by allocator strategies */
|
|
|
|
};
|
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
/* Resource */
|
|
|
|
|
2023-04-21 13:26:06 +00:00
|
|
|
#define TLIST_PREFIX resource
|
|
|
|
#define TLIST_TYPE struct resource
|
|
|
|
#define TLIST_ITEM n
|
|
|
|
#define TLIST_WANT_WALK
|
|
|
|
#define TLIST_WANT_ADD_TAIL
|
|
|
|
#define TLIST_WANT_UPDATE_NODE
|
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
typedef struct resource {
|
2023-04-21 13:26:06 +00:00
|
|
|
TLIST_DEFAULT_NODE; /* Inside resource pool */
|
|
|
|
const struct resclass *class; /* Resource class */
|
1998-04-22 12:58:34 +00:00
|
|
|
} resource;
|
|
|
|
|
2023-04-21 13:26:06 +00:00
|
|
|
#include "lib/tlists.h"
|
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
/* Resource class */
|
|
|
|
|
|
|
|
struct resclass {
|
1998-05-03 16:43:39 +00:00
|
|
|
char *name; /* Resource class name */
|
|
|
|
unsigned size; /* Standard size of single resource */
|
|
|
|
void (*free)(resource *); /* Freeing function */
|
2024-11-14 19:43:35 +00:00
|
|
|
void (*dump)(struct dump_request *, resource *); /* Dump to debug output */
|
2000-05-08 22:33:38 +00:00
|
|
|
resource *(*lookup)(resource *, unsigned long); /* Look up address (only for debugging) */
|
2021-11-26 23:21:12 +00:00
|
|
|
struct resmem (*memsize)(resource *); /* Return size of memory used by the resource, may be NULL */
|
1998-04-22 12:58:34 +00:00
|
|
|
};
|
|
|
|
|
2010-06-02 20:20:40 +00:00
|
|
|
/* Estimate of system allocator overhead per item, for memory consumtion stats */
|
2021-11-26 23:21:12 +00:00
|
|
|
#define ALLOC_OVERHEAD 16
|
2010-06-02 20:20:40 +00:00
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
/* Generic resource manipulation */
|
|
|
|
|
2022-07-18 10:33:00 +00:00
|
|
|
typedef struct pool {
|
|
|
|
resource r;
|
2023-04-21 13:26:06 +00:00
|
|
|
TLIST_LIST(resource) inside;
|
|
|
|
struct domain_generic *domain;
|
2022-07-18 10:33:00 +00:00
|
|
|
const char *name;
|
|
|
|
} pool;
|
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
|
1998-05-03 16:43:39 +00:00
|
|
|
void resource_init(void);
|
1998-04-22 12:58:34 +00:00
|
|
|
void rfree(void *); /* Free single resource */
|
2024-11-14 19:43:35 +00:00
|
|
|
|
|
|
|
struct dump_request;
|
|
|
|
void rdump(struct dump_request *, void *); /* Dump to debug output */
|
|
|
|
void resource_dump(struct dump_request *); /* Dump the root pool */
|
2021-11-26 23:21:12 +00:00
|
|
|
struct resmem rmemsize(void *res); /* Return size of memory used by the resource */
|
2000-05-08 22:33:38 +00:00
|
|
|
void rlookup(unsigned long); /* Look up address (only for debugging) */
|
2004-05-31 18:47:19 +00:00
|
|
|
void rmove(void *, pool *); /* Move to a different pool */
|
1998-04-22 12:58:34 +00:00
|
|
|
|
1998-05-03 16:43:39 +00:00
|
|
|
void *ralloc(pool *, struct resclass *);
|
|
|
|
|
2023-04-21 13:26:06 +00:00
|
|
|
pool *rp_new(pool *, struct domain_generic *, const char *); /* Create a new pool */
|
|
|
|
pool *rp_newf(pool *, struct domain_generic *, const char *, ...); /* Create a new pool with a formatted string as its name */
|
|
|
|
pool *rp_vnewf(pool *, struct domain_generic *, const char *, va_list); /* Create a new pool with a formatted string as its name */
|
|
|
|
void rp_free(pool *p); /* Free the whole pool */
|
2023-04-20 19:08:38 +00:00
|
|
|
|
1998-05-03 16:43:39 +00:00
|
|
|
extern pool root_pool;
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2023-04-28 21:48:03 +00:00
|
|
|
static inline pool *resource_parent(resource *r)
|
|
|
|
{ return SKIP_BACK(pool, inside, resource_enlisted(r)); }
|
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
/* Normal memory blocks */
|
|
|
|
|
2024-05-28 14:41:24 +00:00
|
|
|
void *mb_alloc(pool *, unsigned size) ALLOC_SIZE(2);
|
|
|
|
void *mb_allocz(pool *, unsigned size) ALLOC_SIZE(2);
|
|
|
|
void *mb_realloc(void *m, unsigned size) ALLOC_SIZE(2);
|
1998-05-03 16:43:39 +00:00
|
|
|
void mb_free(void *);
|
1998-04-22 12:58:34 +00:00
|
|
|
|
|
|
|
/* Memory pools with linear allocation */
|
|
|
|
|
1998-12-06 11:59:18 +00:00
|
|
|
typedef struct linpool linpool;
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2017-12-12 18:51:36 +00:00
|
|
|
typedef struct lp_state {
|
2024-05-08 16:55:01 +00:00
|
|
|
struct linpool *p;
|
2017-12-12 18:51:36 +00:00
|
|
|
void *current, *large;
|
2022-04-04 20:34:14 +00:00
|
|
|
uint total_large;
|
2017-12-12 18:51:36 +00:00
|
|
|
} lp_state;
|
|
|
|
|
2022-04-04 20:34:14 +00:00
|
|
|
linpool *lp_new(pool *);
|
2024-05-28 14:41:24 +00:00
|
|
|
void *lp_alloc(linpool *, unsigned size) ALLOC_SIZE(2); /* Aligned */
|
|
|
|
void *lp_allocu(linpool *, unsigned size) ALLOC_SIZE(2); /* Unaligned */
|
|
|
|
void *lp_allocz(linpool *, unsigned size) ALLOC_SIZE(2); /* With clear */
|
1999-03-29 19:35:47 +00:00
|
|
|
void lp_flush(linpool *); /* Free everything, but leave linpool */
|
2023-05-01 13:10:53 +00:00
|
|
|
lp_state *lp_save(linpool *m); /* Save state */
|
2017-12-12 18:51:36 +00:00
|
|
|
void lp_restore(linpool *m, lp_state *p); /* Restore state */
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2024-05-08 16:55:01 +00:00
|
|
|
static inline void lp_saved_cleanup(struct lp_state **lps)
|
|
|
|
{
|
|
|
|
if (*lps)
|
|
|
|
lp_restore((*lps)->p, (*lps));
|
|
|
|
}
|
2023-05-01 13:10:53 +00:00
|
|
|
|
2024-05-08 16:55:01 +00:00
|
|
|
#define LP_SAVED(m) for (CLEANUP(lp_saved_cleanup) struct lp_state *_lp_state = lp_save(m); _lp_state; lp_restore(m, _lp_state), _lp_state = NULL)
|
|
|
|
|
|
|
|
#define lp_new_default lp_new
|
2022-03-19 15:23:42 +00:00
|
|
|
|
2024-05-08 16:55:01 +00:00
|
|
|
/* Thread-local temporary linpools */
|
2022-03-02 09:35:21 +00:00
|
|
|
|
2024-05-08 16:55:01 +00:00
|
|
|
extern _Thread_local linpool *tmp_linpool;
|
2022-03-02 09:35:21 +00:00
|
|
|
#define tmp_alloc(sz) lp_alloc(tmp_linpool, sz)
|
|
|
|
#define tmp_allocu(sz) lp_allocu(tmp_linpool, sz)
|
|
|
|
#define tmp_allocz(sz) lp_allocz(tmp_linpool, sz)
|
2024-05-08 16:55:01 +00:00
|
|
|
#define TMP_SAVED LP_SAVED(tmp_linpool)
|
2022-03-02 09:35:21 +00:00
|
|
|
|
2024-05-08 16:55:01 +00:00
|
|
|
void tmp_init(pool *p);
|
2022-03-19 15:23:42 +00:00
|
|
|
void tmp_flush(void);
|
|
|
|
|
2022-03-02 09:35:21 +00:00
|
|
|
|
1998-04-22 12:58:34 +00:00
|
|
|
/* Slabs */
|
|
|
|
|
|
|
|
typedef struct slab slab;
|
|
|
|
|
|
|
|
slab *sl_new(pool *, unsigned size);
|
|
|
|
void *sl_alloc(slab *);
|
2020-11-24 01:32:13 +00:00
|
|
|
void *sl_allocz(slab *);
|
2022-04-04 18:31:14 +00:00
|
|
|
void sl_free(void *);
|
2023-04-20 19:08:38 +00:00
|
|
|
void sl_delete(slab *);
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2024-12-24 11:18:39 +00:00
|
|
|
/* A whole stonehenge of slabs */
|
|
|
|
|
|
|
|
typedef struct stonehenge stonehenge;
|
|
|
|
typedef struct sth_block {
|
|
|
|
void *block;
|
|
|
|
bool large;
|
|
|
|
} sth_block;
|
|
|
|
|
|
|
|
stonehenge *sth_new(pool *);
|
|
|
|
sth_block sth_alloc(stonehenge *, uint size);
|
|
|
|
sth_block sth_allocz(stonehenge *, uint size);
|
|
|
|
void sth_free(sth_block);
|
|
|
|
void sth_delete(stonehenge *);
|
|
|
|
|
1999-03-04 11:36:26 +00:00
|
|
|
/*
|
|
|
|
* Low-level memory allocation functions, please don't use
|
|
|
|
* outside resource manager and possibly sysdep code.
|
|
|
|
*/
|
1998-04-22 12:58:34 +00:00
|
|
|
|
2013-09-16 21:57:40 +00:00
|
|
|
void buffer_realloc(void **buf, unsigned *size, unsigned need, unsigned item_size);
|
|
|
|
|
2020-07-21 22:09:15 +00:00
|
|
|
/* Allocator of whole pages; for use in slabs and other high-level allocators. */
|
2022-07-15 12:57:02 +00:00
|
|
|
#define PAGE_HEAD(x) ((void *) (((uintptr_t) (x)) & ~(page_size-1)))
|
2022-03-09 08:10:44 +00:00
|
|
|
extern long page_size;
|
2022-09-09 18:57:59 +00:00
|
|
|
extern _Atomic int pages_kept;
|
2022-09-21 09:49:35 +00:00
|
|
|
extern _Atomic int pages_kept_locally;
|
2024-06-26 18:45:54 +00:00
|
|
|
extern _Atomic int pages_kept_cold;
|
|
|
|
extern _Atomic int pages_kept_cold_index;
|
2024-12-13 17:40:11 +00:00
|
|
|
extern _Atomic int pages_total;
|
2024-12-13 11:35:02 +00:00
|
|
|
extern _Atomic int alloc_locking_in_rcu;
|
2020-07-21 22:09:15 +00:00
|
|
|
void *alloc_page(void);
|
|
|
|
void free_page(void *);
|
2022-09-21 09:49:35 +00:00
|
|
|
void flush_local_pages(void);
|
2022-03-09 08:10:44 +00:00
|
|
|
|
|
|
|
void resource_sys_init(void);
|
2013-09-16 21:57:40 +00:00
|
|
|
|
1999-03-04 11:36:26 +00:00
|
|
|
#ifdef HAVE_LIBDMALLOC
|
|
|
|
/*
|
|
|
|
* The standard dmalloc macros tend to produce lots of namespace
|
2009-06-10 21:45:08 +00:00
|
|
|
* conflicts and we use only xmalloc, xrealloc and xfree, so we
|
|
|
|
* can define the stubs ourselves.
|
1999-03-04 11:36:26 +00:00
|
|
|
*/
|
|
|
|
#define DMALLOC_DISABLE
|
|
|
|
#include <dmalloc.h>
|
2019-07-02 08:45:53 +00:00
|
|
|
#define xmalloc(size) \
|
|
|
|
dmalloc_malloc(__FILE__, __LINE__, (size), DMALLOC_FUNC_MALLOC, 0, 1)
|
|
|
|
#define xrealloc(ptr, size) \
|
|
|
|
dmalloc_realloc(__FILE__, __LINE__, (ptr), (size), DMALLOC_FUNC_REALLOC, 1)
|
|
|
|
#define xfree(ptr) \
|
|
|
|
dmalloc_free(__FILE__, __LINE__, (ptr), DMALLOC_FUNC_FREE)
|
|
|
|
|
1999-03-04 11:36:26 +00:00
|
|
|
#else
|
2000-03-29 22:57:46 +00:00
|
|
|
/*
|
|
|
|
* Unfortunately, several libraries we might want to link to define
|
|
|
|
* their own xmalloc and we don't want to interfere with them, hence
|
|
|
|
* the renaming.
|
|
|
|
*/
|
|
|
|
#define xmalloc bird_xmalloc
|
2009-06-10 21:45:08 +00:00
|
|
|
#define xrealloc bird_xrealloc
|
1998-04-22 12:58:34 +00:00
|
|
|
void *xmalloc(unsigned);
|
2009-06-10 21:45:08 +00:00
|
|
|
void *xrealloc(void *, unsigned);
|
2024-05-31 13:16:41 +00:00
|
|
|
void xfree(void *);
|
1999-03-04 11:36:26 +00:00
|
|
|
#endif
|
1998-04-22 12:58:34 +00:00
|
|
|
|
|
|
|
#endif
|
2013-09-16 21:57:40 +00:00
|
|
|
|