- Timestamp:
- 2006-02-06T14:18:28Z (20 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- f5935ed
- Parents:
- 214f5bb
- Location:
- genarch
- Files:
- 
      - 3 edited
 
 - 
          
  include/mm/page_ht.h (modified) (2 diffs)
- 
          
  src/mm/as_ht.c (modified) (2 diffs)
- 
          
  src/mm/page_ht.c (modified) (6 diffs)
 
Legend:
- Unmodified
- Added
- Removed
- 
      genarch/include/mm/page_ht.hr214f5bb rc7ec94a4 29 29 /* 30 30 * This is the generic page hash table interface. 31 * Architectures that use single page hash table for32 * storing page translations must implement it.33 31 */ 34 32 … … 38 36 #include <mm/page.h> 39 37 #include <typedefs.h> 38 #include <arch/types.h> 39 #include <adt/list.h> 40 #include <adt/hash_table.h> 40 41 41 /** Page hash table size. */ 42 #define HT_WIDTH HT_WIDTH_ARCH 43 #define HT_SIZE (1<<HT_WIDTH) 44 #define HT_ENTRIES (HT_SIZE/sizeof(pte_t)) 42 #define PAGE_HT_KEYS 2 43 #define KEY_AS 0 44 #define KEY_PAGE 1 45 45 46 /** Hash function. 47 * 48 * @param page Virtual address. Only vpn bits will be used. 49 * @param asid Address space identifier. 50 * 51 * @return Pointer to hash table typed pte_t *. 52 */ 53 #define HT_HASH(page, asid) HT_HASH_ARCH(page, asid) 46 #define PAGE_HT_ENTRIES_BITS 13 47 #define PAGE_HT_ENTRIES (1<<PAGE_HT_ENTRIES_BITS) 54 48 55 /** Compare PTE with page and asid. 56 * 57 * @param page Virtual address. Only vpn bits will be used. 58 * @param asid Address space identifier. 59 * @param t PTE. 60 * 61 * @return 1 on match, 0 otherwise. 62 */ 63 #define HT_COMPARE(page, asid, t) HT_COMPARE_ARCH(page, asid, t) 64 65 /** Identify empty page hash table slots. 66 * 67 * @param t Pointer ro hash table typed pte_t *. 68 * 69 * @return 1 if the slot is empty, 0 otherwise. 70 */ 71 #define HT_SLOT_EMPTY(t) HT_SLOT_EMPTY_ARCH(t) 72 73 /** Invalidate/empty page hash table slot. 74 * 75 * @param t Address of the slot to be invalidated. 76 */ 77 #define HT_INVALIDATE_SLOT(t) HT_INVALIDATE_SLOT_ARCH(t) 78 79 /** Return next record in collision chain. 80 * 81 * @param t PTE. 82 * 83 * @return Successor of PTE or NULL. 84 */ 85 #define HT_GET_NEXT(t) HT_GET_NEXT_ARCH(t) 86 87 /** Set successor in collision chain. 88 * 89 * @param t PTE. 90 * @param s Successor or NULL. 91 */ 92 #define HT_SET_NEXT(t, s) HT_SET_NEXT_ARCH(t, s) 93 94 /** Set page hash table record. 95 * 96 * @param t PTE. 97 * @param page Virtual address. Only vpn bits will be used. 98 * @param asid Address space identifier. 99 * @param frame Physical address. Only pfn bits will be used. 100 * @param flags Flags. See mm/page.h. 101 */ 102 #define HT_SET_RECORD(t, page, asid, frame, flags) HT_SET_RECORD_ARCH(t, page, asid, frame, flags) 49 struct pte { 50 link_t link; /**< Page hash table link. */ 51 as_t *as; /**< Address space. */ 52 __address page; /**< Virtual memory page. */ 53 __address frame; /**< Physical memory frame. */ 54 int flags; 55 unsigned a : 1; /**< Accessed. */ 56 unsigned d : 1; /**< Dirty. */ 57 unsigned p : 1; /**< Present. */ 58 }; 103 59 104 60 extern page_operations_t page_ht_operations; 105 61 extern spinlock_t page_ht_lock; 106 62 107 extern pte_t *page_ht; 108 109 extern void ht_invalidate_all(void); 63 extern hash_table_t page_ht; 64 extern hash_table_operations_t ht_operations; 110 65 111 66 #endif 
- 
      genarch/src/mm/as_ht.cr214f5bb rc7ec94a4 34 34 #include <typedefs.h> 35 35 #include <memstr.h> 36 #include <adt/hash_table.h> 36 37 37 38 static pte_t *ht_create(int flags); … … 49 50 * @param flags Ignored. 50 51 * 51 * @return Address of global page hash table.52 * @return Returns NULL. 52 53 */ 53 54 pte_t *ht_create(int flags) 54 55 { 55 if (!page_ht) { 56 page_ht = (pte_t *) frame_alloc(HT_WIDTH - FRAME_WIDTH, FRAME_KA | FRAME_PANIC); 57 memsetb((__address) page_ht, HT_SIZE, 0); 56 if (flags & FLAG_AS_KERNEL) { 57 hash_table_create(&page_ht, PAGE_HT_ENTRIES, 2, &ht_operations); 58 58 } 59 return page_ht;59 return NULL; 60 60 } 
- 
      genarch/src/mm/page_ht.cr214f5bb rc7ec94a4 29 29 #include <genarch/mm/page_ht.h> 30 30 #include <mm/page.h> 31 #include <arch/mm/page.h> 31 32 #include <mm/frame.h> 32 33 #include <mm/heap.h> … … 40 41 #include <debug.h> 41 42 #include <memstr.h> 43 #include <adt/hash_table.h> 44 45 static index_t hash(__native key[]); 46 static bool compare(__native key[], count_t keys, link_t *item); 47 static void remove_callback(link_t *item); 48 49 static void ht_mapping_insert(as_t *as, __address page, __address frame, int flags); 50 static pte_t *ht_mapping_find(as_t *as, __address page); 42 51 43 52 /** … … 47 56 48 57 /** 49 * Page hash table pointer.58 * Page hash table. 50 59 * The page hash table may be accessed only when page_ht_lock is held. 51 60 */ 52 pte_t *page_ht = NULL; 53 54 static void ht_mapping_insert(as_t *as, __address page, __address frame, int flags); 55 static pte_t *ht_mapping_find(as_t *as, __address page); 61 hash_table_t page_ht; 62 63 /** Hash table operations for page hash table. */ 64 hash_table_operations_t ht_operations = { 65 .hash = hash, 66 .compare = compare, 67 .remove_callback = remove_callback 68 }; 56 69 57 70 page_operations_t page_ht_operations = { … … 60 73 }; 61 74 75 /** Compute page hash table index. 76 * 77 * @param key Array of two keys (i.e. page and address space). 78 * 79 * @return Index into page hash table. 80 */ 81 index_t hash(__native key[]) 82 { 83 as_t *as = (as_t *) key[KEY_AS]; 84 __address page = (__address) key[KEY_PAGE]; 85 index_t index; 86 87 /* 88 * Virtual page addresses have roughly the same probability 89 * of occurring. Least significant bits of VPN compose the 90 * hash index. 91 */ 92 index = ((page >> PAGE_WIDTH) & (PAGE_HT_ENTRIES-1)); 93 94 /* 95 * Address space structures are likely to be allocated from 96 * similar addresses. Least significant bits compose the 97 * hash index. 98 */ 99 index |= ((__native) as) & (PAGE_HT_ENTRIES-1); 100 101 return index; 102 } 103 104 /** Compare page hash table item with page and/or address space. 105 * 106 * @param key Array of one or two keys (i.e. page and/or address space). 107 * @param keys Number of keys passed. 108 * @param item Item to compare the keys with. 109 * 110 * @return true on match, false otherwise. 111 */ 112 bool compare(__native key[], count_t keys, link_t *item) 113 { 114 pte_t *t; 115 116 ASSERT(item); 117 ASSERT((keys > 0) && (keys <= PAGE_HT_KEYS)); 118 119 /* 120 * Convert item to PTE. 121 */ 122 t = list_get_instance(item, pte_t, link); 123 124 if (keys == PAGE_HT_KEYS) { 125 return (key[KEY_AS] == (__address) t->as) && (key[KEY_PAGE] == t->page); 126 } else { 127 return (key[KEY_AS] == (__address) t->as); 128 } 129 } 130 131 /** Callback on page hash table item removal. 132 * 133 * @param item Page hash table item being removed. 134 */ 135 void remove_callback(link_t *item) 136 { 137 pte_t *t; 138 139 ASSERT(item); 140 141 /* 142 * Convert item to PTE. 143 */ 144 t = list_get_instance(item, pte_t, link); 145 146 free(t); 147 } 148 62 149 /** Map page to frame using page hash table. 63 150 * … … 74 161 void ht_mapping_insert(as_t *as, __address page, __address frame, int flags) 75 162 { 76 pte_t *t , *u;163 pte_t *t; 77 164 ipl_t ipl; 165 __native key[2] = { (__address) as, page }; 78 166 79 167 ipl = interrupts_disable(); 80 168 spinlock_lock(&page_ht_lock); 81 169 82 t = HT_HASH(page, as->asid); 83 if (!HT_SLOT_EMPTY(t)) { 84 85 /* 86 * The slot is occupied. 87 * Walk through the collision chain and append the mapping to its end. 88 */ 89 90 do { 91 u = t; 92 if (HT_COMPARE(page, as->asid, t)) { 93 /* 94 * Nothing to do, 95 * the record is already there. 96 */ 97 spinlock_unlock(&page_ht_lock); 98 interrupts_restore(ipl); 99 return; 100 } 101 } while ((t = HT_GET_NEXT(t))); 102 103 t = (pte_t *) malloc(sizeof(pte_t)); /* FIXME: use slab allocator for this */ 104 if (!t) 105 panic("could not allocate memory\n"); 106 107 HT_SET_NEXT(u, t); 170 if (!hash_table_find(&page_ht, key)) { 171 t = (pte_t *) malloc(sizeof(pte_t)); 172 ASSERT(t != NULL); 173 174 hash_table_insert(&page_ht, key, &t->link); 108 175 } 109 110 HT_SET_RECORD(t, page, as->asid, frame, flags);111 HT_SET_NEXT(t, NULL);112 176 113 177 spinlock_unlock(&page_ht_lock); … … 128 192 pte_t *ht_mapping_find(as_t *as, __address page) 129 193 { 130 pte_t *t; 194 link_t *hlp; 195 pte_t *t = NULL; 196 __native key[2] = { (__address) as, page }; 131 197 132 198 spinlock_lock(&page_ht_lock); 133 t = HT_HASH(page, as->asid); 134 if (!HT_SLOT_EMPTY(t)) { 135 while (!HT_COMPARE(page, as->asid, t) && HT_GET_NEXT(t)) 136 t = HT_GET_NEXT(t); 137 t = HT_COMPARE(page, as->asid, t) ? t : NULL; 138 } else { 139 t = NULL; 140 } 199 200 hlp = hash_table_find(&page_ht, key); 201 if (hlp) 202 t = list_get_instance(hlp, pte_t, link); 203 141 204 spinlock_unlock(&page_ht_lock); 142 205 return t; 143 206 } 144 145 /** Invalidate page hash table.146 *147 * Interrupts must be disabled.148 */149 void ht_invalidate_all(void)150 {151 pte_t *t, *u;152 int i;153 154 spinlock_lock(&page_ht_lock);155 for (i = 0; i < HT_ENTRIES; i++) {156 if (!HT_SLOT_EMPTY(&page_ht[i])) {157 t = HT_GET_NEXT(&page_ht[i]);158 while (t) {159 u = t;160 t = HT_GET_NEXT(t);161 free(u); /* FIXME: use slab allocator for this */162 }163 HT_INVALIDATE_SLOT(&page_ht[i]);164 }165 }166 spinlock_unlock(&page_ht_lock);167 }
  Note:
 See   TracChangeset
 for help on using the changeset viewer.
  
