David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 1 | /* |
| 2 | * Copyright IBM Corp. 1999, 2009 |
| 3 | * |
| 4 | * Author(s): Martin Schwidefsky <schwidefsky@de.ibm.com> |
| 5 | */ |
| 6 | |
| 7 | #ifndef __ASM_CTL_REG_H |
| 8 | #define __ASM_CTL_REG_H |
| 9 | |
Michael Holzheu | acf6a00 | 2013-11-13 10:38:27 +0100 | [diff] [blame^] | 10 | #include <linux/bug.h> |
| 11 | |
Heiko Carstens | f4815ac | 2012-05-23 16:24:51 +0200 | [diff] [blame] | 12 | #ifdef CONFIG_64BIT |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 13 | # define __CTL_LOAD "lctlg" |
| 14 | # define __CTL_STORE "stctg" |
| 15 | #else |
| 16 | # define __CTL_LOAD "lctl" |
| 17 | # define __CTL_STORE "stctl" |
| 18 | #endif |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 19 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 20 | #define __ctl_load(array, low, high) { \ |
| 21 | typedef struct { char _[sizeof(array)]; } addrtype; \ |
| 22 | \ |
| 23 | BUILD_BUG_ON(sizeof(addrtype) != (high - low + 1) * sizeof(long));\ |
| 24 | asm volatile( \ |
| 25 | __CTL_LOAD " %1,%2,%0\n" \ |
| 26 | : : "Q" (*(addrtype *)(&array)), "i" (low), "i" (high));\ |
| 27 | } |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 28 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 29 | #define __ctl_store(array, low, high) { \ |
| 30 | typedef struct { char _[sizeof(array)]; } addrtype; \ |
| 31 | \ |
| 32 | BUILD_BUG_ON(sizeof(addrtype) != (high - low + 1) * sizeof(long));\ |
| 33 | asm volatile( \ |
| 34 | __CTL_STORE " %1,%2,%0\n" \ |
| 35 | : "=Q" (*(addrtype *)(&array)) \ |
| 36 | : "i" (low), "i" (high)); \ |
| 37 | } |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 38 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 39 | static inline void __ctl_set_bit(unsigned int cr, unsigned int bit) |
| 40 | { |
| 41 | unsigned long reg; |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 42 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 43 | __ctl_store(reg, cr, cr); |
| 44 | reg |= 1UL << bit; |
| 45 | __ctl_load(reg, cr, cr); |
| 46 | } |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 47 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 48 | static inline void __ctl_clear_bit(unsigned int cr, unsigned int bit) |
| 49 | { |
| 50 | unsigned long reg; |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 51 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 52 | __ctl_store(reg, cr, cr); |
| 53 | reg &= ~(1UL << bit); |
| 54 | __ctl_load(reg, cr, cr); |
| 55 | } |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 56 | |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 57 | void smp_ctl_set_bit(int cr, int bit); |
| 58 | void smp_ctl_clear_bit(int cr, int bit); |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 59 | |
| 60 | #ifdef CONFIG_SMP |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 61 | # define ctl_set_bit(cr, bit) smp_ctl_set_bit(cr, bit) |
| 62 | # define ctl_clear_bit(cr, bit) smp_ctl_clear_bit(cr, bit) |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 63 | #else |
Heiko Carstens | 12325f0 | 2013-09-30 14:47:46 +0200 | [diff] [blame] | 64 | # define ctl_set_bit(cr, bit) __ctl_set_bit(cr, bit) |
| 65 | # define ctl_clear_bit(cr, bit) __ctl_clear_bit(cr, bit) |
| 66 | #endif |
David Howells | a0616cd | 2012-03-28 18:30:02 +0100 | [diff] [blame] | 67 | |
| 68 | #endif /* __ASM_CTL_REG_H */ |