blob: 0267fb6bec87bae60da5beb27ef11f25afd4dedf [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 Glass3ba929a2020-10-30 21:38:53 -060014#include <asm/global_data.h>
Simon Glass94890462014-11-10 17:16:43 -070015#include <asm/io.h>
16
17DECLARE_GLOBAL_DATA_PTR;
18
Simon Glass8ed64a42018-11-18 08:14:26 -070019static void *alloc_simple(size_t bytes, int align)
Simon Glass94890462014-11-10 17:16:43 -070020{
Simon Glass8ed64a42018-11-18 08:14:26 -070021 ulong addr, new_ptr;
Simon Glass94890462014-11-10 17:16:43 -070022 void *ptr;
23
Simon Glass8ed64a42018-11-18 08:14:26 -070024 addr = ALIGN(gd->malloc_base + gd->malloc_ptr, align);
25 new_ptr = addr + bytes - gd->malloc_base;
26 log_debug("size=%zx, ptr=%lx, limit=%lx: ", bytes, new_ptr,
27 gd->malloc_limit);
Simon Glasse8211b42016-03-06 19:27:55 -070028 if (new_ptr > gd->malloc_limit) {
Simon Glass8ed64a42018-11-18 08:14:26 -070029 log_err("alloc space exhausted\n");
Hans de Goede8b4e7282015-02-04 13:05:50 +010030 return NULL;
Simon Glasse8211b42016-03-06 19:27:55 -070031 }
Simon Glass8ed64a42018-11-18 08:14:26 -070032
33 ptr = map_sysmem(addr, bytes);
Simon Glass94890462014-11-10 17:16:43 -070034 gd->malloc_ptr = ALIGN(new_ptr, sizeof(new_ptr));
Simon Glass65ba4122015-09-08 17:52:46 -060035
Simon Glass94890462014-11-10 17:16:43 -070036 return ptr;
37}
38
Simon Glass8ed64a42018-11-18 08:14:26 -070039void *malloc_simple(size_t bytes)
Simon Glassde9d70f2015-05-12 14:55:06 -060040{
Simon Glassde9d70f2015-05-12 14:55:06 -060041 void *ptr;
42
Simon Glass8ed64a42018-11-18 08:14:26 -070043 ptr = alloc_simple(bytes, 1);
44 if (!ptr)
45 return ptr;
Andrew F. Davis34597e42017-01-27 10:39:18 -060046
Simon Glass8ed64a42018-11-18 08:14:26 -070047 log_debug("%lx\n", (ulong)ptr);
48
49 return ptr;
50}
51
52void *memalign_simple(size_t align, size_t bytes)
53{
54 void *ptr;
55
56 ptr = alloc_simple(bytes, align);
57 if (!ptr)
58 return ptr;
59 log_debug("aligned to %lx\n", (ulong)ptr);
Simon Glass65ba4122015-09-08 17:52:46 -060060
Simon Glassde9d70f2015-05-12 14:55:06 -060061 return ptr;
62}
63
Hans de Goede9f9df6f2015-09-13 14:45:15 +020064#if CONFIG_IS_ENABLED(SYS_MALLOC_SIMPLE)
Simon Glass94890462014-11-10 17:16:43 -070065void *calloc(size_t nmemb, size_t elem_size)
66{
67 size_t size = nmemb * elem_size;
68 void *ptr;
69
70 ptr = malloc(size);
Simon Glass8ed64a42018-11-18 08:14:26 -070071 if (!ptr)
72 return ptr;
73 memset(ptr, '\0', size);
Simon Glass94890462014-11-10 17:16:43 -070074
75 return ptr;
76}
77#endif
Simon Glass8ed64a42018-11-18 08:14:26 -070078
79void malloc_simple_info(void)
80{
81 log_info("malloc_simple: %lx bytes used, %lx remain\n", gd->malloc_ptr,
82 CONFIG_VAL(SYS_MALLOC_F_LEN) - gd->malloc_ptr);
83}