X-Git-Url: http://ftp.safe.ca/?a=blobdiff_plain;f=include%2Fnet%2Fnet_namespace.h;h=ded434b032a44ec406a0fa8bcb11d688d95b7efa;hb=8d2f9e81169b8120cf2b4872930ae491b17c27b8;hp=f880b0f9f1070ea1e6dae1fc17c343bc87806d8f;hpb=dec827d174d7f76c457238800183ca864a639365;p=safe%2Fjmp%2Flinux-2.6 diff --git a/include/net/net_namespace.h b/include/net/net_namespace.h index f880b0f..ded434b 100644 --- a/include/net/net_namespace.h +++ b/include/net/net_namespace.h @@ -9,12 +9,17 @@ #include #include +#include #include #include #include #include #include #include +#if defined(CONFIG_NF_CONNTRACK) || defined(CONFIG_NF_CONNTRACK_MODULE) +#include +#endif +#include struct proc_dir_entry; struct net_device; @@ -26,16 +31,20 @@ struct net { atomic_t count; /* To decided when the network * namespace should be freed. */ +#ifdef NETNS_REFCNT_DEBUG atomic_t use_count; /* To track references we * destroy on demand */ +#endif struct list_head list; /* list of network namespaces */ struct work_struct work; /* work struct for freeing */ struct proc_dir_entry *proc_net; struct proc_dir_entry *proc_net_stat; - struct list_head sysctl_table_headers; +#ifdef CONFIG_SYSCTL + struct ctl_table_set sysctls; +#endif struct net_device *loopback_dev; /* The loopback */ @@ -50,6 +59,7 @@ struct net { struct sock *rtnl; /* rtnetlink socket */ struct netns_core core; + struct netns_mib mib; struct netns_packet packet; struct netns_unix unx; struct netns_ipv4 ipv4; @@ -61,6 +71,12 @@ struct net { #endif #ifdef CONFIG_NETFILTER struct netns_xt xt; +#if defined(CONFIG_NF_CONNTRACK) || defined(CONFIG_NF_CONNTRACK_MODULE) + struct netns_ct ct; +#endif +#endif +#ifdef CONFIG_XFRM + struct netns_xfrm xfrm; #endif struct net_generic *gen; }; @@ -117,32 +133,49 @@ static inline void put_net(struct net *net) __put_net(net); } -static inline struct net *hold_net(struct net *net) +static inline +int net_eq(const struct net *net1, const struct net *net2) +{ + return net1 == net2; +} +#else + +static inline struct net *get_net(struct net *net) { - atomic_inc(&net->use_count); return net; } -static inline void release_net(struct net *net) +static inline void put_net(struct net *net) +{ +} + +static inline struct net *maybe_get_net(struct net *net) { - atomic_dec(&net->use_count); + return net; } static inline int net_eq(const struct net *net1, const struct net *net2) { - return net1 == net2; + return 1; } -#else -static inline struct net *get_net(struct net *net) +#endif + + +#ifdef NETNS_REFCNT_DEBUG +static inline struct net *hold_net(struct net *net) { + if (net) + atomic_inc(&net->use_count); return net; } -static inline void put_net(struct net *net) +static inline void release_net(struct net *net) { + if (net) + atomic_dec(&net->use_count); } - +#else static inline struct net *hold_net(struct net *net) { return net; @@ -151,17 +184,25 @@ static inline struct net *hold_net(struct net *net) static inline void release_net(struct net *net) { } +#endif -static inline struct net *maybe_get_net(struct net *net) +#ifdef CONFIG_NET_NS + +static inline void write_pnet(struct net **pnet, struct net *net) { - return net; + *pnet = net; } -static inline -int net_eq(const struct net *net1, const struct net *net2) +static inline struct net *read_pnet(struct net * const *pnet) { - return 1; + return *pnet; } + +#else + +#define write_pnet(pnet, net) do { (void)(net);} while (0) +#define read_pnet(pnet) (&init_net) + #endif #define for_each_net(VAR) \ @@ -183,8 +224,27 @@ struct pernet_operations { void (*exit)(struct net *net); }; +/* + * Use these carefully. If you implement a network device and it + * needs per network namespace operations use device pernet operations, + * otherwise use pernet subsys operations. + * + * This is critically important. Most of the network code cleanup + * runs with the assumption that dev_remove_pack has been called so no + * new packets will arrive during and after the cleanup functions have + * been called. dev_remove_pack is not per namespace so instead the + * guarantee of no more packets arriving in a network namespace is + * provided by ensuring that all network devices and all sockets have + * left the network namespace before the cleanup methods are called. + * + * For the longest time the ipv4 icmp code was registered as a pernet + * device which caused kernel oops, and panics during network + * namespace cleanup. So please don't get this wrong. + */ extern int register_pernet_subsys(struct pernet_operations *); extern void unregister_pernet_subsys(struct pernet_operations *); +extern int register_pernet_gen_subsys(int *id, struct pernet_operations *); +extern void unregister_pernet_gen_subsys(int id, struct pernet_operations *); extern int register_pernet_device(struct pernet_operations *); extern void unregister_pernet_device(struct pernet_operations *); extern int register_pernet_gen_device(int *id, struct pernet_operations *); @@ -193,8 +253,11 @@ extern void unregister_pernet_gen_device(int id, struct pernet_operations *); struct ctl_path; struct ctl_table; struct ctl_table_header; + extern struct ctl_table_header *register_net_sysctl_table(struct net *net, const struct ctl_path *path, struct ctl_table *table); +extern struct ctl_table_header *register_net_sysctl_rotable( + const struct ctl_path *path, struct ctl_table *table); extern void unregister_net_sysctl_table(struct ctl_table_header *header); #endif /* __NET_NET_NAMESPACE_H */