blob: 791327219531c8a907a23cb54e4091401d15a7c2 [file] [log] [blame]
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +00001/*
Sven Eckelmann567db7b2012-01-01 00:41:38 +01002 * Copyright (C) 2007-2012 B.A.T.M.A.N. contributors:
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +00003 *
4 * Marek Lindner, Simon Wunderlich
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of version 2 of the GNU General Public
8 * License as published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
18 * 02110-1301, USA
19 *
20 */
21
22#include "main.h"
23#include "bat_sysfs.h"
24#include "bat_debugfs.h"
25#include "routing.h"
26#include "send.h"
27#include "originator.h"
28#include "soft-interface.h"
29#include "icmp_socket.h"
30#include "translation-table.h"
31#include "hard-interface.h"
32#include "gateway_client.h"
Simon Wunderlich23721382012-01-22 20:00:19 +010033#include "bridge_loop_avoidance.h"
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000034#include "vis.h"
35#include "hash.h"
Marek Lindner1c280472011-11-28 17:40:17 +080036#include "bat_algo.h"
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000037
Sven Eckelmannc3caf512011-05-03 11:51:38 +020038
39/* List manipulations on hardif_list have to be rtnl_lock()'ed,
40 * list traversals just rcu-locked */
Marek Lindner4389e472011-02-18 12:33:19 +000041struct list_head hardif_list;
Marek Lindner1c280472011-11-28 17:40:17 +080042char bat_routing_algo[20] = "BATMAN IV";
43static struct hlist_head bat_algo_list;
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000044
45unsigned char broadcast_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
46
47struct workqueue_struct *bat_event_workqueue;
48
49static int __init batman_init(void)
50{
Marek Lindner4389e472011-02-18 12:33:19 +000051 INIT_LIST_HEAD(&hardif_list);
Marek Lindner1c280472011-11-28 17:40:17 +080052 INIT_HLIST_HEAD(&bat_algo_list);
53
54 bat_iv_init();
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000055
56 /* the name should not be longer than 10 chars - see
57 * http://lwn.net/Articles/23634/ */
58 bat_event_workqueue = create_singlethread_workqueue("bat_events");
59
60 if (!bat_event_workqueue)
61 return -ENOMEM;
62
63 bat_socket_init();
64 debugfs_init();
65
66 register_netdevice_notifier(&hard_if_notifier);
67
Sven Eckelmann86ceb362012-03-07 09:07:45 +010068 pr_info("B.A.T.M.A.N. advanced %s (compatibility version %i) loaded\n",
69 SOURCE_VERSION, COMPAT_VERSION);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000070
71 return 0;
72}
73
74static void __exit batman_exit(void)
75{
76 debugfs_destroy();
77 unregister_netdevice_notifier(&hard_if_notifier);
78 hardif_remove_interfaces();
79
80 flush_workqueue(bat_event_workqueue);
81 destroy_workqueue(bat_event_workqueue);
82 bat_event_workqueue = NULL;
83
84 rcu_barrier();
85}
86
87int mesh_init(struct net_device *soft_iface)
88{
89 struct bat_priv *bat_priv = netdev_priv(soft_iface);
90
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000091 spin_lock_init(&bat_priv->forw_bat_list_lock);
92 spin_lock_init(&bat_priv->forw_bcast_list_lock);
Antonio Quartullia73105b2011-04-27 14:27:44 +020093 spin_lock_init(&bat_priv->tt_changes_list_lock);
94 spin_lock_init(&bat_priv->tt_req_list_lock);
Antonio Quartullicc47f662011-04-27 14:27:57 +020095 spin_lock_init(&bat_priv->tt_roam_list_lock);
Antonio Quartullia73105b2011-04-27 14:27:44 +020096 spin_lock_init(&bat_priv->tt_buff_lock);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +000097 spin_lock_init(&bat_priv->gw_list_lock);
98 spin_lock_init(&bat_priv->vis_hash_lock);
99 spin_lock_init(&bat_priv->vis_list_lock);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000100
101 INIT_HLIST_HEAD(&bat_priv->forw_bat_list);
102 INIT_HLIST_HEAD(&bat_priv->forw_bcast_list);
103 INIT_HLIST_HEAD(&bat_priv->gw_list);
Antonio Quartullia73105b2011-04-27 14:27:44 +0200104 INIT_LIST_HEAD(&bat_priv->tt_changes_list);
105 INIT_LIST_HEAD(&bat_priv->tt_req_list);
Antonio Quartullicc47f662011-04-27 14:27:57 +0200106 INIT_LIST_HEAD(&bat_priv->tt_roam_list);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000107
108 if (originator_init(bat_priv) < 1)
109 goto err;
110
Antonio Quartullia73105b2011-04-27 14:27:44 +0200111 if (tt_init(bat_priv) < 1)
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000112 goto err;
113
Antonio Quartullibc279082011-07-07 15:35:35 +0200114 tt_local_add(soft_iface, soft_iface->dev_addr, NULL_IFINDEX);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000115
116 if (vis_init(bat_priv) < 1)
117 goto err;
118
Simon Wunderlich23721382012-01-22 20:00:19 +0100119 if (bla_init(bat_priv) < 1)
120 goto err;
121
Antonio Quartulli2265c142011-04-27 00:22:00 +0200122 atomic_set(&bat_priv->gw_reselect, 0);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000123 atomic_set(&bat_priv->mesh_state, MESH_ACTIVE);
124 goto end;
125
126err:
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000127 mesh_free(soft_iface);
128 return -1;
129
130end:
131 return 0;
132}
133
134void mesh_free(struct net_device *soft_iface)
135{
136 struct bat_priv *bat_priv = netdev_priv(soft_iface);
137
138 atomic_set(&bat_priv->mesh_state, MESH_DEACTIVATING);
139
140 purge_outstanding_packets(bat_priv, NULL);
141
142 vis_quit(bat_priv);
143
144 gw_node_purge(bat_priv);
145 originator_free(bat_priv);
146
Antonio Quartullia73105b2011-04-27 14:27:44 +0200147 tt_free(bat_priv);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000148
Simon Wunderlich23721382012-01-22 20:00:19 +0100149 bla_free(bat_priv);
150
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000151 atomic_set(&bat_priv->mesh_state, MESH_INACTIVE);
152}
153
154void inc_module_count(void)
155{
156 try_module_get(THIS_MODULE);
157}
158
159void dec_module_count(void)
160{
161 module_put(THIS_MODULE);
162}
163
Sven Eckelmann747e4222011-05-14 23:14:50 +0200164int is_my_mac(const uint8_t *addr)
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000165{
Sven Eckelmann747e4222011-05-14 23:14:50 +0200166 const struct hard_iface *hard_iface;
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000167
168 rcu_read_lock();
Marek Lindnere6c10f42011-02-18 12:33:20 +0000169 list_for_each_entry_rcu(hard_iface, &hardif_list, list) {
170 if (hard_iface->if_status != IF_ACTIVE)
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000171 continue;
172
Marek Lindnere6c10f42011-02-18 12:33:20 +0000173 if (compare_eth(hard_iface->net_dev->dev_addr, addr)) {
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000174 rcu_read_unlock();
175 return 1;
176 }
177 }
178 rcu_read_unlock();
179 return 0;
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000180}
181
Marek Lindner1c280472011-11-28 17:40:17 +0800182static struct bat_algo_ops *bat_algo_get(char *name)
183{
184 struct bat_algo_ops *bat_algo_ops = NULL, *bat_algo_ops_tmp;
185 struct hlist_node *node;
186
187 hlist_for_each_entry(bat_algo_ops_tmp, node, &bat_algo_list, list) {
188 if (strcmp(bat_algo_ops_tmp->name, name) != 0)
189 continue;
190
191 bat_algo_ops = bat_algo_ops_tmp;
192 break;
193 }
194
195 return bat_algo_ops;
196}
197
198int bat_algo_register(struct bat_algo_ops *bat_algo_ops)
199{
200 struct bat_algo_ops *bat_algo_ops_tmp;
201 int ret = -1;
202
203 bat_algo_ops_tmp = bat_algo_get(bat_algo_ops->name);
204 if (bat_algo_ops_tmp) {
Sven Eckelmann86ceb362012-03-07 09:07:45 +0100205 pr_info("Trying to register already registered routing algorithm: %s\n",
206 bat_algo_ops->name);
Marek Lindner1c280472011-11-28 17:40:17 +0800207 goto out;
208 }
209
Marek Lindner01c42242011-11-28 21:31:55 +0800210 /* all algorithms must implement all ops (for now) */
Marek Lindnerc2aca022012-02-07 17:20:45 +0800211 if (!bat_algo_ops->bat_iface_enable ||
Marek Lindner00a50072012-02-07 17:20:47 +0800212 !bat_algo_ops->bat_iface_disable ||
Marek Lindnercd8b78e2012-02-07 17:20:49 +0800213 !bat_algo_ops->bat_primary_iface_set ||
Marek Lindner01c42242011-11-28 21:31:55 +0800214 !bat_algo_ops->bat_ogm_update_mac ||
215 !bat_algo_ops->bat_ogm_schedule ||
216 !bat_algo_ops->bat_ogm_emit ||
217 !bat_algo_ops->bat_ogm_receive) {
218 pr_info("Routing algo '%s' does not implement required ops\n",
219 bat_algo_ops->name);
220 goto out;
221 }
222
Marek Lindner1c280472011-11-28 17:40:17 +0800223 INIT_HLIST_NODE(&bat_algo_ops->list);
224 hlist_add_head(&bat_algo_ops->list, &bat_algo_list);
225 ret = 0;
226
227out:
228 return ret;
229}
230
231int bat_algo_select(struct bat_priv *bat_priv, char *name)
232{
233 struct bat_algo_ops *bat_algo_ops;
234 int ret = -1;
235
236 bat_algo_ops = bat_algo_get(name);
237 if (!bat_algo_ops)
238 goto out;
239
240 bat_priv->bat_algo_ops = bat_algo_ops;
241 ret = 0;
242
243out:
244 return ret;
245}
246
247int bat_algo_seq_print_text(struct seq_file *seq, void *offset)
248{
249 struct bat_algo_ops *bat_algo_ops;
250 struct hlist_node *node;
251
252 seq_printf(seq, "Available routing algorithms:\n");
253
254 hlist_for_each_entry(bat_algo_ops, node, &bat_algo_list, list) {
255 seq_printf(seq, "%s\n", bat_algo_ops->name);
256 }
257
258 return 0;
259}
260
Marek Lindnerd419be12011-12-10 19:45:53 +0800261static int param_set_ra(const char *val, const struct kernel_param *kp)
262{
263 struct bat_algo_ops *bat_algo_ops;
264
265 bat_algo_ops = bat_algo_get((char *)val);
266 if (!bat_algo_ops) {
267 pr_err("Routing algorithm '%s' is not supported\n", val);
268 return -EINVAL;
269 }
270
271 return param_set_copystring(val, kp);
272}
273
274static const struct kernel_param_ops param_ops_ra = {
275 .set = param_set_ra,
276 .get = param_get_string,
277};
278
279static struct kparam_string __param_string_ra = {
280 .maxlen = sizeof(bat_routing_algo),
281 .string = bat_routing_algo,
282};
283
284module_param_cb(routing_algo, &param_ops_ra, &__param_string_ra, 0644);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000285module_init(batman_init);
286module_exit(batman_exit);
287
288MODULE_LICENSE("GPL");
289
290MODULE_AUTHOR(DRIVER_AUTHOR);
291MODULE_DESCRIPTION(DRIVER_DESC);
292MODULE_SUPPORTED_DEVICE(DRIVER_DEVICE);
Sven Eckelmannc6c8fea2010-12-13 11:19:28 +0000293MODULE_VERSION(SOURCE_VERSION);