diff options
author | Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> | 2011-10-03 14:01:46 -0400 |
---|---|---|
committer | Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> | 2011-10-25 14:54:31 -0400 |
commit | 20284aa77c0f6227da4783a920b72dc61d4bcc09 (patch) | |
tree | 46fb708a9508e3d718be7eb432367547e71208d3 | |
parent | 92e02396d7214fdc7c25239eb647eb48e6fe6ba9 (diff) |
jump_label: add arch_jump_label_transform_static() to optimise non-live code updates
When updating a newly loaded module, the code is definitely not yet
executing on any processor, so it can be updated with no need for any
heavyweight synchronization.
This patch adds arch_jump_label_static() which is implemented as
arch_jump_label_transform() by default, but architectures can override
it if it avoids, say, a call to stop_machine().
Signed-off-by: Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com>
Acked-by: Jason Baron <jbaron@redhat.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
-rw-r--r-- | include/linux/jump_label.h | 2 | ||||
-rw-r--r-- | kernel/jump_label.c | 18 |
2 files changed, 17 insertions, 3 deletions
diff --git a/include/linux/jump_label.h b/include/linux/jump_label.h index 12e804ea32ab..56594e45b011 100644 --- a/include/linux/jump_label.h +++ b/include/linux/jump_label.h | |||
@@ -45,6 +45,8 @@ extern void jump_label_lock(void); | |||
45 | extern void jump_label_unlock(void); | 45 | extern void jump_label_unlock(void); |
46 | extern void arch_jump_label_transform(struct jump_entry *entry, | 46 | extern void arch_jump_label_transform(struct jump_entry *entry, |
47 | enum jump_label_type type); | 47 | enum jump_label_type type); |
48 | extern void arch_jump_label_transform_static(struct jump_entry *entry, | ||
49 | enum jump_label_type type); | ||
48 | extern int jump_label_text_reserved(void *start, void *end); | 50 | extern int jump_label_text_reserved(void *start, void *end); |
49 | extern void jump_label_inc(struct jump_label_key *key); | 51 | extern void jump_label_inc(struct jump_label_key *key); |
50 | extern void jump_label_dec(struct jump_label_key *key); | 52 | extern void jump_label_dec(struct jump_label_key *key); |
diff --git a/kernel/jump_label.c b/kernel/jump_label.c index 059202d5b77a..ff2028f35aa8 100644 --- a/kernel/jump_label.c +++ b/kernel/jump_label.c | |||
@@ -104,6 +104,18 @@ static int __jump_label_text_reserved(struct jump_entry *iter_start, | |||
104 | return 0; | 104 | return 0; |
105 | } | 105 | } |
106 | 106 | ||
107 | /* | ||
108 | * Update code which is definitely not currently executing. | ||
109 | * Architectures which need heavyweight synchronization to modify | ||
110 | * running code can override this to make the non-live update case | ||
111 | * cheaper. | ||
112 | */ | ||
113 | void __weak arch_jump_label_transform_static(struct jump_entry *entry, | ||
114 | enum jump_label_type type) | ||
115 | { | ||
116 | arch_jump_label_transform(entry, type); | ||
117 | } | ||
118 | |||
107 | static void __jump_label_update(struct jump_label_key *key, | 119 | static void __jump_label_update(struct jump_label_key *key, |
108 | struct jump_entry *entry, | 120 | struct jump_entry *entry, |
109 | struct jump_entry *stop, int enable) | 121 | struct jump_entry *stop, int enable) |
@@ -135,8 +147,8 @@ static __init int jump_label_init(void) | |||
135 | struct jump_label_key *iterk; | 147 | struct jump_label_key *iterk; |
136 | 148 | ||
137 | iterk = (struct jump_label_key *)(unsigned long)iter->key; | 149 | iterk = (struct jump_label_key *)(unsigned long)iter->key; |
138 | arch_jump_label_transform(iter, jump_label_enabled(iterk) ? | 150 | arch_jump_label_transform_static(iter, jump_label_enabled(iterk) ? |
139 | JUMP_LABEL_ENABLE : JUMP_LABEL_DISABLE); | 151 | JUMP_LABEL_ENABLE : JUMP_LABEL_DISABLE); |
140 | if (iterk == key) | 152 | if (iterk == key) |
141 | continue; | 153 | continue; |
142 | 154 | ||
@@ -208,7 +220,7 @@ void jump_label_apply_nops(struct module *mod) | |||
208 | return; | 220 | return; |
209 | 221 | ||
210 | for (iter = iter_start; iter < iter_stop; iter++) | 222 | for (iter = iter_start; iter < iter_stop; iter++) |
211 | arch_jump_label_transform(iter, JUMP_LABEL_DISABLE); | 223 | arch_jump_label_transform_static(iter, JUMP_LABEL_DISABLE); |
212 | } | 224 | } |
213 | 225 | ||
214 | static int jump_label_add_module(struct module *mod) | 226 | static int jump_label_add_module(struct module *mod) |