mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00
ppp: Fix false xmit recursion detect with two ppp devices
The global percpu variable ppp_xmit_recursion is used to detect the ppp
xmit recursion to avoid the deadlock, which is caused by one CPU tries to
lock the xmit lock twice. But it would report false recursion when one CPU
wants to send the skb from two different PPP devices, like one L2TP on the
PPPoE. It is a normal case actually.
Now use one percpu member of struct ppp instead of the gloable variable to
detect the xmit recursion of one ppp device.
Fixes: 55454a5658
("ppp: avoid dealock on recursive xmit")
Signed-off-by: Gao Feng <gfree.wind@vip.163.com>
Signed-off-by: Liu Jianying <jianying.liu@ikuai8.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
df39a9f106
commit
e5dadc65f9
@ -120,6 +120,7 @@ struct ppp {
|
|||||||
int n_channels; /* how many channels are attached 54 */
|
int n_channels; /* how many channels are attached 54 */
|
||||||
spinlock_t rlock; /* lock for receive side 58 */
|
spinlock_t rlock; /* lock for receive side 58 */
|
||||||
spinlock_t wlock; /* lock for transmit side 5c */
|
spinlock_t wlock; /* lock for transmit side 5c */
|
||||||
|
int *xmit_recursion __percpu; /* xmit recursion detect */
|
||||||
int mru; /* max receive unit 60 */
|
int mru; /* max receive unit 60 */
|
||||||
unsigned int flags; /* control bits 64 */
|
unsigned int flags; /* control bits 64 */
|
||||||
unsigned int xstate; /* transmit state bits 68 */
|
unsigned int xstate; /* transmit state bits 68 */
|
||||||
@ -1025,6 +1026,7 @@ static int ppp_dev_configure(struct net *src_net, struct net_device *dev,
|
|||||||
struct ppp *ppp = netdev_priv(dev);
|
struct ppp *ppp = netdev_priv(dev);
|
||||||
int indx;
|
int indx;
|
||||||
int err;
|
int err;
|
||||||
|
int cpu;
|
||||||
|
|
||||||
ppp->dev = dev;
|
ppp->dev = dev;
|
||||||
ppp->ppp_net = src_net;
|
ppp->ppp_net = src_net;
|
||||||
@ -1039,6 +1041,15 @@ static int ppp_dev_configure(struct net *src_net, struct net_device *dev,
|
|||||||
INIT_LIST_HEAD(&ppp->channels);
|
INIT_LIST_HEAD(&ppp->channels);
|
||||||
spin_lock_init(&ppp->rlock);
|
spin_lock_init(&ppp->rlock);
|
||||||
spin_lock_init(&ppp->wlock);
|
spin_lock_init(&ppp->wlock);
|
||||||
|
|
||||||
|
ppp->xmit_recursion = alloc_percpu(int);
|
||||||
|
if (!ppp->xmit_recursion) {
|
||||||
|
err = -ENOMEM;
|
||||||
|
goto err1;
|
||||||
|
}
|
||||||
|
for_each_possible_cpu(cpu)
|
||||||
|
(*per_cpu_ptr(ppp->xmit_recursion, cpu)) = 0;
|
||||||
|
|
||||||
#ifdef CONFIG_PPP_MULTILINK
|
#ifdef CONFIG_PPP_MULTILINK
|
||||||
ppp->minseq = -1;
|
ppp->minseq = -1;
|
||||||
skb_queue_head_init(&ppp->mrq);
|
skb_queue_head_init(&ppp->mrq);
|
||||||
@ -1050,11 +1061,15 @@ static int ppp_dev_configure(struct net *src_net, struct net_device *dev,
|
|||||||
|
|
||||||
err = ppp_unit_register(ppp, conf->unit, conf->ifname_is_set);
|
err = ppp_unit_register(ppp, conf->unit, conf->ifname_is_set);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
goto err2;
|
||||||
|
|
||||||
conf->file->private_data = &ppp->file;
|
conf->file->private_data = &ppp->file;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
err2:
|
||||||
|
free_percpu(ppp->xmit_recursion);
|
||||||
|
err1:
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct nla_policy ppp_nl_policy[IFLA_PPP_MAX + 1] = {
|
static const struct nla_policy ppp_nl_policy[IFLA_PPP_MAX + 1] = {
|
||||||
@ -1400,18 +1415,16 @@ static void __ppp_xmit_process(struct ppp *ppp)
|
|||||||
ppp_xmit_unlock(ppp);
|
ppp_xmit_unlock(ppp);
|
||||||
}
|
}
|
||||||
|
|
||||||
static DEFINE_PER_CPU(int, ppp_xmit_recursion);
|
|
||||||
|
|
||||||
static void ppp_xmit_process(struct ppp *ppp)
|
static void ppp_xmit_process(struct ppp *ppp)
|
||||||
{
|
{
|
||||||
local_bh_disable();
|
local_bh_disable();
|
||||||
|
|
||||||
if (unlikely(__this_cpu_read(ppp_xmit_recursion)))
|
if (unlikely(*this_cpu_ptr(ppp->xmit_recursion)))
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
__this_cpu_inc(ppp_xmit_recursion);
|
(*this_cpu_ptr(ppp->xmit_recursion))++;
|
||||||
__ppp_xmit_process(ppp);
|
__ppp_xmit_process(ppp);
|
||||||
__this_cpu_dec(ppp_xmit_recursion);
|
(*this_cpu_ptr(ppp->xmit_recursion))--;
|
||||||
|
|
||||||
local_bh_enable();
|
local_bh_enable();
|
||||||
|
|
||||||
@ -1905,7 +1918,7 @@ static void __ppp_channel_push(struct channel *pch)
|
|||||||
read_lock(&pch->upl);
|
read_lock(&pch->upl);
|
||||||
ppp = pch->ppp;
|
ppp = pch->ppp;
|
||||||
if (ppp)
|
if (ppp)
|
||||||
__ppp_xmit_process(ppp);
|
ppp_xmit_process(ppp);
|
||||||
read_unlock(&pch->upl);
|
read_unlock(&pch->upl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1914,9 +1927,7 @@ static void ppp_channel_push(struct channel *pch)
|
|||||||
{
|
{
|
||||||
local_bh_disable();
|
local_bh_disable();
|
||||||
|
|
||||||
__this_cpu_inc(ppp_xmit_recursion);
|
|
||||||
__ppp_channel_push(pch);
|
__ppp_channel_push(pch);
|
||||||
__this_cpu_dec(ppp_xmit_recursion);
|
|
||||||
|
|
||||||
local_bh_enable();
|
local_bh_enable();
|
||||||
}
|
}
|
||||||
@ -3057,6 +3068,7 @@ static void ppp_destroy_interface(struct ppp *ppp)
|
|||||||
#endif /* CONFIG_PPP_FILTER */
|
#endif /* CONFIG_PPP_FILTER */
|
||||||
|
|
||||||
kfree_skb(ppp->xmit_pending);
|
kfree_skb(ppp->xmit_pending);
|
||||||
|
free_percpu(ppp->xmit_recursion);
|
||||||
|
|
||||||
free_netdev(ppp->dev);
|
free_netdev(ppp->dev);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user