mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-28 09:03:52 +01:00 
			
		
		
		
	arm64: Reduce add_map() complexity
In the add_map() function, for each level it populates, it iterates from the root of the PT tree, making it ineficient if a mapping needs to occur past level 1. Instead, replace it with a recursive (and much simpler) algorithm that keeps the complexity as low as possible. With this, mapping 512GB at level 2 goes from several seconds down to not measurable on an A55 machine. We keep the block mappings at level 1 for now though. Signed-off-by: Marc Zyngier <maz@kernel.org> Signed-off-by: Pierre-Clément Tosi <ptosi@google.com> [ Paul: pick from the Android tree. Fixup Pierre's commit. Rebase to the upstream ] Signed-off-by: Ying-Chun Liu (PaulLiu) <paul.liu@linaro.org> Cc: Tom Rini <trini@konsulko.com> Link:96ad729cf4Link:6be9330601
This commit is contained in:
		| @@ -299,59 +299,57 @@ static void split_block(u64 *pte, int level) | ||||
| 	set_pte_table(pte, new_table); | ||||
| } | ||||
|  | ||||
| /* Add one mm_region map entry to the page tables */ | ||||
| static void map_range(u64 virt, u64 phys, u64 size, int level, | ||||
| 		      u64 *table, u64 attrs) | ||||
| { | ||||
| 	u64 map_size = BIT_ULL(level2shift(level)); | ||||
| 	int i, idx; | ||||
|  | ||||
| 	idx = (virt >> level2shift(level)) & (MAX_PTE_ENTRIES - 1); | ||||
| 	for (i = idx; size; i++) { | ||||
| 		u64 next_size, *next_table; | ||||
|  | ||||
| 		if (level >= 1 && | ||||
| 		    size >= map_size && !(virt & (map_size - 1))) { | ||||
| 			if (level == 3) | ||||
| 				table[i] = phys | attrs | PTE_TYPE_PAGE; | ||||
| 			else | ||||
| 				table[i] = phys | attrs; | ||||
|  | ||||
| 			virt += map_size; | ||||
| 			phys += map_size; | ||||
| 			size -= map_size; | ||||
|  | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		/* Going one level down */ | ||||
| 		if (pte_type(&table[i]) == PTE_TYPE_FAULT) | ||||
| 			set_pte_table(&table[i], create_table()); | ||||
|  | ||||
| 		next_table = (u64 *)(table[i] & GENMASK_ULL(47, PAGE_SHIFT)); | ||||
| 		next_size = min(map_size - (virt & (map_size - 1)), size); | ||||
|  | ||||
| 		map_range(virt, phys, next_size, level + 1, next_table, attrs); | ||||
|  | ||||
| 		virt += next_size; | ||||
| 		phys += next_size; | ||||
| 		size -= next_size; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void add_map(struct mm_region *map) | ||||
| { | ||||
| 	u64 *pte; | ||||
| 	u64 virt = map->virt; | ||||
| 	u64 phys = map->phys; | ||||
| 	u64 size = map->size; | ||||
| 	u64 attrs = map->attrs | PTE_TYPE_BLOCK | PTE_BLOCK_AF; | ||||
| 	u64 blocksize; | ||||
| 	int level; | ||||
| 	u64 *new_table; | ||||
| 	u64 va_bits; | ||||
| 	int level = 0; | ||||
|  | ||||
| 	while (size) { | ||||
| 		pte = find_pte(virt, 0); | ||||
| 		if (pte && (pte_type(pte) == PTE_TYPE_FAULT)) { | ||||
| 			debug("Creating table for virt 0x%llx\n", virt); | ||||
| 			new_table = create_table(); | ||||
| 			set_pte_table(pte, new_table); | ||||
| 		} | ||||
| 	get_tcr(NULL, &va_bits); | ||||
| 	if (va_bits < 39) | ||||
| 		level = 1; | ||||
|  | ||||
| 		for (level = 1; level < 4; level++) { | ||||
| 			pte = find_pte(virt, level); | ||||
| 			if (!pte) | ||||
| 				panic("pte not found\n"); | ||||
|  | ||||
| 			blocksize = 1ULL << level2shift(level); | ||||
| 			debug("Checking if pte fits for virt=%llx size=%llx blocksize=%llx\n", | ||||
| 			      virt, size, blocksize); | ||||
| 			if (size >= blocksize && !(virt & (blocksize - 1))) { | ||||
| 				/* Page fits, create block PTE */ | ||||
| 				debug("Setting PTE %p to block virt=%llx\n", | ||||
| 				      pte, virt); | ||||
| 				if (level == 3) | ||||
| 					*pte = phys | attrs | PTE_TYPE_PAGE; | ||||
| 				else | ||||
| 					*pte = phys | attrs; | ||||
| 				virt += blocksize; | ||||
| 				phys += blocksize; | ||||
| 				size -= blocksize; | ||||
| 				break; | ||||
| 			} else if (pte_type(pte) == PTE_TYPE_FAULT) { | ||||
| 				/* Page doesn't fit, create subpages */ | ||||
| 				debug("Creating subtable for virt 0x%llx blksize=%llx\n", | ||||
| 				      virt, blocksize); | ||||
| 				new_table = create_table(); | ||||
| 				set_pte_table(pte, new_table); | ||||
| 			} else if (pte_type(pte) == PTE_TYPE_BLOCK) { | ||||
| 				debug("Split block into subtable for virt 0x%llx blksize=0x%llx\n", | ||||
| 				      virt, blocksize); | ||||
| 				split_block(pte, level); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	map_range(map->virt, map->phys, map->size, level, | ||||
| 		  (u64 *)gd->arch.tlb_addr, attrs); | ||||
| } | ||||
|  | ||||
| enum pte_type { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user