2021-03-02 09:03:00 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* mm/mm_heap/mm.h
|
|
|
|
*
|
|
|
|
* 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>
|
|
|
|
|
2021-07-03 15:25:14 +02:00
|
|
|
#include <nuttx/fs/procfs.h>
|
|
|
|
|
2022-01-27 05:17:20 +01:00
|
|
|
#include <assert.h>
|
2022-01-20 10:17:29 +01:00
|
|
|
#include <execinfo.h>
|
2021-03-02 09:03:00 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <semaphore.h>
|
2022-01-20 10:17:29 +01:00
|
|
|
#include <unistd.h>
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Pre-processor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* Configuration ************************************************************/
|
|
|
|
|
|
|
|
/* Chunk Header Definitions *************************************************/
|
|
|
|
|
2021-07-31 17:17:21 +02:00
|
|
|
/* These definitions define the characteristics of the allocator:
|
2021-03-02 09:03:00 +01:00
|
|
|
*
|
|
|
|
* MM_MIN_SHIFT is used to define MM_MIN_CHUNK.
|
2021-07-31 17:17:21 +02:00
|
|
|
* 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
|
2021-03-02 09:03:00 +01:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(CONFIG_MM_SMALL) && UINTPTR_MAX <= UINT32_MAX
|
|
|
|
/* Two byte offsets; Pointers may be 2 or 4 bytes;
|
|
|
|
* sizeof(struct mm_freenode_s) is 8 or 12 bytes.
|
|
|
|
* REVISIT: We could do better on machines with 16-bit addressing.
|
|
|
|
*/
|
|
|
|
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MIN_SHIFT_ ( 4) /* 16 bytes */
|
|
|
|
# define MM_MAX_SHIFT (15) /* 32 Kb */
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
#elif defined(CONFIG_HAVE_LONG_LONG)
|
|
|
|
/* Four byte offsets; Pointers may be 4 or 8 bytes
|
|
|
|
* sizeof(struct mm_freenode_s) is 16 or 24 bytes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
# if UINTPTR_MAX <= UINT32_MAX
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MIN_SHIFT_ ( 4) /* 16 bytes */
|
2021-03-02 09:03:00 +01:00
|
|
|
# elif UINTPTR_MAX <= UINT64_MAX
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MIN_SHIFT_ ( 5) /* 32 bytes */
|
2021-03-02 09:03:00 +01:00
|
|
|
# endif
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MAX_SHIFT (22) /* 4 Mb */
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
#else
|
|
|
|
/* Four byte offsets; Pointers must be 4 bytes.
|
|
|
|
* sizeof(struct mm_freenode_s) is 16 bytes.
|
|
|
|
*/
|
|
|
|
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MIN_SHIFT_ ( 4) /* 16 bytes */
|
|
|
|
# define MM_MAX_SHIFT (22) /* 4 Mb */
|
|
|
|
#endif
|
|
|
|
|
2022-07-04 09:41:06 +02:00
|
|
|
#if CONFIG_MM_BACKTRACE == 0
|
|
|
|
# define MM_MIN_SHIFT (MM_MIN_SHIFT_ + 1)
|
|
|
|
# define MM_ADD_BACKTRACE(heap, ptr) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
FAR struct mm_allocnode_s *tmp = (FAR struct mm_allocnode_s *)(ptr); \
|
|
|
|
tmp->pid = getpid(); \
|
|
|
|
} \
|
|
|
|
while (0)
|
|
|
|
#elif CONFIG_MM_BACKTRACE > 0
|
|
|
|
# define MM_MIN_SHIFT (MM_MIN_SHIFT_ + 2)
|
2022-03-17 07:02:37 +01:00
|
|
|
# define MM_ADD_BACKTRACE(heap, ptr) \
|
2022-01-20 10:17:29 +01:00
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
FAR struct mm_allocnode_s *tmp = (FAR struct mm_allocnode_s *)(ptr); \
|
mm/kasan: node header should updated from unpoisoning memory
| (gdb) bt
| #0 up_assert (filename=0x7fffffffdc6c "\001", lineno=0) at sim/up_assert.c:75
| #1 0x00005555555e636b in _assert (filename=0x555555627225 "kasan/kasan.c", linenum=104) at assert/lib_assert.c:36
| #2 0x00005555555a388e in kasan_report (addr=140737284458088, size=1, is_write=true) at kasan/kasan.c:104
| #3 0x00005555555a40a1 in __asan_storeN_noabort (addr=140737284458088, size=1) at kasan/kasan.c:297
| #4 0x00005555555a4519 in __asan_store1_noabort (addr=140737284458088) at kasan/kasan.c:348
| #5 0x00005555555a26d7 in memset (s=0x7ffff3d8c668, c=0, n=63) at string/lib_memset.c:169
| #6 0x00005555555a46a4 in mm_addregion (heap=0x7ffff3d8c000, heapstart=0x7ffff3d8c648, heapsize=66058656) at mm_heap/mm_initialize.c:131
| #7 0x00005555555a4a00 in mm_initialize (name=0x555555627068 "Umem", heapstart=0x7ffff3d8c648, heapsize=67107256) at mm_heap/mm_initialize.c:231
| #8 0x00005555555a33b1 in umm_initialize (heap_start=0x7ffff3d8c000, heap_size=67108864) at umm_heap/umm_initialize.c:84
| #9 0x000055555558f17c in nx_start () at init/nx_start.c:469
| #10 0x0000555555589559 in main (argc=1, argv=0x7fffffffdf58, envp=0x7fffffffdf68) at sim/up_head.c:131
| (gdb)
Signed-off-by: chao.an <anchao@xiaomi.com>
2022-08-01 09:53:24 +02:00
|
|
|
kasan_unpoison(tmp, SIZEOF_MM_ALLOCNODE); \
|
2022-01-20 10:17:29 +01:00
|
|
|
tmp->pid = getpid(); \
|
2022-03-17 07:02:37 +01:00
|
|
|
if ((heap)->mm_procfs.backtrace) \
|
|
|
|
{ \
|
|
|
|
memset(tmp->backtrace, 0, sizeof(tmp->backtrace)); \
|
2022-07-04 09:41:06 +02:00
|
|
|
backtrace(tmp->backtrace, CONFIG_MM_BACKTRACE); \
|
2022-03-17 07:02:37 +01:00
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
tmp->backtrace[0] = 0; \
|
|
|
|
} \
|
mm/kasan: node header should updated from unpoisoning memory
| (gdb) bt
| #0 up_assert (filename=0x7fffffffdc6c "\001", lineno=0) at sim/up_assert.c:75
| #1 0x00005555555e636b in _assert (filename=0x555555627225 "kasan/kasan.c", linenum=104) at assert/lib_assert.c:36
| #2 0x00005555555a388e in kasan_report (addr=140737284458088, size=1, is_write=true) at kasan/kasan.c:104
| #3 0x00005555555a40a1 in __asan_storeN_noabort (addr=140737284458088, size=1) at kasan/kasan.c:297
| #4 0x00005555555a4519 in __asan_store1_noabort (addr=140737284458088) at kasan/kasan.c:348
| #5 0x00005555555a26d7 in memset (s=0x7ffff3d8c668, c=0, n=63) at string/lib_memset.c:169
| #6 0x00005555555a46a4 in mm_addregion (heap=0x7ffff3d8c000, heapstart=0x7ffff3d8c648, heapsize=66058656) at mm_heap/mm_initialize.c:131
| #7 0x00005555555a4a00 in mm_initialize (name=0x555555627068 "Umem", heapstart=0x7ffff3d8c648, heapsize=67107256) at mm_heap/mm_initialize.c:231
| #8 0x00005555555a33b1 in umm_initialize (heap_start=0x7ffff3d8c000, heap_size=67108864) at umm_heap/umm_initialize.c:84
| #9 0x000055555558f17c in nx_start () at init/nx_start.c:469
| #10 0x0000555555589559 in main (argc=1, argv=0x7fffffffdf58, envp=0x7fffffffdf68) at sim/up_head.c:131
| (gdb)
Signed-off-by: chao.an <anchao@xiaomi.com>
2022-08-01 09:53:24 +02:00
|
|
|
kasan_poison(tmp, SIZEOF_MM_ALLOCNODE); \
|
2022-01-20 10:17:29 +01:00
|
|
|
} \
|
|
|
|
while (0)
|
|
|
|
#else
|
2022-03-17 07:02:37 +01:00
|
|
|
# define MM_ADD_BACKTRACE(heap, ptr)
|
2022-01-20 10:17:29 +01:00
|
|
|
# define MM_MIN_SHIFT MM_MIN_SHIFT_
|
2021-03-02 09:03:00 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* All other definitions derive from these two */
|
|
|
|
|
|
|
|
#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)
|
|
|
|
|
2021-03-21 05:11:11 +01:00
|
|
|
#define MM_GRAN_MASK (MM_MIN_CHUNK - 1)
|
2021-03-02 09:03:00 +01:00
|
|
|
#define MM_ALIGN_UP(a) (((a) + MM_GRAN_MASK) & ~MM_GRAN_MASK)
|
|
|
|
#define MM_ALIGN_DOWN(a) ((a) & ~MM_GRAN_MASK)
|
|
|
|
|
2022-03-30 14:31:29 +02:00
|
|
|
/* An allocated chunk is distinguished from a free chunk by bit 0
|
2021-03-02 09:03:00 +01:00
|
|
|
* of the 'preceding' chunk size. If set, then this is an allocated chunk.
|
|
|
|
*/
|
|
|
|
|
2022-03-30 14:31:29 +02:00
|
|
|
#define MM_ALLOC_BIT 0x1
|
2021-03-02 09:03:00 +01:00
|
|
|
#ifdef CONFIG_MM_SMALL
|
2022-01-27 05:17:20 +01:00
|
|
|
# define MMSIZE_MAX UINT16_MAX
|
2021-03-02 09:03:00 +01:00
|
|
|
#else
|
2022-01-27 05:17:20 +01:00
|
|
|
# define MMSIZE_MAX UINT32_MAX
|
2021-03-02 09:03:00 +01:00
|
|
|
#endif
|
2022-01-27 05:17:20 +01:00
|
|
|
|
2021-03-02 09:03:00 +01:00
|
|
|
#define MM_IS_ALLOCATED(n) \
|
2021-03-21 05:11:11 +01:00
|
|
|
((int)((FAR struct mm_allocnode_s *)(n)->preceding) < 0)
|
2021-03-02 09:03:00 +01:00
|
|
|
|
2022-01-27 05:17:20 +01:00
|
|
|
/* What is the size of the allocnode? */
|
|
|
|
|
|
|
|
#define SIZEOF_MM_ALLOCNODE sizeof(struct mm_allocnode_s)
|
|
|
|
|
|
|
|
/* What is the size of the freenode? */
|
|
|
|
|
|
|
|
#define SIZEOF_MM_FREENODE sizeof(struct mm_freenode_s)
|
|
|
|
|
2021-03-02 09:03:00 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Public Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* Determines the size of the chunk size/offset type */
|
|
|
|
|
|
|
|
#ifdef CONFIG_MM_SMALL
|
|
|
|
typedef uint16_t mmsize_t;
|
|
|
|
#else
|
|
|
|
typedef uint32_t mmsize_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* This describes an allocated chunk. An allocated chunk is
|
|
|
|
* distinguished from a free chunk by bit 15/31 of the 'preceding' chunk
|
|
|
|
* size. If set, then this is an allocated chunk.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct mm_allocnode_s
|
|
|
|
{
|
2022-07-04 09:41:06 +02:00
|
|
|
#if CONFIG_MM_BACKTRACE >= 0
|
|
|
|
pid_t pid; /* The pid for caller */
|
|
|
|
# if CONFIG_MM_BACKTRACE > 0
|
|
|
|
FAR void *backtrace[CONFIG_MM_BACKTRACE]; /* The backtrace buffer for caller */
|
|
|
|
# endif
|
2022-01-20 10:17:29 +01:00
|
|
|
#endif
|
2022-07-04 09:41:06 +02:00
|
|
|
mmsize_t size; /* Size of this chunk */
|
|
|
|
mmsize_t preceding; /* Size of the preceding chunk */
|
2021-03-02 09:03:00 +01:00
|
|
|
};
|
|
|
|
|
2022-01-27 05:17:20 +01:00
|
|
|
static_assert(SIZEOF_MM_ALLOCNODE <= MM_MIN_CHUNK,
|
|
|
|
"Error size for struct mm_allocnode_s\n");
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
/* This describes a free chunk */
|
|
|
|
|
|
|
|
struct mm_freenode_s
|
|
|
|
{
|
2022-07-04 09:41:06 +02:00
|
|
|
#if CONFIG_MM_BACKTRACE >= 0
|
|
|
|
pid_t pid; /* The pid for caller */
|
|
|
|
# if CONFIG_MM_BACKTRACE > 0
|
|
|
|
FAR void *backtrace[CONFIG_MM_BACKTRACE]; /* The backtrace buffer for caller */
|
|
|
|
# endif
|
2022-01-20 10:17:29 +01:00
|
|
|
#endif
|
2022-07-04 09:41:06 +02:00
|
|
|
mmsize_t size; /* Size of this chunk */
|
|
|
|
mmsize_t preceding; /* Size of the preceding chunk */
|
|
|
|
FAR struct mm_freenode_s *flink; /* Supports a doubly linked list */
|
2021-03-02 09:03:00 +01:00
|
|
|
FAR struct mm_freenode_s *blink;
|
|
|
|
};
|
|
|
|
|
2022-01-27 05:17:20 +01:00
|
|
|
static_assert(SIZEOF_MM_FREENODE <= MM_MIN_CHUNK,
|
|
|
|
"Error size for struct mm_freenode_s\n");
|
|
|
|
|
2021-03-02 09:03:00 +01:00
|
|
|
struct mm_delaynode_s
|
|
|
|
{
|
2021-03-21 05:11:11 +01:00
|
|
|
FAR struct mm_delaynode_s *flink;
|
2021-03-02 09:03:00 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/* This describes one heap (possibly with multiple regions) */
|
|
|
|
|
2021-07-03 16:16:25 +02:00
|
|
|
struct mm_heap_s
|
2021-03-02 09:03:00 +01:00
|
|
|
{
|
|
|
|
/* Mutually exclusive access to this data set is enforced with
|
|
|
|
* the following un-named semaphore.
|
|
|
|
*/
|
|
|
|
|
|
|
|
sem_t mm_semaphore;
|
|
|
|
|
|
|
|
/* This is the size of the heap provided to mm */
|
|
|
|
|
|
|
|
size_t mm_heapsize;
|
|
|
|
|
|
|
|
/* This is the first and last nodes of the heap */
|
|
|
|
|
|
|
|
FAR struct mm_allocnode_s *mm_heapstart[CONFIG_MM_REGIONS];
|
|
|
|
FAR struct mm_allocnode_s *mm_heapend[CONFIG_MM_REGIONS];
|
|
|
|
|
|
|
|
#if CONFIG_MM_REGIONS > 1
|
|
|
|
int mm_nregions;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* All free nodes are maintained in a doubly linked list. This
|
|
|
|
* array provides some hooks into the list at various points to
|
|
|
|
* speed searches for free nodes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct mm_freenode_s mm_nodelist[MM_NNODES];
|
|
|
|
|
2021-07-31 17:17:21 +02:00
|
|
|
/* Free delay list, for some situations where we can't do free
|
|
|
|
* immdiately.
|
|
|
|
*/
|
2021-03-02 09:03:00 +01:00
|
|
|
|
2021-06-21 10:20:44 +02:00
|
|
|
FAR struct mm_delaynode_s *mm_delaylist[CONFIG_SMP_NCPUS];
|
2021-07-03 15:25:14 +02:00
|
|
|
|
|
|
|
#if defined(CONFIG_FS_PROCFS) && !defined(CONFIG_FS_PROCFS_EXCLUDE_MEMINFO)
|
|
|
|
struct procfs_meminfo_entry_s mm_procfs;
|
|
|
|
#endif
|
2021-03-02 09:03:00 +01:00
|
|
|
};
|
|
|
|
|
2022-01-04 15:10:32 +01:00
|
|
|
/* This describes the callback for mm_foreach */
|
|
|
|
|
|
|
|
typedef CODE void (*mmchunk_handler_t)(FAR struct mm_allocnode_s *node,
|
|
|
|
FAR void *arg);
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Function Prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
|
2021-03-08 14:59:32 +01:00
|
|
|
/* Functions contained in mm_sem.c ******************************************/
|
2021-03-02 09:03:00 +01:00
|
|
|
|
2021-03-08 14:59:32 +01:00
|
|
|
void mm_seminitialize(FAR struct mm_heap_s *heap);
|
2022-06-10 03:47:41 +02:00
|
|
|
void mm_semuninitialize(FAR struct mm_heap_s *heap);
|
2021-07-07 15:21:42 +02:00
|
|
|
bool mm_takesemaphore(FAR struct mm_heap_s *heap);
|
2021-03-08 14:59:32 +01:00
|
|
|
void mm_givesemaphore(FAR struct mm_heap_s *heap);
|
|
|
|
|
|
|
|
/* Functions contained in mm_shrinkchunk.c **********************************/
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
void mm_shrinkchunk(FAR struct mm_heap_s *heap,
|
|
|
|
FAR struct mm_allocnode_s *node, size_t size);
|
|
|
|
|
|
|
|
/* Functions contained in mm_addfreechunk.c *********************************/
|
|
|
|
|
|
|
|
void mm_addfreechunk(FAR struct mm_heap_s *heap,
|
|
|
|
FAR struct mm_freenode_s *node);
|
|
|
|
|
2022-01-04 15:10:32 +01:00
|
|
|
/* Functions contained in mm_size2ndx.c *************************************/
|
2021-03-02 09:03:00 +01:00
|
|
|
|
|
|
|
int mm_size2ndx(size_t size);
|
|
|
|
|
2022-01-04 15:10:32 +01:00
|
|
|
/* Functions contained in mm_foreach.c **************************************/
|
|
|
|
|
|
|
|
void mm_foreach(FAR struct mm_heap_s *heap, mmchunk_handler_t handler,
|
|
|
|
FAR void *arg);
|
|
|
|
|
2021-03-02 09:03:00 +01:00
|
|
|
#endif /* __MM_MM_HEAP_MM_H */
|