diff options
Diffstat (limited to 'net/can')
-rw-r--r-- | net/can/af_can.c | 144 | ||||
-rw-r--r-- | net/can/af_can.h | 4 | ||||
-rw-r--r-- | net/can/bcm.c | 13 | ||||
-rw-r--r-- | net/can/proc.c | 93 | ||||
-rw-r--r-- | net/can/raw.c | 8 |
5 files changed, 121 insertions, 141 deletions
diff --git a/net/can/af_can.c b/net/can/af_can.c index 606832115674..702be5a2c956 100644 --- a/net/can/af_can.c +++ b/net/can/af_can.c | |||
@@ -77,8 +77,8 @@ static int stats_timer __read_mostly = 1; | |||
77 | module_param(stats_timer, int, S_IRUGO); | 77 | module_param(stats_timer, int, S_IRUGO); |
78 | MODULE_PARM_DESC(stats_timer, "enable timer for statistics (default:on)"); | 78 | MODULE_PARM_DESC(stats_timer, "enable timer for statistics (default:on)"); |
79 | 79 | ||
80 | HLIST_HEAD(can_rx_dev_list); | 80 | /* receive filters subscribed for 'all' CAN devices */ |
81 | static struct dev_rcv_lists can_rx_alldev_list; | 81 | struct dev_rcv_lists can_rx_alldev_list; |
82 | static DEFINE_SPINLOCK(can_rcvlists_lock); | 82 | static DEFINE_SPINLOCK(can_rcvlists_lock); |
83 | 83 | ||
84 | static struct kmem_cache *rcv_cache __read_mostly; | 84 | static struct kmem_cache *rcv_cache __read_mostly; |
@@ -114,7 +114,8 @@ static void can_sock_destruct(struct sock *sk) | |||
114 | skb_queue_purge(&sk->sk_receive_queue); | 114 | skb_queue_purge(&sk->sk_receive_queue); |
115 | } | 115 | } |
116 | 116 | ||
117 | static int can_create(struct net *net, struct socket *sock, int protocol) | 117 | static int can_create(struct net *net, struct socket *sock, int protocol, |
118 | int kern) | ||
118 | { | 119 | { |
119 | struct sock *sk; | 120 | struct sock *sk; |
120 | struct can_proto *cp; | 121 | struct can_proto *cp; |
@@ -125,7 +126,7 @@ static int can_create(struct net *net, struct socket *sock, int protocol) | |||
125 | if (protocol < 0 || protocol >= CAN_NPROTO) | 126 | if (protocol < 0 || protocol >= CAN_NPROTO) |
126 | return -EINVAL; | 127 | return -EINVAL; |
127 | 128 | ||
128 | if (net != &init_net) | 129 | if (!net_eq(net, &init_net)) |
129 | return -EAFNOSUPPORT; | 130 | return -EAFNOSUPPORT; |
130 | 131 | ||
131 | #ifdef CONFIG_MODULES | 132 | #ifdef CONFIG_MODULES |
@@ -160,11 +161,6 @@ static int can_create(struct net *net, struct socket *sock, int protocol) | |||
160 | goto errout; | 161 | goto errout; |
161 | } | 162 | } |
162 | 163 | ||
163 | if (cp->capability >= 0 && !capable(cp->capability)) { | ||
164 | err = -EPERM; | ||
165 | goto errout; | ||
166 | } | ||
167 | |||
168 | sock->ops = cp->ops; | 164 | sock->ops = cp->ops; |
169 | 165 | ||
170 | sk = sk_alloc(net, PF_CAN, GFP_KERNEL, cp->prot); | 166 | sk = sk_alloc(net, PF_CAN, GFP_KERNEL, cp->prot); |
@@ -296,28 +292,10 @@ EXPORT_SYMBOL(can_send); | |||
296 | 292 | ||
297 | static struct dev_rcv_lists *find_dev_rcv_lists(struct net_device *dev) | 293 | static struct dev_rcv_lists *find_dev_rcv_lists(struct net_device *dev) |
298 | { | 294 | { |
299 | struct dev_rcv_lists *d = NULL; | 295 | if (!dev) |
300 | struct hlist_node *n; | 296 | return &can_rx_alldev_list; |
301 | 297 | else | |
302 | /* | 298 | return (struct dev_rcv_lists *)dev->ml_priv; |
303 | * find receive list for this device | ||
304 | * | ||
305 | * The hlist_for_each_entry*() macros curse through the list | ||
306 | * using the pointer variable n and set d to the containing | ||
307 | * struct in each list iteration. Therefore, after list | ||
308 | * iteration, d is unmodified when the list is empty, and it | ||
309 | * points to last list element, when the list is non-empty | ||
310 | * but no match in the loop body is found. I.e. d is *not* | ||
311 | * NULL when no match is found. We can, however, use the | ||
312 | * cursor variable n to decide if a match was found. | ||
313 | */ | ||
314 | |||
315 | hlist_for_each_entry_rcu(d, n, &can_rx_dev_list, list) { | ||
316 | if (d->dev == dev) | ||
317 | break; | ||
318 | } | ||
319 | |||
320 | return n ? d : NULL; | ||
321 | } | 299 | } |
322 | 300 | ||
323 | /** | 301 | /** |
@@ -379,8 +357,8 @@ static struct hlist_head *find_rcv_list(canid_t *can_id, canid_t *mask, | |||
379 | return &d->rx[RX_ALL]; | 357 | return &d->rx[RX_ALL]; |
380 | 358 | ||
381 | /* extra filterlists for the subscription of a single non-RTR can_id */ | 359 | /* extra filterlists for the subscription of a single non-RTR can_id */ |
382 | if (((*mask & CAN_EFF_RTR_FLAGS) == CAN_EFF_RTR_FLAGS) | 360 | if (((*mask & CAN_EFF_RTR_FLAGS) == CAN_EFF_RTR_FLAGS) && |
383 | && !(*can_id & CAN_RTR_FLAG)) { | 361 | !(*can_id & CAN_RTR_FLAG)) { |
384 | 362 | ||
385 | if (*can_id & CAN_EFF_FLAG) { | 363 | if (*can_id & CAN_EFF_FLAG) { |
386 | if (*mask == (CAN_EFF_MASK | CAN_EFF_RTR_FLAGS)) { | 364 | if (*mask == (CAN_EFF_MASK | CAN_EFF_RTR_FLAGS)) { |
@@ -437,6 +415,9 @@ int can_rx_register(struct net_device *dev, canid_t can_id, canid_t mask, | |||
437 | 415 | ||
438 | /* insert new receiver (dev,canid,mask) -> (func,data) */ | 416 | /* insert new receiver (dev,canid,mask) -> (func,data) */ |
439 | 417 | ||
418 | if (dev && dev->type != ARPHRD_CAN) | ||
419 | return -ENODEV; | ||
420 | |||
440 | r = kmem_cache_alloc(rcv_cache, GFP_KERNEL); | 421 | r = kmem_cache_alloc(rcv_cache, GFP_KERNEL); |
441 | if (!r) | 422 | if (!r) |
442 | return -ENOMEM; | 423 | return -ENOMEM; |
@@ -472,16 +453,6 @@ int can_rx_register(struct net_device *dev, canid_t can_id, canid_t mask, | |||
472 | EXPORT_SYMBOL(can_rx_register); | 453 | EXPORT_SYMBOL(can_rx_register); |
473 | 454 | ||
474 | /* | 455 | /* |
475 | * can_rx_delete_device - rcu callback for dev_rcv_lists structure removal | ||
476 | */ | ||
477 | static void can_rx_delete_device(struct rcu_head *rp) | ||
478 | { | ||
479 | struct dev_rcv_lists *d = container_of(rp, struct dev_rcv_lists, rcu); | ||
480 | |||
481 | kfree(d); | ||
482 | } | ||
483 | |||
484 | /* | ||
485 | * can_rx_delete_receiver - rcu callback for single receiver entry removal | 456 | * can_rx_delete_receiver - rcu callback for single receiver entry removal |
486 | */ | 457 | */ |
487 | static void can_rx_delete_receiver(struct rcu_head *rp) | 458 | static void can_rx_delete_receiver(struct rcu_head *rp) |
@@ -510,6 +481,9 @@ void can_rx_unregister(struct net_device *dev, canid_t can_id, canid_t mask, | |||
510 | struct hlist_node *next; | 481 | struct hlist_node *next; |
511 | struct dev_rcv_lists *d; | 482 | struct dev_rcv_lists *d; |
512 | 483 | ||
484 | if (dev && dev->type != ARPHRD_CAN) | ||
485 | return; | ||
486 | |||
513 | spin_lock(&can_rcvlists_lock); | 487 | spin_lock(&can_rcvlists_lock); |
514 | 488 | ||
515 | d = find_dev_rcv_lists(dev); | 489 | d = find_dev_rcv_lists(dev); |
@@ -529,8 +503,8 @@ void can_rx_unregister(struct net_device *dev, canid_t can_id, canid_t mask, | |||
529 | */ | 503 | */ |
530 | 504 | ||
531 | hlist_for_each_entry_rcu(r, next, rl, list) { | 505 | hlist_for_each_entry_rcu(r, next, rl, list) { |
532 | if (r->can_id == can_id && r->mask == mask | 506 | if (r->can_id == can_id && r->mask == mask && |
533 | && r->func == func && r->data == data) | 507 | r->func == func && r->data == data) |
534 | break; | 508 | break; |
535 | } | 509 | } |
536 | 510 | ||
@@ -545,7 +519,6 @@ void can_rx_unregister(struct net_device *dev, canid_t can_id, canid_t mask, | |||
545 | "dev %s, id %03X, mask %03X\n", | 519 | "dev %s, id %03X, mask %03X\n", |
546 | DNAME(dev), can_id, mask); | 520 | DNAME(dev), can_id, mask); |
547 | r = NULL; | 521 | r = NULL; |
548 | d = NULL; | ||
549 | goto out; | 522 | goto out; |
550 | } | 523 | } |
551 | 524 | ||
@@ -556,10 +529,10 @@ void can_rx_unregister(struct net_device *dev, canid_t can_id, canid_t mask, | |||
556 | can_pstats.rcv_entries--; | 529 | can_pstats.rcv_entries--; |
557 | 530 | ||
558 | /* remove device structure requested by NETDEV_UNREGISTER */ | 531 | /* remove device structure requested by NETDEV_UNREGISTER */ |
559 | if (d->remove_on_zero_entries && !d->entries) | 532 | if (d->remove_on_zero_entries && !d->entries) { |
560 | hlist_del_rcu(&d->list); | 533 | kfree(d); |
561 | else | 534 | dev->ml_priv = NULL; |
562 | d = NULL; | 535 | } |
563 | 536 | ||
564 | out: | 537 | out: |
565 | spin_unlock(&can_rcvlists_lock); | 538 | spin_unlock(&can_rcvlists_lock); |
@@ -567,10 +540,6 @@ void can_rx_unregister(struct net_device *dev, canid_t can_id, canid_t mask, | |||
567 | /* schedule the receiver item for deletion */ | 540 | /* schedule the receiver item for deletion */ |
568 | if (r) | 541 | if (r) |
569 | call_rcu(&r->rcu, can_rx_delete_receiver); | 542 | call_rcu(&r->rcu, can_rx_delete_receiver); |
570 | |||
571 | /* schedule the device structure for deletion */ | ||
572 | if (d) | ||
573 | call_rcu(&d->rcu, can_rx_delete_device); | ||
574 | } | 543 | } |
575 | EXPORT_SYMBOL(can_rx_unregister); | 544 | EXPORT_SYMBOL(can_rx_unregister); |
576 | 545 | ||
@@ -784,48 +753,35 @@ static int can_notifier(struct notifier_block *nb, unsigned long msg, | |||
784 | 753 | ||
785 | case NETDEV_REGISTER: | 754 | case NETDEV_REGISTER: |
786 | 755 | ||
787 | /* | 756 | /* create new dev_rcv_lists for this device */ |
788 | * create new dev_rcv_lists for this device | ||
789 | * | ||
790 | * N.B. zeroing the struct is the correct initialization | ||
791 | * for the embedded hlist_head structs. | ||
792 | * Another list type, e.g. list_head, would require | ||
793 | * explicit initialization. | ||
794 | */ | ||
795 | |||
796 | d = kzalloc(sizeof(*d), GFP_KERNEL); | 757 | d = kzalloc(sizeof(*d), GFP_KERNEL); |
797 | if (!d) { | 758 | if (!d) { |
798 | printk(KERN_ERR | 759 | printk(KERN_ERR |
799 | "can: allocation of receive list failed\n"); | 760 | "can: allocation of receive list failed\n"); |
800 | return NOTIFY_DONE; | 761 | return NOTIFY_DONE; |
801 | } | 762 | } |
802 | d->dev = dev; | 763 | BUG_ON(dev->ml_priv); |
803 | 764 | dev->ml_priv = d; | |
804 | spin_lock(&can_rcvlists_lock); | ||
805 | hlist_add_head_rcu(&d->list, &can_rx_dev_list); | ||
806 | spin_unlock(&can_rcvlists_lock); | ||
807 | 765 | ||
808 | break; | 766 | break; |
809 | 767 | ||
810 | case NETDEV_UNREGISTER: | 768 | case NETDEV_UNREGISTER: |
811 | spin_lock(&can_rcvlists_lock); | 769 | spin_lock(&can_rcvlists_lock); |
812 | 770 | ||
813 | d = find_dev_rcv_lists(dev); | 771 | d = dev->ml_priv; |
814 | if (d) { | 772 | if (d) { |
815 | if (d->entries) { | 773 | if (d->entries) |
816 | d->remove_on_zero_entries = 1; | 774 | d->remove_on_zero_entries = 1; |
817 | d = NULL; | 775 | else { |
818 | } else | 776 | kfree(d); |
819 | hlist_del_rcu(&d->list); | 777 | dev->ml_priv = NULL; |
778 | } | ||
820 | } else | 779 | } else |
821 | printk(KERN_ERR "can: notifier: receive list not " | 780 | printk(KERN_ERR "can: notifier: receive list not " |
822 | "found for dev %s\n", dev->name); | 781 | "found for dev %s\n", dev->name); |
823 | 782 | ||
824 | spin_unlock(&can_rcvlists_lock); | 783 | spin_unlock(&can_rcvlists_lock); |
825 | 784 | ||
826 | if (d) | ||
827 | call_rcu(&d->rcu, can_rx_delete_device); | ||
828 | |||
829 | break; | 785 | break; |
830 | } | 786 | } |
831 | 787 | ||
@@ -842,7 +798,7 @@ static struct packet_type can_packet __read_mostly = { | |||
842 | .func = can_rcv, | 798 | .func = can_rcv, |
843 | }; | 799 | }; |
844 | 800 | ||
845 | static struct net_proto_family can_family_ops __read_mostly = { | 801 | static const struct net_proto_family can_family_ops = { |
846 | .family = PF_CAN, | 802 | .family = PF_CAN, |
847 | .create = can_create, | 803 | .create = can_create, |
848 | .owner = THIS_MODULE, | 804 | .owner = THIS_MODULE, |
@@ -857,21 +813,13 @@ static __init int can_init(void) | |||
857 | { | 813 | { |
858 | printk(banner); | 814 | printk(banner); |
859 | 815 | ||
816 | memset(&can_rx_alldev_list, 0, sizeof(can_rx_alldev_list)); | ||
817 | |||
860 | rcv_cache = kmem_cache_create("can_receiver", sizeof(struct receiver), | 818 | rcv_cache = kmem_cache_create("can_receiver", sizeof(struct receiver), |
861 | 0, 0, NULL); | 819 | 0, 0, NULL); |
862 | if (!rcv_cache) | 820 | if (!rcv_cache) |
863 | return -ENOMEM; | 821 | return -ENOMEM; |
864 | 822 | ||
865 | /* | ||
866 | * Insert can_rx_alldev_list for reception on all devices. | ||
867 | * This struct is zero initialized which is correct for the | ||
868 | * embedded hlist heads, the dev pointer, and the entries counter. | ||
869 | */ | ||
870 | |||
871 | spin_lock(&can_rcvlists_lock); | ||
872 | hlist_add_head_rcu(&can_rx_alldev_list.list, &can_rx_dev_list); | ||
873 | spin_unlock(&can_rcvlists_lock); | ||
874 | |||
875 | if (stats_timer) { | 823 | if (stats_timer) { |
876 | /* the statistics are updated every second (timer triggered) */ | 824 | /* the statistics are updated every second (timer triggered) */ |
877 | setup_timer(&can_stattimer, can_stat_update, 0); | 825 | setup_timer(&can_stattimer, can_stat_update, 0); |
@@ -891,8 +839,7 @@ static __init int can_init(void) | |||
891 | 839 | ||
892 | static __exit void can_exit(void) | 840 | static __exit void can_exit(void) |
893 | { | 841 | { |
894 | struct dev_rcv_lists *d; | 842 | struct net_device *dev; |
895 | struct hlist_node *n, *next; | ||
896 | 843 | ||
897 | if (stats_timer) | 844 | if (stats_timer) |
898 | del_timer(&can_stattimer); | 845 | del_timer(&can_stattimer); |
@@ -904,14 +851,19 @@ static __exit void can_exit(void) | |||
904 | unregister_netdevice_notifier(&can_netdev_notifier); | 851 | unregister_netdevice_notifier(&can_netdev_notifier); |
905 | sock_unregister(PF_CAN); | 852 | sock_unregister(PF_CAN); |
906 | 853 | ||
907 | /* remove can_rx_dev_list */ | 854 | /* remove created dev_rcv_lists from still registered CAN devices */ |
908 | spin_lock(&can_rcvlists_lock); | 855 | rcu_read_lock(); |
909 | hlist_del(&can_rx_alldev_list.list); | 856 | for_each_netdev_rcu(&init_net, dev) { |
910 | hlist_for_each_entry_safe(d, n, next, &can_rx_dev_list, list) { | 857 | if (dev->type == ARPHRD_CAN && dev->ml_priv){ |
911 | hlist_del(&d->list); | 858 | |
912 | kfree(d); | 859 | struct dev_rcv_lists *d = dev->ml_priv; |
860 | |||
861 | BUG_ON(d->entries); | ||
862 | kfree(d); | ||
863 | dev->ml_priv = NULL; | ||
864 | } | ||
913 | } | 865 | } |
914 | spin_unlock(&can_rcvlists_lock); | 866 | rcu_read_unlock(); |
915 | 867 | ||
916 | rcu_barrier(); /* Wait for completion of call_rcu()'s */ | 868 | rcu_barrier(); /* Wait for completion of call_rcu()'s */ |
917 | 869 | ||
diff --git a/net/can/af_can.h b/net/can/af_can.h index 18f91e37cc30..34253b84e30f 100644 --- a/net/can/af_can.h +++ b/net/can/af_can.h | |||
@@ -63,10 +63,8 @@ struct receiver { | |||
63 | 63 | ||
64 | enum { RX_ERR, RX_ALL, RX_FIL, RX_INV, RX_EFF, RX_MAX }; | 64 | enum { RX_ERR, RX_ALL, RX_FIL, RX_INV, RX_EFF, RX_MAX }; |
65 | 65 | ||
66 | /* per device receive filters linked at dev->ml_priv */ | ||
66 | struct dev_rcv_lists { | 67 | struct dev_rcv_lists { |
67 | struct hlist_node list; | ||
68 | struct rcu_head rcu; | ||
69 | struct net_device *dev; | ||
70 | struct hlist_head rx[RX_MAX]; | 68 | struct hlist_head rx[RX_MAX]; |
71 | struct hlist_head rx_sff[0x800]; | 69 | struct hlist_head rx_sff[0x800]; |
72 | int remove_on_zero_entries; | 70 | int remove_on_zero_entries; |
diff --git a/net/can/bcm.c b/net/can/bcm.c index e8d58f33fe09..907dc871fac8 100644 --- a/net/can/bcm.c +++ b/net/can/bcm.c | |||
@@ -56,6 +56,7 @@ | |||
56 | #include <linux/can.h> | 56 | #include <linux/can.h> |
57 | #include <linux/can/core.h> | 57 | #include <linux/can/core.h> |
58 | #include <linux/can/bcm.h> | 58 | #include <linux/can/bcm.h> |
59 | #include <linux/slab.h> | ||
59 | #include <net/sock.h> | 60 | #include <net/sock.h> |
60 | #include <net/net_namespace.h> | 61 | #include <net/net_namespace.h> |
61 | 62 | ||
@@ -139,13 +140,13 @@ static char *bcm_proc_getifname(char *result, int ifindex) | |||
139 | if (!ifindex) | 140 | if (!ifindex) |
140 | return "any"; | 141 | return "any"; |
141 | 142 | ||
142 | read_lock(&dev_base_lock); | 143 | rcu_read_lock(); |
143 | dev = __dev_get_by_index(&init_net, ifindex); | 144 | dev = dev_get_by_index_rcu(&init_net, ifindex); |
144 | if (dev) | 145 | if (dev) |
145 | strcpy(result, dev->name); | 146 | strcpy(result, dev->name); |
146 | else | 147 | else |
147 | strcpy(result, "???"); | 148 | strcpy(result, "???"); |
148 | read_unlock(&dev_base_lock); | 149 | rcu_read_unlock(); |
149 | 150 | ||
150 | return result; | 151 | return result; |
151 | } | 152 | } |
@@ -1478,6 +1479,9 @@ static int bcm_connect(struct socket *sock, struct sockaddr *uaddr, int len, | |||
1478 | struct sock *sk = sock->sk; | 1479 | struct sock *sk = sock->sk; |
1479 | struct bcm_sock *bo = bcm_sk(sk); | 1480 | struct bcm_sock *bo = bcm_sk(sk); |
1480 | 1481 | ||
1482 | if (len < sizeof(*addr)) | ||
1483 | return -EINVAL; | ||
1484 | |||
1481 | if (bo->bound) | 1485 | if (bo->bound) |
1482 | return -EISCONN; | 1486 | return -EISCONN; |
1483 | 1487 | ||
@@ -1539,7 +1543,7 @@ static int bcm_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
1539 | return err; | 1543 | return err; |
1540 | } | 1544 | } |
1541 | 1545 | ||
1542 | sock_recv_timestamp(msg, sk, skb); | 1546 | sock_recv_ts_and_drops(msg, sk, skb); |
1543 | 1547 | ||
1544 | if (msg->msg_name) { | 1548 | if (msg->msg_name) { |
1545 | msg->msg_namelen = sizeof(struct sockaddr_can); | 1549 | msg->msg_namelen = sizeof(struct sockaddr_can); |
@@ -1581,7 +1585,6 @@ static struct proto bcm_proto __read_mostly = { | |||
1581 | static struct can_proto bcm_can_proto __read_mostly = { | 1585 | static struct can_proto bcm_can_proto __read_mostly = { |
1582 | .type = SOCK_DGRAM, | 1586 | .type = SOCK_DGRAM, |
1583 | .protocol = CAN_BCM, | 1587 | .protocol = CAN_BCM, |
1584 | .capability = -1, | ||
1585 | .ops = &bcm_ops, | 1588 | .ops = &bcm_ops, |
1586 | .prot = &bcm_proto, | 1589 | .prot = &bcm_proto, |
1587 | }; | 1590 | }; |
diff --git a/net/can/proc.c b/net/can/proc.c index 9b9ad29be567..f4265cc9c3fb 100644 --- a/net/can/proc.c +++ b/net/can/proc.c | |||
@@ -45,6 +45,7 @@ | |||
45 | #include <linux/proc_fs.h> | 45 | #include <linux/proc_fs.h> |
46 | #include <linux/list.h> | 46 | #include <linux/list.h> |
47 | #include <linux/rcupdate.h> | 47 | #include <linux/rcupdate.h> |
48 | #include <linux/if_arp.h> | ||
48 | #include <linux/can/core.h> | 49 | #include <linux/can/core.h> |
49 | 50 | ||
50 | #include "af_can.h" | 51 | #include "af_can.h" |
@@ -84,6 +85,9 @@ static const char rx_list_name[][8] = { | |||
84 | [RX_EFF] = "rx_eff", | 85 | [RX_EFF] = "rx_eff", |
85 | }; | 86 | }; |
86 | 87 | ||
88 | /* receive filters subscribed for 'all' CAN devices */ | ||
89 | extern struct dev_rcv_lists can_rx_alldev_list; | ||
90 | |||
87 | /* | 91 | /* |
88 | * af_can statistics stuff | 92 | * af_can statistics stuff |
89 | */ | 93 | */ |
@@ -190,10 +194,6 @@ void can_stat_update(unsigned long data) | |||
190 | 194 | ||
191 | /* | 195 | /* |
192 | * proc read functions | 196 | * proc read functions |
193 | * | ||
194 | * From known use-cases we expect about 10 entries in a receive list to be | ||
195 | * printed in the proc_fs. So PAGE_SIZE is definitely enough space here. | ||
196 | * | ||
197 | */ | 197 | */ |
198 | 198 | ||
199 | static void can_print_rcvlist(struct seq_file *m, struct hlist_head *rx_list, | 199 | static void can_print_rcvlist(struct seq_file *m, struct hlist_head *rx_list, |
@@ -202,7 +202,6 @@ static void can_print_rcvlist(struct seq_file *m, struct hlist_head *rx_list, | |||
202 | struct receiver *r; | 202 | struct receiver *r; |
203 | struct hlist_node *n; | 203 | struct hlist_node *n; |
204 | 204 | ||
205 | rcu_read_lock(); | ||
206 | hlist_for_each_entry_rcu(r, n, rx_list, list) { | 205 | hlist_for_each_entry_rcu(r, n, rx_list, list) { |
207 | char *fmt = (r->can_id & CAN_EFF_FLAG)? | 206 | char *fmt = (r->can_id & CAN_EFF_FLAG)? |
208 | " %-5s %08X %08x %08x %08x %8ld %s\n" : | 207 | " %-5s %08X %08x %08x %08x %8ld %s\n" : |
@@ -212,7 +211,6 @@ static void can_print_rcvlist(struct seq_file *m, struct hlist_head *rx_list, | |||
212 | (unsigned long)r->func, (unsigned long)r->data, | 211 | (unsigned long)r->func, (unsigned long)r->data, |
213 | r->matches, r->ident); | 212 | r->matches, r->ident); |
214 | } | 213 | } |
215 | rcu_read_unlock(); | ||
216 | } | 214 | } |
217 | 215 | ||
218 | static void can_print_recv_banner(struct seq_file *m) | 216 | static void can_print_recv_banner(struct seq_file *m) |
@@ -346,24 +344,39 @@ static const struct file_operations can_version_proc_fops = { | |||
346 | .release = single_release, | 344 | .release = single_release, |
347 | }; | 345 | }; |
348 | 346 | ||
347 | static inline void can_rcvlist_proc_show_one(struct seq_file *m, int idx, | ||
348 | struct net_device *dev, | ||
349 | struct dev_rcv_lists *d) | ||
350 | { | ||
351 | if (!hlist_empty(&d->rx[idx])) { | ||
352 | can_print_recv_banner(m); | ||
353 | can_print_rcvlist(m, &d->rx[idx], dev); | ||
354 | } else | ||
355 | seq_printf(m, " (%s: no entry)\n", DNAME(dev)); | ||
356 | |||
357 | } | ||
358 | |||
349 | static int can_rcvlist_proc_show(struct seq_file *m, void *v) | 359 | static int can_rcvlist_proc_show(struct seq_file *m, void *v) |
350 | { | 360 | { |
351 | /* double cast to prevent GCC warning */ | 361 | /* double cast to prevent GCC warning */ |
352 | int idx = (int)(long)m->private; | 362 | int idx = (int)(long)m->private; |
363 | struct net_device *dev; | ||
353 | struct dev_rcv_lists *d; | 364 | struct dev_rcv_lists *d; |
354 | struct hlist_node *n; | ||
355 | 365 | ||
356 | seq_printf(m, "\nreceive list '%s':\n", rx_list_name[idx]); | 366 | seq_printf(m, "\nreceive list '%s':\n", rx_list_name[idx]); |
357 | 367 | ||
358 | rcu_read_lock(); | 368 | rcu_read_lock(); |
359 | hlist_for_each_entry_rcu(d, n, &can_rx_dev_list, list) { | ||
360 | 369 | ||
361 | if (!hlist_empty(&d->rx[idx])) { | 370 | /* receive list for 'all' CAN devices (dev == NULL) */ |
362 | can_print_recv_banner(m); | 371 | d = &can_rx_alldev_list; |
363 | can_print_rcvlist(m, &d->rx[idx], d->dev); | 372 | can_rcvlist_proc_show_one(m, idx, NULL, d); |
364 | } else | 373 | |
365 | seq_printf(m, " (%s: no entry)\n", DNAME(d->dev)); | 374 | /* receive list for registered CAN devices */ |
375 | for_each_netdev_rcu(&init_net, dev) { | ||
376 | if (dev->type == ARPHRD_CAN && dev->ml_priv) | ||
377 | can_rcvlist_proc_show_one(m, idx, dev, dev->ml_priv); | ||
366 | } | 378 | } |
379 | |||
367 | rcu_read_unlock(); | 380 | rcu_read_unlock(); |
368 | 381 | ||
369 | seq_putc(m, '\n'); | 382 | seq_putc(m, '\n'); |
@@ -383,34 +396,50 @@ static const struct file_operations can_rcvlist_proc_fops = { | |||
383 | .release = single_release, | 396 | .release = single_release, |
384 | }; | 397 | }; |
385 | 398 | ||
399 | static inline void can_rcvlist_sff_proc_show_one(struct seq_file *m, | ||
400 | struct net_device *dev, | ||
401 | struct dev_rcv_lists *d) | ||
402 | { | ||
403 | int i; | ||
404 | int all_empty = 1; | ||
405 | |||
406 | /* check wether at least one list is non-empty */ | ||
407 | for (i = 0; i < 0x800; i++) | ||
408 | if (!hlist_empty(&d->rx_sff[i])) { | ||
409 | all_empty = 0; | ||
410 | break; | ||
411 | } | ||
412 | |||
413 | if (!all_empty) { | ||
414 | can_print_recv_banner(m); | ||
415 | for (i = 0; i < 0x800; i++) { | ||
416 | if (!hlist_empty(&d->rx_sff[i])) | ||
417 | can_print_rcvlist(m, &d->rx_sff[i], dev); | ||
418 | } | ||
419 | } else | ||
420 | seq_printf(m, " (%s: no entry)\n", DNAME(dev)); | ||
421 | } | ||
422 | |||
386 | static int can_rcvlist_sff_proc_show(struct seq_file *m, void *v) | 423 | static int can_rcvlist_sff_proc_show(struct seq_file *m, void *v) |
387 | { | 424 | { |
425 | struct net_device *dev; | ||
388 | struct dev_rcv_lists *d; | 426 | struct dev_rcv_lists *d; |
389 | struct hlist_node *n; | ||
390 | 427 | ||
391 | /* RX_SFF */ | 428 | /* RX_SFF */ |
392 | seq_puts(m, "\nreceive list 'rx_sff':\n"); | 429 | seq_puts(m, "\nreceive list 'rx_sff':\n"); |
393 | 430 | ||
394 | rcu_read_lock(); | 431 | rcu_read_lock(); |
395 | hlist_for_each_entry_rcu(d, n, &can_rx_dev_list, list) { | 432 | |
396 | int i, all_empty = 1; | 433 | /* sff receive list for 'all' CAN devices (dev == NULL) */ |
397 | /* check wether at least one list is non-empty */ | 434 | d = &can_rx_alldev_list; |
398 | for (i = 0; i < 0x800; i++) | 435 | can_rcvlist_sff_proc_show_one(m, NULL, d); |
399 | if (!hlist_empty(&d->rx_sff[i])) { | 436 | |
400 | all_empty = 0; | 437 | /* sff receive list for registered CAN devices */ |
401 | break; | 438 | for_each_netdev_rcu(&init_net, dev) { |
402 | } | 439 | if (dev->type == ARPHRD_CAN && dev->ml_priv) |
403 | 440 | can_rcvlist_sff_proc_show_one(m, dev, dev->ml_priv); | |
404 | if (!all_empty) { | ||
405 | can_print_recv_banner(m); | ||
406 | for (i = 0; i < 0x800; i++) { | ||
407 | if (!hlist_empty(&d->rx_sff[i])) | ||
408 | can_print_rcvlist(m, &d->rx_sff[i], | ||
409 | d->dev); | ||
410 | } | ||
411 | } else | ||
412 | seq_printf(m, " (%s: no entry)\n", DNAME(d->dev)); | ||
413 | } | 441 | } |
442 | |||
414 | rcu_read_unlock(); | 443 | rcu_read_unlock(); |
415 | 444 | ||
416 | seq_putc(m, '\n'); | 445 | seq_putc(m, '\n'); |
diff --git a/net/can/raw.c b/net/can/raw.c index b5e897922d32..da99cf153b33 100644 --- a/net/can/raw.c +++ b/net/can/raw.c | |||
@@ -45,6 +45,7 @@ | |||
45 | #include <linux/init.h> | 45 | #include <linux/init.h> |
46 | #include <linux/uio.h> | 46 | #include <linux/uio.h> |
47 | #include <linux/net.h> | 47 | #include <linux/net.h> |
48 | #include <linux/slab.h> | ||
48 | #include <linux/netdevice.h> | 49 | #include <linux/netdevice.h> |
49 | #include <linux/socket.h> | 50 | #include <linux/socket.h> |
50 | #include <linux/if_arp.h> | 51 | #include <linux/if_arp.h> |
@@ -424,8 +425,6 @@ static int raw_setsockopt(struct socket *sock, int level, int optname, | |||
424 | 425 | ||
425 | if (level != SOL_CAN_RAW) | 426 | if (level != SOL_CAN_RAW) |
426 | return -EINVAL; | 427 | return -EINVAL; |
427 | if (optlen < 0) | ||
428 | return -EINVAL; | ||
429 | 428 | ||
430 | switch (optname) { | 429 | switch (optname) { |
431 | 430 | ||
@@ -446,7 +445,7 @@ static int raw_setsockopt(struct socket *sock, int level, int optname, | |||
446 | return -EFAULT; | 445 | return -EFAULT; |
447 | } | 446 | } |
448 | } else if (count == 1) { | 447 | } else if (count == 1) { |
449 | if (copy_from_user(&sfilter, optval, optlen)) | 448 | if (copy_from_user(&sfilter, optval, sizeof(sfilter))) |
450 | return -EFAULT; | 449 | return -EFAULT; |
451 | } | 450 | } |
452 | 451 | ||
@@ -702,7 +701,7 @@ static int raw_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
702 | return err; | 701 | return err; |
703 | } | 702 | } |
704 | 703 | ||
705 | sock_recv_timestamp(msg, sk, skb); | 704 | sock_recv_ts_and_drops(msg, sk, skb); |
706 | 705 | ||
707 | if (msg->msg_name) { | 706 | if (msg->msg_name) { |
708 | msg->msg_namelen = sizeof(struct sockaddr_can); | 707 | msg->msg_namelen = sizeof(struct sockaddr_can); |
@@ -744,7 +743,6 @@ static struct proto raw_proto __read_mostly = { | |||
744 | static struct can_proto raw_can_proto __read_mostly = { | 743 | static struct can_proto raw_can_proto __read_mostly = { |
745 | .type = SOCK_RAW, | 744 | .type = SOCK_RAW, |
746 | .protocol = CAN_RAW, | 745 | .protocol = CAN_RAW, |
747 | .capability = -1, | ||
748 | .ops = &raw_ops, | 746 | .ops = &raw_ops, |
749 | .prot = &raw_proto, | 747 | .prot = &raw_proto, |
750 | }; | 748 | }; |