xref: /haiku/src/system/kernel/arch/m68k/paging/040/M68KVMTranslationMap040.cpp (revision caed67a8cba83913b9c21ac2b06ebc6bd1cb3111)
1 /*
2  * Copyright 2008-2010, Ingo Weinhold, ingo_weinhold@gmx.de.
3  * Copyright 2002-2007, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
4  * Distributed under the terms of the MIT License.
5  *
6  * Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
7  * Distributed under the terms of the NewOS License.
8  */
9 
10 
11 #include "paging/040/M68KVMTranslationMap040.h"
12 
13 #include <stdlib.h>
14 #include <string.h>
15 
16 #include <int.h>
17 #include <thread.h>
18 #include <slab/Slab.h>
19 #include <smp.h>
20 #include <util/AutoLock.h>
21 #include <util/ThreadAutoLock.h>
22 #include <util/queue.h>
23 #include <vm/vm_page.h>
24 #include <vm/vm_priv.h>
25 #include <vm/VMAddressSpace.h>
26 #include <vm/VMCache.h>
27 
28 #include "paging/040/M68KPagingMethod040.h"
29 #include "paging/040/M68KPagingStructures040.h"
30 #include "paging/m68k_physical_page_mapper.h"
31 
32 
33 #define TRACE_M68K_VM_TRANSLATION_MAP_040
34 #ifdef TRACE_M68K_VM_TRANSLATION_MAP_040
35 #	define TRACE(x...) dprintf(x)
36 #else
37 #	define TRACE(x...) ;
38 #endif
39 
40 
41 M68KVMTranslationMap040::M68KVMTranslationMap040()
42 	:
43 	fPagingStructures(NULL)
44 {
45 }
46 
47 
48 M68KVMTranslationMap040::~M68KVMTranslationMap040()
49 {
50 	if (fPagingStructures == NULL)
51 		return;
52 
53 	if (fPageMapper != NULL)
54 		fPageMapper->Delete();
55 
56 	if (fPagingStructures->pgroot_virt != NULL) {
57 		page_root_entry *pgroot_virt = fPagingStructures->pgroot_virt;
58 
59 		// cycle through and free all of the user space pgdirs & pgtables
60 		// since the size of tables don't match B_PAGE_SIZE,
61 		// we alloc several at once, based on modulos,
62 		// we make sure they are either all in the tree or none.
63 		for (uint32 i = VADDR_TO_PRENT(USER_BASE);
64 				i <= VADDR_TO_PRENT(USER_BASE + (USER_SIZE - 1)); i++) {
65 			addr_t pgdir_pn;
66 			page_directory_entry *pgdir;
67 			vm_page *dirpage;
68 
69 			if (PRE_TYPE(pgroot_virt[i]) == DT_INVALID)
70 				continue;
71 			if (PRE_TYPE(pgroot_virt[i]) != DT_ROOT) {
72 				panic("rtdir[%ld]: buggy descriptor type", i);
73 				return;
74 			}
75 			// XXX:suboptimal (done 8 times)
76 			pgdir_pn = PRE_TO_PN(pgroot_virt[i]);
77 			dirpage = vm_lookup_page(pgdir_pn);
78 			pgdir = &(((page_directory_entry *)dirpage)[i%NUM_DIRTBL_PER_PAGE]);
79 
80 			for (uint32 j = 0; j <= NUM_DIRENT_PER_TBL;
81 					j+=NUM_PAGETBL_PER_PAGE) {
82 				addr_t pgtbl_pn;
83 				page_table_entry *pgtbl;
84 				vm_page *page;
85 				if (PDE_TYPE(pgdir[j]) == DT_INVALID)
86 					continue;
87 				if (PDE_TYPE(pgdir[j]) != DT_DIR) {
88 					panic("pgroot[%ld][%ld]: buggy descriptor type", i, j);
89 					return;
90 				}
91 				pgtbl_pn = PDE_TO_PN(pgdir[j]);
92 				page = vm_lookup_page(pgtbl_pn);
93 				pgtbl = (page_table_entry *)page;
94 
95 				if (!page) {
96 					panic("destroy_tmap: didn't find pgtable page\n");
97 					return;
98 				}
99 				DEBUG_PAGE_ACCESS_START(page);
100 				vm_page_set_state(page, PAGE_STATE_FREE);
101 			}
102 			if (((i + 1) % NUM_DIRTBL_PER_PAGE) == 0) {
103 				DEBUG_PAGE_ACCESS_END(dirpage);
104 				vm_page_set_state(dirpage, PAGE_STATE_FREE);
105 			}
106 		}
107 
108 
109 
110 #if 0
111 //X86
112 		for (uint32 i = VADDR_TO_PDENT(USER_BASE);
113 				i <= VADDR_TO_PDENT(USER_BASE + (USER_SIZE - 1)); i++) {
114 			if ((fPagingStructures->pgdir_virt[i] & M68K_PDE_PRESENT) != 0) {
115 				addr_t address = fPagingStructures->pgdir_virt[i]
116 					& M68K_PDE_ADDRESS_MASK;
117 				vm_page* page = vm_lookup_page(address / B_PAGE_SIZE);
118 				if (!page)
119 					panic("destroy_tmap: didn't find pgtable page\n");
120 				DEBUG_PAGE_ACCESS_START(page);
121 				vm_page_set_state(page, PAGE_STATE_FREE);
122 			}
123 		}
124 #endif
125 	}
126 
127 	fPagingStructures->RemoveReference();
128 }
129 
130 
131 status_t
132 M68KVMTranslationMap040::Init(bool kernel)
133 {
134 	TRACE("M68KVMTranslationMap040::Init()\n");
135 
136 	M68KVMTranslationMap::Init(kernel);
137 
138 	fPagingStructures = new(std::nothrow) M68KPagingStructures040;
139 	if (fPagingStructures == NULL)
140 		return B_NO_MEMORY;
141 
142 	M68KPagingMethod040* method = M68KPagingMethod040::Method();
143 
144 	if (!kernel) {
145 		// user
146 		// allocate a physical page mapper
147 		status_t error = method->PhysicalPageMapper()
148 			->CreateTranslationMapPhysicalPageMapper(&fPageMapper);
149 		if (error != B_OK)
150 			return error;
151 
152 		// allocate the page root
153 		page_root_entry* virtualPageRoot = (page_root_entry*)memalign(
154 			SIZ_ROOTTBL, SIZ_ROOTTBL);
155 		if (virtualPageRoot == NULL)
156 			return B_NO_MEMORY;
157 
158 		// look up the page directory's physical address
159 		phys_addr_t physicalPageRoot;
160 		vm_get_page_mapping(VMAddressSpace::KernelID(),
161 			(addr_t)virtualPageRoot, &physicalPageRoot);
162 
163 		fPagingStructures->Init(virtualPageRoot, physicalPageRoot,
164 			method->KernelVirtualPageRoot());
165 	} else {
166 		// kernel
167 		// get the physical page mapper
168 		fPageMapper = method->KernelPhysicalPageMapper();
169 
170 		// we already know the kernel pgdir mapping
171 		fPagingStructures->Init(method->KernelVirtualPageRoot(),
172 			method->KernelPhysicalPageRoot(), NULL);
173 	}
174 
175 	return B_OK;
176 }
177 
178 
179 size_t
180 M68KVMTranslationMap040::MaxPagesNeededToMap(addr_t start, addr_t end) const
181 {
182 	size_t need;
183 	size_t pgdirs;
184 
185 	// If start == 0, the actual base address is not yet known to the caller and
186 	// we shall assume the worst case.
187 	if (start == 0) {
188 		// offset the range so it has the worst possible alignment
189 #warning M68K: FIXME?
190 		start = 1023 * B_PAGE_SIZE;
191 		end += 1023 * B_PAGE_SIZE;
192 	}
193 
194 	pgdirs = VADDR_TO_PRENT(end) + 1 - VADDR_TO_PRENT(start);
195 	// how much for page directories
196 	need = (pgdirs + NUM_DIRTBL_PER_PAGE - 1) / NUM_DIRTBL_PER_PAGE;
197 	// and page tables themselves
198 	need = ((pgdirs * NUM_DIRENT_PER_TBL) + NUM_PAGETBL_PER_PAGE - 1) / NUM_PAGETBL_PER_PAGE;
199 
200 	// better rounding when only 1 pgdir
201 	// XXX: do better for other cases
202 	if (pgdirs == 1) {
203 		need = 1;
204 		need += (VADDR_TO_PDENT(end) + 1 - VADDR_TO_PDENT(start) + NUM_PAGETBL_PER_PAGE - 1) / NUM_PAGETBL_PER_PAGE;
205 	}
206 
207 	return need;
208 }
209 
210 
211 status_t
212 M68KVMTranslationMap040::Map(addr_t va, phys_addr_t pa, uint32 attributes,
213 	uint32 memoryType, vm_page_reservation* reservation)
214 {
215 	TRACE("M68KVMTranslationMap040::Map: entry pa 0x%lx va 0x%lx\n", pa, va);
216 
217 /*
218 	dprintf("pgdir at 0x%x\n", pgdir);
219 	dprintf("index is %d\n", va / B_PAGE_SIZE / 1024);
220 	dprintf("final at 0x%x\n", &pgdir[va / B_PAGE_SIZE / 1024]);
221 	dprintf("value is 0x%x\n", *(int *)&pgdir[va / B_PAGE_SIZE / 1024]);
222 	dprintf("present bit is %d\n", pgdir[va / B_PAGE_SIZE / 1024].present);
223 	dprintf("addr is %d\n", pgdir[va / B_PAGE_SIZE / 1024].addr);
224 */
225 	page_root_entry *pr = fPagingStructures->pgroot_virt;
226 	page_directory_entry *pd;
227 	page_table_entry *pt;
228 	addr_t pd_pg, pt_pg;
229 	uint32 rindex, dindex, pindex;
230 
231 
232 	// check to see if a page directory exists for this range
233 	rindex = VADDR_TO_PRENT(va);
234 	if (PRE_TYPE(pr[rindex]) != DT_ROOT) {
235 		phys_addr_t pgdir;
236 		vm_page *page;
237 		uint32 i;
238 
239 		// we need to allocate a pgdir group
240 		page = vm_page_allocate_page(reservation,
241 			PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
242 
243 		DEBUG_PAGE_ACCESS_END(page);
244 
245 		pgdir = (phys_addr_t)page->physical_page_number * B_PAGE_SIZE;
246 
247 		TRACE("::Map: asked for free page for pgdir. 0x%lx\n", pgdir);
248 
249 		// for each pgdir on the allocated page:
250 		for (i = 0; i < NUM_DIRTBL_PER_PAGE; i++) {
251 			uint32 aindex = rindex & ~(NUM_DIRTBL_PER_PAGE-1); /* aligned */
252 			page_root_entry *apr = &pr[aindex + i];
253 
254 			// put in the pgroot
255 			M68KPagingMethod040::PutPageDirInPageRoot(apr, pgdir, attributes
256 				| ((attributes & B_USER_PROTECTION) != 0
257 						? B_WRITE_AREA : B_KERNEL_WRITE_AREA));
258 
259 			// update any other page roots, if it maps kernel space
260 			//XXX: suboptimal, should batch them
261 			if ((aindex+i) >= FIRST_KERNEL_PGDIR_ENT && (aindex+i)
262 					< (FIRST_KERNEL_PGDIR_ENT + NUM_KERNEL_PGDIR_ENTS))
263 				M68KPagingStructures040::UpdateAllPageDirs((aindex+i),
264 					pr[aindex+i]);
265 
266 			pgdir += SIZ_DIRTBL;
267 		}
268 		fMapCount++;
269 	}
270 	// now, fill in the pentry
271 	//XXX: is this required?
272 	Thread* thread = thread_get_current_thread();
273 	ThreadCPUPinner pinner(thread);
274 
275 	pd = (page_directory_entry*)MapperGetPageTableAt(
276 		PRE_TO_PA(pr[rindex]));
277 
278 	//pinner.Unlock();
279 
280 	// we want the table at rindex, not at rindex%(tbl/page)
281 	//pd += (rindex % NUM_DIRTBL_PER_PAGE) * NUM_DIRENT_PER_TBL;
282 
283 	// check to see if a page table exists for this range
284 	dindex = VADDR_TO_PDENT(va);
285 	if (PDE_TYPE(pd[dindex]) != DT_DIR) {
286 		phys_addr_t pgtable;
287 		vm_page *page;
288 		uint32 i;
289 
290 		// we need to allocate a pgtable group
291 		page = vm_page_allocate_page(reservation,
292 			PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
293 
294 		DEBUG_PAGE_ACCESS_END(page);
295 
296 		pgtable = (phys_addr_t)page->physical_page_number * B_PAGE_SIZE;
297 
298 		TRACE("::Map: asked for free page for pgtable. 0x%lx\n", pgtable);
299 
300 		// for each pgtable on the allocated page:
301 		for (i = 0; i < NUM_PAGETBL_PER_PAGE; i++) {
302 			uint32 aindex = dindex & ~(NUM_PAGETBL_PER_PAGE-1); /* aligned */
303 			page_directory_entry *apd = &pd[aindex + i];
304 
305 			// put in the pgdir
306 			M68KPagingMethod040::PutPageTableInPageDir(apd, pgtable, attributes
307 				| ((attributes & B_USER_PROTECTION) != 0
308 						? B_WRITE_AREA : B_KERNEL_WRITE_AREA));
309 
310 			// no need to update other page directories for kernel space;
311 			// the root-level already point to us.
312 
313 			pgtable += SIZ_PAGETBL;
314 		}
315 
316 #warning M68K: really mean map_count++ ??
317 		fMapCount++;
318 	}
319 
320 	// now, fill in the pentry
321 	//ThreadCPUPinner pinner(thread);
322 
323 	pt = (page_table_entry*)MapperGetPageTableAt(PDE_TO_PA(pd[dindex]));
324 	// we want the table at rindex, not at rindex%(tbl/page)
325 	//pt += (dindex % NUM_PAGETBL_PER_PAGE) * NUM_PAGEENT_PER_TBL;
326 
327 	pindex = VADDR_TO_PTENT(va);
328 
329 	ASSERT_PRINT((PTE_TYPE(pt[pindex]) != DT_INVALID) == 0,
330 		"virtual address: %#" B_PRIxADDR ", existing pte: %#" B_PRIx32, va,
331 		pt[pindex]);
332 
333 	M68KPagingMethod040::PutPageTableEntryInTable(&pt[pindex], pa, attributes,
334 		memoryType, fIsKernelMap);
335 
336 	pinner.Unlock();
337 
338 	// Note: We don't need to invalidate the TLB for this address, as previously
339 	// the entry was not present and the TLB doesn't cache those entries.
340 
341 	fMapCount++;
342 
343 	return B_OK;
344 }
345 
346 
347 status_t
348 M68KVMTranslationMap040::Unmap(addr_t start, addr_t end)
349 {
350 	start = ROUNDDOWN(start, B_PAGE_SIZE);
351 	if (start >= end)
352 		return B_OK;
353 
354 	TRACE("M68KVMTranslationMap040::Unmap: asked to free pages 0x%lx to 0x%lx\n", start, end);
355 
356 	page_root_entry *pr = fPagingStructures->pgroot_virt;
357 	page_directory_entry *pd;
358 	page_table_entry *pt;
359 	int index;
360 
361 	do {
362 		index = VADDR_TO_PRENT(start);
363 		if (PRE_TYPE(pr[index]) != DT_ROOT) {
364 			// no pagedir here, move the start up to access the next page
365 			// dir group
366 			start = ROUNDUP(start + 1, kPageDirAlignment);
367 			continue;
368 		}
369 
370 		Thread* thread = thread_get_current_thread();
371 		ThreadCPUPinner pinner(thread);
372 
373 		pd = (page_directory_entry*)MapperGetPageTableAt(
374 			PRE_TO_PA(pr[index]));
375 		// we want the table at rindex, not at rindex%(tbl/page)
376 		//pd += (index % NUM_DIRTBL_PER_PAGE) * NUM_DIRENT_PER_TBL;
377 
378 
379 		index = VADDR_TO_PDENT(start);
380 		if (PDE_TYPE(pd[index]) != DT_DIR) {
381 			// no pagedir here, move the start up to access the next page
382 			// table group
383 			start = ROUNDUP(start + 1, kPageTableAlignment);
384 			continue;
385 		}
386 
387 		pt = (page_table_entry*)MapperGetPageTableAt(
388 			PDE_TO_PA(pd[index]));
389 		// we want the table at rindex, not at rindex%(tbl/page)
390 		//pt += (index % NUM_PAGETBL_PER_PAGE) * NUM_PAGEENT_PER_TBL;
391 
392 		for (index = VADDR_TO_PTENT(start);
393 				(index < NUM_PAGEENT_PER_TBL) && (start < end);
394 				index++, start += B_PAGE_SIZE) {
395 			if (PTE_TYPE(pt[index]) != DT_PAGE
396 				&& PTE_TYPE(pt[index]) != DT_INDIRECT) {
397 				// page mapping not valid
398 				continue;
399 			}
400 
401 			TRACE("::Unmap: removing page 0x%lx\n", start);
402 
403 			page_table_entry oldEntry
404 				= M68KPagingMethod040::ClearPageTableEntry(&pt[index]);
405 			fMapCount--;
406 
407 			if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
408 				// Note, that we only need to invalidate the address, if the
409 				// accessed flags was set, since only then the entry could have
410 				// been in any TLB.
411 				InvalidatePage(start);
412 			}
413 		}
414 	} while (start != 0 && start < end);
415 
416 	return B_OK;
417 }
418 
419 
420 /*!	Caller must have locked the cache of the page to be unmapped.
421 	This object shouldn't be locked.
422 */
423 status_t
424 M68KVMTranslationMap040::UnmapPage(VMArea* area, addr_t address,
425 	bool updatePageQueue)
426 {
427 	ASSERT(address % B_PAGE_SIZE == 0);
428 
429 	page_root_entry* pr = fPagingStructures->pgroot_virt;
430 
431 	TRACE("M68KVMTranslationMap040::UnmapPage(%#" B_PRIxADDR ")\n", address);
432 
433 	RecursiveLocker locker(fLock);
434 
435 	int index;
436 
437 	index = VADDR_TO_PRENT(address);
438 	if (PRE_TYPE(pr[index]) == DT_ROOT)
439 		return B_ENTRY_NOT_FOUND;
440 
441 	ThreadCPUPinner pinner(thread_get_current_thread());
442 
443 	page_table_entry* pd = (page_table_entry*)MapperGetPageTableAt(
444 		pr[index] & M68K_PRE_ADDRESS_MASK);
445 
446 	index = VADDR_TO_PDENT(address);
447 	if (PDE_TYPE(pd[index]) == DT_DIR)
448 		return B_ENTRY_NOT_FOUND;
449 
450 	page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
451 		pd[index] & M68K_PDE_ADDRESS_MASK);
452 
453 	index = VADDR_TO_PTENT(address);
454 	if (PTE_TYPE(pt[index]) == DT_INDIRECT) {
455 		phys_addr_t indirectAddress = PIE_TO_TA(pt[index]);
456 		pt = (page_table_entry*)MapperGetPageTableAt(
457 			PIE_TO_TA(pt[index]), true);
458 		index = 0; // single descriptor
459 	}
460 
461 	page_table_entry oldEntry = M68KPagingMethod040::ClearPageTableEntry(
462 		&pt[index]);
463 
464 	pinner.Unlock();
465 
466 	if (PTE_TYPE(oldEntry) != DT_PAGE) {
467 		// page mapping not valid
468 		return B_ENTRY_NOT_FOUND;
469 	}
470 
471 	fMapCount--;
472 
473 	if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
474 		// Note, that we only need to invalidate the address, if the
475 		// accessed flags was set, since only then the entry could have been
476 		// in any TLB.
477 		InvalidatePage(address);
478 		Flush();
479 
480 		// NOTE: Between clearing the page table entry and Flush() other
481 		// processors (actually even this processor with another thread of the
482 		// same team) could still access the page in question via their cached
483 		// entry. We can obviously lose a modified flag in this case, with the
484 		// effect that the page looks unmodified (and might thus be recycled),
485 		// but is actually modified.
486 		// In most cases this is harmless, but for vm_remove_all_page_mappings()
487 		// this is actually a problem.
488 		// Interestingly FreeBSD seems to ignore this problem as well
489 		// (cf. pmap_remove_all()), unless I've missed something.
490 	}
491 
492 	locker.Detach();
493 		// PageUnmapped() will unlock for us
494 
495 	PageUnmapped(area, (oldEntry & M68K_PTE_ADDRESS_MASK) / B_PAGE_SIZE,
496 		(oldEntry & M68K_PTE_ACCESSED) != 0, (oldEntry & M68K_PTE_DIRTY) != 0,
497 		updatePageQueue);
498 
499 	return B_OK;
500 }
501 
502 
503 void
504 M68KVMTranslationMap040::UnmapPages(VMArea* area, addr_t base, size_t size,
505 	bool updatePageQueue)
506 {
507 	int index;
508 
509 	if (size == 0)
510 		return;
511 
512 	addr_t start = base;
513 	addr_t end = base + size - 1;
514 
515 	TRACE("M68KVMTranslationMap040::UnmapPages(%p, %#" B_PRIxADDR ", %#"
516 		B_PRIxADDR ")\n", area, start, end);
517 
518 	page_root_entry* pr = fPagingStructures->pgroot_virt;
519 
520 	VMAreaMappings queue;
521 
522 	RecursiveLocker locker(fLock);
523 
524 	do {
525 		index = VADDR_TO_PRENT(start);
526 		if (PRE_TYPE(pr[index]) != DT_ROOT) {
527 			// no page table here, move the start up to access the next page
528 			// table
529 			start = ROUNDUP(start + 1, kPageDirAlignment);
530 			continue;
531 		}
532 
533 		Thread* thread = thread_get_current_thread();
534 		ThreadCPUPinner pinner(thread);
535 
536 		page_table_entry* pd = (page_directory_entry*)MapperGetPageTableAt(
537 			pr[index] & M68K_PRE_ADDRESS_MASK);
538 
539 		index = VADDR_TO_PDENT(start);
540 		if (PDE_TYPE(pd[index]) != DT_DIR) {
541 			// no page table here, move the start up to access the next page
542 			// table
543 			start = ROUNDUP(start + 1, kPageTableAlignment);
544 			continue;
545 		}
546 
547 		page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
548 			pd[index] & M68K_PDE_ADDRESS_MASK);
549 
550 		for (index = VADDR_TO_PTENT(start); (index < 1024) && (start < end);
551 				index++, start += B_PAGE_SIZE) {
552 			page_table_entry *e = &pt[index];
553 			// fetch indirect descriptor
554 			//XXX:clear the indirect descriptor too??
555 			if (PTE_TYPE(pt[index]) == DT_INDIRECT) {
556 				phys_addr_t indirectAddress = PIE_TO_TA(pt[index]);
557 				e = (page_table_entry*)MapperGetPageTableAt(
558 					PIE_TO_TA(pt[index]));
559 			}
560 
561 			page_table_entry oldEntry
562 				= M68KPagingMethod040::ClearPageTableEntry(e);
563 			if (PTE_TYPE(oldEntry) != DT_PAGE)
564 				continue;
565 
566 			fMapCount--;
567 
568 			if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
569 				// Note, that we only need to invalidate the address, if the
570 				// accessed flags was set, since only then the entry could have
571 				// been in any TLB.
572 				InvalidatePage(start);
573 			}
574 
575 			if (area->cache_type != CACHE_TYPE_DEVICE) {
576 				// get the page
577 				vm_page* page = vm_lookup_page(
578 					(oldEntry & M68K_PTE_ADDRESS_MASK) / B_PAGE_SIZE);
579 				ASSERT(page != NULL);
580 
581 				DEBUG_PAGE_ACCESS_START(page);
582 
583 				// transfer the accessed/dirty flags to the page
584 				if ((oldEntry & M68K_PTE_ACCESSED) != 0)
585 					page->accessed = true;
586 				if ((oldEntry & M68K_PTE_DIRTY) != 0)
587 					page->modified = true;
588 
589 				// remove the mapping object/decrement the wired_count of the
590 				// page
591 				if (area->wiring == B_NO_LOCK) {
592 					vm_page_mapping* mapping = NULL;
593 					vm_page_mappings::Iterator iterator
594 						= page->mappings.GetIterator();
595 					while ((mapping = iterator.Next()) != NULL) {
596 						if (mapping->area == area)
597 							break;
598 					}
599 
600 					ASSERT(mapping != NULL);
601 
602 					area->mappings.Remove(mapping);
603 					page->mappings.Remove(mapping);
604 					queue.Add(mapping);
605 				} else
606 					page->DecrementWiredCount();
607 
608 				if (!page->IsMapped()) {
609 					atomic_add(&gMappedPagesCount, -1);
610 
611 					if (updatePageQueue) {
612 						if (page->Cache()->temporary)
613 							vm_page_set_state(page, PAGE_STATE_INACTIVE);
614 						else if (page->modified)
615 							vm_page_set_state(page, PAGE_STATE_MODIFIED);
616 						else
617 							vm_page_set_state(page, PAGE_STATE_CACHED);
618 					}
619 				}
620 
621 				DEBUG_PAGE_ACCESS_END(page);
622 			}
623 		}
624 
625 		Flush();
626 			// flush explicitly, since we directly use the lock
627 	} while (start != 0 && start < end);
628 
629 	// TODO: As in UnmapPage() we can lose page dirty flags here. ATM it's not
630 	// really critical here, as in all cases this method is used, the unmapped
631 	// area range is unmapped for good (resized/cut) and the pages will likely
632 	// be freed.
633 
634 	locker.Unlock();
635 
636 	// free removed mappings
637 	bool isKernelSpace = area->address_space == VMAddressSpace::Kernel();
638 	uint32 freeFlags = CACHE_DONT_WAIT_FOR_MEMORY
639 		| (isKernelSpace ? CACHE_DONT_LOCK_KERNEL_SPACE : 0);
640 	while (vm_page_mapping* mapping = queue.RemoveHead())
641 		vm_free_page_mapping(mapping->page->physical_page_number, mapping, freeFlags);
642 }
643 
644 
645 void
646 M68KVMTranslationMap040::UnmapArea(VMArea* area, bool deletingAddressSpace,
647 	bool ignoreTopCachePageFlags)
648 {
649 	if (area->cache_type == CACHE_TYPE_DEVICE || area->wiring != B_NO_LOCK) {
650 		M68KVMTranslationMap040::UnmapPages(area, area->Base(), area->Size(),
651 			true);
652 		return;
653 	}
654 
655 	bool unmapPages = !deletingAddressSpace || !ignoreTopCachePageFlags;
656 
657 	page_root_entry* pr = fPagingStructures->pgroot_virt;
658 
659 	RecursiveLocker locker(fLock);
660 
661 	VMAreaMappings mappings;
662 	mappings.MoveFrom(&area->mappings);
663 
664 	for (VMAreaMappings::Iterator it = mappings.GetIterator();
665 			vm_page_mapping* mapping = it.Next();) {
666 		vm_page* page = mapping->page;
667 		page->mappings.Remove(mapping);
668 
669 		VMCache* cache = page->Cache();
670 
671 		bool pageFullyUnmapped = false;
672 		if (!page->IsMapped()) {
673 			atomic_add(&gMappedPagesCount, -1);
674 			pageFullyUnmapped = true;
675 		}
676 
677 		if (unmapPages || cache != area->cache) {
678 			addr_t address = area->Base()
679 				+ ((page->cache_offset * B_PAGE_SIZE) - area->cache_offset);
680 
681 			int index;
682 			index = VADDR_TO_PRENT(address);
683 			if (PRE_TYPE(pr[index]) != DT_ROOT) {
684 				panic("page %p has mapping for area %p (%#" B_PRIxADDR "), but "
685 					"has no page root entry", page, area, address);
686 				continue;
687 			}
688 
689 			ThreadCPUPinner pinner(thread_get_current_thread());
690 
691 			page_directory_entry* pd
692 				= (page_directory_entry*)MapperGetPageTableAt(
693 					pr[index] & M68K_PRE_ADDRESS_MASK);
694 
695 			index = VADDR_TO_PDENT(address);
696 			if (PDE_TYPE(pr[index]) != DT_DIR) {
697 				panic("page %p has mapping for area %p (%#" B_PRIxADDR "), but "
698 					"has no page dir entry", page, area, address);
699 				continue;
700 			}
701 
702 			page_table_entry* pt
703 				= (page_table_entry*)MapperGetPageTableAt(
704 					pd[index] & M68K_PDE_ADDRESS_MASK);
705 
706 			//XXX:M68K: DT_INDIRECT here?
707 
708 			page_table_entry oldEntry
709 				= M68KPagingMethod040::ClearPageTableEntry(
710 					&pt[VADDR_TO_PTENT(address)]);
711 
712 			pinner.Unlock();
713 
714 			if (PTE_TYPE(oldEntry) != DT_PAGE) {
715 				panic("page %p has mapping for area %p (%#" B_PRIxADDR "), but "
716 					"has no page table entry", page, area, address);
717 				continue;
718 			}
719 
720 			// transfer the accessed/dirty flags to the page and invalidate
721 			// the mapping, if necessary
722 			if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
723 				page->accessed = true;
724 
725 				if (!deletingAddressSpace)
726 					InvalidatePage(address);
727 			}
728 
729 			if ((oldEntry & M68K_PTE_DIRTY) != 0)
730 				page->modified = true;
731 
732 			if (pageFullyUnmapped) {
733 				DEBUG_PAGE_ACCESS_START(page);
734 
735 				if (cache->temporary)
736 					vm_page_set_state(page, PAGE_STATE_INACTIVE);
737 				else if (page->modified)
738 					vm_page_set_state(page, PAGE_STATE_MODIFIED);
739 				else
740 					vm_page_set_state(page, PAGE_STATE_CACHED);
741 
742 				DEBUG_PAGE_ACCESS_END(page);
743 			}
744 		}
745 
746 		fMapCount--;
747 	}
748 
749 	Flush();
750 		// flush explicitely, since we directly use the lock
751 
752 	locker.Unlock();
753 
754 	bool isKernelSpace = area->address_space == VMAddressSpace::Kernel();
755 	uint32 freeFlags = CACHE_DONT_WAIT_FOR_MEMORY
756 		| (isKernelSpace ? CACHE_DONT_LOCK_KERNEL_SPACE : 0);
757 	while (vm_page_mapping* mapping = mappings.RemoveHead())
758 		vm_free_page_mapping(mapping->page->physical_page_number, mapping, freeFlags);
759 }
760 
761 
762 status_t
763 M68KVMTranslationMap040::Query(addr_t va, phys_addr_t *_physical,
764 	uint32 *_flags)
765 {
766 	// default the flags to not present
767 	*_flags = 0;
768 	*_physical = 0;
769 	TRACE("040::Query(0x%lx,)\n", va);
770 
771 	int index = VADDR_TO_PRENT(va);
772 	page_root_entry *pr = fPagingStructures->pgroot_virt;
773 	if (PRE_TYPE(pr[index]) != DT_ROOT) {
774 		// no pagetable here
775 		return B_OK;
776 	}
777 
778 	Thread* thread = thread_get_current_thread();
779 	ThreadCPUPinner pinner(thread);
780 
781 	page_directory_entry* pd = (page_directory_entry*)MapperGetPageTableAt(
782 		pr[index] & M68K_PDE_ADDRESS_MASK);
783 
784 	index = VADDR_TO_PDENT(va);
785 	if (PDE_TYPE(pd[index]) != DT_DIR) {
786 		// no pagetable here
787 		return B_OK;
788 	}
789 
790 	page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
791 		pd[index] & M68K_PDE_ADDRESS_MASK);
792 
793 	index = VADDR_TO_PTENT(va);
794 	if (PTE_TYPE(pt[index]) == DT_INDIRECT) {
795 		pt = (page_table_entry*)MapperGetPageTableAt(
796 			pt[index] & M68K_PIE_ADDRESS_MASK);
797 		index = 0;
798 	}
799 
800 	page_table_entry entry = pt[index];
801 
802 	*_physical = entry & M68K_PTE_ADDRESS_MASK;
803 
804 	// read in the page state flags
805 	if ((entry & M68K_PTE_SUPERVISOR) == 0) {
806 		*_flags |= ((entry & M68K_PTE_READONLY) == 0 ? B_WRITE_AREA : 0)
807 			| B_READ_AREA;
808 	}
809 
810 	*_flags |= ((entry & M68K_PTE_READONLY) == 0 ? B_KERNEL_WRITE_AREA : 0)
811 		| B_KERNEL_READ_AREA
812 		| ((entry & M68K_PTE_DIRTY) != 0 ? PAGE_MODIFIED : 0)
813 		| ((entry & M68K_PTE_ACCESSED) != 0 ? PAGE_ACCESSED : 0)
814 		| ((PTE_TYPE(entry) == DT_PAGE) ? PAGE_PRESENT : 0);
815 
816 	pinner.Unlock();
817 
818 	TRACE("query_tmap: returning pa 0x%lx for va 0x%lx\n", *_physical, va);
819 
820 	return B_OK;
821 }
822 
823 
824 status_t
825 M68KVMTranslationMap040::QueryInterrupt(addr_t va, phys_addr_t *_physical,
826 	uint32 *_flags)
827 {
828 	*_flags = 0;
829 	*_physical = 0;
830 	TRACE("040::QueryInterrupt(0x%lx,)\n", va);
831 
832 	int index = VADDR_TO_PRENT(va);
833 	page_root_entry* pr = fPagingStructures->pgroot_virt;
834 	if (PRE_TYPE(pr[index]) != DT_ROOT) {
835 		// no pagetable here
836 		return B_OK;
837 	}
838 
839 	// map page table entry
840 	phys_addr_t ppr = pr[index] & M68K_PRE_ADDRESS_MASK;
841 	page_directory_entry* pd = (page_directory_entry*)((char *)
842 		M68KPagingMethod040::Method()->PhysicalPageMapper()
843 		->InterruptGetPageTableAt(ppr & ~(B_PAGE_SIZE-1))
844 		+ (ppr % B_PAGE_SIZE));
845 
846 	index = VADDR_TO_PDENT(va);
847 	if (PDE_TYPE(pd[index]) != DT_DIR) {
848 		// no pagetable here
849 		return B_OK;
850 	}
851 
852 	phys_addr_t ppd = pd[index] & M68K_PDE_ADDRESS_MASK;
853 	page_table_entry* pt = (page_table_entry*)((char *)
854 		M68KPagingMethod040::Method()->PhysicalPageMapper()
855 		->InterruptGetPageTableAt(ppd & ~(B_PAGE_SIZE-1))
856 		+ (ppd % B_PAGE_SIZE));
857 
858 	index = VADDR_TO_PTENT(va);
859 	if (PTE_TYPE(pt[index]) == DT_INDIRECT) {
860 		phys_addr_t ppt = pt[index] & M68K_PIE_ADDRESS_MASK;
861 		pt = (page_table_entry*)((char *)
862 			M68KPagingMethod040::Method()->PhysicalPageMapper()
863 			->InterruptGetPageTableAt(ppt & ~(B_PAGE_SIZE-1))
864 			+ (ppt % B_PAGE_SIZE));
865 		index = 0;
866 	}
867 
868 	page_table_entry entry = pt[index];
869 
870 	*_physical = entry & M68K_PTE_ADDRESS_MASK;
871 
872 	// read in the page state flags
873 	if ((entry & M68K_PTE_SUPERVISOR) == 0) {
874 		*_flags |= ((entry & M68K_PTE_READONLY) == 0 ? B_WRITE_AREA : 0)
875 			| B_READ_AREA;
876 	}
877 
878 	*_flags |= ((entry & M68K_PTE_READONLY) == 0 ? B_KERNEL_WRITE_AREA : 0)
879 		| B_KERNEL_READ_AREA
880 		| ((entry & M68K_PTE_DIRTY) != 0 ? PAGE_MODIFIED : 0)
881 		| ((entry & M68K_PTE_ACCESSED) != 0 ? PAGE_ACCESSED : 0)
882 		| ((PTE_TYPE(entry) == DT_PAGE) ? PAGE_PRESENT : 0);
883 
884 	return B_OK;
885 }
886 
887 
888 status_t
889 M68KVMTranslationMap040::Protect(addr_t start, addr_t end, uint32 attributes,
890 	uint32 memoryType)
891 {
892 	start = ROUNDDOWN(start, B_PAGE_SIZE);
893 	if (start >= end)
894 		return B_OK;
895 
896 	TRACE("protect_tmap: pages 0x%lx to 0x%lx, attributes %lx\n", start, end,
897 		attributes);
898 
899 	return ENOSYS;
900 #if 0
901 	// compute protection flags
902 	uint32 newProtectionFlags = 0;
903 	if ((attributes & B_USER_PROTECTION) != 0) {
904 		newProtectionFlags = M68K_PTE_USER;
905 		if ((attributes & B_WRITE_AREA) != 0)
906 			newProtectionFlags |= M68K_PTE_WRITABLE;
907 	} else if ((attributes & B_KERNEL_WRITE_AREA) != 0)
908 		newProtectionFlags = M68K_PTE_WRITABLE;
909 
910 	page_directory_entry *pd = fPagingStructures->pgdir_virt;
911 
912 	do {
913 		int index = VADDR_TO_PDENT(start);
914 		if ((pd[index] & M68K_PDE_PRESENT) == 0) {
915 			// no page table here, move the start up to access the next page
916 			// table
917 			start = ROUNDUP(start + 1, kPageTableAlignment);
918 			continue;
919 		}
920 
921 		struct thread* thread = thread_get_current_thread();
922 		ThreadCPUPinner pinner(thread);
923 
924 		page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
925 			pd[index] & M68K_PDE_ADDRESS_MASK);
926 
927 		for (index = VADDR_TO_PTENT(start); index < 1024 && start < end;
928 				index++, start += B_PAGE_SIZE) {
929 			page_table_entry entry = pt[index];
930 			if ((entry & M68K_PTE_PRESENT) == 0) {
931 				// page mapping not valid
932 				continue;
933 			}
934 
935 			TRACE("protect_tmap: protect page 0x%lx\n", start);
936 
937 			// set the new protection flags -- we want to do that atomically,
938 			// without changing the accessed or dirty flag
939 			page_table_entry oldEntry;
940 			while (true) {
941 				oldEntry = M68KPagingMethod040::TestAndSetPageTableEntry(
942 					&pt[index],
943 					(entry & ~(M68K_PTE_PROTECTION_MASK
944 							| M68K_PTE_MEMORY_TYPE_MASK))
945 						| newProtectionFlags
946 						| M68KPagingMethod040::MemoryTypeToPageTableEntryFlags(
947 							memoryType),
948 					entry);
949 				if (oldEntry == entry)
950 					break;
951 				entry = oldEntry;
952 			}
953 
954 			if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
955 				// Note, that we only need to invalidate the address, if the
956 				// accessed flag was set, since only then the entry could have
957 				// been in any TLB.
958 				InvalidatePage(start);
959 			}
960 		}
961 	} while (start != 0 && start < end);
962 	return B_OK;
963 #endif
964 }
965 
966 
967 status_t
968 M68KVMTranslationMap040::ClearFlags(addr_t va, uint32 flags)
969 {
970 	return ENOSYS;
971 #if 0
972 	int index = VADDR_TO_PDENT(va);
973 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
974 	if ((pd[index] & M68K_PDE_PRESENT) == 0) {
975 		// no pagetable here
976 		return B_OK;
977 	}
978 
979 	uint32 flagsToClear = ((flags & PAGE_MODIFIED) ? M68K_PTE_DIRTY : 0)
980 		| ((flags & PAGE_ACCESSED) ? M68K_PTE_ACCESSED : 0);
981 
982 	struct thread* thread = thread_get_current_thread();
983 	ThreadCPUPinner pinner(thread);
984 
985 	page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
986 		pd[index] & M68K_PDE_ADDRESS_MASK);
987 	index = VADDR_TO_PTENT(va);
988 
989 	// clear out the flags we've been requested to clear
990 	page_table_entry oldEntry
991 		= M68KPagingMethod040::ClearPageTableEntryFlags(&pt[index],
992 			flagsToClear);
993 
994 	pinner.Unlock();
995 
996 	if ((oldEntry & flagsToClear) != 0)
997 		InvalidatePage(va);
998 
999 	return B_OK;
1000 #endif
1001 }
1002 
1003 
1004 bool
1005 M68KVMTranslationMap040::ClearAccessedAndModified(VMArea* area, addr_t address,
1006 	bool unmapIfUnaccessed, bool& _modified)
1007 {
1008 	ASSERT(address % B_PAGE_SIZE == 0);
1009 
1010 	page_root_entry* pr = fPagingStructures->pgroot_virt;
1011 
1012 	TRACE("M68KVMTranslationMap040::ClearAccessedAndModified(%#" B_PRIxADDR
1013 		")\n", address);
1014 
1015 #if 0
1016 	RecursiveLocker locker(fLock);
1017 
1018 	int index = VADDR_TO_PDENT(address);
1019 	if ((pd[index] & M68K_PDE_PRESENT) == 0)
1020 		return false;
1021 
1022 	ThreadCPUPinner pinner(thread_get_current_thread());
1023 
1024 	page_table_entry* pt = (page_table_entry*)MapperGetPageTableAt(
1025 		pd[index] & M68K_PDE_ADDRESS_MASK);
1026 
1027 	index = VADDR_TO_PTENT(address);
1028 
1029 	// perform the deed
1030 	page_table_entry oldEntry;
1031 
1032 	if (unmapIfUnaccessed) {
1033 		while (true) {
1034 			oldEntry = pt[index];
1035 			if ((oldEntry & M68K_PTE_PRESENT) == 0) {
1036 				// page mapping not valid
1037 				return false;
1038 			}
1039 
1040 			if (oldEntry & M68K_PTE_ACCESSED) {
1041 				// page was accessed -- just clear the flags
1042 				oldEntry = M68KPagingMethod040::ClearPageTableEntryFlags(
1043 					&pt[index], M68K_PTE_ACCESSED | M68K_PTE_DIRTY);
1044 				break;
1045 			}
1046 
1047 			// page hasn't been accessed -- unmap it
1048 			if (M68KPagingMethod040::TestAndSetPageTableEntry(&pt[index], 0,
1049 					oldEntry) == oldEntry) {
1050 				break;
1051 			}
1052 
1053 			// something changed -- check again
1054 		}
1055 	} else {
1056 		oldEntry = M68KPagingMethod040::ClearPageTableEntryFlags(&pt[index],
1057 			M68K_PTE_ACCESSED | M68K_PTE_DIRTY);
1058 	}
1059 
1060 	pinner.Unlock();
1061 
1062 	_modified = (oldEntry & M68K_PTE_DIRTY) != 0;
1063 
1064 	if ((oldEntry & M68K_PTE_ACCESSED) != 0) {
1065 		// Note, that we only need to invalidate the address, if the
1066 		// accessed flags was set, since only then the entry could have been
1067 		// in any TLB.
1068 		InvalidatePage(address);
1069 
1070 		Flush();
1071 
1072 		return true;
1073 	}
1074 
1075 	if (!unmapIfUnaccessed)
1076 		return false;
1077 
1078 	// We have unmapped the address. Do the "high level" stuff.
1079 
1080 	fMapCount--;
1081 
1082 	locker.Detach();
1083 		// UnaccessedPageUnmapped() will unlock for us
1084 
1085 	UnaccessedPageUnmapped(area,
1086 		(oldEntry & M68K_PTE_ADDRESS_MASK) / B_PAGE_SIZE);
1087 
1088 #endif
1089 	return false;
1090 }
1091 
1092 
1093 M68KPagingStructures*
1094 M68KVMTranslationMap040::PagingStructures() const
1095 {
1096 	return fPagingStructures;
1097 }
1098 
1099 
1100 inline void *
1101 M68KVMTranslationMap040::MapperGetPageTableAt(phys_addr_t physicalAddress,
1102 	bool indirect)
1103 {
1104 	// M68K fits several page tables in a single page...
1105 	uint32 offset = physicalAddress % B_PAGE_SIZE;
1106 	ASSERT((indirect && (offset % 4) == 0) || (offset % SIZ_ROOTTBL) == 0);
1107 	physicalAddress &= ~(B_PAGE_SIZE-1);
1108 	void *va = fPageMapper->GetPageTableAt(physicalAddress);
1109 	return (void *)((addr_t)va + offset);
1110 }
1111 
1112 
1113