blob: 26861df52cc4ebde75c53f18621f42dd53ff71d0 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001#ifndef __ASM_MSR_H
2#define __ASM_MSR_H
3
H. Peter Anvin4bc5aa92007-05-02 19:27:12 +02004#include <asm/msr-index.h>
5
6#ifdef __KERNEL__
7#ifndef __ASSEMBLY__
8
Rusty Russell90a0a062007-05-02 19:27:10 +02009#include <asm/errno.h>
10
11static inline unsigned long long native_read_msr(unsigned int msr)
12{
13 unsigned long long val;
14
15 asm volatile("rdmsr" : "=A" (val) : "c" (msr));
16 return val;
17}
18
19static inline unsigned long long native_read_msr_safe(unsigned int msr,
20 int *err)
21{
22 unsigned long long val;
23
24 asm volatile("2: rdmsr ; xorl %0,%0\n"
25 "1:\n\t"
26 ".section .fixup,\"ax\"\n\t"
27 "3: movl %3,%0 ; jmp 1b\n\t"
28 ".previous\n\t"
29 ".section __ex_table,\"a\"\n"
30 " .align 4\n\t"
31 " .long 2b,3b\n\t"
32 ".previous"
33 : "=r" (*err), "=A" (val)
34 : "c" (msr), "i" (-EFAULT));
35
36 return val;
37}
38
39static inline void native_write_msr(unsigned int msr, unsigned long long val)
40{
41 asm volatile("wrmsr" : : "c" (msr), "A"(val));
42}
43
44static inline int native_write_msr_safe(unsigned int msr,
45 unsigned long long val)
46{
47 int err;
48 asm volatile("2: wrmsr ; xorl %0,%0\n"
49 "1:\n\t"
50 ".section .fixup,\"ax\"\n\t"
51 "3: movl %4,%0 ; jmp 1b\n\t"
52 ".previous\n\t"
53 ".section __ex_table,\"a\"\n"
54 " .align 4\n\t"
55 " .long 2b,3b\n\t"
56 ".previous"
57 : "=a" (err)
58 : "c" (msr), "0" ((u32)val), "d" ((u32)(val>>32)),
59 "i" (-EFAULT));
60 return err;
61}
62
63static inline unsigned long long native_read_tsc(void)
64{
65 unsigned long long val;
66 asm volatile("rdtsc" : "=A" (val));
67 return val;
68}
69
70static inline unsigned long long native_read_pmc(void)
71{
72 unsigned long long val;
73 asm volatile("rdpmc" : "=A" (val));
74 return val;
75}
76
Rusty Russelld3561b72006-12-07 02:14:07 +010077#ifdef CONFIG_PARAVIRT
78#include <asm/paravirt.h>
79#else
Rudolf Marek4e9baad2007-05-08 17:22:01 +020080#include <linux/errno.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070081/*
82 * Access to machine-specific registers (available on 586 and better only)
83 * Note: the rd* operations modify the parameters directly (without using
84 * pointer indirection), this allows gcc to optimize better
85 */
86
Rusty Russell90a0a062007-05-02 19:27:10 +020087#define rdmsr(msr,val1,val2) \
88 do { \
89 unsigned long long __val = native_read_msr(msr); \
90 val1 = __val; \
91 val2 = __val >> 32; \
92 } while(0)
Linus Torvalds1da177e2005-04-16 15:20:36 -070093
Rusty Russell90a0a062007-05-02 19:27:10 +020094#define wrmsr(msr,val1,val2) \
95 native_write_msr(msr, ((unsigned long long)val2 << 32) | val1)
Linus Torvalds1da177e2005-04-16 15:20:36 -070096
Rusty Russell90a0a062007-05-02 19:27:10 +020097#define rdmsrl(msr,val) \
98 do { \
99 (val) = native_read_msr(msr); \
100 } while(0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700101
102static inline void wrmsrl (unsigned long msr, unsigned long long val)
103{
104 unsigned long lo, hi;
105 lo = (unsigned long) val;
106 hi = val >> 32;
107 wrmsr (msr, lo, hi);
108}
109
110/* wrmsr with exception handling */
Rusty Russell90a0a062007-05-02 19:27:10 +0200111#define wrmsr_safe(msr,val1,val2) \
112 (native_write_msr_safe(msr, ((unsigned long long)val2 << 32) | val1))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700113
Zachary Amsdenf2ab4462005-09-03 15:56:42 -0700114/* rdmsr with exception handling */
Rusty Russell90a0a062007-05-02 19:27:10 +0200115#define rdmsr_safe(msr,p1,p2) \
116 ({ \
117 int __err; \
118 unsigned long long __val = native_read_msr_safe(msr, &__err);\
119 (*p1) = __val; \
120 (*p2) = __val >> 32; \
121 __err; \
122 })
Zachary Amsdenf2ab4462005-09-03 15:56:42 -0700123
Rusty Russell90a0a062007-05-02 19:27:10 +0200124#define rdtsc(low,high) \
125 do { \
126 u64 _l = native_read_tsc(); \
127 (low) = (u32)_l; \
128 (high) = _l >> 32; \
129 } while(0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700130
Rusty Russell90a0a062007-05-02 19:27:10 +0200131#define rdtscl(low) \
132 do { \
133 (low) = native_read_tsc(); \
134 } while(0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135
Rusty Russell90a0a062007-05-02 19:27:10 +0200136#define rdtscll(val) ((val) = native_read_tsc())
Linus Torvalds1da177e2005-04-16 15:20:36 -0700137
138#define write_tsc(val1,val2) wrmsr(0x10, val1, val2)
139
Rusty Russell90a0a062007-05-02 19:27:10 +0200140#define rdpmc(counter,low,high) \
141 do { \
142 u64 _l = native_read_pmc(); \
143 low = (u32)_l; \
144 high = _l >> 32; \
145 } while(0)
Rusty Russelld3561b72006-12-07 02:14:07 +0100146#endif /* !CONFIG_PARAVIRT */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700147
Adrian Bunkb44755c2007-02-20 01:07:13 +0100148#ifdef CONFIG_SMP
Alexey Dobriyanb077ffb2007-02-16 01:48:11 -0800149void rdmsr_on_cpu(unsigned int cpu, u32 msr_no, u32 *l, u32 *h);
150void wrmsr_on_cpu(unsigned int cpu, u32 msr_no, u32 l, u32 h);
Rudolf Marek4e9baad2007-05-08 17:22:01 +0200151int rdmsr_safe_on_cpu(unsigned int cpu, u32 msr_no, u32 *l, u32 *h);
152int wrmsr_safe_on_cpu(unsigned int cpu, u32 msr_no, u32 l, u32 h);
Adrian Bunkb44755c2007-02-20 01:07:13 +0100153#else /* CONFIG_SMP */
154static inline void rdmsr_on_cpu(unsigned int cpu, u32 msr_no, u32 *l, u32 *h)
155{
156 rdmsr(msr_no, *l, *h);
157}
158static inline void wrmsr_on_cpu(unsigned int cpu, u32 msr_no, u32 l, u32 h)
159{
160 wrmsr(msr_no, l, h);
161}
Rudolf Marek4e9baad2007-05-08 17:22:01 +0200162static inline int rdmsr_safe_on_cpu(unsigned int cpu, u32 msr_no, u32 *l, u32 *h)
163{
164 return rdmsr_safe(msr_no, l, h);
165}
166static inline int wrmsr_safe_on_cpu(unsigned int cpu, u32 msr_no, u32 l, u32 h)
167{
168 return wrmsr_safe(msr_no, l, h);
169}
Adrian Bunkb44755c2007-02-20 01:07:13 +0100170#endif /* CONFIG_SMP */
H. Peter Anvin4bc5aa92007-05-02 19:27:12 +0200171#endif
172#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700173#endif /* __ASM_MSR_H */