blob: 34f0b490938709d4cb4188fe2d9aa601d3471ee7 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glass94890462014-11-10 17:16:43 -07002/*
3 * Simple malloc implementation
4 *
5 * Copyright (c) 2014 Google, Inc
Simon Glass94890462014-11-10 17:16:43 -07006 */
7
Simon Glass8ed64a42018-11-18 08:14:26 -07008#define LOG_CATEGORY LOGC_ALLOC
9
Simon Glass94890462014-11-10 17:16:43 -070010#include <common.h>
Simon Glass0f2af882020-05-10 11:40:05 -060011#include <log.h>
Simon Glass94890462014-11-10 17:16:43 -070012#include <malloc.h>
Joe Hershberger65b905b2015-03-22 17:08:59 -050013#include <mapmem.h>
Simon Glass94890462014-11-10 17:16:43 -070014#include <asm/io.h>
15
16DECLARE_GLOBAL_DATA_PTR;
17
Simon Glass8ed64a42018-11-18 08:14:26 -070018static void *alloc_simple(size_t bytes, int align)
Simon Glass94890462014-11-10 17:16:43 -070019{
Simon Glass8ed64a42018-11-18 08:14:26 -070020 ulong addr, new_ptr;
Simon Glass94890462014-11-10 17:16:43 -070021 void *ptr;
22
Simon Glass8ed64a42018-11-18 08:14:26 -070023 addr = ALIGN(gd->malloc_base + gd->malloc_ptr, align);
24 new_ptr = addr + bytes - gd->malloc_base;
25 log_debug("size=%zx, ptr=%lx, limit=%lx: ", bytes, new_ptr,
26 gd->malloc_limit);
Simon Glasse8211b42016-03-06 19:27:55 -070027 if (new_ptr > gd->malloc_limit) {
Simon Glass8ed64a42018-11-18 08:14:26 -070028 log_err("alloc space exhausted\n");
Hans de Goede8b4e7282015-02-04 13:05:50 +010029 return NULL;
Simon Glasse8211b42016-03-06 19:27:55 -070030 }
Simon Glass8ed64a42018-11-18 08:14:26 -070031
32 ptr = map_sysmem(addr, bytes);
Simon Glass94890462014-11-10 17:16:43 -070033 gd->malloc_ptr = ALIGN(new_ptr, sizeof(new_ptr));
Simon Glass65ba4122015-09-08 17:52:46 -060034
Simon Glass94890462014-11-10 17:16:43 -070035 return ptr;
36}
37
Simon Glass8ed64a42018-11-18 08:14:26 -070038void *malloc_simple(size_t bytes)
Simon Glassde9d70f2015-05-12 14:55:06 -060039{
Simon Glassde9d70f2015-05-12 14:55:06 -060040 void *ptr;
41
Simon Glass8ed64a42018-11-18 08:14:26 -070042 ptr = alloc_simple(bytes, 1);
43 if (!ptr)
44 return ptr;
Andrew F. Davis34597e42017-01-27 10:39:18 -060045
Simon Glass8ed64a42018-11-18 08:14:26 -070046 log_debug("%lx\n", (ulong)ptr);
47
48 return ptr;
49}
50
51void *memalign_simple(size_t align, size_t bytes)
52{
53 void *ptr;
54
55 ptr = alloc_simple(bytes, align);
56 if (!ptr)
57 return ptr;
58 log_debug("aligned to %lx\n", (ulong)ptr);
Simon Glass65ba4122015-09-08 17:52:46 -060059
Simon Glassde9d70f2015-05-12 14:55:06 -060060 return ptr;
61}
62
Hans de Goede9f9df6f2015-09-13 14:45:15 +020063#if CONFIG_IS_ENABLED(SYS_MALLOC_SIMPLE)
Simon Glass94890462014-11-10 17:16:43 -070064void *calloc(size_t nmemb, size_t elem_size)
65{
66 size_t size = nmemb * elem_size;
67 void *ptr;
68
69 ptr = malloc(size);
Simon Glass8ed64a42018-11-18 08:14:26 -070070 if (!ptr)
71 return ptr;
72 memset(ptr, '\0', size);
Simon Glass94890462014-11-10 17:16:43 -070073
74 return ptr;
75}
76#endif
Simon Glass8ed64a42018-11-18 08:14:26 -070077
78void malloc_simple_info(void)
79{
80 log_info("malloc_simple: %lx bytes used, %lx remain\n", gd->malloc_ptr,
81 CONFIG_VAL(SYS_MALLOC_F_LEN) - gd->malloc_ptr);
82}