2020-03-08 12:14:49 +08:00
|
|
|
#ifndef JEMALLOC_INTERNAL_PA_H
|
|
|
|
#define JEMALLOC_INTERNAL_PA_H
|
|
|
|
|
2020-03-11 03:04:16 +08:00
|
|
|
#include "jemalloc/internal/base.h"
|
2020-03-10 04:11:35 +08:00
|
|
|
#include "jemalloc/internal/decay.h"
|
2020-03-09 04:08:15 +08:00
|
|
|
#include "jemalloc/internal/ecache.h"
|
|
|
|
#include "jemalloc/internal/edata_cache.h"
|
2020-03-15 01:49:34 +08:00
|
|
|
#include "jemalloc/internal/emap.h"
|
2020-08-15 04:36:41 +08:00
|
|
|
#include "jemalloc/internal/hpa.h"
|
2020-03-10 01:40:37 +08:00
|
|
|
#include "jemalloc/internal/lockedint.h"
|
2020-05-30 07:57:31 +08:00
|
|
|
#include "jemalloc/internal/pac.h"
|
2020-05-30 06:02:19 +08:00
|
|
|
#include "jemalloc/internal/pai.h"
|
2020-10-17 04:14:59 +08:00
|
|
|
#include "jemalloc/internal/sec.h"
|
2020-03-09 04:08:15 +08:00
|
|
|
|
2020-05-30 07:57:31 +08:00
|
|
|
/*
|
|
|
|
* The page allocator; responsible for acquiring pages of memory for
|
|
|
|
* allocations. It picks the implementation of the page allocator interface
|
|
|
|
* (i.e. a pai_t) to handle a given page-level allocation request. For now, the
|
|
|
|
* only such implementation is the PAC code ("page allocator classic"), but
|
|
|
|
* others will be coming soon.
|
|
|
|
*/
|
|
|
|
|
2021-05-08 04:54:26 +08:00
|
|
|
typedef struct pa_central_s pa_central_t;
|
|
|
|
struct pa_central_s {
|
|
|
|
hpa_central_t hpa;
|
|
|
|
};
|
|
|
|
|
2020-03-10 02:26:15 +08:00
|
|
|
/*
|
|
|
|
* The stats for a particular pa_shard. Because of the way the ctl module
|
|
|
|
* handles stats epoch data collection (it has its own arena_stats, and merges
|
|
|
|
* the stats from each arena into it), this needs to live in the arena_stats_t;
|
|
|
|
* hence we define it here and let the pa_shard have a pointer (rather than the
|
|
|
|
* more natural approach of just embedding it in the pa_shard itself).
|
|
|
|
*
|
|
|
|
* We follow the arena_stats_t approach of marking the derived fields. These
|
|
|
|
* are the ones that are not maintained on their own; instead, their values are
|
|
|
|
* derived during those stats merges.
|
|
|
|
*/
|
2020-03-09 04:08:15 +08:00
|
|
|
typedef struct pa_shard_stats_s pa_shard_stats_t;
|
|
|
|
struct pa_shard_stats_s {
|
2020-03-12 09:14:53 +08:00
|
|
|
/* Number of edata_t structs allocated by base, but not being used. */
|
|
|
|
size_t edata_avail; /* Derived. */
|
2020-06-02 08:42:27 +08:00
|
|
|
/*
|
|
|
|
* Stats specific to the PAC. For now, these are the only stats that
|
|
|
|
* exist, but there will eventually be other page allocators. Things
|
|
|
|
* like edata_avail make sense in a cross-PA sense, but things like
|
|
|
|
* npurges don't.
|
|
|
|
*/
|
|
|
|
pac_stats_t pac_stats;
|
2020-03-09 04:08:15 +08:00
|
|
|
};
|
|
|
|
|
2020-03-12 07:13:36 +08:00
|
|
|
/*
|
|
|
|
* The local allocator handle. Keeps the state necessary to satisfy page-sized
|
|
|
|
* allocations.
|
|
|
|
*
|
|
|
|
* The contents are mostly internal to the PA module. The key exception is that
|
|
|
|
* arena decay code is allowed to grab pointers to the dirty and muzzy ecaches
|
|
|
|
* decay_ts, for a couple of queries, passing them back to a PA function, or
|
|
|
|
* acquiring decay.mtx and looking at decay.purging. The reasoning is that,
|
|
|
|
* while PA decides what and how to purge, the arena code decides when and where
|
|
|
|
* (e.g. on what thread). It's allowed to use the presence of another purger to
|
|
|
|
* decide.
|
|
|
|
* (The background thread code also touches some other decay internals, but
|
|
|
|
* that's not fundamental; its' just an artifact of a partial refactoring, and
|
|
|
|
* its accesses could be straightforwardly moved inside the decay module).
|
|
|
|
*/
|
2020-03-09 01:35:56 +08:00
|
|
|
typedef struct pa_shard_s pa_shard_t;
|
|
|
|
struct pa_shard_s {
|
2021-05-08 04:54:26 +08:00
|
|
|
/* The central PA this shard is associated with. */
|
|
|
|
pa_central_t *central;
|
|
|
|
|
2020-03-12 08:40:17 +08:00
|
|
|
/*
|
|
|
|
* Number of pages in active extents.
|
|
|
|
*
|
|
|
|
* Synchronization: atomic.
|
|
|
|
*/
|
|
|
|
atomic_zu_t nactive;
|
|
|
|
|
2020-08-15 04:36:41 +08:00
|
|
|
/*
|
|
|
|
* Whether or not we should prefer the hugepage allocator. Atomic since
|
|
|
|
* it may be concurrently modified by a thread setting extent hooks.
|
|
|
|
* Note that we still may do HPA operations in this arena; if use_hpa is
|
|
|
|
* changed from true to false, we'll free back to the hugepage allocator
|
|
|
|
* for those allocations.
|
|
|
|
*/
|
|
|
|
atomic_b_t use_hpa;
|
2021-05-08 04:54:26 +08:00
|
|
|
|
2020-08-15 04:36:41 +08:00
|
|
|
/*
|
|
|
|
* If we never used the HPA to begin with, it wasn't initialized, and so
|
|
|
|
* we shouldn't try to e.g. acquire its mutexes during fork. This
|
|
|
|
* tracks that knowledge.
|
|
|
|
*/
|
|
|
|
bool ever_used_hpa;
|
|
|
|
|
2020-06-12 02:53:30 +08:00
|
|
|
/* Allocates from a PAC. */
|
2020-05-30 07:57:31 +08:00
|
|
|
pac_t pac;
|
2020-03-09 02:41:19 +08:00
|
|
|
|
2020-10-17 04:14:59 +08:00
|
|
|
/*
|
|
|
|
* We place a small extent cache in front of the HPA, since we intend
|
|
|
|
* these configurations to use many fewer arenas, and therefore have a
|
|
|
|
* higher risk of hot locks.
|
|
|
|
*/
|
|
|
|
sec_t hpa_sec;
|
2020-08-15 04:36:41 +08:00
|
|
|
hpa_shard_t hpa_shard;
|
|
|
|
|
2020-03-09 02:41:19 +08:00
|
|
|
/* The source of edata_t objects. */
|
|
|
|
edata_cache_t edata_cache;
|
2020-03-09 04:08:15 +08:00
|
|
|
|
2020-08-15 04:36:41 +08:00
|
|
|
unsigned ind;
|
|
|
|
|
2020-03-10 03:06:19 +08:00
|
|
|
malloc_mutex_t *stats_mtx;
|
2020-03-09 04:08:15 +08:00
|
|
|
pa_shard_stats_t *stats;
|
2020-03-10 04:11:35 +08:00
|
|
|
|
2020-03-15 01:49:34 +08:00
|
|
|
/* The emap this shard is tied to. */
|
|
|
|
emap_t *emap;
|
|
|
|
|
2020-03-11 03:04:16 +08:00
|
|
|
/* The base from which we get the ehooks and allocate metadat. */
|
|
|
|
base_t *base;
|
2020-03-09 01:35:56 +08:00
|
|
|
};
|
|
|
|
|
2020-03-13 02:21:22 +08:00
|
|
|
static inline bool
|
|
|
|
pa_shard_dont_decay_muzzy(pa_shard_t *shard) {
|
2020-05-30 07:57:31 +08:00
|
|
|
return ecache_npages_get(&shard->pac.ecache_muzzy) == 0 &&
|
2020-06-04 09:30:33 +08:00
|
|
|
pac_decay_ms_get(&shard->pac, extent_state_muzzy) <= 0;
|
2020-03-13 02:21:22 +08:00
|
|
|
}
|
|
|
|
|
2020-03-11 03:04:16 +08:00
|
|
|
static inline ehooks_t *
|
|
|
|
pa_shard_ehooks_get(pa_shard_t *shard) {
|
|
|
|
return base_ehooks_get(shard->base);
|
|
|
|
}
|
|
|
|
|
2020-03-09 02:19:41 +08:00
|
|
|
/* Returns true on error. */
|
2021-05-08 04:54:26 +08:00
|
|
|
bool pa_central_init(pa_central_t *central, base_t *base, bool hpa,
|
|
|
|
hpa_hooks_t *hpa_hooks);
|
|
|
|
|
|
|
|
/* Returns true on error. */
|
|
|
|
bool pa_shard_init(tsdn_t *tsdn, pa_shard_t *shard, pa_central_t *central,
|
|
|
|
emap_t *emap, base_t *base, unsigned ind, pa_shard_stats_t *stats,
|
|
|
|
malloc_mutex_t *stats_mtx, nstime_t *cur_time, size_t oversize_threshold,
|
|
|
|
ssize_t dirty_decay_ms, ssize_t muzzy_decay_ms);
|
2020-03-15 01:49:34 +08:00
|
|
|
|
2020-08-15 04:36:41 +08:00
|
|
|
/*
|
|
|
|
* This isn't exposed to users; we allow late enablement of the HPA shard so
|
|
|
|
* that we can boot without worrying about the HPA, then turn it on in a0.
|
|
|
|
*/
|
2021-05-19 05:52:46 +08:00
|
|
|
bool pa_shard_enable_hpa(tsdn_t *tsdn, pa_shard_t *shard,
|
2021-05-08 04:54:26 +08:00
|
|
|
const hpa_shard_opts_t *hpa_opts, const sec_opts_t *hpa_sec_opts);
|
2021-06-15 05:18:08 +08:00
|
|
|
|
2020-08-15 04:36:41 +08:00
|
|
|
/*
|
|
|
|
* We stop using the HPA when custom extent hooks are installed, but still
|
|
|
|
* redirect deallocations to it.
|
|
|
|
*/
|
2020-10-17 04:14:59 +08:00
|
|
|
void pa_shard_disable_hpa(tsdn_t *tsdn, pa_shard_t *shard);
|
2020-08-15 04:36:41 +08:00
|
|
|
|
2020-03-13 02:21:22 +08:00
|
|
|
/*
|
|
|
|
* This does the PA-specific parts of arena reset (i.e. freeing all active
|
|
|
|
* allocations).
|
|
|
|
*/
|
2020-10-17 04:14:59 +08:00
|
|
|
void pa_shard_reset(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
|
2020-03-13 02:21:22 +08:00
|
|
|
/*
|
|
|
|
* Destroy all the remaining retained extents. Should only be called after
|
|
|
|
* decaying all active, dirty, and muzzy extents to the retained state, as the
|
|
|
|
* last step in destroying the shard.
|
|
|
|
*/
|
2020-06-11 08:42:49 +08:00
|
|
|
void pa_shard_destroy(tsdn_t *tsdn, pa_shard_t *shard);
|
2020-03-13 02:21:22 +08:00
|
|
|
|
2020-03-12 02:36:38 +08:00
|
|
|
/* Gets an edata for the given allocation. */
|
2020-03-11 05:38:55 +08:00
|
|
|
edata_t *pa_alloc(tsdn_t *tsdn, pa_shard_t *shard, size_t size,
|
2021-04-27 05:22:25 +08:00
|
|
|
size_t alignment, bool slab, szind_t szind, bool zero, bool guarded,
|
2021-08-19 10:24:37 +08:00
|
|
|
bool *deferred_work_generated);
|
2020-03-11 05:38:55 +08:00
|
|
|
/* Returns true on error, in which case nothing changed. */
|
2020-03-11 08:27:31 +08:00
|
|
|
bool pa_expand(tsdn_t *tsdn, pa_shard_t *shard, edata_t *edata, size_t old_size,
|
2021-08-19 10:24:37 +08:00
|
|
|
size_t new_size, szind_t szind, bool zero, bool *deferred_work_generated);
|
2020-03-12 02:36:38 +08:00
|
|
|
/*
|
|
|
|
* The same. Sets *generated_dirty to true if we produced new dirty pages, and
|
|
|
|
* false otherwise.
|
|
|
|
*/
|
|
|
|
bool pa_shrink(tsdn_t *tsdn, pa_shard_t *shard, edata_t *edata, size_t old_size,
|
2021-08-19 10:24:37 +08:00
|
|
|
size_t new_size, szind_t szind, bool *deferred_work_generated);
|
2020-03-12 02:36:38 +08:00
|
|
|
/*
|
|
|
|
* Frees the given edata back to the pa. Sets *generated_dirty if we produced
|
2022-04-24 23:32:44 +08:00
|
|
|
* new dirty pages (well, we always set it for now; but this need not be the
|
2020-03-12 02:36:38 +08:00
|
|
|
* case).
|
|
|
|
* (We could make generated_dirty the return value of course, but this is more
|
|
|
|
* consistent with the shrink pathway and our error codes here).
|
|
|
|
*/
|
|
|
|
void pa_dalloc(tsdn_t *tsdn, pa_shard_t *shard, edata_t *edata,
|
2021-08-19 10:24:37 +08:00
|
|
|
bool *deferred_work_generated);
|
2020-06-04 09:30:33 +08:00
|
|
|
bool pa_decay_ms_set(tsdn_t *tsdn, pa_shard_t *shard, extent_state_t state,
|
|
|
|
ssize_t decay_ms, pac_purge_eagerness_t eagerness);
|
|
|
|
ssize_t pa_decay_ms_get(pa_shard_t *shard, extent_state_t state);
|
|
|
|
|
2021-06-08 02:45:57 +08:00
|
|
|
/*
|
|
|
|
* Do deferred work on this PA shard.
|
|
|
|
*
|
|
|
|
* Morally, this should do both PAC decay and the HPA deferred work. For now,
|
|
|
|
* though, the arena, background thread, and PAC modules are tightly interwoven
|
|
|
|
* in a way that's tricky to extricate, so we only do the HPA-specific parts.
|
|
|
|
*/
|
|
|
|
void pa_shard_set_deferral_allowed(tsdn_t *tsdn, pa_shard_t *shard,
|
|
|
|
bool deferral_allowed);
|
|
|
|
void pa_shard_do_deferred_work(tsdn_t *tsdn, pa_shard_t *shard);
|
2021-08-07 05:53:05 +08:00
|
|
|
void pa_shard_try_deferred_work(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
uint64_t pa_shard_time_until_deferred_work(tsdn_t *tsdn, pa_shard_t *shard);
|
2021-06-08 02:45:57 +08:00
|
|
|
|
2020-03-13 00:20:37 +08:00
|
|
|
/******************************************************************************/
|
|
|
|
/*
|
|
|
|
* Various bits of "boring" functionality that are still part of this module,
|
|
|
|
* but that we relegate to pa_extra.c, to keep the core logic in pa.c as
|
|
|
|
* readable as possible.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These fork phases are synchronized with the arena fork phase numbering to
|
|
|
|
* make it easy to keep straight. That's why there's no prefork1.
|
|
|
|
*/
|
|
|
|
void pa_shard_prefork0(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
void pa_shard_prefork2(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
void pa_shard_prefork3(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
void pa_shard_prefork4(tsdn_t *tsdn, pa_shard_t *shard);
|
2020-10-17 04:14:59 +08:00
|
|
|
void pa_shard_prefork5(tsdn_t *tsdn, pa_shard_t *shard);
|
2020-03-13 00:20:37 +08:00
|
|
|
void pa_shard_postfork_parent(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
void pa_shard_postfork_child(tsdn_t *tsdn, pa_shard_t *shard);
|
|
|
|
|
2020-03-13 00:28:13 +08:00
|
|
|
void pa_shard_basic_stats_merge(pa_shard_t *shard, size_t *nactive,
|
|
|
|
size_t *ndirty, size_t *nmuzzy);
|
|
|
|
|
2020-03-13 01:28:18 +08:00
|
|
|
void pa_shard_stats_merge(tsdn_t *tsdn, pa_shard_t *shard,
|
2020-06-02 08:42:27 +08:00
|
|
|
pa_shard_stats_t *pa_shard_stats_out, pac_estats_t *estats_out,
|
2020-10-17 04:14:59 +08:00
|
|
|
hpa_shard_stats_t *hpa_stats_out, sec_stats_t *sec_stats_out,
|
|
|
|
size_t *resident);
|
2020-03-13 01:28:18 +08:00
|
|
|
|
2020-03-13 06:26:50 +08:00
|
|
|
/*
|
|
|
|
* Reads the PA-owned mutex stats into the output stats array, at the
|
|
|
|
* appropriate positions. Morally, these stats should really live in
|
|
|
|
* pa_shard_stats_t, but the indices are sort of baked into the various mutex
|
|
|
|
* prof macros. This would be a good thing to do at some point.
|
|
|
|
*/
|
|
|
|
void pa_shard_mtx_stats_read(tsdn_t *tsdn, pa_shard_t *shard,
|
|
|
|
mutex_prof_data_t mutex_prof_data[mutex_prof_num_arena_mutexes]);
|
|
|
|
|
2020-03-08 12:14:49 +08:00
|
|
|
#endif /* JEMALLOC_INTERNAL_PA_H */
|