lib: add memrange infrastructure
The memrange infrastructure allows for keeping track of the machine's physical address space. Each memory_range entry in a memory_ranges structure can be tagged with an arbitrary value. It supports merging and deleting ranges as well as filling in holes in the address space with a particular tag. The memrange infrastructure will serve as a shared implementation for address tracking by the MTRR and coreboot mem table code. Change-Id: Id5bea9d2a419114fca55c59af0fdca063551110e Signed-off-by: Aaron Durbin <adurbin@chromium.org> Reviewed-on: http://review.coreboot.org/2888 Tested-by: build bot (Jenkins) Reviewed-by: Stefan Reinauer <stefan.reinauer@coreboot.org>
This commit is contained in:
		
				
					committed by
					
						 Stefan Reinauer
						Stefan Reinauer
					
				
			
			
				
	
			
			
			
						parent
						
							3ece5ac40c
						
					
				
				
					commit
					a05a8522ce
				
			
							
								
								
									
										107
									
								
								src/include/memrange.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								src/include/memrange.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | ||||
| /* | ||||
|  * This file is part of the coreboot project. | ||||
|  * | ||||
|  * Copyright (C) 2013 Google, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; version 2 of the License. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA | ||||
|  */ | ||||
| #ifndef MEMRANGE_H_ | ||||
| #define MEMRANGE_H_ | ||||
|  | ||||
| #include <device/resource.h> | ||||
|  | ||||
| /* A memranges structure consists of a list of range_entry(s). The structure | ||||
|  * is exposed so that a memranges can be used on the stack if needed. */ | ||||
| struct memranges { | ||||
| 	struct range_entry *entries; | ||||
| }; | ||||
|  | ||||
| /* Each region within a memranges structure is represented by a | ||||
|  * range_entry structure. Use the associated range_entry_(base|end|size|tag) | ||||
|  * functions to interrogate its properties. i.e. don't rely on one's own | ||||
|  * interpretation of the fields. */ | ||||
| struct range_entry { | ||||
| 	resource_t begin; | ||||
| 	resource_t end; | ||||
| 	unsigned long tag; | ||||
| 	struct range_entry *next; | ||||
| }; | ||||
|  | ||||
| /* Return inclusive base address of memory range. */ | ||||
| static inline resource_t range_entry_base(const struct range_entry *r) | ||||
| { | ||||
| 	return r->begin; | ||||
| } | ||||
|  | ||||
| /* Return exclusive end address of memory range. */ | ||||
| static inline resource_t range_entry_end(const struct range_entry *r) | ||||
| { | ||||
| 	return r->end + 1; | ||||
| } | ||||
|  | ||||
| /* Return size of of memory range. */ | ||||
| static inline resource_t range_entry_size(const struct range_entry *r) | ||||
| { | ||||
| 	return r->end - r->begin + 1; | ||||
| } | ||||
|  | ||||
| static inline unsigned long range_entry_tag(const struct range_entry *r) | ||||
| { | ||||
| 	return r->tag; | ||||
| } | ||||
|  | ||||
| /* Iterate over each entry in a memranges structure. Ranges cannot | ||||
|  * be deleted while processing each entry as the list cannot be safely | ||||
|  * traversed after such an operation. | ||||
|  * r - range_entry pointer. | ||||
|  * ranges - memranges pointer */ | ||||
| #define memranges_each_entry(r, ranges) \ | ||||
| 	for (r = (ranges)->entries; r != NULL; r = r->next) | ||||
|  | ||||
| /* Initialize and fill a memranges structure according to the | ||||
|  * mask and match type for all memory resources. Tag each entry with the | ||||
|  * specified type. */ | ||||
| void memranges_init(struct memranges *ranges, | ||||
|                     unsigned long mask, unsigned long match, | ||||
|                     unsigned long tag); | ||||
|  | ||||
| /* Remove and free all entries within the memranges structure. */ | ||||
| void memranges_teardown(struct memranges *ranges); | ||||
|  | ||||
| /* Add memory resources that match with the corresponding mask and match. | ||||
|  * Each entry will be tagged with the provided tag. e.g.  To populate | ||||
|  * all cacheable memory resources in the range: | ||||
|  * memranges_add_resources(range, IORESOURCE_CACHEABLE, | ||||
|  *                            IORESROUCE_CACHEABLE, my_cacheable_tag); */ | ||||
| void memranges_add_resources(struct memranges *ranges, | ||||
|                              unsigned long mask, unsigned long match, | ||||
|                              unsigned long tag); | ||||
|  | ||||
| /* Fill all address ranges up to limit (exclusive) not covered by an entry by | ||||
|  * inserting new entries with the provided tag. */ | ||||
| void memranges_fill_holes_up_to(struct memranges *ranges, | ||||
|                                 resource_t limit, unsigned long tag); | ||||
|  | ||||
| /* Create a hole in the range by deleting/modifying entries that overlap with | ||||
|  * the region specified by base and size. */ | ||||
| void memranges_create_hole(struct memranges *ranges, | ||||
|                            resource_t base, resource_t size); | ||||
|  | ||||
| /* Insert a resource to the given memranges.  All existing ranges | ||||
|  * covered by range specified by base and size will be removed before a | ||||
|  * new one is added. */ | ||||
| void memranges_insert(struct memranges *ranges, | ||||
|                       resource_t base, resource_t size, unsigned long tag); | ||||
|  | ||||
| #endif /* MEMRANGE_H_ */ | ||||
| @@ -86,6 +86,7 @@ ramstage-$(CONFIG_TRACE) += trace.c | ||||
| ramstage-$(CONFIG_COLLECT_TIMESTAMPS) += timestamp.c | ||||
| ramstage-$(CONFIG_COVERAGE) += libgcov.c | ||||
| ramstage-$(CONFIG_MAINBOARD_DO_NATIVE_VGA_INIT) += edid.c | ||||
| ramstage-y += memrange.c | ||||
|  | ||||
| # The CBMEM implementations are chosen based on CONFIG_DYNAMIC_CBMEM. | ||||
| ifeq ($(CONFIG_DYNAMIC_CBMEM),y) | ||||
|   | ||||
							
								
								
									
										323
									
								
								src/lib/memrange.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										323
									
								
								src/lib/memrange.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,323 @@ | ||||
| /* | ||||
|  * This file is part of the coreboot project. | ||||
|  * | ||||
|  * Copyright (C) 2013 Google, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; version 2 of the License. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA | ||||
|  */ | ||||
| #include <stdlib.h> | ||||
| #include <console/console.h> | ||||
| #include <memrange.h> | ||||
|  | ||||
| /* Coreboot doesn't have a free() function. Therefore, keep a cache of | ||||
|  * free'd entries.  */ | ||||
| static struct range_entry *free_list; | ||||
|  | ||||
| static inline void range_entry_link(struct range_entry **prev_ptr, | ||||
|                                     struct range_entry *r) | ||||
| { | ||||
| 	r->next = *prev_ptr; | ||||
| 	*prev_ptr = r; | ||||
| } | ||||
|  | ||||
| static inline void range_entry_unlink(struct range_entry **prev_ptr, | ||||
|                                       struct range_entry *r) | ||||
| { | ||||
| 	*prev_ptr = r->next; | ||||
| 	r->next = NULL; | ||||
| } | ||||
|  | ||||
| static inline void range_entry_unlink_and_free(struct range_entry **prev_ptr, | ||||
|                                                struct range_entry *r) | ||||
| { | ||||
| 	range_entry_unlink(prev_ptr, r); | ||||
| 	range_entry_link(&free_list, r); | ||||
| } | ||||
|  | ||||
| static struct range_entry *alloc_range(void) | ||||
| { | ||||
| 	if (free_list != NULL) { | ||||
| 		struct range_entry *r; | ||||
|  | ||||
| 		r = free_list; | ||||
| 		range_entry_unlink(&free_list, r); | ||||
| 		return r; | ||||
| 	} | ||||
| 	return malloc(sizeof(struct range_entry)); | ||||
| } | ||||
|  | ||||
| static inline struct range_entry * | ||||
| range_list_add(struct range_entry **prev_ptr, resource_t begin, resource_t end, | ||||
|                unsigned long tag) | ||||
| { | ||||
| 	struct range_entry *new_entry; | ||||
|  | ||||
| 	new_entry = alloc_range(); | ||||
| 	if (new_entry == NULL) { | ||||
| 		printk(BIOS_ERR, "Could not allocate range_entry!\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	new_entry->begin = begin; | ||||
| 	new_entry->end = end; | ||||
| 	new_entry->tag = tag; | ||||
| 	range_entry_link(prev_ptr, new_entry); | ||||
|  | ||||
| 	return new_entry; | ||||
| } | ||||
|  | ||||
| static void merge_neighbor_entries(struct memranges *ranges) | ||||
| { | ||||
| 	struct range_entry *cur; | ||||
| 	struct range_entry *prev; | ||||
|  | ||||
| 	prev = NULL; | ||||
| 	/* Merge all neighbors and delete/free the leftover entry. */ | ||||
| 	for (cur = ranges->entries; cur != NULL; cur = cur->next) { | ||||
| 		/* First entry. Just set prev. */ | ||||
| 		if (prev == NULL) { | ||||
| 			prev = cur; | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		/* If the previous entry merges with the current update the | ||||
| 		 * previous entry to cover full range and delete current from | ||||
| 		 * the list. */ | ||||
| 		if (prev->end + 1 >= cur->begin && prev->tag == cur->tag) { | ||||
| 			prev->end = cur->end; | ||||
| 			range_entry_unlink_and_free(&prev->next, cur); | ||||
| 			/* Set cur to prev so cur->next is valid since cur | ||||
| 			 * was just unlinked and free. */ | ||||
| 			cur = prev; | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		prev = cur; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void remove_memranges(struct memranges *ranges, | ||||
|                              resource_t begin, resource_t end, | ||||
|                              unsigned long unused) | ||||
| { | ||||
| 	struct range_entry *cur; | ||||
| 	struct range_entry *next; | ||||
| 	struct range_entry **prev_ptr; | ||||
|  | ||||
| 	prev_ptr = &ranges->entries; | ||||
| 	for (cur = ranges->entries; cur != NULL; cur = next) { | ||||
| 		resource_t tmp_end; | ||||
|  | ||||
| 		/* Cache the next value to handle unlinks. */ | ||||
| 		next = cur->next; | ||||
|  | ||||
| 		/* No other ranges are affected. */ | ||||
| 		if (end < cur->begin) | ||||
| 			break; | ||||
|  | ||||
| 		/* The removal range starts after this one. */ | ||||
| 		if (begin > cur->end) { | ||||
| 			prev_ptr = &cur->next; | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		/* The removal range overlaps with the current entry either | ||||
| 		 * partially or fully. However, we need to adjust the removal | ||||
| 		 * range for any holes. */ | ||||
| 		if (begin <= cur->begin) { | ||||
| 			begin = cur->begin; | ||||
|  | ||||
| 			/* Full removal. */ | ||||
| 			if (end >= cur->end) { | ||||
| 				begin = cur->end + 1; | ||||
| 				range_entry_unlink_and_free(prev_ptr, cur); | ||||
| 				continue; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		/* prev_ptr can be set now that the unlink path wasn't taken. */ | ||||
| 		prev_ptr = &cur->next; | ||||
|  | ||||
| 		/* Clip the end fragment to do proper splitting. */ | ||||
| 		tmp_end = end; | ||||
| 		if (end > cur->end) | ||||
| 			tmp_end = cur->end; | ||||
|  | ||||
| 		/* Hole punched in middle of entry. */ | ||||
| 		if (begin > cur->begin && tmp_end < cur->end) { | ||||
| 			range_list_add(&cur->next, end + 1, cur->end, cur->tag); | ||||
| 			cur->end = begin - 1; | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		/* Removal at beginning. */ | ||||
| 		if (begin == cur->begin) | ||||
| 			cur->begin = tmp_end + 1; | ||||
|  | ||||
| 		/* Removal at end. */ | ||||
| 		if (tmp_end == cur->end) | ||||
| 			cur->end = begin - 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void merge_add_memranges(struct memranges *ranges, | ||||
|                                    resource_t begin, resource_t end, | ||||
|                                    unsigned long tag) | ||||
| { | ||||
| 	struct range_entry *cur; | ||||
| 	struct range_entry **prev_ptr; | ||||
|  | ||||
| 	prev_ptr = &ranges->entries; | ||||
|  | ||||
| 	/* Remove all existing entries covered by the range. */ | ||||
| 	remove_memranges(ranges, begin, end, -1); | ||||
|  | ||||
| 	/* Find the entry to place the new entry after. Since | ||||
| 	 * remove_memranges() was called above there is a guranteed | ||||
| 	 * spot for this new entry. */ | ||||
| 	for (cur = ranges->entries; cur != NULL; cur = cur->next) { | ||||
| 		/* Found insertion spot before current entry. */ | ||||
| 		if (end < cur->begin) | ||||
| 			break; | ||||
|  | ||||
| 		/* Keep track of previous entry to insert new entry after it. */ | ||||
| 		prev_ptr = &cur->next; | ||||
|  | ||||
| 		/* The new entry starts after this one. */ | ||||
| 		if (begin > cur->end) | ||||
| 			continue; | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	/* Add new entry and merge with neighbors. */ | ||||
| 	range_list_add(prev_ptr, begin, end, tag); | ||||
| 	merge_neighbor_entries(ranges); | ||||
| } | ||||
|  | ||||
| typedef void (*range_action_t)(struct memranges *ranges, | ||||
|                                resource_t begin, resource_t end, | ||||
|                                unsigned long tag); | ||||
|  | ||||
| static void do_action(struct memranges *ranges, | ||||
|                       resource_t base, resource_t size, unsigned long tag, | ||||
|                       range_action_t action) | ||||
| { | ||||
| 	resource_t end; | ||||
| 	resource_t begin; | ||||
|  | ||||
| 	/* The addresses are aligned to 4096 bytes: the begin address is | ||||
| 	 * aligned down while the end address is aligned up to be conservative | ||||
| 	 * about the full range covered. */ | ||||
| 	begin = ALIGN_DOWN(base, 4096); | ||||
| 	end = begin + size + (base - begin); | ||||
| 	end = ALIGN_UP(end, 4096) - 1; | ||||
| 	action(ranges, begin, end, tag); | ||||
| } | ||||
|  | ||||
| void memranges_create_hole(struct memranges *ranges, | ||||
|                               resource_t base, resource_t size) | ||||
| { | ||||
| 	do_action(ranges, base, size, -1, remove_memranges); | ||||
| } | ||||
|  | ||||
| void memranges_insert(struct memranges *ranges, | ||||
|                       resource_t base, resource_t size, unsigned long tag) | ||||
| { | ||||
| 	do_action(ranges, base, size, tag, merge_add_memranges); | ||||
| } | ||||
|  | ||||
| struct collect_context { | ||||
| 	struct memranges *ranges; | ||||
| 	unsigned long tag; | ||||
| }; | ||||
|  | ||||
| static void collect_ranges(void *gp, struct device *dev, struct resource *res) | ||||
| { | ||||
| 	struct collect_context *ctx = gp; | ||||
|  | ||||
| 	memranges_insert(ctx->ranges, res->base, res->size, ctx->tag); | ||||
| } | ||||
|  | ||||
| void memranges_add_resources(struct memranges *ranges, | ||||
|                                  unsigned long mask, unsigned long match, | ||||
|                                  unsigned long tag) | ||||
| { | ||||
| 	struct collect_context context; | ||||
|  | ||||
| 	/* Only deal with MEM resources. */ | ||||
| 	mask |= IORESOURCE_MEM; | ||||
| 	match |= IORESOURCE_MEM; | ||||
|  | ||||
| 	context.ranges = ranges; | ||||
| 	context.tag = tag; | ||||
| 	search_global_resources(mask, match, collect_ranges, &context); | ||||
| } | ||||
|  | ||||
| void memranges_init(struct memranges *ranges, | ||||
|                     unsigned long mask, unsigned long match, | ||||
|                     unsigned long tag) | ||||
| { | ||||
| 	ranges->entries = NULL; | ||||
| 	memranges_add_resources(ranges, mask, match, tag); | ||||
| } | ||||
|  | ||||
| void memranges_teardown(struct memranges *ranges) | ||||
| { | ||||
| 	while (ranges->entries != NULL) { | ||||
| 		range_entry_unlink_and_free(&ranges->entries, ranges->entries); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void memranges_fill_holes_up_to(struct memranges *ranges, | ||||
|                                 resource_t limit, unsigned long tag) | ||||
| { | ||||
| 	struct range_entry *cur; | ||||
| 	struct range_entry *prev; | ||||
|  | ||||
| 	prev = NULL; | ||||
| 	for (cur = ranges->entries; cur != NULL; cur = cur->next) { | ||||
| 		/* First entry. Just set prev. */ | ||||
| 		if (prev == NULL) { | ||||
| 			prev = cur; | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		/* If the previous entry does not directly preceed the current | ||||
| 		 * entry then add a new entry just after the previous one. */ | ||||
| 		if (range_entry_end(prev) != cur->begin) { | ||||
| 			resource_t end; | ||||
|  | ||||
| 			end = cur->begin - 1; | ||||
| 			if (end >= limit) | ||||
| 				end = limit - 1; | ||||
| 			range_list_add(&prev->next, range_entry_end(prev), | ||||
| 			               end, tag); | ||||
| 		} | ||||
|  | ||||
| 		prev = cur; | ||||
|  | ||||
| 		/* Hit the requested range limit. No other entries after this | ||||
| 		 * are affected. */ | ||||
| 		if (cur->begin >= limit) | ||||
| 			break; | ||||
| 	} | ||||
|  | ||||
| 	/* Handle the case where the limit was never reached. A new entry needs | ||||
| 	 * to be added to cover the range up to the limit. */ | ||||
| 	if (prev != NULL && range_entry_end(prev) < limit) | ||||
| 		range_list_add(&prev->next, range_entry_end(prev), | ||||
| 		               limit - 1, tag); | ||||
|  | ||||
| 	/* Merge all entries that were newly added. */ | ||||
| 	merge_neighbor_entries(ranges); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user