diff options
author | Robert Richter <robert.richter@amd.com> | 2011-02-11 11:31:44 -0500 |
---|---|---|
committer | Robert Richter <robert.richter@amd.com> | 2011-02-15 05:10:20 -0500 |
commit | a0d76247e07abd14968adc4486aaa8e270e9c209 (patch) | |
tree | bec9fa0f5a3c7b3d1f3c3032900d74f41941ed2d /drivers | |
parent | 997dbb4967da248808850c250182ef2528fff2d1 (diff) |
oprofile, s390: Rework hwsampler implementation
This patch is a rework of the hwsampler oprofile implementation that
has been applied recently. Now there are less non-architectural
changes. The only changes are:
* introduction of oprofile_add_ext_hw_sample(), and
* removal of section attributes of oprofile_timer_init/_exit().
To setup hwsampler for oprofile we need to modify start()/stop()
callbacks and additional hwsampler control files in oprofilefs. We do
not reinitialize the timer or hwsampler mode by restarting calling
init/exit() anymore, instead hwsampler_running is used to switch the
mode directly in oprofile_hwsampler_start/_stop(). For locking reasons
there is also hwsampler_file that reflects the value in oprofilefs.
The overall diffstat of the oprofile s390 hwsampler implemenation
shows the low impact to non-architectural code:
arch/Kconfig | 3 +
arch/s390/Kconfig | 1 +
arch/s390/oprofile/Makefile | 2 +-
arch/s390/oprofile/hwsampler.c | 1256 ++++++++++++++++++++++++++++++++++
arch/s390/oprofile/hwsampler.h | 113 +++
arch/s390/oprofile/hwsampler_files.c | 162 +++++
arch/s390/oprofile/init.c | 6 +-
drivers/oprofile/cpu_buffer.c | 24 +-
drivers/oprofile/timer_int.c | 4 +-
include/linux/oprofile.h | 7 +
10 files changed, 1567 insertions(+), 11 deletions(-)
Acked-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Robert Richter <robert.richter@amd.com>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/oprofile/oprof.c | 32 | ||||
-rw-r--r-- | drivers/oprofile/oprof.h | 2 | ||||
-rw-r--r-- | drivers/oprofile/timer_int.c | 15 |
3 files changed, 3 insertions, 46 deletions
diff --git a/drivers/oprofile/oprof.c b/drivers/oprofile/oprof.c index 43b01daa91e1..f9bda64fcd1b 100644 --- a/drivers/oprofile/oprof.c +++ b/drivers/oprofile/oprof.c | |||
@@ -239,38 +239,6 @@ int oprofile_set_ulong(unsigned long *addr, unsigned long val) | |||
239 | return err; | 239 | return err; |
240 | } | 240 | } |
241 | 241 | ||
242 | #ifdef CONFIG_HAVE_HWSAMPLER | ||
243 | int oprofile_set_hwsampler(unsigned long val) | ||
244 | { | ||
245 | int err = 0; | ||
246 | |||
247 | mutex_lock(&start_mutex); | ||
248 | |||
249 | if (oprofile_started) { | ||
250 | err = -EBUSY; | ||
251 | goto out; | ||
252 | } | ||
253 | |||
254 | switch (val) { | ||
255 | case 1: | ||
256 | /* Switch to hardware sampling. */ | ||
257 | __oprofile_timer_exit(); | ||
258 | err = oprofile_arch_set_hwsampler(&oprofile_ops); | ||
259 | break; | ||
260 | case 0: | ||
261 | printk(KERN_INFO "oprofile: using timer interrupt.\n"); | ||
262 | err = __oprofile_timer_init(&oprofile_ops); | ||
263 | break; | ||
264 | default: | ||
265 | err = -EINVAL; | ||
266 | } | ||
267 | |||
268 | out: | ||
269 | mutex_unlock(&start_mutex); | ||
270 | return err; | ||
271 | } | ||
272 | #endif /* CONFIG_HAVE_HWSAMPLER */ | ||
273 | |||
274 | static int __init oprofile_init(void) | 242 | static int __init oprofile_init(void) |
275 | { | 243 | { |
276 | int err; | 244 | int err; |
diff --git a/drivers/oprofile/oprof.h b/drivers/oprofile/oprof.h index 5a6ceb1954a2..177b73de5e5f 100644 --- a/drivers/oprofile/oprof.h +++ b/drivers/oprofile/oprof.h | |||
@@ -35,9 +35,7 @@ struct dentry; | |||
35 | 35 | ||
36 | void oprofile_create_files(struct super_block *sb, struct dentry *root); | 36 | void oprofile_create_files(struct super_block *sb, struct dentry *root); |
37 | int oprofile_timer_init(struct oprofile_operations *ops); | 37 | int oprofile_timer_init(struct oprofile_operations *ops); |
38 | int __oprofile_timer_init(struct oprofile_operations *ops); | ||
39 | void oprofile_timer_exit(void); | 38 | void oprofile_timer_exit(void); |
40 | void __oprofile_timer_exit(void); | ||
41 | 39 | ||
42 | int oprofile_set_ulong(unsigned long *addr, unsigned long val); | 40 | int oprofile_set_ulong(unsigned long *addr, unsigned long val); |
43 | int oprofile_set_timeout(unsigned long time); | 41 | int oprofile_set_timeout(unsigned long time); |
diff --git a/drivers/oprofile/timer_int.c b/drivers/oprofile/timer_int.c index 0099a458fd37..3ef44624f510 100644 --- a/drivers/oprofile/timer_int.c +++ b/drivers/oprofile/timer_int.c | |||
@@ -97,13 +97,14 @@ static struct notifier_block __refdata oprofile_cpu_notifier = { | |||
97 | .notifier_call = oprofile_cpu_notify, | 97 | .notifier_call = oprofile_cpu_notify, |
98 | }; | 98 | }; |
99 | 99 | ||
100 | int __oprofile_timer_init(struct oprofile_operations *ops) | 100 | int oprofile_timer_init(struct oprofile_operations *ops) |
101 | { | 101 | { |
102 | int rc; | 102 | int rc; |
103 | 103 | ||
104 | rc = register_hotcpu_notifier(&oprofile_cpu_notifier); | 104 | rc = register_hotcpu_notifier(&oprofile_cpu_notifier); |
105 | if (rc) | 105 | if (rc) |
106 | return rc; | 106 | return rc; |
107 | ops->create_files = NULL; | ||
107 | ops->setup = NULL; | 108 | ops->setup = NULL; |
108 | ops->shutdown = NULL; | 109 | ops->shutdown = NULL; |
109 | ops->start = oprofile_hrtimer_start; | 110 | ops->start = oprofile_hrtimer_start; |
@@ -112,17 +113,7 @@ int __oprofile_timer_init(struct oprofile_operations *ops) | |||
112 | return 0; | 113 | return 0; |
113 | } | 114 | } |
114 | 115 | ||
115 | int __init oprofile_timer_init(struct oprofile_operations *ops) | 116 | void oprofile_timer_exit(void) |
116 | { | ||
117 | return __oprofile_timer_init(ops); | ||
118 | } | ||
119 | |||
120 | void __oprofile_timer_exit(void) | ||
121 | { | 117 | { |
122 | unregister_hotcpu_notifier(&oprofile_cpu_notifier); | 118 | unregister_hotcpu_notifier(&oprofile_cpu_notifier); |
123 | } | 119 | } |
124 | |||
125 | void __exit oprofile_timer_exit(void) | ||
126 | { | ||
127 | __oprofile_timer_exit(); | ||
128 | } | ||