net/hsr: Switch from dev_add_pack() to netdev_rx_handler_register()
[firefly-linux-kernel-4.4.55.git] / net / hsr / hsr_main.c
1 /* Copyright 2011-2014 Autronica Fire and Security AS
2  *
3  * This program is free software; you can redistribute it and/or modify it
4  * under the terms of the GNU General Public License as published by the Free
5  * Software Foundation; either version 2 of the License, or (at your option)
6  * any later version.
7  *
8  * Author(s):
9  *      2011-2014 Arvid Brodin, arvid.brodin@alten.se
10  */
11
12 #include <linux/netdevice.h>
13 #include <linux/rculist.h>
14 #include <linux/timer.h>
15 #include <linux/etherdevice.h>
16 #include "hsr_main.h"
17 #include "hsr_device.h"
18 #include "hsr_netlink.h"
19 #include "hsr_framereg.h"
20
21
22 /* List of all registered virtual HSR devices */
23 static LIST_HEAD(hsr_list);
24
25 void register_hsr_master(struct hsr_priv *hsr)
26 {
27         list_add_tail_rcu(&hsr->hsr_list, &hsr_list);
28 }
29
30 void unregister_hsr_master(struct hsr_priv *hsr)
31 {
32         struct hsr_priv *hsr_it;
33
34         list_for_each_entry(hsr_it, &hsr_list, hsr_list)
35                 if (hsr_it == hsr) {
36                         list_del_rcu(&hsr_it->hsr_list);
37                         return;
38                 }
39 }
40
41 bool is_hsr_slave(struct net_device *dev)
42 {
43         struct hsr_priv *hsr_it;
44
45         list_for_each_entry_rcu(hsr_it, &hsr_list, hsr_list) {
46                 if (dev == hsr_it->slave[0])
47                         return true;
48                 if (dev == hsr_it->slave[1])
49                         return true;
50         }
51
52         return false;
53 }
54
55 /* If dev is a HSR slave device, return the virtual master device. Return NULL
56  * otherwise.
57  */
58 struct hsr_priv *get_hsr_master(struct net_device *dev)
59 {
60         struct hsr_priv *hsr;
61
62         rcu_read_lock();
63         list_for_each_entry_rcu(hsr, &hsr_list, hsr_list)
64                 if ((dev == hsr->slave[0]) ||
65                     (dev == hsr->slave[1])) {
66                         rcu_read_unlock();
67                         return hsr;
68                 }
69
70         rcu_read_unlock();
71         return NULL;
72 }
73
74 /* If dev is a HSR slave device, return the other slave device. Return NULL
75  * otherwise.
76  */
77 struct net_device *get_other_slave(struct hsr_priv *hsr,
78                                    struct net_device *dev)
79 {
80         if (dev == hsr->slave[0])
81                 return hsr->slave[1];
82         if (dev == hsr->slave[1])
83                 return hsr->slave[0];
84
85         return NULL;
86 }
87
88
89 static int hsr_netdev_notify(struct notifier_block *nb, unsigned long event,
90                              void *ptr)
91 {
92         struct net_device *slave, *other_slave;
93         struct hsr_priv *hsr;
94         int old_operstate;
95         int mtu_max;
96         int res;
97         struct net_device *dev;
98
99         dev = netdev_notifier_info_to_dev(ptr);
100
101         hsr = get_hsr_master(dev);
102         if (hsr) {
103                 /* dev is a slave device */
104                 slave = dev;
105                 other_slave = get_other_slave(hsr, slave);
106         } else {
107                 if (!is_hsr_master(dev))
108                         return NOTIFY_DONE;
109                 hsr = netdev_priv(dev);
110                 slave = hsr->slave[0];
111                 other_slave = hsr->slave[1];
112         }
113
114         switch (event) {
115         case NETDEV_UP:         /* Administrative state DOWN */
116         case NETDEV_DOWN:       /* Administrative state UP */
117         case NETDEV_CHANGE:     /* Link (carrier) state changes */
118                 old_operstate = hsr->dev->operstate;
119                 hsr_set_carrier(hsr->dev, slave, other_slave);
120                 /* netif_stacked_transfer_operstate() cannot be used here since
121                  * it doesn't set IF_OPER_LOWERLAYERDOWN (?)
122                  */
123                 hsr_set_operstate(hsr->dev, slave, other_slave);
124                 hsr_check_announce(hsr->dev, old_operstate);
125                 break;
126         case NETDEV_CHANGEADDR:
127
128                 /* This should not happen since there's no ndo_set_mac_address()
129                  * for HSR devices - i.e. not supported.
130                  */
131                 if (dev == hsr->dev)
132                         break;
133
134                 if (dev == hsr->slave[0])
135                         ether_addr_copy(hsr->dev->dev_addr,
136                                         hsr->slave[0]->dev_addr);
137
138                 /* Make sure we recognize frames from ourselves in hsr_rcv() */
139                 res = hsr_create_self_node(&hsr->self_node_db,
140                                            hsr->dev->dev_addr,
141                                            hsr->slave[1] ?
142                                                 hsr->slave[1]->dev_addr :
143                                                 hsr->dev->dev_addr);
144                 if (res)
145                         netdev_warn(hsr->dev,
146                                     "Could not update HSR node address.\n");
147
148                 if (dev == hsr->slave[0])
149                         call_netdevice_notifiers(NETDEV_CHANGEADDR, hsr->dev);
150                 break;
151         case NETDEV_CHANGEMTU:
152                 if (dev == hsr->dev)
153                         break; /* Handled in ndo_change_mtu() */
154                 mtu_max = hsr_get_max_mtu(hsr);
155                 if (hsr->dev->mtu > mtu_max)
156                         dev_set_mtu(hsr->dev, mtu_max);
157                 break;
158         case NETDEV_UNREGISTER:
159                 if (dev == hsr->slave[0])
160                         hsr->slave[0] = NULL;
161                 if (dev == hsr->slave[1])
162                         hsr->slave[1] = NULL;
163
164                 /* There should really be a way to set a new slave device... */
165
166                 break;
167         case NETDEV_PRE_TYPE_CHANGE:
168                 /* HSR works only on Ethernet devices. Refuse slave to change
169                  * its type.
170                  */
171                 return NOTIFY_BAD;
172         }
173
174         return NOTIFY_DONE;
175 }
176
177
178 static struct timer_list prune_timer;
179
180 static void prune_nodes_all(unsigned long data)
181 {
182         struct hsr_priv *hsr;
183
184         rcu_read_lock();
185         list_for_each_entry_rcu(hsr, &hsr_list, hsr_list)
186                 hsr_prune_nodes(hsr);
187         rcu_read_unlock();
188
189         prune_timer.expires = jiffies + msecs_to_jiffies(PRUNE_PERIOD);
190         add_timer(&prune_timer);
191 }
192
193
194 static struct notifier_block hsr_nb = {
195         .notifier_call = hsr_netdev_notify,     /* Slave event notifications */
196 };
197
198
199 static int __init hsr_init(void)
200 {
201         int res;
202
203         BUILD_BUG_ON(sizeof(struct hsr_tag) != HSR_HLEN);
204
205         init_timer(&prune_timer);
206         prune_timer.function = prune_nodes_all;
207         prune_timer.data = 0;
208         prune_timer.expires = jiffies + msecs_to_jiffies(PRUNE_PERIOD);
209         add_timer(&prune_timer);
210
211         register_netdevice_notifier(&hsr_nb);
212
213         res = hsr_netlink_init();
214
215         return res;
216 }
217
218 static void __exit hsr_exit(void)
219 {
220         unregister_netdevice_notifier(&hsr_nb);
221         del_timer_sync(&prune_timer);
222         hsr_netlink_exit();
223 }
224
225 module_init(hsr_init);
226 module_exit(hsr_exit);
227 MODULE_LICENSE("GPL");