9
0
Fork 0
barebox/common/memory.c

237 lines
5.1 KiB
C

/*
* memory.c
*
* Copyright (c) 2011 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
*
* See file CREDITS for list of people who contributed to this
* project.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <common.h>
#include <memory.h>
#include <of.h>
#include <init.h>
#include <libfdt.h>
#include <linux/ioport.h>
#include <asm-generic/memory_layout.h>
#include <asm/sections.h>
#include <malloc.h>
/*
* Begin and End of memory area for malloc(), and current "brk"
*/
static unsigned long malloc_start;
static unsigned long malloc_end;
static unsigned long malloc_brk;
unsigned long mem_malloc_start(void)
{
return malloc_start;
}
unsigned long mem_malloc_end(void)
{
return malloc_end;
}
#ifdef CONFIG_MALLOC_TLSF
#include <tlsf.h>
tlsf_pool tlsf_mem_pool;
#endif
void mem_malloc_init(void *start, void *end)
{
malloc_start = (unsigned long)start;
malloc_end = (unsigned long)end;
malloc_brk = malloc_start;
#ifdef CONFIG_MALLOC_TLSF
tlsf_mem_pool = tlsf_create(start, end - start + 1);
#endif
}
#ifndef __SANDBOX__
static int mem_malloc_resource(void)
{
/*
* Normally it's a bug when one of these fails,
* but we have some setups where some of these
* regions are outside of sdram in which case
* the following fails.
*/
request_sdram_region("malloc space",
malloc_start,
malloc_end - malloc_start + 1);
request_sdram_region("barebox",
(unsigned long)&_stext,
(unsigned long)&_etext -
(unsigned long)&_stext + 1);
request_sdram_region("bss",
(unsigned long)&__bss_start,
(unsigned long)&__bss_stop -
(unsigned long)&__bss_start + 1);
#ifdef STACK_BASE
request_sdram_region("stack", STACK_BASE, STACK_SIZE);
#endif
return 0;
}
coredevice_initcall(mem_malloc_resource);
#endif
static void *sbrk_no_zero(ptrdiff_t increment)
{
unsigned long old = malloc_brk;
unsigned long new = old + increment;
if ((new < malloc_start) || (new > malloc_end))
return NULL;
malloc_brk = new;
return (void *)old;
}
void *sbrk(ptrdiff_t increment)
{
void *old = sbrk_no_zero(increment);
/* Only clear increment, if valid address was returned */
if (old != NULL)
memset(old, 0, increment);
return old;
}
LIST_HEAD(memory_banks);
int barebox_add_memory_bank(const char *name, resource_size_t start,
resource_size_t size)
{
struct memory_bank *bank = xzalloc(sizeof(*bank));
struct device_d *dev;
bank->res = request_iomem_region(name, start, start + size - 1);
if (!bank->res)
return -EBUSY;
dev = add_mem_device(name, start, size, IORESOURCE_MEM_WRITEABLE);
bank->dev = dev;
bank->start = start;
bank->size = size;
list_add_tail(&bank->list, &memory_banks);
return 0;
}
/*
* Request a region from the registered sdram
*/
struct resource *request_sdram_region(const char *name, resource_size_t start,
resource_size_t size)
{
struct memory_bank *bank;
for_each_memory_bank(bank) {
struct resource *res;
res = request_region(bank->res, name, start, start + size - 1);
if (res)
return res;
}
return NULL;
}
int release_sdram_region(struct resource *res)
{
return release_region(res);
}
#ifdef CONFIG_OFTREE
/*
* Get cells len in bytes
* if #NNNN-cells property is 2 then len is 8
* otherwise len is 4
*/
static int get_cells_len(struct fdt_header *fdt, char *nr_cells_name)
{
const u32 *cell;
cell = fdt_getprop(fdt, 0, nr_cells_name, NULL);
if (cell && *cell == 2)
return 8;
return 4;
}
/*
* Write a 4 or 8 byte big endian cell
*/
static void write_cell(u8 *addr, u64 val, int size)
{
int shift = (size - 1) * 8;
while (size-- > 0) {
*addr++ = (val >> shift) & 0xff;
shift -= 8;
}
}
static int of_memory_fixup(struct fdt_header *fdt)
{
struct memory_bank *bank;
int err, nodeoffset;
int addr_cell_len, size_cell_len, len = 0;
u8 tmp[16 * 16]; /* Up to 64-bit address + 64-bit size */
/* update, or add and update /memory node */
nodeoffset = fdt_get_path_or_create(fdt, "/memory");
if (nodeoffset < 0)
return nodeoffset;
err = fdt_setprop(fdt, nodeoffset, "device_type", "memory",
sizeof("memory"));
if (err < 0) {
printf("WARNING: could not set %s %s.\n", "device_type",
fdt_strerror(err));
return err;
}
addr_cell_len = get_cells_len(fdt, "#address-cells");
size_cell_len = get_cells_len(fdt, "#size-cells");
for_each_memory_bank(bank) {
write_cell(tmp + len, bank->start, addr_cell_len);
len += addr_cell_len;
write_cell(tmp + len, bank->size, size_cell_len);
len += size_cell_len;
}
err = fdt_setprop(fdt, nodeoffset, "reg", tmp, len);
if (err < 0) {
printf("WARNING: could not set %s %s.\n",
"reg", fdt_strerror(err));
return err;
}
return 0;
}
static int of_register_memory_fixup(void)
{
return of_register_fixup(of_memory_fixup);
}
late_initcall(of_register_memory_fixup);
#endif