| 
									
										
										
										
											2012-03-28 18:30:02 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright IBM Corp. 1999, 2009 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Author(s): Martin Schwidefsky <schwidefsky@de.ibm.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef __ASM_CTL_REG_H
 | 
					
						
							|  |  |  | #define __ASM_CTL_REG_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-13 10:38:27 +01:00
										 |  |  | #include <linux/bug.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-23 16:24:51 +02:00
										 |  |  | #ifdef CONFIG_64BIT
 | 
					
						
							| 
									
										
										
										
											2013-09-30 14:47:46 +02:00
										 |  |  | # define __CTL_LOAD	"lctlg"
 | 
					
						
							|  |  |  | # define __CTL_STORE	"stctg"
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | # define __CTL_LOAD	"lctl"
 | 
					
						
							|  |  |  | # define __CTL_STORE	"stctl"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define __ctl_load(array, low, high) {					\
 | 
					
						
							|  |  |  | 	typedef struct { char _[sizeof(array)]; } addrtype;		\ | 
					
						
							|  |  |  | 									\ | 
					
						
							|  |  |  | 	BUILD_BUG_ON(sizeof(addrtype) != (high - low + 1) * sizeof(long));\ | 
					
						
							|  |  |  | 	asm volatile(							\ | 
					
						
							|  |  |  | 		__CTL_LOAD " %1,%2,%0\n"				\ | 
					
						
							|  |  |  | 		: : "Q" (*(addrtype *)(&array)), "i" (low), "i" (high));\ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define __ctl_store(array, low, high) {					\
 | 
					
						
							|  |  |  | 	typedef struct { char _[sizeof(array)]; } addrtype;		\ | 
					
						
							|  |  |  | 									\ | 
					
						
							|  |  |  | 	BUILD_BUG_ON(sizeof(addrtype) != (high - low + 1) * sizeof(long));\ | 
					
						
							|  |  |  | 	asm volatile(							\ | 
					
						
							|  |  |  | 		__CTL_STORE " %1,%2,%0\n"				\ | 
					
						
							|  |  |  | 		: "=Q" (*(addrtype *)(&array))				\ | 
					
						
							|  |  |  | 		: "i" (low), "i" (high));				\ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void __ctl_set_bit(unsigned int cr, unsigned int bit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long reg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	__ctl_store(reg, cr, cr); | 
					
						
							|  |  |  | 	reg |= 1UL << bit; | 
					
						
							|  |  |  | 	__ctl_load(reg, cr, cr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void __ctl_clear_bit(unsigned int cr, unsigned int bit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long reg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	__ctl_store(reg, cr, cr); | 
					
						
							|  |  |  | 	reg &= ~(1UL << bit); | 
					
						
							|  |  |  | 	__ctl_load(reg, cr, cr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void smp_ctl_set_bit(int cr, int bit); | 
					
						
							|  |  |  | void smp_ctl_clear_bit(int cr, int bit); | 
					
						
							| 
									
										
										
										
											2012-03-28 18:30:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-01 16:08:37 +01:00
										 |  |  | union ctlreg0 { | 
					
						
							|  |  |  | 	unsigned long val; | 
					
						
							|  |  |  | 	struct { | 
					
						
							|  |  |  | #ifdef CONFIG_64BIT
 | 
					
						
							|  |  |  | 		unsigned long	   : 32; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		unsigned long	   : 3; | 
					
						
							|  |  |  | 		unsigned long lap  : 1; /* Low-address-protection control */ | 
					
						
							|  |  |  | 		unsigned long	   : 4; | 
					
						
							|  |  |  | 		unsigned long edat : 1; /* Enhanced-DAT-enablement control */ | 
					
						
							|  |  |  | 		unsigned long	   : 23; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:30:02 +01:00
										 |  |  | #ifdef CONFIG_SMP
 | 
					
						
							| 
									
										
										
										
											2013-09-30 14:47:46 +02:00
										 |  |  | # define ctl_set_bit(cr, bit) smp_ctl_set_bit(cr, bit)
 | 
					
						
							|  |  |  | # define ctl_clear_bit(cr, bit) smp_ctl_clear_bit(cr, bit)
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:30:02 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-09-30 14:47:46 +02:00
										 |  |  | # define ctl_set_bit(cr, bit) __ctl_set_bit(cr, bit)
 | 
					
						
							|  |  |  | # define ctl_clear_bit(cr, bit) __ctl_clear_bit(cr, bit)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:30:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* __ASM_CTL_REG_H */
 |