diff options
| author | André Fabian Silva Delgado <emulatorman@parabola.nu> | 2015-09-08 01:01:14 -0300 | 
|---|---|---|
| committer | André Fabian Silva Delgado <emulatorman@parabola.nu> | 2015-09-08 01:01:14 -0300 | 
| commit | e5fd91f1ef340da553f7a79da9540c3db711c937 (patch) | |
| tree | b11842027dc6641da63f4bcc524f8678263304a3 /kernel/bpf/syscall.c | |
| parent | 2a9b0348e685a63d97486f6749622b61e9e3292f (diff) | |
Linux-libre 4.2-gnu
Diffstat (limited to 'kernel/bpf/syscall.c')
| -rw-r--r-- | kernel/bpf/syscall.c | 42 | 
1 files changed, 40 insertions, 2 deletions
| diff --git a/kernel/bpf/syscall.c b/kernel/bpf/syscall.c index 3bae6c591..a1b14d197 100644 --- a/kernel/bpf/syscall.c +++ b/kernel/bpf/syscall.c @@ -68,6 +68,12 @@ static int bpf_map_release(struct inode *inode, struct file *filp)  {  	struct bpf_map *map = filp->private_data; +	if (map->map_type == BPF_MAP_TYPE_PROG_ARRAY) +		/* prog_array stores refcnt-ed bpf_prog pointers +		 * release them all when user space closes prog_array_fd +		 */ +		bpf_prog_array_map_clear(map); +  	bpf_map_put(map);  	return 0;  } @@ -392,6 +398,19 @@ static void fixup_bpf_calls(struct bpf_prog *prog)  			 */  			BUG_ON(!prog->aux->ops->get_func_proto); +			if (insn->imm == BPF_FUNC_tail_call) { +				/* mark bpf_tail_call as different opcode +				 * to avoid conditional branch in +				 * interpeter for every normal call +				 * and to prevent accidental JITing by +				 * JIT compiler that doesn't support +				 * bpf_tail_call yet +				 */ +				insn->imm = 0; +				insn->code |= BPF_X; +				continue; +			} +  			fn = prog->aux->ops->get_func_proto(insn->imm);  			/* all functions that have prototype and verifier allowed  			 * programs to call them, must be real in-kernel functions @@ -413,6 +432,23 @@ static void free_used_maps(struct bpf_prog_aux *aux)  	kfree(aux->used_maps);  } +static void __prog_put_rcu(struct rcu_head *rcu) +{ +	struct bpf_prog_aux *aux = container_of(rcu, struct bpf_prog_aux, rcu); + +	free_used_maps(aux); +	bpf_prog_free(aux->prog); +} + +/* version of bpf_prog_put() that is called after a grace period */ +void bpf_prog_put_rcu(struct bpf_prog *prog) +{ +	if (atomic_dec_and_test(&prog->aux->refcnt)) { +		prog->aux->prog = prog; +		call_rcu(&prog->aux->rcu, __prog_put_rcu); +	} +} +  void bpf_prog_put(struct bpf_prog *prog)  {  	if (atomic_dec_and_test(&prog->aux->refcnt)) { @@ -426,7 +462,7 @@ static int bpf_prog_release(struct inode *inode, struct file *filp)  {  	struct bpf_prog *prog = filp->private_data; -	bpf_prog_put(prog); +	bpf_prog_put_rcu(prog);  	return 0;  } @@ -532,7 +568,9 @@ static int bpf_prog_load(union bpf_attr *attr)  	fixup_bpf_calls(prog);  	/* eBPF program is ready to be JITed */ -	bpf_prog_select_runtime(prog); +	err = bpf_prog_select_runtime(prog); +	if (err < 0) +		goto free_used_maps;  	err = anon_inode_getfd("bpf-prog", &bpf_prog_fops, prog, O_RDWR | O_CLOEXEC);  	if (err < 0) | 
