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 /arch/s390/oprofile | |
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 'arch/s390/oprofile')
-rw-r--r-- | arch/s390/oprofile/hwsampler_files.c | 60 | ||||
-rw-r--r-- | arch/s390/oprofile/init.c | 1 |
2 files changed, 38 insertions, 23 deletions
diff --git a/arch/s390/oprofile/hwsampler_files.c b/arch/s390/oprofile/hwsampler_files.c index 493f7cc4e861..2e1da2449ba9 100644 --- a/arch/s390/oprofile/hwsampler_files.c +++ b/arch/s390/oprofile/hwsampler_files.c | |||
@@ -8,6 +8,7 @@ | |||
8 | #include <linux/errno.h> | 8 | #include <linux/errno.h> |
9 | #include <linux/fs.h> | 9 | #include <linux/fs.h> |
10 | 10 | ||
11 | #include "../../../drivers/oprofile/oprof.h" | ||
11 | #include "hwsampler.h" | 12 | #include "hwsampler.h" |
12 | 13 | ||
13 | #define DEFAULT_INTERVAL 4096 | 14 | #define DEFAULT_INTERVAL 4096 |
@@ -22,12 +23,20 @@ static unsigned long oprofile_max_interval; | |||
22 | static unsigned long oprofile_sdbt_blocks = DEFAULT_SDBT_BLOCKS; | 23 | static unsigned long oprofile_sdbt_blocks = DEFAULT_SDBT_BLOCKS; |
23 | static unsigned long oprofile_sdb_blocks = DEFAULT_SDB_BLOCKS; | 24 | static unsigned long oprofile_sdb_blocks = DEFAULT_SDB_BLOCKS; |
24 | 25 | ||
25 | static unsigned long oprofile_hwsampler; | 26 | static int hwsampler_file; |
27 | static int hwsampler_running; /* start_mutex must be held to change */ | ||
28 | |||
29 | static struct oprofile_operations timer_ops; | ||
26 | 30 | ||
27 | static int oprofile_hwsampler_start(void) | 31 | static int oprofile_hwsampler_start(void) |
28 | { | 32 | { |
29 | int retval; | 33 | int retval; |
30 | 34 | ||
35 | hwsampler_running = hwsampler_file; | ||
36 | |||
37 | if (!hwsampler_running) | ||
38 | return timer_ops.start(); | ||
39 | |||
31 | retval = hwsampler_allocate(oprofile_sdbt_blocks, oprofile_sdb_blocks); | 40 | retval = hwsampler_allocate(oprofile_sdbt_blocks, oprofile_sdb_blocks); |
32 | if (retval) | 41 | if (retval) |
33 | return retval; | 42 | return retval; |
@@ -41,25 +50,20 @@ static int oprofile_hwsampler_start(void) | |||
41 | 50 | ||
42 | static void oprofile_hwsampler_stop(void) | 51 | static void oprofile_hwsampler_stop(void) |
43 | { | 52 | { |
53 | if (!hwsampler_running) { | ||
54 | timer_ops.stop(); | ||
55 | return; | ||
56 | } | ||
57 | |||
44 | hwsampler_stop_all(); | 58 | hwsampler_stop_all(); |
45 | hwsampler_deallocate(); | 59 | hwsampler_deallocate(); |
46 | return; | 60 | return; |
47 | } | 61 | } |
48 | 62 | ||
49 | int oprofile_arch_set_hwsampler(struct oprofile_operations *ops) | ||
50 | { | ||
51 | printk(KERN_INFO "oprofile: using hardware sampling\n"); | ||
52 | ops->start = oprofile_hwsampler_start; | ||
53 | ops->stop = oprofile_hwsampler_stop; | ||
54 | ops->cpu_type = "timer"; | ||
55 | |||
56 | return 0; | ||
57 | } | ||
58 | |||
59 | static ssize_t hwsampler_read(struct file *file, char __user *buf, | 63 | static ssize_t hwsampler_read(struct file *file, char __user *buf, |
60 | size_t count, loff_t *offset) | 64 | size_t count, loff_t *offset) |
61 | { | 65 | { |
62 | return oprofilefs_ulong_to_user(oprofile_hwsampler, buf, count, offset); | 66 | return oprofilefs_ulong_to_user(hwsampler_file, buf, count, offset); |
63 | } | 67 | } |
64 | 68 | ||
65 | static ssize_t hwsampler_write(struct file *file, char const __user *buf, | 69 | static ssize_t hwsampler_write(struct file *file, char const __user *buf, |
@@ -75,15 +79,16 @@ static ssize_t hwsampler_write(struct file *file, char const __user *buf, | |||
75 | if (retval) | 79 | if (retval) |
76 | return retval; | 80 | return retval; |
77 | 81 | ||
78 | if (oprofile_hwsampler == val) | 82 | if (oprofile_started) |
79 | return -EINVAL; | 83 | /* |
80 | 84 | * save to do without locking as we set | |
81 | retval = oprofile_set_hwsampler(val); | 85 | * hwsampler_running in start() when start_mutex is |
86 | * held | ||
87 | */ | ||
88 | return -EBUSY; | ||
82 | 89 | ||
83 | if (retval) | 90 | hwsampler_file = val; |
84 | return retval; | ||
85 | 91 | ||
86 | oprofile_hwsampler = val; | ||
87 | return count; | 92 | return count; |
88 | } | 93 | } |
89 | 94 | ||
@@ -98,7 +103,7 @@ static int oprofile_create_hwsampling_files(struct super_block *sb, | |||
98 | struct dentry *hw_dir; | 103 | struct dentry *hw_dir; |
99 | 104 | ||
100 | /* reinitialize default values */ | 105 | /* reinitialize default values */ |
101 | oprofile_hwsampler = 1; | 106 | hwsampler_file = 1; |
102 | 107 | ||
103 | hw_dir = oprofilefs_mkdir(sb, root, "hwsampling"); | 108 | hw_dir = oprofilefs_mkdir(sb, root, "hwsampling"); |
104 | if (!hw_dir) | 109 | if (!hw_dir) |
@@ -125,7 +130,6 @@ int oprofile_hwsampler_init(struct oprofile_operations* ops) | |||
125 | /* | 130 | /* |
126 | * create hwsampler files only if hwsampler_setup() succeeds. | 131 | * create hwsampler files only if hwsampler_setup() succeeds. |
127 | */ | 132 | */ |
128 | ops->create_files = oprofile_create_hwsampling_files; | ||
129 | oprofile_min_interval = hwsampler_query_min_interval(); | 133 | oprofile_min_interval = hwsampler_query_min_interval(); |
130 | if (oprofile_min_interval < 0) { | 134 | if (oprofile_min_interval < 0) { |
131 | oprofile_min_interval = 0; | 135 | oprofile_min_interval = 0; |
@@ -136,11 +140,23 @@ int oprofile_hwsampler_init(struct oprofile_operations* ops) | |||
136 | oprofile_max_interval = 0; | 140 | oprofile_max_interval = 0; |
137 | return -ENODEV; | 141 | return -ENODEV; |
138 | } | 142 | } |
139 | oprofile_arch_set_hwsampler(ops); | 143 | |
144 | if (oprofile_timer_init(ops)) | ||
145 | return -ENODEV; | ||
146 | |||
147 | printk(KERN_INFO "oprofile: using hardware sampling\n"); | ||
148 | |||
149 | memcpy(&timer_ops, ops, sizeof(timer_ops)); | ||
150 | |||
151 | ops->start = oprofile_hwsampler_start; | ||
152 | ops->stop = oprofile_hwsampler_stop; | ||
153 | ops->create_files = oprofile_create_hwsampling_files; | ||
154 | |||
140 | return 0; | 155 | return 0; |
141 | } | 156 | } |
142 | 157 | ||
143 | void oprofile_hwsampler_exit(void) | 158 | void oprofile_hwsampler_exit(void) |
144 | { | 159 | { |
160 | oprofile_timer_exit(); | ||
145 | hwsampler_shutdown(); | 161 | hwsampler_shutdown(); |
146 | } | 162 | } |
diff --git a/arch/s390/oprofile/init.c b/arch/s390/oprofile/init.c index f6b3f724f590..059b44b9f171 100644 --- a/arch/s390/oprofile/init.c +++ b/arch/s390/oprofile/init.c | |||
@@ -11,7 +11,6 @@ | |||
11 | #include <linux/oprofile.h> | 11 | #include <linux/oprofile.h> |
12 | #include <linux/init.h> | 12 | #include <linux/init.h> |
13 | #include <linux/errno.h> | 13 | #include <linux/errno.h> |
14 | #include <linux/fs.h> | ||
15 | 14 | ||
16 | extern int oprofile_hwsampler_init(struct oprofile_operations* ops); | 15 | extern int oprofile_hwsampler_init(struct oprofile_operations* ops); |
17 | extern void oprofile_hwsampler_exit(void); | 16 | extern void oprofile_hwsampler_exit(void); |