1eb0616c2d
The conversion to the generic kmap_atomic() implementation missed the fact
that xtensa's fixmap works bottom up while all other implementations work
top down. There is no real reason why xtensa needs to work that way.
Cure it by:
- Using the generic fix_to_virt()/virt_to_fix() functions which work top
down
- Adjusting the mapping defines
- Using the generic index calculation for the non cache aliasing case
- Making the cache colour offset reverse so the effective index is correct
While at it, remove the outdated and misleading comment above the fixmap
enum which originates from the initial copy&pasta of this code from i386.
[ Max: Fixed the off by one in the index calculation ]
Fixes: 629ed3f7da
("xtensa/mm/highmem: Switch to generic kmap atomic")
Reported-by: Max Filippov <jcmvbkbc@gmail.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Max Filippov <jcmvbkbc@gmail.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Max Filippov <jcmvbkbc@gmail.com>
Link: https://lore.kernel.org/r/20201116193253.23875-1-jcmvbkbc@gmail.com
39 lines
1.1 KiB
C
39 lines
1.1 KiB
C
/*
|
|
* fixmap.h: compile-time virtual memory allocation
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
* License. See the file "COPYING" in the main directory of this archive
|
|
* for more details.
|
|
*
|
|
* Copyright (C) 1998 Ingo Molnar
|
|
*
|
|
* Support of BIGMEM added by Gerhard Wichert, Siemens AG, July 1999
|
|
*/
|
|
|
|
#ifndef _ASM_FIXMAP_H
|
|
#define _ASM_FIXMAP_H
|
|
|
|
#ifdef CONFIG_HIGHMEM
|
|
#include <linux/threads.h>
|
|
#include <linux/pgtable.h>
|
|
#include <asm/kmap_size.h>
|
|
|
|
/* The map slots for temporary mappings via kmap_atomic/local(). */
|
|
enum fixed_addresses {
|
|
FIX_KMAP_BEGIN,
|
|
FIX_KMAP_END = FIX_KMAP_BEGIN +
|
|
(KM_MAX_IDX * NR_CPUS * DCACHE_N_COLORS) - 1,
|
|
__end_of_fixed_addresses
|
|
};
|
|
|
|
#define FIXADDR_END (XCHAL_KSEG_CACHED_VADDR - PAGE_SIZE)
|
|
#define FIXADDR_SIZE (__end_of_fixed_addresses << PAGE_SHIFT)
|
|
/* Enforce that FIXADDR_START is PMD aligned to handle cache aliasing */
|
|
#define FIXADDR_START ((FIXADDR_END - FIXADDR_SIZE) & PMD_MASK)
|
|
#define FIXADDR_TOP (FIXADDR_START + FIXADDR_SIZE - PAGE_SIZE)
|
|
|
|
#include <asm-generic/fixmap.h>
|
|
|
|
#endif /* CONFIG_HIGHMEM */
|
|
#endif
|