diff options
-rw-r--r-- | include/net/ip_fib.h | 2 | ||||
-rw-r--r-- | net/ipv4/fib_frontend.c | 18 | ||||
-rw-r--r-- | net/ipv4/fib_rules.c | 14 |
3 files changed, 28 insertions, 6 deletions
diff --git a/include/net/ip_fib.h b/include/net/ip_fib.h index f74cbb21af45..cbff18d748dd 100644 --- a/include/net/ip_fib.h +++ b/include/net/ip_fib.h | |||
@@ -186,7 +186,7 @@ static inline void fib_select_default(const struct flowi *flp, struct fib_result | |||
186 | } | 186 | } |
187 | 187 | ||
188 | #else /* CONFIG_IP_MULTIPLE_TABLES */ | 188 | #else /* CONFIG_IP_MULTIPLE_TABLES */ |
189 | extern void __init fib4_rules_init(void); | 189 | extern int __init fib4_rules_init(void); |
190 | 190 | ||
191 | #ifdef CONFIG_NET_CLS_ROUTE | 191 | #ifdef CONFIG_NET_CLS_ROUTE |
192 | extern u32 fib_rules_tclass(struct fib_result *res); | 192 | extern u32 fib_rules_tclass(struct fib_result *res); |
diff --git a/net/ipv4/fib_frontend.c b/net/ipv4/fib_frontend.c index ac6238a3b0fd..1bb956b3be28 100644 --- a/net/ipv4/fib_frontend.c +++ b/net/ipv4/fib_frontend.c | |||
@@ -59,12 +59,24 @@ struct fib_table *ip_fib_main_table; | |||
59 | #define FIB_TABLE_HASHSZ 1 | 59 | #define FIB_TABLE_HASHSZ 1 |
60 | static struct hlist_head fib_table_hash[FIB_TABLE_HASHSZ]; | 60 | static struct hlist_head fib_table_hash[FIB_TABLE_HASHSZ]; |
61 | 61 | ||
62 | static void __init fib4_rules_init(void) | 62 | static int __init fib4_rules_init(void) |
63 | { | 63 | { |
64 | ip_fib_local_table = fib_hash_init(RT_TABLE_LOCAL); | 64 | ip_fib_local_table = fib_hash_init(RT_TABLE_LOCAL); |
65 | hlist_add_head_rcu(&ip_fib_local_table->tb_hlist, &fib_table_hash[0]); | 65 | if (ip_fib_local_table == NULL) |
66 | return -ENOMEM; | ||
67 | |||
66 | ip_fib_main_table = fib_hash_init(RT_TABLE_MAIN); | 68 | ip_fib_main_table = fib_hash_init(RT_TABLE_MAIN); |
69 | if (ip_fib_main_table == NULL) | ||
70 | goto fail; | ||
71 | |||
72 | hlist_add_head_rcu(&ip_fib_local_table->tb_hlist, &fib_table_hash[0]); | ||
67 | hlist_add_head_rcu(&ip_fib_main_table->tb_hlist, &fib_table_hash[0]); | 73 | hlist_add_head_rcu(&ip_fib_main_table->tb_hlist, &fib_table_hash[0]); |
74 | return 0; | ||
75 | |||
76 | fail: | ||
77 | kfree(ip_fib_local_table); | ||
78 | ip_fib_local_table = NULL; | ||
79 | return -ENOMEM; | ||
68 | } | 80 | } |
69 | #else | 81 | #else |
70 | 82 | ||
@@ -944,7 +956,7 @@ void __init ip_fib_init(void) | |||
944 | for (i = 0; i < FIB_TABLE_HASHSZ; i++) | 956 | for (i = 0; i < FIB_TABLE_HASHSZ; i++) |
945 | INIT_HLIST_HEAD(&fib_table_hash[i]); | 957 | INIT_HLIST_HEAD(&fib_table_hash[i]); |
946 | 958 | ||
947 | fib4_rules_init(); | 959 | BUG_ON(fib4_rules_init()); |
948 | 960 | ||
949 | register_netdevice_notifier(&fib_netdev_notifier); | 961 | register_netdevice_notifier(&fib_netdev_notifier); |
950 | register_inetaddr_notifier(&fib_inetaddr_notifier); | 962 | register_inetaddr_notifier(&fib_inetaddr_notifier); |
diff --git a/net/ipv4/fib_rules.c b/net/ipv4/fib_rules.c index afe669dd1bd5..0751734ecf41 100644 --- a/net/ipv4/fib_rules.c +++ b/net/ipv4/fib_rules.c | |||
@@ -311,8 +311,18 @@ static int __init fib_default_rules_init(void) | |||
311 | return 0; | 311 | return 0; |
312 | } | 312 | } |
313 | 313 | ||
314 | void __init fib4_rules_init(void) | 314 | int __init fib4_rules_init() |
315 | { | 315 | { |
316 | BUG_ON(fib_default_rules_init()); | 316 | int err; |
317 | |||
317 | fib_rules_register(&init_net, &fib4_rules_ops); | 318 | fib_rules_register(&init_net, &fib4_rules_ops); |
319 | err = fib_default_rules_init(); | ||
320 | if (err < 0) | ||
321 | goto fail; | ||
322 | return 0; | ||
323 | |||
324 | fail: | ||
325 | /* also cleans all rules already added */ | ||
326 | fib_rules_unregister(&init_net, &fib4_rules_ops); | ||
327 | return err; | ||
318 | } | 328 | } |