xref: /haiku/src/system/kernel/arch/arm/paging/32bit/ARMVMTranslationMap32Bit.cpp (revision 4c8e85b316c35a9161f5a1c50ad70bc91c83a76f)
1 /*
2  * Copyright 2010, Ithamar R. Adema, ithamar.adema@team-embedded.nl
3  * Copyright 2008-2011, Ingo Weinhold, ingo_weinhold@gmx.de.
4  * Copyright 2002-2007, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
5  * Distributed under the terms of the MIT License.
6  *
7  * Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
8  * Distributed under the terms of the NewOS License.
9  */
10 
11 
12 #include "paging/32bit/ARMVMTranslationMap32Bit.h"
13 
14 #include <stdlib.h>
15 #include <string.h>
16 
17 #include <int.h>
18 #include <thread.h>
19 #include <slab/Slab.h>
20 #include <smp.h>
21 #include <util/AutoLock.h>
22 #include <util/ThreadAutoLock.h>
23 #include <util/queue.h>
24 #include <vm/vm_page.h>
25 #include <vm/vm_priv.h>
26 #include <vm/VMAddressSpace.h>
27 #include <vm/VMCache.h>
28 
29 #include "paging/32bit/ARMPagingMethod32Bit.h"
30 #include "paging/32bit/ARMPagingStructures32Bit.h"
31 #include "paging/arm_physical_page_mapper.h"
32 
33 
34 //#define TRACE_ARM_VM_TRANSLATION_MAP_32_BIT
35 #ifdef TRACE_ARM_VM_TRANSLATION_MAP_32_BIT
36 #	define TRACE(x...) dprintf(x)
37 #else
38 #	define TRACE(x...) ;
39 #endif
40 
41 
42 #define PAGEDIR_SIZE	ARM_MMU_L1_TABLE_SIZE
43 #define PAGEDIR_ALIGN	(4 * B_PAGE_SIZE)
44 
45 
46 ARMVMTranslationMap32Bit::ARMVMTranslationMap32Bit()
47 	:
48 	fPagingStructures(NULL)
49 {
50 }
51 
52 
53 ARMVMTranslationMap32Bit::~ARMVMTranslationMap32Bit()
54 {
55 	if (fPagingStructures == NULL)
56 		return;
57 
58 	if (fPageMapper != NULL)
59 		fPageMapper->Delete();
60 
61 	if (fPagingStructures->pgdir_virt != NULL) {
62 		// cycle through and free all of the user space pgtables
63 		for (uint32 i = VADDR_TO_PDENT(USER_BASE);
64 				i <= VADDR_TO_PDENT(USER_BASE + (USER_SIZE - 1)); i++) {
65 			if ((fPagingStructures->pgdir_virt[i] & ARM_PDE_TYPE_MASK) != 0) {
66 				addr_t address = fPagingStructures->pgdir_virt[i]
67 					& ARM_PDE_ADDRESS_MASK;
68 				vm_page* page = vm_lookup_page(address / B_PAGE_SIZE);
69 				if (!page)
70 					panic("destroy_tmap: didn't find pgtable page\n");
71 				DEBUG_PAGE_ACCESS_START(page);
72 				vm_page_set_state(page, PAGE_STATE_FREE);
73 			}
74 		}
75 	}
76 
77 	fPagingStructures->RemoveReference();
78 }
79 
80 
81 status_t
82 ARMVMTranslationMap32Bit::Init(bool kernel)
83 {
84 	TRACE("ARMVMTranslationMap32Bit::Init()\n");
85 
86 	ARMVMTranslationMap::Init(kernel);
87 
88 	fPagingStructures = new(std::nothrow) ARMPagingStructures32Bit;
89 	if (fPagingStructures == NULL)
90 		return B_NO_MEMORY;
91 
92 	ARMPagingMethod32Bit* method = ARMPagingMethod32Bit::Method();
93 
94 	if (!kernel) {
95 		// user
96 		// allocate a physical page mapper
97 		status_t error = method->PhysicalPageMapper()
98 			->CreateTranslationMapPhysicalPageMapper(&fPageMapper);
99 		if (error != B_OK)
100 			return error;
101 
102 		// allocate the page directory
103 		page_directory_entry* virtualPageDir = (page_directory_entry*)memalign(
104 			PAGEDIR_ALIGN, PAGEDIR_SIZE);
105 		if (virtualPageDir == NULL)
106 			return B_NO_MEMORY;
107 
108 		// look up the page directory's physical address
109 		phys_addr_t physicalPageDir;
110 		vm_get_page_mapping(VMAddressSpace::KernelID(),
111 			(addr_t)virtualPageDir, &physicalPageDir);
112 
113 		fPagingStructures->Init(virtualPageDir, physicalPageDir,
114 			method->KernelVirtualPageDirectory());
115 	} else {
116 		// kernel
117 		// get the physical page mapper
118 		fPageMapper = method->KernelPhysicalPageMapper();
119 
120 		// we already know the kernel pgdir mapping
121 		fPagingStructures->Init(method->KernelVirtualPageDirectory(),
122 			method->KernelPhysicalPageDirectory(), NULL);
123 	}
124 
125 	return B_OK;
126 }
127 
128 
129 size_t
130 ARMVMTranslationMap32Bit::MaxPagesNeededToMap(addr_t start, addr_t end) const
131 {
132 	// If start == 0, the actual base address is not yet known to the caller and
133 	// we shall assume the worst case.
134 	if (start == 0) {
135 		// offset the range so it has the worst possible alignment
136 		start = 1023 * B_PAGE_SIZE;
137 		end += 1023 * B_PAGE_SIZE;
138 	}
139 
140 	return VADDR_TO_PDENT(end) + 1 - VADDR_TO_PDENT(start);
141 }
142 
143 
144 status_t
145 ARMVMTranslationMap32Bit::Map(addr_t va, phys_addr_t pa, uint32 attributes,
146 	uint32 memoryType, vm_page_reservation* reservation)
147 {
148 	TRACE("map_tmap: entry pa 0x%lx va 0x%lx\n", pa, va);
149 
150 /*
151 	dprintf("pgdir at 0x%x\n", pgdir);
152 	dprintf("index is %d\n", va / B_PAGE_SIZE / 1024);
153 	dprintf("final at 0x%x\n", &pgdir[va / B_PAGE_SIZE / 1024]);
154 	dprintf("value is 0x%x\n", *(int *)&pgdir[va / B_PAGE_SIZE / 1024]);
155 	dprintf("present bit is %d\n", pgdir[va / B_PAGE_SIZE / 1024].present);
156 	dprintf("addr is %d\n", pgdir[va / B_PAGE_SIZE / 1024].addr);
157 */
158 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
159 
160 	// check to see if a page table exists for this range
161 	uint32 index = VADDR_TO_PDENT(va);
162 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
163 		phys_addr_t pgtable;
164 		vm_page *page;
165 
166 		// we need to allocate a pgtable
167 		page = vm_page_allocate_page(reservation,
168 			PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
169 
170 		DEBUG_PAGE_ACCESS_END(page);
171 
172 		pgtable = (phys_addr_t)page->physical_page_number * B_PAGE_SIZE;
173 
174 		TRACE("map_tmap: asked for free page for pgtable. 0x%lx\n", pgtable);
175 
176 		// put it in the pgdir
177 		ARMPagingMethod32Bit::PutPageTableInPageDir(&pd[index], pgtable,
178 			attributes
179 				| ((attributes & B_USER_PROTECTION) != 0
180 						? B_WRITE_AREA : B_KERNEL_WRITE_AREA));
181 
182 		// update any other page directories, if it maps kernel space
183 		if (index >= FIRST_KERNEL_PGDIR_ENT
184 			&& index < (FIRST_KERNEL_PGDIR_ENT + NUM_KERNEL_PGDIR_ENTS)) {
185 			ARMPagingStructures32Bit::UpdateAllPageDirs(index, pd[index]);
186 		}
187 
188 		fMapCount++;
189 	}
190 
191 	// now, fill in the pentry
192 	Thread* thread = thread_get_current_thread();
193 	ThreadCPUPinner pinner(thread);
194 
195 	page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
196 		pd[index] & ARM_PDE_ADDRESS_MASK);
197 	index = VADDR_TO_PTENT(va);
198 
199 	ASSERT_PRINT((pt[index] & ARM_PTE_TYPE_MASK) == 0,
200 		"virtual address: %#" B_PRIxADDR ", existing pte: %#" B_PRIx32, va,
201 		pt[index]);
202 
203 	ARMPagingMethod32Bit::PutPageTableEntryInTable(&pt[index], pa, attributes,
204 		memoryType, fIsKernelMap);
205 
206 	pinner.Unlock();
207 
208 	// Note: We don't need to invalidate the TLB for this address, as previously
209 	// the entry was not present and the TLB doesn't cache those entries.
210 
211 	fMapCount++;
212 
213 	return 0;
214 }
215 
216 
217 status_t
218 ARMVMTranslationMap32Bit::Unmap(addr_t start, addr_t end)
219 {
220 	start = ROUNDDOWN(start, B_PAGE_SIZE);
221 	if (start >= end)
222 		return B_OK;
223 
224 	TRACE("unmap_tmap: asked to free pages 0x%lx to 0x%lx\n", start, end);
225 
226 	page_directory_entry *pd = fPagingStructures->pgdir_virt;
227 
228 	do {
229 		int index = VADDR_TO_PDENT(start);
230 		if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
231 			// no page table here, move the start up to access the next page
232 			// table
233 			start = ROUNDUP(start + 1, kPageTableAlignment);
234 			continue;
235 		}
236 
237 		Thread* thread = thread_get_current_thread();
238 		ThreadCPUPinner pinner(thread);
239 
240 		page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
241 			pd[index] & ARM_PDE_ADDRESS_MASK);
242 
243 		for (index = VADDR_TO_PTENT(start); (index < 256) && (start < end);
244 				index++, start += B_PAGE_SIZE) {
245 			if ((pt[index] & ARM_PTE_TYPE_MASK) == 0) {
246 				// page mapping not valid
247 				continue;
248 			}
249 
250 			TRACE("unmap_tmap: removing page 0x%lx\n", start);
251 
252 			page_table_entry oldEntry
253 				= ARMPagingMethod32Bit::ClearPageTableEntryFlags(&pt[index],
254 					ARM_PTE_TYPE_MASK);
255 			fMapCount--;
256 
257 			if (true /* (oldEntry & ARM_PTE_ACCESSED) != 0*/) {
258 				// Note, that we only need to invalidate the address, if the
259 				// accessed flags was set, since only then the entry could have
260 				// been in any TLB.
261 				InvalidatePage(start);
262 			}
263 		}
264 	} while (start != 0 && start < end);
265 
266 	return B_OK;
267 }
268 
269 
270 status_t
271 ARMVMTranslationMap32Bit::DebugMarkRangePresent(addr_t start, addr_t end,
272 	bool markPresent)
273 {
274 #if 0
275 	start = ROUNDDOWN(start, B_PAGE_SIZE);
276 	if (start >= end)
277 		return B_OK;
278 
279 	page_directory_entry *pd = fPagingStructures->pgdir_virt;
280 
281 	do {
282 		int index = VADDR_TO_PDENT(start);
283 		if ((pd[index] & X86_PDE_PRESENT) == 0) {
284 			// no page table here, move the start up to access the next page
285 			// table
286 			start = ROUNDUP(start + 1, kPageTableAlignment);
287 			continue;
288 		}
289 
290 		Thread* thread = thread_get_current_thread();
291 		ThreadCPUPinner pinner(thread);
292 
293 		page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
294 			pd[index] & X86_PDE_ADDRESS_MASK);
295 
296 		for (index = VADDR_TO_PTENT(start); (index < 1024) && (start < end);
297 				index++, start += B_PAGE_SIZE) {
298 			if ((pt[index] & X86_PTE_PRESENT) == 0) {
299 				if (!markPresent)
300 					continue;
301 
302 				X86PagingMethod32Bit::SetPageTableEntryFlags(&pt[index],
303 					X86_PTE_PRESENT);
304 			} else {
305 				if (markPresent)
306 					continue;
307 
308 				page_table_entry oldEntry
309 					= X86PagingMethod32Bit::ClearPageTableEntryFlags(&pt[index],
310 						X86_PTE_PRESENT);
311 
312 				if ((oldEntry & X86_PTE_ACCESSED) != 0) {
313 					// Note, that we only need to invalidate the address, if the
314 					// accessed flags was set, since only then the entry could
315 					// have been in any TLB.
316 					InvalidatePage(start);
317 				}
318 			}
319 		}
320 	} while (start != 0 && start < end);
321 #endif
322 	return B_OK;
323 }
324 
325 
326 /*!	Caller must have locked the cache of the page to be unmapped.
327 	This object shouldn't be locked.
328 */
329 status_t
330 ARMVMTranslationMap32Bit::UnmapPage(VMArea* area, addr_t address,
331 	bool updatePageQueue)
332 {
333 	ASSERT(address % B_PAGE_SIZE == 0);
334 
335 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
336 
337 	TRACE("ARMVMTranslationMap32Bit::UnmapPage(%#" B_PRIxADDR ")\n", address);
338 
339 	RecursiveLocker locker(fLock);
340 
341 	int index = VADDR_TO_PDENT(address);
342 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0)
343 		return B_ENTRY_NOT_FOUND;
344 
345 	ThreadCPUPinner pinner(thread_get_current_thread());
346 
347 	page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
348 		pd[index] & ARM_PDE_ADDRESS_MASK);
349 
350 	index = VADDR_TO_PTENT(address);
351 	page_table_entry oldEntry = ARMPagingMethod32Bit::ClearPageTableEntry(
352 		&pt[index]);
353 
354 	pinner.Unlock();
355 
356 	if ((oldEntry & ARM_PTE_TYPE_MASK) == 0) {
357 		// page mapping not valid
358 		return B_ENTRY_NOT_FOUND;
359 	}
360 
361 	fMapCount--;
362 
363 
364 	if (true /*(oldEntry & ARM_PTE_ACCESSED) != 0*/) { // XXX IRA
365 		// Note, that we only need to invalidate the address, if the
366 		// accessed flags was set, since only then the entry could have been
367 		// in any TLB.
368 		InvalidatePage(address);
369 		Flush();
370 
371 		// NOTE: Between clearing the page table entry and Flush() other
372 		// processors (actually even this processor with another thread of the
373 		// same team) could still access the page in question via their cached
374 		// entry. We can obviously lose a modified flag in this case, with the
375 		// effect that the page looks unmodified (and might thus be recycled),
376 		// but is actually modified.
377 		// In most cases this is harmless, but for vm_remove_all_page_mappings()
378 		// this is actually a problem.
379 		// Interestingly FreeBSD seems to ignore this problem as well
380 		// (cf. pmap_remove_all()), unless I've missed something.
381 	}
382 
383 	locker.Detach();
384 		// PageUnmapped() will unlock for us
385 
386 	PageUnmapped(area, (oldEntry & ARM_PTE_ADDRESS_MASK) / B_PAGE_SIZE,
387 		true /*(oldEntry & ARM_PTE_ACCESSED) != 0*/, true /*(oldEntry & ARM_PTE_DIRTY) != 0*/,
388 		updatePageQueue);
389 
390 	return B_OK;
391 }
392 
393 
394 void
395 ARMVMTranslationMap32Bit::UnmapPages(VMArea* area, addr_t base, size_t size,
396 	bool updatePageQueue)
397 {
398 	if (size == 0)
399 		return;
400 
401 	addr_t start = base;
402 	addr_t end = base + size - 1;
403 
404 	TRACE("ARMVMTranslationMap32Bit::UnmapPages(%p, %#" B_PRIxADDR ", %#"
405 		B_PRIxADDR ")\n", area, start, end);
406 
407 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
408 
409 	VMAreaMappings queue;
410 
411 	RecursiveLocker locker(fLock);
412 
413 	do {
414 		int index = VADDR_TO_PDENT(start);
415 		if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
416 			// no page table here, move the start up to access the next page
417 			// table
418 			start = ROUNDUP(start + 1, kPageTableAlignment);
419 			continue;
420 		}
421 
422 		Thread* thread = thread_get_current_thread();
423 		ThreadCPUPinner pinner(thread);
424 
425 		page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
426 			pd[index] & ARM_PDE_ADDRESS_MASK);
427 
428 		for (index = VADDR_TO_PTENT(start); (index < 256) && (start < end);
429 				index++, start += B_PAGE_SIZE) {
430 			page_table_entry oldEntry
431 				= ARMPagingMethod32Bit::ClearPageTableEntry(&pt[index]);
432 			if ((oldEntry & ARM_PTE_TYPE_MASK) == 0)
433 				continue;
434 
435 			fMapCount--;
436 
437 			if (true /*(oldEntry & ARM_PTE_ACCESSED) != 0*/) { // XXX IRA
438 				// Note, that we only need to invalidate the address, if the
439 				// accessed flags was set, since only then the entry could have
440 				// been in any TLB.
441 				InvalidatePage(start);
442 			}
443 
444 			if (area->cache_type != CACHE_TYPE_DEVICE) {
445 				// get the page
446 				vm_page* page = vm_lookup_page(
447 					(oldEntry & ARM_PTE_ADDRESS_MASK) / B_PAGE_SIZE);
448 				ASSERT(page != NULL);
449 
450 				DEBUG_PAGE_ACCESS_START(page);
451 
452 				// transfer the accessed/dirty flags to the page
453 				if (/*(oldEntry & ARM_PTE_ACCESSED) != 0*/ true) // XXX IRA
454 					page->accessed = true;
455 				if (/*(oldEntry & ARM_PTE_DIRTY) != 0 */ true)
456 					page->modified = true;
457 
458 				// remove the mapping object/decrement the wired_count of the
459 				// page
460 				if (area->wiring == B_NO_LOCK) {
461 					vm_page_mapping* mapping = NULL;
462 					vm_page_mappings::Iterator iterator
463 						= page->mappings.GetIterator();
464 					while ((mapping = iterator.Next()) != NULL) {
465 						if (mapping->area == area)
466 							break;
467 					}
468 
469 					ASSERT(mapping != NULL);
470 
471 					area->mappings.Remove(mapping);
472 					page->mappings.Remove(mapping);
473 					queue.Add(mapping);
474 				} else
475 					page->DecrementWiredCount();
476 
477 				if (!page->IsMapped()) {
478 					atomic_add(&gMappedPagesCount, -1);
479 
480 					if (updatePageQueue) {
481 						if (page->Cache()->temporary)
482 							vm_page_set_state(page, PAGE_STATE_INACTIVE);
483 						else if (page->modified)
484 							vm_page_set_state(page, PAGE_STATE_MODIFIED);
485 						else
486 							vm_page_set_state(page, PAGE_STATE_CACHED);
487 					}
488 				}
489 
490 				DEBUG_PAGE_ACCESS_END(page);
491 			}
492 		}
493 
494 		Flush();
495 			// flush explicitly, since we directly use the lock
496 	} while (start != 0 && start < end);
497 
498 	// TODO: As in UnmapPage() we can lose page dirty flags here. ATM it's not
499 	// really critical here, as in all cases this method is used, the unmapped
500 	// area range is unmapped for good (resized/cut) and the pages will likely
501 	// be freed.
502 
503 	locker.Unlock();
504 
505 	// free removed mappings
506 	bool isKernelSpace = area->address_space == VMAddressSpace::Kernel();
507 	uint32 freeFlags = CACHE_DONT_WAIT_FOR_MEMORY
508 		| (isKernelSpace ? CACHE_DONT_LOCK_KERNEL_SPACE : 0);
509 	while (vm_page_mapping* mapping = queue.RemoveHead())
510 		object_cache_free(gPageMappingsObjectCache, mapping, freeFlags);
511 }
512 
513 
514 void
515 ARMVMTranslationMap32Bit::UnmapArea(VMArea* area, bool deletingAddressSpace,
516 	bool ignoreTopCachePageFlags)
517 {
518 	if (area->cache_type == CACHE_TYPE_DEVICE || area->wiring != B_NO_LOCK) {
519 		ARMVMTranslationMap32Bit::UnmapPages(area, area->Base(), area->Size(),
520 			true);
521 		return;
522 	}
523 
524 	bool unmapPages = !deletingAddressSpace || !ignoreTopCachePageFlags;
525 
526 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
527 
528 	RecursiveLocker locker(fLock);
529 
530 	VMAreaMappings mappings;
531 	mappings.MoveFrom(&area->mappings);
532 
533 	for (VMAreaMappings::Iterator it = mappings.GetIterator();
534 			vm_page_mapping* mapping = it.Next();) {
535 		vm_page* page = mapping->page;
536 		page->mappings.Remove(mapping);
537 
538 		VMCache* cache = page->Cache();
539 
540 		bool pageFullyUnmapped = false;
541 		if (!page->IsMapped()) {
542 			atomic_add(&gMappedPagesCount, -1);
543 			pageFullyUnmapped = true;
544 		}
545 
546 		if (unmapPages || cache != area->cache) {
547 			addr_t address = area->Base()
548 				+ ((page->cache_offset * B_PAGE_SIZE) - area->cache_offset);
549 
550 			int index = VADDR_TO_PDENT(address);
551 			if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
552 				panic("page %p has mapping for area %p (%#" B_PRIxADDR "), but "
553 					"has no page dir entry", page, area, address);
554 				continue;
555 			}
556 
557 			ThreadCPUPinner pinner(thread_get_current_thread());
558 
559 			page_table_entry* pt
560 				= (page_table_entry*)fPageMapper->GetPageTableAt(
561 					pd[index] & ARM_PDE_ADDRESS_MASK);
562 			page_table_entry oldEntry
563 				= ARMPagingMethod32Bit::ClearPageTableEntry(
564 					&pt[VADDR_TO_PTENT(address)]);
565 
566 			pinner.Unlock();
567 
568 			if ((oldEntry & ARM_PTE_TYPE_MASK) == 0) {
569 				panic("page %p has mapping for area %p (%#" B_PRIxADDR "), but "
570 					"has no page table entry", page, area, address);
571 				continue;
572 			}
573 
574 			// transfer the accessed/dirty flags to the page and invalidate
575 			// the mapping, if necessary
576 			if (true /*(oldEntry & ARM_PTE_ACCESSED) != 0*/) { // XXX IRA
577 				page->accessed = true;
578 
579 				if (!deletingAddressSpace)
580 					InvalidatePage(address);
581 			}
582 
583 			if (true /*(oldEntry & ARM_PTE_DIRTY) != 0*/)
584 				page->modified = true;
585 
586 			if (pageFullyUnmapped) {
587 				DEBUG_PAGE_ACCESS_START(page);
588 
589 				if (cache->temporary)
590 					vm_page_set_state(page, PAGE_STATE_INACTIVE);
591 				else if (page->modified)
592 					vm_page_set_state(page, PAGE_STATE_MODIFIED);
593 				else
594 					vm_page_set_state(page, PAGE_STATE_CACHED);
595 
596 				DEBUG_PAGE_ACCESS_END(page);
597 			}
598 		}
599 
600 		fMapCount--;
601 	}
602 
603 	Flush();
604 		// flush explicitely, since we directly use the lock
605 
606 	locker.Unlock();
607 
608 	bool isKernelSpace = area->address_space == VMAddressSpace::Kernel();
609 	uint32 freeFlags = CACHE_DONT_WAIT_FOR_MEMORY
610 		| (isKernelSpace ? CACHE_DONT_LOCK_KERNEL_SPACE : 0);
611 	while (vm_page_mapping* mapping = mappings.RemoveHead())
612 		object_cache_free(gPageMappingsObjectCache, mapping, freeFlags);
613 }
614 
615 
616 status_t
617 ARMVMTranslationMap32Bit::Query(addr_t va, phys_addr_t *_physical,
618 	uint32 *_flags)
619 {
620 	// default the flags to not present
621 	*_flags = 0;
622 	*_physical = 0;
623 
624 	int index = VADDR_TO_PDENT(va);
625 	page_directory_entry *pd = fPagingStructures->pgdir_virt;
626 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
627 		// no pagetable here
628 		return B_OK;
629 	}
630 
631 	Thread* thread = thread_get_current_thread();
632 	ThreadCPUPinner pinner(thread);
633 
634 	page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
635 		pd[index] & ARM_PDE_ADDRESS_MASK);
636 	page_table_entry entry = pt[VADDR_TO_PTENT(va)];
637 
638 	if ((entry & ARM_PTE_TYPE_MASK) != 0)
639 		*_physical = (entry & ARM_PTE_ADDRESS_MASK);
640 
641 #if 0 //IRA
642 	// read in the page state flags
643 	if ((entry & X86_PTE_USER) != 0) {
644 		*_flags |= ((entry & X86_PTE_WRITABLE) != 0 ? B_WRITE_AREA : 0)
645 			| B_READ_AREA;
646 	}
647 
648 	*_flags |= ((entry & X86_PTE_WRITABLE) != 0 ? B_KERNEL_WRITE_AREA : 0)
649 		| B_KERNEL_READ_AREA
650 		| ((entry & ARM_PTE_DIRTY) != 0 ? PAGE_MODIFIED : 0)
651 		| ((entry & ARM_PTE_ACCESSED) != 0 ? PAGE_ACCESSED : 0)
652 		| ((entry & ARM_PTE_PRESENT) != 0 ? PAGE_PRESENT : 0);
653 #else
654 	*_flags = B_KERNEL_WRITE_AREA | B_KERNEL_READ_AREA;
655 	if (*_physical != 0)
656 		*_flags |= PAGE_PRESENT;
657 #endif
658 	pinner.Unlock();
659 
660 	TRACE("query_tmap: returning pa 0x%lx for va 0x%lx\n", *_physical, va);
661 
662 	return B_OK;
663 }
664 
665 
666 status_t
667 ARMVMTranslationMap32Bit::QueryInterrupt(addr_t va, phys_addr_t *_physical,
668 	uint32 *_flags)
669 {
670 	*_flags = 0;
671 	*_physical = 0;
672 
673 	int index = VADDR_TO_PDENT(va);
674 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
675 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
676 		// no pagetable here
677 		return B_OK;
678 	}
679 
680 	// map page table entry
681 	page_table_entry* pt = (page_table_entry*)ARMPagingMethod32Bit::Method()
682 		->PhysicalPageMapper()->InterruptGetPageTableAt(
683 			pd[index] & ARM_PDE_ADDRESS_MASK);
684 	page_table_entry entry = pt[VADDR_TO_PTENT(va)];
685 
686 	if ((entry & ARM_PTE_TYPE_MASK) != 0)
687 		*_physical = (entry & ARM_PTE_ADDRESS_MASK);
688 
689 #if 0
690 	// read in the page state flags
691 	if ((entry & X86_PTE_USER) != 0) {
692 		*_flags |= ((entry & X86_PTE_WRITABLE) != 0 ? B_WRITE_AREA : 0)
693 			| B_READ_AREA;
694 	}
695 
696 	*_flags |= ((entry & X86_PTE_WRITABLE) != 0 ? B_KERNEL_WRITE_AREA : 0)
697 		| B_KERNEL_READ_AREA
698 		| ((entry & X86_PTE_DIRTY) != 0 ? PAGE_MODIFIED : 0)
699 		| ((entry & X86_PTE_ACCESSED) != 0 ? PAGE_ACCESSED : 0)
700 		| ((entry & X86_PTE_PRESENT) != 0 ? PAGE_PRESENT : 0);
701 #else
702 	*_flags = B_KERNEL_WRITE_AREA | B_KERNEL_READ_AREA;
703 	if (*_physical != 0)
704 		*_flags |= PAGE_PRESENT;
705 #endif
706 	return B_OK;
707 }
708 
709 
710 status_t
711 ARMVMTranslationMap32Bit::Protect(addr_t start, addr_t end, uint32 attributes,
712 	uint32 memoryType)
713 {
714 	start = ROUNDDOWN(start, B_PAGE_SIZE);
715 	if (start >= end)
716 		return B_OK;
717 
718 	TRACE("protect_tmap: pages 0x%lx to 0x%lx, attributes %lx\n", start, end,
719 		attributes);
720 #if 0 //IRA
721 	// compute protection flags
722 	uint32 newProtectionFlags = 0;
723 	if ((attributes & B_USER_PROTECTION) != 0) {
724 		newProtectionFlags = ARM_PTE_USER;
725 		if ((attributes & B_WRITE_AREA) != 0)
726 			newProtectionFlags |= ARM_PTE_WRITABLE;
727 	} else if ((attributes & B_KERNEL_WRITE_AREA) != 0)
728 		newProtectionFlags = ARM_PTE_WRITABLE;
729 
730 	page_directory_entry *pd = fPagingStructures->pgdir_virt;
731 
732 	do {
733 		int index = VADDR_TO_PDENT(start);
734 		if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
735 			// no page table here, move the start up to access the next page
736 			// table
737 			start = ROUNDUP(start + 1, kPageTableAlignment);
738 			continue;
739 		}
740 
741 		Thread* thread = thread_get_current_thread();
742 		ThreadCPUPinner pinner(thread);
743 
744 		page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
745 			pd[index] & ARM_PDE_ADDRESS_MASK);
746 
747 		for (index = VADDR_TO_PTENT(start); index < 256 && start < end;
748 				index++, start += B_PAGE_SIZE) {
749 			page_table_entry entry = pt[index];
750 			if ((entry & ARM_PTE_PRESENT) == 0) {
751 				// page mapping not valid
752 				continue;
753 			}
754 
755 			TRACE("protect_tmap: protect page 0x%lx\n", start);
756 
757 			// set the new protection flags -- we want to do that atomically,
758 			// without changing the accessed or dirty flag
759 			page_table_entry oldEntry;
760 			while (true) {
761 				oldEntry = ARMPagingMethod32Bit::TestAndSetPageTableEntry(
762 					&pt[index],
763 					(entry & ~(ARM_PTE_PROTECTION_MASK
764 							| ARM_PTE_MEMORY_TYPE_MASK))
765 						| newProtectionFlags
766 						| ARMPagingMethod32Bit::MemoryTypeToPageTableEntryFlags(
767 							memoryType),
768 					entry);
769 				if (oldEntry == entry)
770 					break;
771 				entry = oldEntry;
772 			}
773 
774 			if ((oldEntry & ARM_PTE_ACCESSED) != 0) {
775 				// Note, that we only need to invalidate the address, if the
776 				// accessed flag was set, since only then the entry could have
777 				// been in any TLB.
778 				InvalidatePage(start);
779 			}
780 		}
781 	} while (start != 0 && start < end);
782 #endif
783 	return B_OK;
784 }
785 
786 
787 status_t
788 ARMVMTranslationMap32Bit::ClearFlags(addr_t va, uint32 flags)
789 {
790 	int index = VADDR_TO_PDENT(va);
791 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
792 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0) {
793 		// no pagetable here
794 		return B_OK;
795 	}
796 #if 0 //IRA
797 	uint32 flagsToClear = ((flags & PAGE_MODIFIED) ? X86_PTE_DIRTY : 0)
798 		| ((flags & PAGE_ACCESSED) ? X86_PTE_ACCESSED : 0);
799 #else
800 	uint32 flagsToClear = 0;
801 #endif
802 	Thread* thread = thread_get_current_thread();
803 	ThreadCPUPinner pinner(thread);
804 
805 	page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
806 		pd[index] & ARM_PDE_ADDRESS_MASK);
807 	index = VADDR_TO_PTENT(va);
808 
809 	// clear out the flags we've been requested to clear
810 	page_table_entry oldEntry
811 		= ARMPagingMethod32Bit::ClearPageTableEntryFlags(&pt[index],
812 			flagsToClear);
813 
814 	pinner.Unlock();
815 
816 	//XXX IRA if ((oldEntry & flagsToClear) != 0)
817 		InvalidatePage(va);
818 
819 	return B_OK;
820 }
821 
822 
823 bool
824 ARMVMTranslationMap32Bit::ClearAccessedAndModified(VMArea* area, addr_t address,
825 	bool unmapIfUnaccessed, bool& _modified)
826 {
827 	ASSERT(address % B_PAGE_SIZE == 0);
828 
829 	page_directory_entry* pd = fPagingStructures->pgdir_virt;
830 
831 	TRACE("ARMVMTranslationMap32Bit::ClearAccessedAndModified(%#" B_PRIxADDR
832 		")\n", address);
833 
834 	RecursiveLocker locker(fLock);
835 
836 	int index = VADDR_TO_PDENT(address);
837 	if ((pd[index] & ARM_PDE_TYPE_MASK) == 0)
838 		return false;
839 
840 	ThreadCPUPinner pinner(thread_get_current_thread());
841 
842 	page_table_entry* pt = (page_table_entry*)fPageMapper->GetPageTableAt(
843 		pd[index] & ARM_PDE_ADDRESS_MASK);
844 
845 	index = VADDR_TO_PTENT(address);
846 
847 	// perform the deed
848 	page_table_entry oldEntry;
849 
850 	if (unmapIfUnaccessed) {
851 		while (true) {
852 			oldEntry = pt[index];
853 			if ((oldEntry & ARM_PTE_TYPE_MASK) == 0) {
854 				// page mapping not valid
855 				return false;
856 			}
857 #if 0 //IRA
858 			if (oldEntry & ARM_PTE_ACCESSED) {
859 				// page was accessed -- just clear the flags
860 				oldEntry = ARMPagingMethod32Bit::ClearPageTableEntryFlags(
861 					&pt[index], ARM_PTE_ACCESSED | ARM_PTE_DIRTY);
862 				break;
863 			}
864 #endif
865 			// page hasn't been accessed -- unmap it
866 			if (ARMPagingMethod32Bit::TestAndSetPageTableEntry(&pt[index], 0,
867 					oldEntry) == oldEntry) {
868 				break;
869 			}
870 
871 			// something changed -- check again
872 		}
873 	} else {
874 #if 0 //IRA
875 		oldEntry = ARMPagingMethod32Bit::ClearPageTableEntryFlags(&pt[index],
876 			ARM_PTE_ACCESSED | ARM_PTE_DIRTY);
877 #else
878 		oldEntry = pt[index];
879 #endif
880 	}
881 
882 	pinner.Unlock();
883 
884 	_modified = true /* (oldEntry & X86_PTE_DIRTY) != 0 */; // XXX IRA
885 
886 	if (true /*(oldEntry & ARM_PTE_ACCESSED) != 0*/) {
887 		// Note, that we only need to invalidate the address, if the
888 		// accessed flags was set, since only then the entry could have been
889 		// in any TLB.
890 		InvalidatePage(address);
891 
892 		Flush();
893 
894 		return true;
895 	}
896 
897 	if (!unmapIfUnaccessed)
898 		return false;
899 
900 	// We have unmapped the address. Do the "high level" stuff.
901 
902 	fMapCount--;
903 
904 	locker.Detach();
905 		// UnaccessedPageUnmapped() will unlock for us
906 
907 	UnaccessedPageUnmapped(area,
908 		(oldEntry & ARM_PTE_ADDRESS_MASK) / B_PAGE_SIZE);
909 
910 	return false;
911 }
912 
913 
914 ARMPagingStructures*
915 ARMVMTranslationMap32Bit::PagingStructures() const
916 {
917 	return fPagingStructures;
918 }
919