diff options
Diffstat (limited to 'kernel/bpf/devmap.c')
| -rw-r--r-- | kernel/bpf/devmap.c | 19 | 
1 files changed, 8 insertions, 11 deletions
| diff --git a/kernel/bpf/devmap.c b/kernel/bpf/devmap.c index 1e525d70f833..cd8297b3bdb9 100644 --- a/kernel/bpf/devmap.c +++ b/kernel/bpf/devmap.c @@ -1,13 +1,5 @@ +// SPDX-License-Identifier: GPL-2.0-only  /* Copyright (c) 2017 Covalent IO, Inc. http://covalent.io - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of version 2 of the GNU General Public - * License as published by the Free Software Foundation. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - * General Public License for more details.   */  /* Devmaps primary use is as a backend map for XDP BPF helper call @@ -186,6 +178,7 @@ static void dev_map_free(struct bpf_map *map)  		if (!dev)  			continue; +		free_percpu(dev->bulkq);  		dev_put(dev->dev);  		kfree(dev);  	} @@ -281,6 +274,7 @@ void __dev_map_flush(struct bpf_map *map)  	unsigned long *bitmap = this_cpu_ptr(dtab->flush_needed);  	u32 bit; +	rcu_read_lock();  	for_each_set_bit(bit, bitmap, map->max_entries) {  		struct bpf_dtab_netdev *dev = READ_ONCE(dtab->netdev_map[bit]);  		struct xdp_bulk_queue *bq; @@ -291,11 +285,12 @@ void __dev_map_flush(struct bpf_map *map)  		if (unlikely(!dev))  			continue; -		__clear_bit(bit, bitmap); -  		bq = this_cpu_ptr(dev->bulkq);  		bq_xmit_all(dev, bq, XDP_XMIT_FLUSH, true); + +		__clear_bit(bit, bitmap);  	} +	rcu_read_unlock();  }  /* rcu_read_lock (from syscall and BPF contexts) ensures that if a delete and/or @@ -388,6 +383,7 @@ static void dev_map_flush_old(struct bpf_dtab_netdev *dev)  		int cpu; +		rcu_read_lock();  		for_each_online_cpu(cpu) {  			bitmap = per_cpu_ptr(dev->dtab->flush_needed, cpu);  			__clear_bit(dev->bit, bitmap); @@ -395,6 +391,7 @@ static void dev_map_flush_old(struct bpf_dtab_netdev *dev)  			bq = per_cpu_ptr(dev->bulkq, cpu);  			bq_xmit_all(dev, bq, XDP_XMIT_FLUSH, false);  		} +		rcu_read_unlock();  	}  } | 
