| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | /* MN10300 Virtual kernel memory mappings for high memory
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved. | 
					
						
							|  |  |  |  * Written by David Howells (dhowells@redhat.com) | 
					
						
							|  |  |  |  * - Derived from include/asm-i386/highmem.h | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public Licence | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version | 
					
						
							|  |  |  |  * 2 of the Licence, or (at your option) any later version. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #ifndef _ASM_HIGHMEM_H
 | 
					
						
							|  |  |  | #define _ASM_HIGHMEM_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __KERNEL__
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/init.h>
 | 
					
						
							|  |  |  | #include <linux/interrupt.h>
 | 
					
						
							| 
									
										
										
										
											2009-03-31 15:23:25 -07:00
										 |  |  | #include <linux/highmem.h>
 | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | #include <asm/kmap_types.h>
 | 
					
						
							|  |  |  | #include <asm/pgtable.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* undef for production */ | 
					
						
							|  |  |  | #undef HIGHMEM_DEBUG
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* declarations for highmem.c */ | 
					
						
							|  |  |  | extern unsigned long highstart_pfn, highend_pfn; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern pte_t *kmap_pte; | 
					
						
							|  |  |  | extern pgprot_t kmap_prot; | 
					
						
							|  |  |  | extern pte_t *pkmap_page_table; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern void __init kmap_init(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Right now we initialize only a single pte table. It can be extended | 
					
						
							|  |  |  |  * easily, subsequent pte tables have to be allocated in one physical | 
					
						
							|  |  |  |  * chunk of RAM. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define PKMAP_BASE	0xfe000000UL
 | 
					
						
							|  |  |  | #define LAST_PKMAP	1024
 | 
					
						
							|  |  |  | #define LAST_PKMAP_MASK (LAST_PKMAP - 1)
 | 
					
						
							|  |  |  | #define PKMAP_NR(virt)  ((virt - PKMAP_BASE) >> PAGE_SHIFT)
 | 
					
						
							|  |  |  | #define PKMAP_ADDR(nr)  (PKMAP_BASE + ((nr) << PAGE_SHIFT))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-13 15:03:17 -08:00
										 |  |  | extern unsigned long kmap_high(struct page *page); | 
					
						
							|  |  |  | extern void kunmap_high(struct page *page); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline unsigned long kmap(struct page *page) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (in_interrupt()) | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	if (page < highmem_start_page) | 
					
						
							|  |  |  | 		return page_address(page); | 
					
						
							|  |  |  | 	return kmap_high(page); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void kunmap(struct page *page) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (in_interrupt()) | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	if (page < highmem_start_page) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	kunmap_high(page); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * The use of kmap_atomic/kunmap_atomic is discouraged - kmap/kunmap | 
					
						
							|  |  |  |  * gives a more generic (and caching) interface. But kmap_atomic can | 
					
						
							|  |  |  |  * be used in IRQ contexts, so in some (very limited) cases we need | 
					
						
							|  |  |  |  * it. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-11-26 10:53:39 +08:00
										 |  |  | static inline unsigned long kmap_atomic(struct page *page) | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned long vaddr; | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	int idx, type; | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	pagefault_disable(); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 	if (page < highmem_start_page) | 
					
						
							|  |  |  | 		return page_address(page); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	type = kmap_atomic_idx_push(); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 	idx = type + KM_TYPE_NR * smp_processor_id(); | 
					
						
							|  |  |  | 	vaddr = __fix_to_virt(FIX_KMAP_BEGIN + idx); | 
					
						
							|  |  |  | #if HIGHMEM_DEBUG
 | 
					
						
							|  |  |  | 	if (!pte_none(*(kmap_pte - idx))) | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	set_pte(kmap_pte - idx, mk_pte(page, kmap_prot)); | 
					
						
							| 
									
										
										
										
											2010-10-27 17:28:49 +01:00
										 |  |  | 	local_flush_tlb_one(vaddr); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return vaddr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | static inline void __kunmap_atomic(unsigned long vaddr) | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	int type; | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	if (vaddr < FIXADDR_START) { /* FIXME */ | 
					
						
							|  |  |  | 		pagefault_enable(); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-27 15:32:58 -07:00
										 |  |  | 	type = kmap_atomic_idx(); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | #if HIGHMEM_DEBUG
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		unsigned int idx; | 
					
						
							|  |  |  | 		idx = type + KM_TYPE_NR * smp_processor_id(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (vaddr != __fix_to_virt(FIX_KMAP_BEGIN + idx)) | 
					
						
							|  |  |  | 			BUG(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * force other mappings to Oops if they'll try to access | 
					
						
							|  |  |  | 		 * this pte without first remap it | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		pte_clear(kmap_pte - idx); | 
					
						
							| 
									
										
										
										
											2010-10-27 17:28:49 +01:00
										 |  |  | 		local_flush_tlb_one(vaddr); | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-10-27 15:32:58 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	kmap_atomic_idx_pop(); | 
					
						
							| 
									
										
										
										
											2010-10-26 14:21:51 -07:00
										 |  |  | 	pagefault_enable(); | 
					
						
							| 
									
										
										
										
											2008-02-08 04:19:31 -08:00
										 |  |  | } | 
					
						
							|  |  |  | #endif /* __KERNEL__ */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* _ASM_HIGHMEM_H */
 |