Commit bd304a6c authored by Kai Germaschewski's avatar Kai Germaschewski

ISDN: Protect ipc_head list

Make sure that the ipc_head list cannot change under us by
protecting it with a spin lock.
parent 6565c3f9
...@@ -557,6 +557,7 @@ ippp_ccp_send_ccp(struct ippp_ccp *ccp, struct sk_buff *skb) ...@@ -557,6 +557,7 @@ ippp_ccp_send_ccp(struct ippp_ccp *ccp, struct sk_buff *skb)
} }
static LIST_HEAD(ipc_head); static LIST_HEAD(ipc_head);
static spinlock_t ipc_head_lock;
int int
ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit, ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit,
...@@ -571,25 +572,28 @@ ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit, ...@@ -571,25 +572,28 @@ ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit,
printk(KERN_DEBUG "[%d] Set %scompressor type %d\n", unit, printk(KERN_DEBUG "[%d] Set %scompressor type %d\n", unit,
data->flags & IPPP_COMP_FLAG_XMIT ? "" : "de", num); data->flags & IPPP_COMP_FLAG_XMIT ? "" : "de", num);
spin_lock(&ipc_head_lock);
list_for_each_entry(ipc, &ipc_head, list) { list_for_each_entry(ipc, &ipc_head, list) {
if (ipc->num != num) if (ipc->num == num &&
continue; try_module_get(ipc->owner))
goto found;
}
spin_unlock(&ipc_head_lock);
return -EINVAL;
if (!try_module_get(ipc->owner)) found:
continue; spin_unlock(&ipc_head_lock);
stat = ipc->alloc(data); stat = ipc->alloc(data);
if (!stat) { if (!stat) {
printk(KERN_ERR "Can't alloc (de)compression!\n"); printk(KERN_ERR "Can't alloc (de)compression!\n");
module_put(ipc->owner); goto err;
break;
} }
ret = ipc->init(stat, data, unit, 0); ret = ipc->init(stat, data, unit, 0);
if(!ret) { if(!ret) {
printk(KERN_ERR "Can't init (de)compression!\n"); printk(KERN_ERR "Can't init (de)compression!\n");
ipc->free(stat); ipc->free(stat);
module_put(ipc->owner); goto err;
break;
} }
if (data->flags & IPPP_COMP_FLAG_XMIT) { if (data->flags & IPPP_COMP_FLAG_XMIT) {
if (ccp->comp_stat) { if (ccp->comp_stat) {
...@@ -607,7 +611,9 @@ ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit, ...@@ -607,7 +611,9 @@ ippp_ccp_set_compressor(struct ippp_ccp *ccp, int unit,
ccp->decompressor = ipc; ccp->decompressor = ipc;
} }
return 0; return 0;
}
err:
module_put(ipc->owner);
return -EINVAL; return -EINVAL;
} }
...@@ -618,25 +624,34 @@ ippp_ccp_get_compressors(unsigned long protos[8]) ...@@ -618,25 +624,34 @@ ippp_ccp_get_compressors(unsigned long protos[8])
int i, j; int i, j;
memset(protos, 0, sizeof(unsigned long) * 8); memset(protos, 0, sizeof(unsigned long) * 8);
spin_lock(&ipc_head_lock);
list_for_each_entry(ipc, &ipc_head, list) { list_for_each_entry(ipc, &ipc_head, list) {
j = ipc->num / (sizeof(long)*8); j = ipc->num / (sizeof(long)*8);
i = ipc->num % (sizeof(long)*8); i = ipc->num % (sizeof(long)*8);
if (j < 8) if (j < 8)
protos[j] |= 1 << i; protos[j] |= 1 << i;
} }
spin_unlock(&ipc_head_lock);
} }
int int
isdn_ppp_register_compressor(struct isdn_ppp_compressor *ipc) isdn_ppp_register_compressor(struct isdn_ppp_compressor *ipc)
{ {
spin_lock(&ipc_head_lock);
list_add_tail(&ipc->list, &ipc_head); list_add_tail(&ipc->list, &ipc_head);
spin_unlock(&ipc_head_lock);
return 0; return 0;
} }
int int
isdn_ppp_unregister_compressor(struct isdn_ppp_compressor *ipc) isdn_ppp_unregister_compressor(struct isdn_ppp_compressor *ipc)
{ {
spin_lock(&ipc_head_lock);
list_del(&ipc->list); list_del(&ipc->list);
spin_unlock(&ipc_head_lock);
return 0; return 0;
} }
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