mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	cache: Add L2 cache management for Andes AX45MP RISC-V core
I/O Coherence Port (IOCP) provides an AXI interface for connecting external non-caching masters, such as DMA controllers. The accesses from IOCP are coherent with D-Caches and L2 Cache. IOCP is a specification option and is disabled on the Renesas RZ/Five SoC due to this reason IP blocks using DMA will fail. The Andes AX45MP core has a Programmable Physical Memory Attributes (PMA) block that allows dynamic adjustment of memory attributes in the runtime. It contains a configurable amount of PMA entries implemented as CSR registers to control the attributes of memory locations in interest. Below are the memory attributes supported: * Device, Non-bufferable * Device, bufferable * Memory, Non-cacheable, Non-bufferable * Memory, Non-cacheable, Bufferable * Memory, Write-back, No-allocate * Memory, Write-back, Read-allocate * Memory, Write-back, Write-allocate * Memory, Write-back, Read and Write-allocate More info about PMA (section 10.3): Link: http://www.andestech.com/wp-content/uploads/AX45MP-1C-Rev.-5.0.0-Datasheet.pdf As a workaround for SoCs with IOCP disabled CMO needs to be handled by software. Firstly OpenSBI configures the memory region as "Memory, Non-cacheable, Bufferable" and passes this region as a global shared dma pool as a DT node. With DMA_GLOBAL_POOL enabled all DMA allocations happen from this region and synchronization callbacks are implemented to synchronize when doing DMA transactions. Example PMA region passes as a DT node from OpenSBI: reserved-memory { #address-cells = <2>; #size-cells = <2>; ranges; pma_resv0@58000000 { compatible = "shared-dma-pool"; reg = <0x0 0x58000000 0x0 0x08000000>; no-map; linux,dma-default; }; }; Signed-off-by: Lad Prabhakar <prabhakar.mahadev-lad.rj@bp.renesas.com> Reviewed-by: Conor Dooley <conor.dooley@microchip.com> Tested-by: Conor Dooley <conor.dooley@microchip.com> # tyre-kicking on a d1 Link: https://lore.kernel.org/r/20230818135723.80612-6-prabhakar.mahadev-lad.rj@bp.renesas.com Signed-off-by: Palmer Dabbelt <palmer@rivosinc.com>
This commit is contained in:
		
							parent
							
								
									3e7bf4685e
								
							
						
					
					
						commit
						d34599bcd2
					
				
					 6 changed files with 237 additions and 0 deletions
				
			
		|  | @ -20241,6 +20241,13 @@ S:	Supported | ||||||
| T:	git git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/staging.git | T:	git git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/staging.git | ||||||
| F:	drivers/staging/ | F:	drivers/staging/ | ||||||
| 
 | 
 | ||||||
|  | STANDALONE CACHE CONTROLLER DRIVERS | ||||||
|  | M:	Conor Dooley <conor@kernel.org> | ||||||
|  | L:	linux-riscv@lists.infradead.org | ||||||
|  | S:	Maintained | ||||||
|  | T:	git https://git.kernel.org/pub/scm/linux/kernel/git/conor/linux.git/ | ||||||
|  | F:	drivers/cache | ||||||
|  | 
 | ||||||
| STARFIRE/DURALAN NETWORK DRIVER | STARFIRE/DURALAN NETWORK DRIVER | ||||||
| M:	Ion Badulescu <ionut@badula.org> | M:	Ion Badulescu <ionut@badula.org> | ||||||
| S:	Odd Fixes | S:	Odd Fixes | ||||||
|  |  | ||||||
|  | @ -15,6 +15,8 @@ source "drivers/base/Kconfig" | ||||||
| 
 | 
 | ||||||
| source "drivers/bus/Kconfig" | source "drivers/bus/Kconfig" | ||||||
| 
 | 
 | ||||||
|  | source "drivers/cache/Kconfig" | ||||||
|  | 
 | ||||||
| source "drivers/connector/Kconfig" | source "drivers/connector/Kconfig" | ||||||
| 
 | 
 | ||||||
| source "drivers/firmware/Kconfig" | source "drivers/firmware/Kconfig" | ||||||
|  |  | ||||||
|  | @ -11,6 +11,7 @@ ifdef building_out_of_srctree | ||||||
| MAKEFLAGS += --include-dir=$(srctree) | MAKEFLAGS += --include-dir=$(srctree) | ||||||
| endif | endif | ||||||
| 
 | 
 | ||||||
|  | obj-y				+= cache/ | ||||||
| obj-y				+= irqchip/ | obj-y				+= irqchip/ | ||||||
| obj-y				+= bus/ | obj-y				+= bus/ | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										11
									
								
								drivers/cache/Kconfig
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								drivers/cache/Kconfig
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | ||||||
|  | # SPDX-License-Identifier: GPL-2.0 | ||||||
|  | menu "Cache Drivers" | ||||||
|  | 
 | ||||||
|  | config AX45MP_L2_CACHE | ||||||
|  | 	bool "Andes Technology AX45MP L2 Cache controller" | ||||||
|  | 	depends on RISCV_DMA_NONCOHERENT | ||||||
|  | 	select RISCV_NONSTANDARD_CACHE_OPS | ||||||
|  | 	help | ||||||
|  | 	  Support for the L2 cache controller on Andes Technology AX45MP platforms. | ||||||
|  | 
 | ||||||
|  | endmenu | ||||||
							
								
								
									
										3
									
								
								drivers/cache/Makefile
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								drivers/cache/Makefile
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | ||||||
|  | # SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | 
 | ||||||
|  | obj-$(CONFIG_AX45MP_L2_CACHE) += ax45mp_cache.o | ||||||
							
								
								
									
										213
									
								
								drivers/cache/ax45mp_cache.c
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								drivers/cache/ax45mp_cache.c
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,213 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * non-coherent cache functions for Andes AX45MP | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2023 Renesas Electronics Corp. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/cacheflush.h> | ||||||
|  | #include <linux/cacheinfo.h> | ||||||
|  | #include <linux/dma-direction.h> | ||||||
|  | #include <linux/of_address.h> | ||||||
|  | #include <linux/of_platform.h> | ||||||
|  | 
 | ||||||
|  | #include <asm/dma-noncoherent.h> | ||||||
|  | 
 | ||||||
|  | /* L2 cache registers */ | ||||||
|  | #define AX45MP_L2C_REG_CTL_OFFSET		0x8 | ||||||
|  | 
 | ||||||
|  | #define AX45MP_L2C_REG_C0_CMD_OFFSET		0x40 | ||||||
|  | #define AX45MP_L2C_REG_C0_ACC_OFFSET		0x48 | ||||||
|  | #define AX45MP_L2C_REG_STATUS_OFFSET		0x80 | ||||||
|  | 
 | ||||||
|  | /* D-cache operation */ | ||||||
|  | #define AX45MP_CCTL_L1D_VA_INVAL		0 /* Invalidate an L1 cache entry */ | ||||||
|  | #define AX45MP_CCTL_L1D_VA_WB			1 /* Write-back an L1 cache entry */ | ||||||
|  | 
 | ||||||
|  | /* L2 CCTL status */ | ||||||
|  | #define AX45MP_CCTL_L2_STATUS_IDLE		0 | ||||||
|  | 
 | ||||||
|  | /* L2 CCTL status cores mask */ | ||||||
|  | #define AX45MP_CCTL_L2_STATUS_C0_MASK		0xf | ||||||
|  | 
 | ||||||
|  | /* L2 cache operation */ | ||||||
|  | #define AX45MP_CCTL_L2_PA_INVAL			0x8 /* Invalidate an L2 cache entry */ | ||||||
|  | #define AX45MP_CCTL_L2_PA_WB			0x9 /* Write-back an L2 cache entry */ | ||||||
|  | 
 | ||||||
|  | #define AX45MP_L2C_REG_PER_CORE_OFFSET		0x10 | ||||||
|  | #define AX45MP_CCTL_L2_STATUS_PER_CORE_OFFSET	4 | ||||||
|  | 
 | ||||||
|  | #define AX45MP_L2C_REG_CN_CMD_OFFSET(n)	\ | ||||||
|  | 	(AX45MP_L2C_REG_C0_CMD_OFFSET + ((n) * AX45MP_L2C_REG_PER_CORE_OFFSET)) | ||||||
|  | #define AX45MP_L2C_REG_CN_ACC_OFFSET(n)	\ | ||||||
|  | 	(AX45MP_L2C_REG_C0_ACC_OFFSET + ((n) * AX45MP_L2C_REG_PER_CORE_OFFSET)) | ||||||
|  | #define AX45MP_CCTL_L2_STATUS_CN_MASK(n)	\ | ||||||
|  | 	(AX45MP_CCTL_L2_STATUS_C0_MASK << ((n) * AX45MP_CCTL_L2_STATUS_PER_CORE_OFFSET)) | ||||||
|  | 
 | ||||||
|  | #define AX45MP_CCTL_REG_UCCTLBEGINADDR_NUM	0x80b | ||||||
|  | #define AX45MP_CCTL_REG_UCCTLCOMMAND_NUM	0x80c | ||||||
|  | 
 | ||||||
|  | #define AX45MP_CACHE_LINE_SIZE			64 | ||||||
|  | 
 | ||||||
|  | struct ax45mp_priv { | ||||||
|  | 	void __iomem *l2c_base; | ||||||
|  | 	u32 ax45mp_cache_line_size; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static struct ax45mp_priv ax45mp_priv; | ||||||
|  | 
 | ||||||
|  | /* L2 Cache operations */ | ||||||
|  | static inline uint32_t ax45mp_cpu_l2c_get_cctl_status(void) | ||||||
|  | { | ||||||
|  | 	return readl(ax45mp_priv.l2c_base + AX45MP_L2C_REG_STATUS_OFFSET); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ax45mp_cpu_cache_operation(unsigned long start, unsigned long end, | ||||||
|  | 				       unsigned int l1_op, unsigned int l2_op) | ||||||
|  | { | ||||||
|  | 	unsigned long line_size = ax45mp_priv.ax45mp_cache_line_size; | ||||||
|  | 	void __iomem *base = ax45mp_priv.l2c_base; | ||||||
|  | 	int mhartid = smp_processor_id(); | ||||||
|  | 	unsigned long pa; | ||||||
|  | 
 | ||||||
|  | 	while (end > start) { | ||||||
|  | 		csr_write(AX45MP_CCTL_REG_UCCTLBEGINADDR_NUM, start); | ||||||
|  | 		csr_write(AX45MP_CCTL_REG_UCCTLCOMMAND_NUM, l1_op); | ||||||
|  | 
 | ||||||
|  | 		pa = virt_to_phys((void *)start); | ||||||
|  | 		writel(pa, base + AX45MP_L2C_REG_CN_ACC_OFFSET(mhartid)); | ||||||
|  | 		writel(l2_op, base + AX45MP_L2C_REG_CN_CMD_OFFSET(mhartid)); | ||||||
|  | 		while ((ax45mp_cpu_l2c_get_cctl_status() & | ||||||
|  | 			AX45MP_CCTL_L2_STATUS_CN_MASK(mhartid)) != | ||||||
|  | 			AX45MP_CCTL_L2_STATUS_IDLE) | ||||||
|  | 			; | ||||||
|  | 
 | ||||||
|  | 		start += line_size; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Write-back L1 and L2 cache entry */ | ||||||
|  | static inline void ax45mp_cpu_dcache_wb_range(unsigned long start, unsigned long end) | ||||||
|  | { | ||||||
|  | 	ax45mp_cpu_cache_operation(start, end, AX45MP_CCTL_L1D_VA_WB, | ||||||
|  | 				   AX45MP_CCTL_L2_PA_WB); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Invalidate the L1 and L2 cache entry */ | ||||||
|  | static inline void ax45mp_cpu_dcache_inval_range(unsigned long start, unsigned long end) | ||||||
|  | { | ||||||
|  | 	ax45mp_cpu_cache_operation(start, end, AX45MP_CCTL_L1D_VA_INVAL, | ||||||
|  | 				   AX45MP_CCTL_L2_PA_INVAL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ax45mp_dma_cache_inv(phys_addr_t paddr, size_t size) | ||||||
|  | { | ||||||
|  | 	unsigned long start = (unsigned long)phys_to_virt(paddr); | ||||||
|  | 	unsigned long end = start + size; | ||||||
|  | 	unsigned long line_size; | ||||||
|  | 	unsigned long flags; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(start == end)) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	line_size = ax45mp_priv.ax45mp_cache_line_size; | ||||||
|  | 
 | ||||||
|  | 	start = start & (~(line_size - 1)); | ||||||
|  | 	end = ((end + line_size - 1) & (~(line_size - 1))); | ||||||
|  | 
 | ||||||
|  | 	local_irq_save(flags); | ||||||
|  | 
 | ||||||
|  | 	ax45mp_cpu_dcache_inval_range(start, end); | ||||||
|  | 
 | ||||||
|  | 	local_irq_restore(flags); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ax45mp_dma_cache_wback(phys_addr_t paddr, size_t size) | ||||||
|  | { | ||||||
|  | 	unsigned long start = (unsigned long)phys_to_virt(paddr); | ||||||
|  | 	unsigned long end = start + size; | ||||||
|  | 	unsigned long line_size; | ||||||
|  | 	unsigned long flags; | ||||||
|  | 
 | ||||||
|  | 	line_size = ax45mp_priv.ax45mp_cache_line_size; | ||||||
|  | 	start = start & (~(line_size - 1)); | ||||||
|  | 	local_irq_save(flags); | ||||||
|  | 	ax45mp_cpu_dcache_wb_range(start, end); | ||||||
|  | 	local_irq_restore(flags); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ax45mp_dma_cache_wback_inv(phys_addr_t paddr, size_t size) | ||||||
|  | { | ||||||
|  | 	ax45mp_dma_cache_wback(paddr, size); | ||||||
|  | 	ax45mp_dma_cache_inv(paddr, size); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ax45mp_get_l2_line_size(struct device_node *np) | ||||||
|  | { | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	ret = of_property_read_u32(np, "cache-line-size", &ax45mp_priv.ax45mp_cache_line_size); | ||||||
|  | 	if (ret) { | ||||||
|  | 		pr_err("Failed to get cache-line-size, defaulting to 64 bytes\n"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (ax45mp_priv.ax45mp_cache_line_size != AX45MP_CACHE_LINE_SIZE) { | ||||||
|  | 		pr_err("Expected cache-line-size to be 64 bytes (found:%u)\n", | ||||||
|  | 		       ax45mp_priv.ax45mp_cache_line_size); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct riscv_nonstd_cache_ops ax45mp_cmo_ops __initdata = { | ||||||
|  | 	.wback = &ax45mp_dma_cache_wback, | ||||||
|  | 	.inv = &ax45mp_dma_cache_inv, | ||||||
|  | 	.wback_inv = &ax45mp_dma_cache_wback_inv, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct of_device_id ax45mp_cache_ids[] = { | ||||||
|  | 	{ .compatible = "andestech,ax45mp-cache" }, | ||||||
|  | 	{ /* sentinel */ } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int __init ax45mp_cache_init(void) | ||||||
|  | { | ||||||
|  | 	struct device_node *np; | ||||||
|  | 	struct resource res; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	np = of_find_matching_node(NULL, ax45mp_cache_ids); | ||||||
|  | 	if (!of_device_is_available(np)) | ||||||
|  | 		return -ENODEV; | ||||||
|  | 
 | ||||||
|  | 	ret = of_address_to_resource(np, 0, &res); | ||||||
|  | 	if (ret) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * If IOCP is present on the Andes AX45MP core riscv_cbom_block_size | ||||||
|  | 	 * will be 0 for sure, so we can definitely rely on it. If | ||||||
|  | 	 * riscv_cbom_block_size = 0 we don't need to handle CMO using SW any | ||||||
|  | 	 * more so we just return success here and only if its being set we | ||||||
|  | 	 * continue further in the probe path. | ||||||
|  | 	 */ | ||||||
|  | 	if (!riscv_cbom_block_size) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	ax45mp_priv.l2c_base = ioremap(res.start, resource_size(&res)); | ||||||
|  | 	if (!ax45mp_priv.l2c_base) | ||||||
|  | 		return -ENOMEM; | ||||||
|  | 
 | ||||||
|  | 	ret = ax45mp_get_l2_line_size(np); | ||||||
|  | 	if (ret) { | ||||||
|  | 		iounmap(ax45mp_priv.l2c_base); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	riscv_noncoherent_register_cache_ops(&ax45mp_cmo_ops); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | early_initcall(ax45mp_cache_init); | ||||||
		Loading…
	
		Reference in a new issue
	
	 Lad Prabhakar
						Lad Prabhakar