diff options
Diffstat (limited to 'arch')
-rw-r--r-- | arch/s390/include/asm/cpu_mf.h | 138 | ||||
-rw-r--r-- | arch/s390/oprofile/hwsampler.c | 67 | ||||
-rw-r--r-- | arch/s390/oprofile/hwsampler.h | 52 |
3 files changed, 142 insertions, 115 deletions
diff --git a/arch/s390/include/asm/cpu_mf.h b/arch/s390/include/asm/cpu_mf.h index c879fad404c8..f6dddeaad965 100644 --- a/arch/s390/include/asm/cpu_mf.h +++ b/arch/s390/include/asm/cpu_mf.h | |||
@@ -56,6 +56,78 @@ struct cpumf_ctr_info { | |||
56 | u32 reserved2[12]; | 56 | u32 reserved2[12]; |
57 | } __packed; | 57 | } __packed; |
58 | 58 | ||
59 | /* QUERY SAMPLING INFORMATION block */ | ||
60 | struct hws_qsi_info_block { /* Bit(s) */ | ||
61 | unsigned int b0_13:14; /* 0-13: zeros */ | ||
62 | unsigned int as:1; /* 14: sampling authorisation control*/ | ||
63 | unsigned int b15_21:7; /* 15-21: zeros */ | ||
64 | unsigned int es:1; /* 22: sampling enable control */ | ||
65 | unsigned int b23_29:7; /* 23-29: zeros */ | ||
66 | unsigned int cs:1; /* 30: sampling activation control */ | ||
67 | unsigned int:1; /* 31: reserved */ | ||
68 | unsigned int bsdes:16; /* 4-5: size of basic sampling entry */ | ||
69 | unsigned int dsdes:16; /* 6-7: size of diagnostic sampling entry */ | ||
70 | unsigned long min_sampl_rate; /* 8-15: minimum sampling interval */ | ||
71 | unsigned long max_sampl_rate; /* 16-23: maximum sampling interval*/ | ||
72 | unsigned long tear; /* 24-31: TEAR contents */ | ||
73 | unsigned long dear; /* 32-39: DEAR contents */ | ||
74 | unsigned int rsvrd0; /* 40-43: reserved */ | ||
75 | unsigned int cpu_speed; /* 44-47: CPU speed */ | ||
76 | unsigned long long rsvrd1; /* 48-55: reserved */ | ||
77 | unsigned long long rsvrd2; /* 56-63: reserved */ | ||
78 | } __packed; | ||
79 | |||
80 | /* SET SAMPLING CONTROLS request block */ | ||
81 | struct hws_lsctl_request_block { | ||
82 | unsigned int s:1; /* 0: maximum buffer indicator */ | ||
83 | unsigned int h:1; /* 1: part. level reserved for VM use*/ | ||
84 | unsigned long long b2_53:52;/* 2-53: zeros */ | ||
85 | unsigned int es:1; /* 54: sampling enable control */ | ||
86 | unsigned int b55_61:7; /* 55-61: - zeros */ | ||
87 | unsigned int cs:1; /* 62: sampling activation control */ | ||
88 | unsigned int b63:1; /* 63: zero */ | ||
89 | unsigned long interval; /* 8-15: sampling interval */ | ||
90 | unsigned long tear; /* 16-23: TEAR contents */ | ||
91 | unsigned long dear; /* 24-31: DEAR contents */ | ||
92 | /* 32-63: */ | ||
93 | unsigned long rsvrd1; /* reserved */ | ||
94 | unsigned long rsvrd2; /* reserved */ | ||
95 | unsigned long rsvrd3; /* reserved */ | ||
96 | unsigned long rsvrd4; /* reserved */ | ||
97 | } __packed; | ||
98 | |||
99 | |||
100 | struct hws_data_entry { | ||
101 | unsigned int def:16; /* 0-15 Data Entry Format */ | ||
102 | unsigned int R:4; /* 16-19 reserved */ | ||
103 | unsigned int U:4; /* 20-23 Number of unique instruct. */ | ||
104 | unsigned int z:2; /* zeros */ | ||
105 | unsigned int T:1; /* 26 PSW DAT mode */ | ||
106 | unsigned int W:1; /* 27 PSW wait state */ | ||
107 | unsigned int P:1; /* 28 PSW Problem state */ | ||
108 | unsigned int AS:2; /* 29-30 PSW address-space control */ | ||
109 | unsigned int I:1; /* 31 entry valid or invalid */ | ||
110 | unsigned int:16; | ||
111 | unsigned int prim_asn:16; /* primary ASN */ | ||
112 | unsigned long long ia; /* Instruction Address */ | ||
113 | unsigned long long gpp; /* Guest Program Parameter */ | ||
114 | unsigned long long hpp; /* Host Program Parameter */ | ||
115 | } __packed; | ||
116 | |||
117 | struct hws_trailer_entry { | ||
118 | unsigned int f:1; /* 0 - Block Full Indicator */ | ||
119 | unsigned int a:1; /* 1 - Alert request control */ | ||
120 | unsigned int t:1; /* 2 - Timestamp format */ | ||
121 | unsigned long long:61; /* 3 - 63: Reserved */ | ||
122 | unsigned long long overflow; /* 64 - sample Overflow count */ | ||
123 | unsigned long long timestamp; /* 16 - time-stamp */ | ||
124 | unsigned long long timestamp1; /* */ | ||
125 | unsigned long long reserved1; /* 32 -Reserved */ | ||
126 | unsigned long long reserved2; /* */ | ||
127 | unsigned long long progusage1; /* 48 - reserved for programming use */ | ||
128 | unsigned long long progusage2; /* */ | ||
129 | } __packed; | ||
130 | |||
59 | /* Query counter information */ | 131 | /* Query counter information */ |
60 | static inline int qctri(struct cpumf_ctr_info *info) | 132 | static inline int qctri(struct cpumf_ctr_info *info) |
61 | { | 133 | { |
@@ -99,4 +171,70 @@ static inline int ecctr(u64 ctr, u64 *val) | |||
99 | return cc; | 171 | return cc; |
100 | } | 172 | } |
101 | 173 | ||
174 | /* Query sampling information */ | ||
175 | static inline int qsi(struct hws_qsi_info_block *info) | ||
176 | { | ||
177 | int cc; | ||
178 | cc = 1; | ||
179 | |||
180 | asm volatile( | ||
181 | "0: .insn s,0xb2860000,0(%1)\n" | ||
182 | "1: lhi %0,0\n" | ||
183 | "2:\n" | ||
184 | EX_TABLE(0b, 2b) EX_TABLE(1b, 2b) | ||
185 | : "=d" (cc), "+a" (info) | ||
186 | : "m" (*info) | ||
187 | : "cc", "memory"); | ||
188 | |||
189 | return cc ? -EINVAL : 0; | ||
190 | } | ||
191 | |||
192 | /* Load sampling controls */ | ||
193 | static inline int lsctl(struct hws_lsctl_request_block *req) | ||
194 | { | ||
195 | int cc; | ||
196 | |||
197 | cc = 1; | ||
198 | asm volatile( | ||
199 | "0: .insn s,0xb2870000,0(%1)\n" | ||
200 | "1: ipm %0\n" | ||
201 | " srl %0,28\n" | ||
202 | "2:\n" | ||
203 | EX_TABLE(0b, 2b) EX_TABLE(1b, 2b) | ||
204 | : "+d" (cc), "+a" (req) | ||
205 | : "m" (*req) | ||
206 | : "cc", "memory"); | ||
207 | |||
208 | return cc ? -EINVAL : 0; | ||
209 | } | ||
210 | |||
211 | /* Sampling control helper functions */ | ||
212 | |||
213 | #define SDB_TE_ALERT_REQ_MASK 0x4000000000000000UL | ||
214 | #define SDB_TE_BUFFER_FULL_MASK 0x8000000000000000UL | ||
215 | |||
216 | /* Return pointer to trailer entry of an sample data block */ | ||
217 | static inline unsigned long *trailer_entry_ptr(unsigned long v) | ||
218 | { | ||
219 | void *ret; | ||
220 | |||
221 | ret = (void *) v; | ||
222 | ret += PAGE_SIZE; | ||
223 | ret -= sizeof(struct hws_trailer_entry); | ||
224 | |||
225 | return (unsigned long *) ret; | ||
226 | } | ||
227 | |||
228 | /* Return if the entry in the sample data block table (sdbt) | ||
229 | * is a link to the next sdbt */ | ||
230 | static inline int is_link_entry(unsigned long *s) | ||
231 | { | ||
232 | return *s & 0x1ul ? 1 : 0; | ||
233 | } | ||
234 | |||
235 | /* Return pointer to the linked sdbt */ | ||
236 | static inline unsigned long *get_next_sdbt(unsigned long *s) | ||
237 | { | ||
238 | return (unsigned long *) (*s & ~0x1ul); | ||
239 | } | ||
102 | #endif /* _ASM_S390_CPU_MF_H */ | 240 | #endif /* _ASM_S390_CPU_MF_H */ |
diff --git a/arch/s390/oprofile/hwsampler.c b/arch/s390/oprofile/hwsampler.c index 231cecafc2f1..bbca76ad6e1b 100644 --- a/arch/s390/oprofile/hwsampler.c +++ b/arch/s390/oprofile/hwsampler.c | |||
@@ -26,9 +26,6 @@ | |||
26 | #define MAX_NUM_SDB 511 | 26 | #define MAX_NUM_SDB 511 |
27 | #define MIN_NUM_SDB 1 | 27 | #define MIN_NUM_SDB 1 |
28 | 28 | ||
29 | #define ALERT_REQ_MASK 0x4000000000000000ul | ||
30 | #define BUFFER_FULL_MASK 0x8000000000000000ul | ||
31 | |||
32 | DECLARE_PER_CPU(struct hws_cpu_buffer, sampler_cpu_buffer); | 29 | DECLARE_PER_CPU(struct hws_cpu_buffer, sampler_cpu_buffer); |
33 | 30 | ||
34 | struct hws_execute_parms { | 31 | struct hws_execute_parms { |
@@ -65,43 +62,6 @@ static unsigned long interval; | |||
65 | static unsigned long min_sampler_rate; | 62 | static unsigned long min_sampler_rate; |
66 | static unsigned long max_sampler_rate; | 63 | static unsigned long max_sampler_rate; |
67 | 64 | ||
68 | static int ssctl(void *buffer) | ||
69 | { | ||
70 | int cc; | ||
71 | |||
72 | /* set in order to detect a program check */ | ||
73 | cc = 1; | ||
74 | |||
75 | asm volatile( | ||
76 | "0: .insn s,0xB2870000,0(%1)\n" | ||
77 | "1: ipm %0\n" | ||
78 | " srl %0,28\n" | ||
79 | "2:\n" | ||
80 | EX_TABLE(0b, 2b) EX_TABLE(1b, 2b) | ||
81 | : "+d" (cc), "+a" (buffer) | ||
82 | : "m" (*((struct hws_ssctl_request_block *)buffer)) | ||
83 | : "cc", "memory"); | ||
84 | |||
85 | return cc ? -EINVAL : 0 ; | ||
86 | } | ||
87 | |||
88 | static int qsi(void *buffer) | ||
89 | { | ||
90 | int cc; | ||
91 | cc = 1; | ||
92 | |||
93 | asm volatile( | ||
94 | "0: .insn s,0xB2860000,0(%1)\n" | ||
95 | "1: lhi %0,0\n" | ||
96 | "2:\n" | ||
97 | EX_TABLE(0b, 2b) EX_TABLE(1b, 2b) | ||
98 | : "=d" (cc), "+a" (buffer) | ||
99 | : "m" (*((struct hws_qsi_info_block *)buffer)) | ||
100 | : "cc", "memory"); | ||
101 | |||
102 | return cc ? -EINVAL : 0; | ||
103 | } | ||
104 | |||
105 | static void execute_qsi(void *parms) | 65 | static void execute_qsi(void *parms) |
106 | { | 66 | { |
107 | struct hws_execute_parms *ep = parms; | 67 | struct hws_execute_parms *ep = parms; |
@@ -113,7 +73,7 @@ static void execute_ssctl(void *parms) | |||
113 | { | 73 | { |
114 | struct hws_execute_parms *ep = parms; | 74 | struct hws_execute_parms *ep = parms; |
115 | 75 | ||
116 | ep->rc = ssctl(ep->buffer); | 76 | ep->rc = lsctl(ep->buffer); |
117 | } | 77 | } |
118 | 78 | ||
119 | static int smp_ctl_ssctl_stop(int cpu) | 79 | static int smp_ctl_ssctl_stop(int cpu) |
@@ -214,17 +174,6 @@ static int smp_ctl_qsi(int cpu) | |||
214 | return ep.rc; | 174 | return ep.rc; |
215 | } | 175 | } |
216 | 176 | ||
217 | static inline unsigned long *trailer_entry_ptr(unsigned long v) | ||
218 | { | ||
219 | void *ret; | ||
220 | |||
221 | ret = (void *)v; | ||
222 | ret += PAGE_SIZE; | ||
223 | ret -= sizeof(struct hws_trailer_entry); | ||
224 | |||
225 | return (unsigned long *) ret; | ||
226 | } | ||
227 | |||
228 | static void hws_ext_handler(struct ext_code ext_code, | 177 | static void hws_ext_handler(struct ext_code ext_code, |
229 | unsigned int param32, unsigned long param64) | 178 | unsigned int param32, unsigned long param64) |
230 | { | 179 | { |
@@ -256,16 +205,6 @@ static void init_all_cpu_buffers(void) | |||
256 | } | 205 | } |
257 | } | 206 | } |
258 | 207 | ||
259 | static int is_link_entry(unsigned long *s) | ||
260 | { | ||
261 | return *s & 0x1ul ? 1 : 0; | ||
262 | } | ||
263 | |||
264 | static unsigned long *get_next_sdbt(unsigned long *s) | ||
265 | { | ||
266 | return (unsigned long *) (*s & ~0x1ul); | ||
267 | } | ||
268 | |||
269 | static int prepare_cpu_buffers(void) | 208 | static int prepare_cpu_buffers(void) |
270 | { | 209 | { |
271 | int cpu; | 210 | int cpu; |
@@ -353,7 +292,7 @@ static int allocate_sdbt(int cpu) | |||
353 | } | 292 | } |
354 | *sdbt = sdb; | 293 | *sdbt = sdb; |
355 | trailer = trailer_entry_ptr(*sdbt); | 294 | trailer = trailer_entry_ptr(*sdbt); |
356 | *trailer = ALERT_REQ_MASK; | 295 | *trailer = SDB_TE_ALERT_REQ_MASK; |
357 | sdbt++; | 296 | sdbt++; |
358 | mutex_unlock(&hws_sem_oom); | 297 | mutex_unlock(&hws_sem_oom); |
359 | } | 298 | } |
@@ -829,7 +768,7 @@ static void worker_on_interrupt(unsigned int cpu) | |||
829 | 768 | ||
830 | trailer = trailer_entry_ptr(*sdbt); | 769 | trailer = trailer_entry_ptr(*sdbt); |
831 | /* leave loop if no more work to do */ | 770 | /* leave loop if no more work to do */ |
832 | if (!(*trailer & BUFFER_FULL_MASK)) { | 771 | if (!(*trailer & SDB_TE_BUFFER_FULL_MASK)) { |
833 | done = 1; | 772 | done = 1; |
834 | if (!hws_flush_all) | 773 | if (!hws_flush_all) |
835 | continue; | 774 | continue; |
diff --git a/arch/s390/oprofile/hwsampler.h b/arch/s390/oprofile/hwsampler.h index 0022e1ebfbde..a483d06f2fa7 100644 --- a/arch/s390/oprofile/hwsampler.h +++ b/arch/s390/oprofile/hwsampler.h | |||
@@ -9,27 +9,7 @@ | |||
9 | #define HWSAMPLER_H_ | 9 | #define HWSAMPLER_H_ |
10 | 10 | ||
11 | #include <linux/workqueue.h> | 11 | #include <linux/workqueue.h> |
12 | 12 | #include <asm/cpu_mf.h> | |
13 | struct hws_qsi_info_block /* QUERY SAMPLING information block */ | ||
14 | { /* Bit(s) */ | ||
15 | unsigned int b0_13:14; /* 0-13: zeros */ | ||
16 | unsigned int as:1; /* 14: sampling authorisation control*/ | ||
17 | unsigned int b15_21:7; /* 15-21: zeros */ | ||
18 | unsigned int es:1; /* 22: sampling enable control */ | ||
19 | unsigned int b23_29:7; /* 23-29: zeros */ | ||
20 | unsigned int cs:1; /* 30: sampling activation control */ | ||
21 | unsigned int:1; /* 31: reserved */ | ||
22 | unsigned int bsdes:16; /* 4-5: size of sampling entry */ | ||
23 | unsigned int:16; /* 6-7: reserved */ | ||
24 | unsigned long min_sampl_rate; /* 8-15: minimum sampling interval */ | ||
25 | unsigned long max_sampl_rate; /* 16-23: maximum sampling interval*/ | ||
26 | unsigned long tear; /* 24-31: TEAR contents */ | ||
27 | unsigned long dear; /* 32-39: DEAR contents */ | ||
28 | unsigned int rsvrd0; /* 40-43: reserved */ | ||
29 | unsigned int cpu_speed; /* 44-47: CPU speed */ | ||
30 | unsigned long long rsvrd1; /* 48-55: reserved */ | ||
31 | unsigned long long rsvrd2; /* 56-63: reserved */ | ||
32 | }; | ||
33 | 13 | ||
34 | struct hws_ssctl_request_block /* SET SAMPLING CONTROLS req block */ | 14 | struct hws_ssctl_request_block /* SET SAMPLING CONTROLS req block */ |
35 | { /* bytes 0 - 7 Bit(s) */ | 15 | { /* bytes 0 - 7 Bit(s) */ |
@@ -68,36 +48,6 @@ struct hws_cpu_buffer { | |||
68 | unsigned int stop_mode:1; | 48 | unsigned int stop_mode:1; |
69 | }; | 49 | }; |
70 | 50 | ||
71 | struct hws_data_entry { | ||
72 | unsigned int def:16; /* 0-15 Data Entry Format */ | ||
73 | unsigned int R:4; /* 16-19 reserved */ | ||
74 | unsigned int U:4; /* 20-23 Number of unique instruct. */ | ||
75 | unsigned int z:2; /* zeros */ | ||
76 | unsigned int T:1; /* 26 PSW DAT mode */ | ||
77 | unsigned int W:1; /* 27 PSW wait state */ | ||
78 | unsigned int P:1; /* 28 PSW Problem state */ | ||
79 | unsigned int AS:2; /* 29-30 PSW address-space control */ | ||
80 | unsigned int I:1; /* 31 entry valid or invalid */ | ||
81 | unsigned int:16; | ||
82 | unsigned int prim_asn:16; /* primary ASN */ | ||
83 | unsigned long long ia; /* Instruction Address */ | ||
84 | unsigned long long gpp; /* Guest Program Parameter */ | ||
85 | unsigned long long hpp; /* Host Program Parameter */ | ||
86 | }; | ||
87 | |||
88 | struct hws_trailer_entry { | ||
89 | unsigned int f:1; /* 0 - Block Full Indicator */ | ||
90 | unsigned int a:1; /* 1 - Alert request control */ | ||
91 | unsigned long:62; /* 2 - 63: Reserved */ | ||
92 | unsigned long overflow; /* 64 - sample Overflow count */ | ||
93 | unsigned long timestamp; /* 16 - time-stamp */ | ||
94 | unsigned long timestamp1; /* */ | ||
95 | unsigned long reserved1; /* 32 -Reserved */ | ||
96 | unsigned long reserved2; /* */ | ||
97 | unsigned long progusage1; /* 48 - reserved for programming use */ | ||
98 | unsigned long progusage2; /* */ | ||
99 | }; | ||
100 | |||
101 | int hwsampler_setup(void); | 51 | int hwsampler_setup(void); |
102 | int hwsampler_shutdown(void); | 52 | int hwsampler_shutdown(void); |
103 | int hwsampler_allocate(unsigned long sdbt, unsigned long sdb); | 53 | int hwsampler_allocate(unsigned long sdbt, unsigned long sdb); |