Commit 085fee1a authored by Thomas Gleixner's avatar Thomas Gleixner Committed by Alexei Starovoitov

bpf: Use recursion prevention helpers in hashtab code

The required protection is that the caller cannot be migrated to a
different CPU as these places take either a hash bucket lock or might
trigger a kprobe inside the memory allocator. Both scenarios can lead to
deadlocks. The deadlock prevention is per CPU by incrementing a per CPU
variable which temporarily blocks the invocation of BPF programs from perf
and kprobes.

Replace the open coded preempt_disable/enable() and this_cpu_inc/dec()
pairs with the new recursion prevention helpers to prepare BPF to work on
PREEMPT_RT enabled kernels. On a non-RT kernel the migrate disable/enable
in the helpers map to preempt_disable/enable(), i.e. no functional change.
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
Link: https://lore.kernel.org/bpf/20200224145644.211208533@linutronix.de
parent c518cfa0
...@@ -1333,8 +1333,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map, ...@@ -1333,8 +1333,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map,
} }
again: again:
preempt_disable(); bpf_disable_instrumentation();
this_cpu_inc(bpf_prog_active);
rcu_read_lock(); rcu_read_lock();
again_nocopy: again_nocopy:
dst_key = keys; dst_key = keys;
...@@ -1362,8 +1361,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map, ...@@ -1362,8 +1361,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map,
*/ */
raw_spin_unlock_irqrestore(&b->lock, flags); raw_spin_unlock_irqrestore(&b->lock, flags);
rcu_read_unlock(); rcu_read_unlock();
this_cpu_dec(bpf_prog_active); bpf_enable_instrumentation();
preempt_enable();
goto after_loop; goto after_loop;
} }
...@@ -1374,8 +1372,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map, ...@@ -1374,8 +1372,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map,
*/ */
raw_spin_unlock_irqrestore(&b->lock, flags); raw_spin_unlock_irqrestore(&b->lock, flags);
rcu_read_unlock(); rcu_read_unlock();
this_cpu_dec(bpf_prog_active); bpf_enable_instrumentation();
preempt_enable();
kvfree(keys); kvfree(keys);
kvfree(values); kvfree(values);
goto alloc; goto alloc;
...@@ -1445,8 +1442,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map, ...@@ -1445,8 +1442,7 @@ __htab_map_lookup_and_delete_batch(struct bpf_map *map,
} }
rcu_read_unlock(); rcu_read_unlock();
this_cpu_dec(bpf_prog_active); bpf_enable_instrumentation();
preempt_enable();
if (bucket_cnt && (copy_to_user(ukeys + total * key_size, keys, if (bucket_cnt && (copy_to_user(ukeys + total * key_size, keys,
key_size * bucket_cnt) || key_size * bucket_cnt) ||
copy_to_user(uvalues + total * value_size, values, copy_to_user(uvalues + total * value_size, values,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment