[Lguest] [PATCH 4/5] lguest: use KVM hypercalls

Eric W. Biederman ebiederm at xmission.com
Thu Apr 9 23:59:25 EST 2009


Patrick McHardy <kaber at trash.net> writes:

> Patrick McHardy wrote:
>> which goes away by turning of CONFIG_STACKPROTECTOR. Now it seems
>> to work, but the host crashes before the guests are fully up somewhere
>> in the tun-device. I couldn't capture the oops yet, but I'll try
>> to fix it myself.
>
> This was caused by a local change of mine to attach to existing
> tun-devices, combined with a bug in the tun driver, which looks
> like it was introduced by this patch:
>
> commit c70f182940f988448f3c12a209d18b1edc276e33
> Author: Eric W. Biederman <ebiederm at xmission.com>
> Date:   Tue Jan 20 11:07:17 2009 +0000
>
>     tun: Fix races between tun_net_close and free_netdev.

A combination of my patch and the patches that added a socket
into the tun driver.

> When creating the device using tunctl the sk->sk_sleep poiner is
> set to the read_wait completion of the file opened by tunctl, but
> it is not refreshed when attaching to lguest or released when
> closing the file, causing a stale pointer dereference in
> tun_sock_write_space().
>
> Eric, please review. Thanks.

That looks a little better.  Certainly as the socket currently
lives with the tun_struct instead of the tun_file it make sense.
I'm not at all certain it makes sense for the socket to live in
tun_struct instead of tun_file.

I happened to glance at the code about a week ago, and realized that
the introduction of the socket had done horribly things to the
guarantees I had introduced, and I haven't had a chance to think
through and figure out what the code should be doing.

I am certain that:
opening a tap device and then "ip link del tap0" while holding
the tap open leads into a territory of madness right now.

And apparently so does reattaching to an existing tun device.

Patrick I'm not seeing anything in the particular patch you pointed
out that would cause crashes.

Other lurking bugs aside your patch appears slightly off.

tun->sk->sk_sleep in __tun_detach is correct.

Setting sk_sleep on both paths to tun_attach instead
of in tun_attach is wrong.  You are performing the assignment
before we complete the permission checks into tun_attach, which
means there is no guarantee that the tun_attach will succeed.

Eric

> Signed-off-by: Patrick McHardy <kaber at trash.net>
>
> diff --git a/drivers/net/tun.c b/drivers/net/tun.c
> index a1b0697..0af7ceb 100644
> --- a/drivers/net/tun.c
> +++ b/drivers/net/tun.c
> @@ -173,6 +173,8 @@ static void __tun_detach(struct tun_struct *tun)
>  	tun->tfile = NULL;
>  	netif_tx_unlock_bh(tun->dev);
>  
> +	tun->sk->sk_sleep = NULL;
> +
>  	/* Drop read queue */
>  	skb_queue_purge(&tun->readq);
>  
> @@ -873,6 +875,9 @@ static int tun_set_iff(struct net *net, struct file *file, struct ifreq *ifr)
>  		else
>  			return -EINVAL;
>  
> +		sk = tun->sk;
> +		sk->sk_sleep = &tfile->read_wait;
> +
>  		err = tun_attach(tun, file);
>  		if (err < 0)
>  			return err;



More information about the Lguest mailing list