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