summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDoug Berger <opendmb@gmail.com>2019-07-16 19:26:24 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2019-07-16 22:23:21 -0400
commitc633324e311243586675e732249339685e5d6faa (patch)
tree8bf128fc3f110de26db69c8292049ccc7a52e893
parentc92d2f38563db20c20c8db2f98fa1349290477d5 (diff)
mm/cma.c: fail if fixed declaration can't be honored
The description of cma_declare_contiguous() indicates that if the 'fixed' argument is true the reserved contiguous area must be exactly at the address of the 'base' argument. However, the function currently allows the 'base', 'size', and 'limit' arguments to be silently adjusted to meet alignment constraints. This commit enforces the documented behavior through explicit checks that return an error if the region does not fit within a specified region. Link: http://lkml.kernel.org/r/1561422051-16142-1-git-send-email-opendmb@gmail.com Fixes: 5ea3b1b2f8ad ("cma: add placement specifier for "cma=" kernel parameter") Signed-off-by: Doug Berger <opendmb@gmail.com> Acked-by: Michal Nazarewicz <mina86@mina86.com> Cc: Yue Hu <huyue2@yulong.com> Cc: Mike Rapoport <rppt@linux.ibm.com> Cc: Laura Abbott <labbott@redhat.com> Cc: Peng Fan <peng.fan@nxp.com> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Marek Szyprowski <m.szyprowski@samsung.com> Cc: Andrey Konovalov <andreyknvl@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--mm/cma.c13
1 files changed, 13 insertions, 0 deletions
diff --git a/mm/cma.c b/mm/cma.c
index d415dfc0965e..7fe0b8356775 100644
--- a/mm/cma.c
+++ b/mm/cma.c
@@ -278,6 +278,12 @@ int __init cma_declare_contiguous(phys_addr_t base,
278 */ 278 */
279 alignment = max(alignment, (phys_addr_t)PAGE_SIZE << 279 alignment = max(alignment, (phys_addr_t)PAGE_SIZE <<
280 max_t(unsigned long, MAX_ORDER - 1, pageblock_order)); 280 max_t(unsigned long, MAX_ORDER - 1, pageblock_order));
281 if (fixed && base & (alignment - 1)) {
282 ret = -EINVAL;
283 pr_err("Region at %pa must be aligned to %pa bytes\n",
284 &base, &alignment);
285 goto err;
286 }
281 base = ALIGN(base, alignment); 287 base = ALIGN(base, alignment);
282 size = ALIGN(size, alignment); 288 size = ALIGN(size, alignment);
283 limit &= ~(alignment - 1); 289 limit &= ~(alignment - 1);
@@ -308,6 +314,13 @@ int __init cma_declare_contiguous(phys_addr_t base,
308 if (limit == 0 || limit > memblock_end) 314 if (limit == 0 || limit > memblock_end)
309 limit = memblock_end; 315 limit = memblock_end;
310 316
317 if (base + size > limit) {
318 ret = -EINVAL;
319 pr_err("Size (%pa) of region at %pa exceeds limit (%pa)\n",
320 &size, &base, &limit);
321 goto err;
322 }
323
311 /* Reserve memory */ 324 /* Reserve memory */
312 if (fixed) { 325 if (fixed) {
313 if (memblock_is_region_reserved(base, size) || 326 if (memblock_is_region_reserved(base, size) ||