* mm/mm_heap/mm.h
*
* SPDX-License-Identifier: Apache-2.0
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
#ifndef __MM_MM_HEAP_MM_H
#define __MM_MM_HEAP_MM_H
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <nuttx/mutex.h>
#include <nuttx/sched.h>
#include <nuttx/fs/procfs.h>
#include <nuttx/lib/math32.h>
#include <nuttx/mm/mempool.h>
#include <nuttx/mm/mm.h>
#include <nuttx/tls.h>
#include <assert.h>
#include <execinfo.h>
#include <sys/types.h>
#include <stdbool.h>
#include <string.h>
#include <unistd.h>
* Pre-processor Definitions
****************************************************************************/
*
* MM_MIN_SHIFT is used to define MM_MIN_CHUNK.
* MM_MIN_CHUNK - is the smallest physical chunk that can be allocated.
* It must be at least as large as sizeof(struct mm_freenode_s). Larger
* values may improve performance slightly, but will waste memory due to
* quantization losses.
*
* MM_MAX_SHIFT is used to define MM_MAX_CHUNK
* MM_MAX_CHUNK is the largest, contiguous chunk of memory that can be
* allocated. It can range from 16-bytes to 4Gb. Larger values of
* MM_MAX_SHIFT can cause larger data structure sizes and, perhaps,
* minor performance losses.
*/
#define MM_MIN_SHIFT LOG2_CEIL(sizeof(struct mm_freenode_s))
#if defined(CONFIG_MM_SMALL) && UINTPTR_MAX <= UINT32_MAX
# define MM_MAX_SHIFT (15)
#else
# define MM_MAX_SHIFT (22)
#endif
#ifdef CONFIG_MM_RECORD_PID
# define MM_RECORD_PID(node, tid) ((node)->pid = tid)
#else
# define MM_RECORD_PID(node, tid)
#endif
#ifdef CONFIG_MM_RECORD_STACK
# define MM_RECORD_STACK(node, tid) \
do \
{ \
FAR struct tls_info_s *info = tls_get_info(); \
if (((heap)->mm_procfs && (heap)->mm_procfs->backtrace) || \
(info && info->tl_flags & TLS_FLAG_HEAP_DUMP)) \
{ \
(node)->stack = backtrace_record(CONFIG_MM_RECORD_STACK_SKIP); \
} \
else \
{ \
(node)->stack = NULL; \
} \
} \
while (0)
#else
# define MM_RECORD_STACK(node, tid)
#endif
#ifdef CONFIG_MM_RECORD
# define MM_RECORD(heap, ptr) \
do \
{ \
FAR struct mm_allocnode_s *tmp = (FAR struct mm_allocnode_s *)(ptr); \
pid_t tid = _SCHED_GETTID(); \
UNUSED(tid); \
MM_INCSEQNO(tmp); \
MM_RECORD_PID(tmp, tid); \
MM_RECORD_STACK(tmp, tid); \
} \
while (0)
#else
# define MM_RECORD(heap, ptr)
#endif
#define MM_MIN_CHUNK (1 << MM_MIN_SHIFT)
#define MM_MAX_CHUNK (1 << MM_MAX_SHIFT)
#define MM_NNODES (MM_MAX_SHIFT - MM_MIN_SHIFT + 1)
#define MM_GRAN_MASK (MM_ALIGN - 1)
#define MM_ALIGN_UP(a) (((a) + MM_GRAN_MASK) & ~MM_GRAN_MASK)
#define MM_ALIGN_DOWN(a) ((a) & ~MM_GRAN_MASK)
* zero, thus the two bits are reused to depict allocation status: bit
* 0 depicts the allocation state of current chunk, and bit 1 depicts that
* of the physically preceding chunk.
*/
#define MM_ALLOC_BIT 0x1
#define MM_PREVFREE_BIT 0x2
#define MM_MASK_BIT (MM_ALLOC_BIT | MM_PREVFREE_BIT)
#ifdef CONFIG_MM_SMALL
# define MMSIZE_MAX UINT16_MAX
#else
# define MMSIZE_MAX UINT32_MAX
#endif
#define MM_SIZEOF_ALLOCNODE sizeof(struct mm_allocnode_s)
* Remove the space of preceding field since it locates at the end of the
* previous freenode
*/
#define MM_ALLOCNODE_OVERHEAD (CONFIG_MM_NODE_GUARDSIZE + \
MM_SIZEOF_ALLOCNODE - sizeof(mmsize_t))
#define MM_SIZEOF_NODE(node) ((node)->size & (~MM_MASK_BIT))
#define MM_NODE_IS_ALLOC(node) (((node)->size & MM_ALLOC_BIT) != 0)
#define MM_NODE_IS_FREE(node) (((node)->size & MM_ALLOC_BIT) == 0)
#define MM_PREVNODE_IS_ALLOC(node) (((node)->size & MM_PREVFREE_BIT) == 0)
#define MM_PREVNODE_IS_FREE(node) (((node)->size & MM_PREVFREE_BIT) != 0)
* Public Types
****************************************************************************/
#ifdef CONFIG_MM_SMALL
typedef uint16_t mmsize_t;
#else
typedef size_t mmsize_t;
#endif
struct mm_allocnode_s
{
mmsize_t preceding;
mmsize_t size;
#ifdef CONFIG_MM_RECORD_PID
pid_t pid;
#endif
#ifdef CONFIG_MM_RECORD_SEQNO
unsigned long seqno;
#endif
#ifdef CONFIG_MM_RECORD_STACK
FAR void *stack;
# endif
};
struct mm_freenode_s
{
mmsize_t preceding;
mmsize_t size;
#ifdef CONFIG_MM_RECORD_PID
pid_t pid;
#endif
#ifdef CONFIG_MM_RECORD_SEQNO
unsigned long seqno;
#endif
#ifdef CONFIG_MM_RECORD_STACK
FAR void *stack;
#endif
FAR struct mm_freenode_s *flink;
FAR struct mm_freenode_s *blink;
};
static_assert(MM_SIZEOF_ALLOCNODE <= MM_MIN_CHUNK,
"Error size for struct mm_allocnode_s\n");
static_assert(MM_ALIGN >= sizeof(uintptr_t) &&
(MM_ALIGN & MM_GRAN_MASK) == 0,
"Error memory alignment\n");
struct mm_delaynode_s
{
FAR struct mm_delaynode_s *flink;
#ifdef CONFIG_DEBUG_ASSERTIONS
uintptr_t magic;
#endif
};
struct mm_delayhead_s
{
FAR struct mm_delaynode_s *head;
#if CONFIG_MM_FREE_DELAYCOUNT_MAX > 0
size_t delaycount;
#endif
};
#ifdef CONFIG_SMP
typedef struct mm_delayhead_s mm_delaylist_t[CONFIG_SMP_NCPUS];
#else
typedef struct mm_delayhead_s mm_delaylist_t;
#endif
struct mm_heap_s
{
rmutex_t mm_lock;
size_t mm_heapsize;
size_t mm_maxused;
size_t mm_curused;
FAR struct mm_allocnode_s *mm_heapstart[CONFIG_MM_REGIONS];
FAR struct mm_allocnode_s *mm_heapend[CONFIG_MM_REGIONS];
size_t mm_regionsize[CONFIG_MM_REGIONS];
#if CONFIG_MM_REGIONS > 1
int mm_nregions;
#endif
* array provides some hooks into the list at various points to
* speed up searching of free nodes.
*/
struct mm_freenode_s mm_nodelist[MM_NNODES];
mm_delaylist_t delay;
#ifdef CONFIG_MM_HEAP_MEMPOOL
size_t mm_threshold;
FAR struct mempool_multiple_s *mm_mpool;
#endif
#if defined(CONFIG_FS_PROCFS) && !defined(CONFIG_FS_PROCFS_EXCLUDE_MEMINFO)
FAR struct procfs_meminfo_entry_s *mm_procfs;
#endif
bool mm_nokasan;
};
typedef CODE void (*mm_node_handler_t)(FAR struct mm_allocnode_s *node,
FAR void *arg);
* Public Function Prototypes
****************************************************************************/
void mm_shrinkchunk(FAR struct mm_heap_s *heap,
FAR struct mm_allocnode_s *node, size_t size);
void mm_foreach(FAR struct mm_heap_s *heap, mm_node_handler_t handler,
FAR void *arg);
void mm_forcefree(FAR struct mm_heap_s *heap, FAR void *mem);
void mm_free_delaylist(FAR struct mm_heap_s *heap);
* Inline Functions
****************************************************************************/
static inline_function int mm_size2ndx(size_t size)
{
DEBUGASSERT(size >= MM_MIN_CHUNK);
if (size >= MM_MAX_CHUNK)
{
return MM_NNODES - 1;
}
size >>= MM_MIN_SHIFT;
return flsl(size) - 1;
}
static inline_function void mm_addfreechunk(FAR struct mm_heap_s *heap,
FAR struct mm_freenode_s *node)
{
FAR struct mm_freenode_s *next;
FAR struct mm_freenode_s *prev;
size_t nodesize = MM_SIZEOF_NODE(node);
int ndx;
DEBUGASSERT(nodesize >= MM_MIN_CHUNK);
DEBUGASSERT(MM_NODE_IS_FREE(node));
ndx = mm_size2ndx(nodesize);
for (prev = &heap->mm_nodelist[ndx],
next = heap->mm_nodelist[ndx].flink;
next && next->size && MM_SIZEOF_NODE(next) < nodesize;
prev = next, next = next->flink);
prev->flink = node;
node->blink = prev;
node->flink = next;
if (next)
{
next->blink = node;
}
}
static inline_function
FAR struct mm_delayhead_s *get_delayhead(FAR struct mm_heap_s *heap)
{
#ifdef CONFIG_SMP
return &heap->delay[this_cpu()];
#else
return &heap->delay;
#endif
}
#endif