···1818 unsigned long end;1919};20202121+/*2222+ * Bank array is sorted by .start.2323+ * Banks don't overlap and there's at least one page gap2424+ * between adjacent bank entries.2525+ */2126struct sysmem_info {2227 int nr_banks;2328 struct meminfo bank[SYSMEM_BANKS_MAX];
+98-5
arch/xtensa/mm/init.c
···88 * for more details.99 *1010 * Copyright (C) 2001 - 2005 Tensilica Inc.1111+ * Copyright (C) 2014 Cadence Design Systems Inc.1112 *1213 * Chris Zankel <chris@zankel.net>1314 * Joe Taylor <joe@tensilica.com, joetylr@yahoo.com>···32313332struct sysmem_info sysmem __initdata;34333434+/*3535+ * Find bank with maximal .start such that bank.start <= start3636+ */3737+static inline struct meminfo * __init find_bank(unsigned long start)3838+{3939+ unsigned i;4040+ struct meminfo *it = NULL;4141+4242+ for (i = 0; i < sysmem.nr_banks; ++i)4343+ if (sysmem.bank[i].start <= start)4444+ it = sysmem.bank + i;4545+ else4646+ break;4747+ return it;4848+}4949+5050+/*5151+ * Move all memory banks starting at 'from' to a new place at 'to',5252+ * adjust nr_banks accordingly.5353+ * Both 'from' and 'to' must be inside the sysmem.bank.5454+ *5555+ * Returns: 0 (success), -ENOMEM (not enough space in the sysmem.bank).5656+ */5757+static int __init move_banks(struct meminfo *to, struct meminfo *from)5858+{5959+ unsigned n = sysmem.nr_banks - (from - sysmem.bank);6060+6161+ if (to > from && to - from + sysmem.nr_banks > SYSMEM_BANKS_MAX)6262+ return -ENOMEM;6363+ if (to != from)6464+ memmove(to, from, n * sizeof(struct meminfo));6565+ sysmem.nr_banks += to - from;6666+ return 0;6767+}6868+6969+/*7070+ * Add new bank to sysmem. Resulting sysmem is the union of bytes of the7171+ * original sysmem and the new bank.7272+ *7373+ * Returns: 0 (success), < 0 (error)7474+ */3575int __init add_sysmem_bank(unsigned long start, unsigned long end)3676{3737- if (sysmem.nr_banks >= SYSMEM_BANKS_MAX) {3838- pr_warn("Ignoring memory bank 0x%08lx size %ldKB\n",7777+ unsigned i;7878+ struct meminfo *it = NULL;7979+ unsigned long sz;8080+ unsigned long bank_sz = 0;8181+8282+ if (start == end ||8383+ (start < end) != (PAGE_ALIGN(start) < (end & PAGE_MASK))) {8484+ pr_warn("Ignoring small memory bank 0x%08lx size: %ld bytes\n",3985 start, end - start);4086 return -EINVAL;4187 }4242- sysmem.bank[sysmem.nr_banks].start = PAGE_ALIGN(start);4343- sysmem.bank[sysmem.nr_banks].end = end & PAGE_MASK;4444- sysmem.nr_banks++;45888989+ start = PAGE_ALIGN(start);9090+ end &= PAGE_MASK;9191+ sz = end - start;9292+9393+ it = find_bank(start);9494+9595+ if (it)9696+ bank_sz = it->end - it->start;9797+9898+ if (it && bank_sz >= start - it->start) {9999+ if (end - it->start > bank_sz)100100+ it->end = end;101101+ else102102+ return 0;103103+ } else {104104+ if (!it)105105+ it = sysmem.bank;106106+ else107107+ ++it;108108+109109+ if (it - sysmem.bank < sysmem.nr_banks &&110110+ it->start - start <= sz) {111111+ it->start = start;112112+ if (it->end - it->start < sz)113113+ it->end = end;114114+ else115115+ return 0;116116+ } else {117117+ if (move_banks(it + 1, it) < 0) {118118+ pr_warn("Ignoring memory bank 0x%08lx size %ld bytes\n",119119+ start, end - start);120120+ return -EINVAL;121121+ }122122+ it->start = start;123123+ it->end = end;124124+ return 0;125125+ }126126+ }127127+ sz = it->end - it->start;128128+ for (i = it + 1 - sysmem.bank; i < sysmem.nr_banks; ++i)129129+ if (sysmem.bank[i].start - it->start <= sz) {130130+ if (sz < sysmem.bank[i].end - it->start)131131+ it->end = sysmem.bank[i].end;132132+ } else {133133+ break;134134+ }135135+136136+ move_banks(it + 1, sysmem.bank + i);46137 return 0;47138}48139