2006-06-18 07:37:28 +04:00
/*
* Copyright ( c ) 2005 Voltaire Inc . All rights reserved .
* Copyright ( c ) 2002 - 2005 , Network Appliance , Inc . All rights reserved .
* Copyright ( c ) 1999 - 2005 , Mellanox Technologies , Inc . All rights reserved .
* Copyright ( c ) 2005 Intel Corporation . All rights reserved .
*
2008-07-15 10:48:43 +04:00
* This software is available to you under a choice of one of two
* licenses . You may choose to be licensed under the terms of the GNU
* General Public License ( GPL ) Version 2 , available from the file
* COPYING in the main directory of this source tree , or the
* OpenIB . org BSD license below :
2006-06-18 07:37:28 +04:00
*
2008-07-15 10:48:43 +04:00
* Redistribution and use in source and binary forms , with or
* without modification , are permitted provided that the following
* conditions are met :
2006-06-18 07:37:28 +04:00
*
2008-07-15 10:48:43 +04:00
* - Redistributions of source code must retain the above
* copyright notice , this list of conditions and the following
* disclaimer .
2006-06-18 07:37:28 +04:00
*
2008-07-15 10:48:43 +04:00
* - Redistributions in binary form must reproduce the above
* copyright notice , this list of conditions and the following
* disclaimer in the documentation and / or other materials
* provided with the distribution .
2006-06-18 07:37:28 +04:00
*
2008-07-15 10:48:43 +04:00
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
2006-06-18 07:37:28 +04:00
*/
# include <linux/mutex.h>
# include <linux/inetdevice.h>
# include <linux/workqueue.h>
# include <linux/if_arp.h>
# include <net/arp.h>
# include <net/neighbour.h>
# include <net/route.h>
2006-07-31 07:44:19 +04:00
# include <net/netevent.h>
2008-12-24 21:16:37 +03:00
# include <net/addrconf.h>
# include <net/ip6_route.h>
2006-06-18 07:37:28 +04:00
# include <rdma/ib_addr.h>
MODULE_AUTHOR ( " Sean Hefty " ) ;
MODULE_DESCRIPTION ( " IB Address Translation " ) ;
MODULE_LICENSE ( " Dual BSD/GPL " ) ;
struct addr_req {
struct list_head list ;
2008-12-24 21:16:37 +03:00
struct sockaddr_storage src_addr ;
struct sockaddr_storage dst_addr ;
2006-06-18 07:37:28 +04:00
struct rdma_dev_addr * addr ;
2006-10-31 22:12:59 +03:00
struct rdma_addr_client * client ;
2006-06-18 07:37:28 +04:00
void * context ;
void ( * callback ) ( int status , struct sockaddr * src_addr ,
struct rdma_dev_addr * addr , void * context ) ;
unsigned long timeout ;
int status ;
} ;
2006-11-22 17:57:56 +03:00
static void process_req ( struct work_struct * work ) ;
2006-06-18 07:37:28 +04:00
static DEFINE_MUTEX ( lock ) ;
static LIST_HEAD ( req_list ) ;
2006-11-22 17:57:56 +03:00
static DECLARE_DELAYED_WORK ( work , process_req ) ;
2006-06-18 07:37:28 +04:00
static struct workqueue_struct * addr_wq ;
2006-10-31 22:12:59 +03:00
void rdma_addr_register_client ( struct rdma_addr_client * client )
{
atomic_set ( & client - > refcount , 1 ) ;
init_completion ( & client - > comp ) ;
}
EXPORT_SYMBOL ( rdma_addr_register_client ) ;
static inline void put_client ( struct rdma_addr_client * client )
{
if ( atomic_dec_and_test ( & client - > refcount ) )
complete ( & client - > comp ) ;
}
void rdma_addr_unregister_client ( struct rdma_addr_client * client )
{
put_client ( client ) ;
wait_for_completion ( & client - > comp ) ;
}
EXPORT_SYMBOL ( rdma_addr_unregister_client ) ;
2006-08-04 01:02:42 +04:00
int rdma_copy_addr ( struct rdma_dev_addr * dev_addr , struct net_device * dev ,
const unsigned char * dst_dev_addr )
2006-06-18 07:37:28 +04:00
{
switch ( dev - > type ) {
case ARPHRD_INFINIBAND :
2006-08-04 01:02:42 +04:00
dev_addr - > dev_type = RDMA_NODE_IB_CA ;
break ;
case ARPHRD_ETHER :
dev_addr - > dev_type = RDMA_NODE_RNIC ;
2006-06-18 07:37:28 +04:00
break ;
default :
return - EADDRNOTAVAIL ;
}
memcpy ( dev_addr - > src_dev_addr , dev - > dev_addr , MAX_ADDR_LEN ) ;
memcpy ( dev_addr - > broadcast , dev - > broadcast , MAX_ADDR_LEN ) ;
if ( dst_dev_addr )
memcpy ( dev_addr - > dst_dev_addr , dst_dev_addr , MAX_ADDR_LEN ) ;
2008-07-15 10:48:53 +04:00
dev_addr - > src_dev = dev ;
2006-06-18 07:37:28 +04:00
return 0 ;
}
2006-08-04 01:02:42 +04:00
EXPORT_SYMBOL ( rdma_copy_addr ) ;
2006-06-18 07:37:28 +04:00
int rdma_translate_ip ( struct sockaddr * addr , struct rdma_dev_addr * dev_addr )
{
struct net_device * dev ;
2008-12-24 21:16:37 +03:00
int ret = - EADDRNOTAVAIL ;
2006-06-18 07:37:28 +04:00
2008-12-24 21:16:37 +03:00
switch ( addr - > sa_family ) {
case AF_INET :
dev = ip_dev_find ( & init_net ,
( ( struct sockaddr_in * ) addr ) - > sin_addr . s_addr ) ;
if ( ! dev )
return ret ;
2006-06-18 07:37:28 +04:00
2008-12-24 21:16:37 +03:00
ret = rdma_copy_addr ( dev_addr , dev , NULL ) ;
dev_put ( dev ) ;
break ;
2008-12-30 10:37:14 +03:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
2008-12-24 21:16:37 +03:00
case AF_INET6 :
for_each_netdev ( & init_net , dev ) {
if ( ipv6_chk_addr ( & init_net ,
& ( ( struct sockaddr_in6 * ) addr ) - > sin6_addr ,
dev , 1 ) ) {
ret = rdma_copy_addr ( dev_addr , dev , NULL ) ;
break ;
}
}
break ;
2008-12-30 10:37:14 +03:00
# endif
2008-12-24 21:16:37 +03:00
}
2006-06-18 07:37:28 +04:00
return ret ;
}
EXPORT_SYMBOL ( rdma_translate_ip ) ;
static void set_timeout ( unsigned long time )
{
unsigned long delay ;
cancel_delayed_work ( & work ) ;
delay = time - jiffies ;
if ( ( long ) delay < = 0 )
delay = 1 ;
queue_delayed_work ( addr_wq , & work , delay ) ;
}
static void queue_req ( struct addr_req * req )
{
struct addr_req * temp_req ;
mutex_lock ( & lock ) ;
list_for_each_entry_reverse ( temp_req , & req_list , list ) {
2006-10-17 08:39:09 +04:00
if ( time_after_eq ( req - > timeout , temp_req - > timeout ) )
2006-06-18 07:37:28 +04:00
break ;
}
list_add ( & req - > list , & temp_req - > list ) ;
if ( req_list . next = = & req - > list )
set_timeout ( req - > timeout ) ;
mutex_unlock ( & lock ) ;
}
2008-12-24 21:16:37 +03:00
static void addr_send_arp ( struct sockaddr * dst_in )
2006-06-18 07:37:28 +04:00
{
struct rtable * rt ;
struct flowi fl ;
memset ( & fl , 0 , sizeof fl ) ;
2008-12-30 10:37:14 +03:00
switch ( dst_in - > sa_family ) {
case AF_INET :
2008-12-24 21:16:37 +03:00
fl . nl_u . ip4_u . daddr =
( ( struct sockaddr_in * ) dst_in ) - > sin_addr . s_addr ;
2006-06-18 07:37:28 +04:00
2008-12-24 21:16:37 +03:00
if ( ip_route_output_key ( & init_net , & rt , & fl ) )
return ;
neigh_event_send ( rt - > u . dst . neighbour , NULL ) ;
ip_rt_put ( rt ) ;
2008-12-30 10:37:14 +03:00
break ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
case AF_INET6 :
{
struct dst_entry * dst ;
2008-12-24 21:16:37 +03:00
fl . nl_u . ip6_u . daddr =
( ( struct sockaddr_in6 * ) dst_in ) - > sin6_addr ;
dst = ip6_route_output ( & init_net , NULL , & fl ) ;
if ( ! dst )
return ;
neigh_event_send ( dst - > neighbour , NULL ) ;
dst_release ( dst ) ;
2008-12-30 10:37:14 +03:00
break ;
}
# endif
2008-12-24 21:16:37 +03:00
}
2006-06-18 07:37:28 +04:00
}
2008-12-24 21:16:37 +03:00
static int addr4_resolve_remote ( struct sockaddr_in * src_in ,
2006-06-18 07:37:28 +04:00
struct sockaddr_in * dst_in ,
struct rdma_dev_addr * addr )
{
2008-03-29 06:10:28 +03:00
__be32 src_ip = src_in - > sin_addr . s_addr ;
__be32 dst_ip = dst_in - > sin_addr . s_addr ;
2006-06-18 07:37:28 +04:00
struct flowi fl ;
struct rtable * rt ;
struct neighbour * neigh ;
int ret ;
memset ( & fl , 0 , sizeof fl ) ;
fl . nl_u . ip4_u . daddr = dst_ip ;
fl . nl_u . ip4_u . saddr = src_ip ;
2008-01-23 09:07:34 +03:00
ret = ip_route_output_key ( & init_net , & rt , & fl ) ;
2006-06-18 07:37:28 +04:00
if ( ret )
goto out ;
/* If the device does ARP internally, return 'done' */
if ( rt - > idev - > dev - > flags & IFF_NOARP ) {
2006-08-04 01:02:42 +04:00
rdma_copy_addr ( addr , rt - > idev - > dev , NULL ) ;
2006-06-18 07:37:28 +04:00
goto put ;
}
neigh = neigh_lookup ( & arp_tbl , & rt - > rt_gateway , rt - > idev - > dev ) ;
if ( ! neigh ) {
ret = - ENODATA ;
goto put ;
}
if ( ! ( neigh - > nud_state & NUD_VALID ) ) {
ret = - ENODATA ;
goto release ;
}
if ( ! src_ip ) {
src_in - > sin_family = dst_in - > sin_family ;
src_in - > sin_addr . s_addr = rt - > rt_src ;
}
2006-08-04 01:02:42 +04:00
ret = rdma_copy_addr ( addr , neigh - > dev , neigh - > ha ) ;
2006-06-18 07:37:28 +04:00
release :
neigh_release ( neigh ) ;
put :
ip_rt_put ( rt ) ;
out :
return ret ;
}
2008-12-30 10:37:14 +03:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
2008-12-24 21:16:37 +03:00
static int addr6_resolve_remote ( struct sockaddr_in6 * src_in ,
struct sockaddr_in6 * dst_in ,
struct rdma_dev_addr * addr )
{
struct flowi fl ;
struct neighbour * neigh ;
struct dst_entry * dst ;
int ret = - ENODATA ;
memset ( & fl , 0 , sizeof fl ) ;
fl . nl_u . ip6_u . daddr = dst_in - > sin6_addr ;
fl . nl_u . ip6_u . saddr = src_in - > sin6_addr ;
dst = ip6_route_output ( & init_net , NULL , & fl ) ;
if ( ! dst )
return ret ;
if ( dst - > dev - > flags & IFF_NOARP ) {
ret = rdma_copy_addr ( addr , dst - > dev , NULL ) ;
} else {
neigh = dst - > neighbour ;
if ( neigh & & ( neigh - > nud_state & NUD_VALID ) )
ret = rdma_copy_addr ( addr , neigh - > dev , neigh - > ha ) ;
}
dst_release ( dst ) ;
return ret ;
}
2008-12-30 10:37:14 +03:00
# else
static int addr6_resolve_remote ( struct sockaddr_in6 * src_in ,
struct sockaddr_in6 * dst_in ,
struct rdma_dev_addr * addr )
{
return - EADDRNOTAVAIL ;
}
# endif
2008-12-24 21:16:37 +03:00
static int addr_resolve_remote ( struct sockaddr * src_in ,
struct sockaddr * dst_in ,
struct rdma_dev_addr * addr )
{
if ( src_in - > sa_family = = AF_INET ) {
return addr4_resolve_remote ( ( struct sockaddr_in * ) src_in ,
( struct sockaddr_in * ) dst_in , addr ) ;
} else
return addr6_resolve_remote ( ( struct sockaddr_in6 * ) src_in ,
( struct sockaddr_in6 * ) dst_in , addr ) ;
}
2006-11-22 17:57:56 +03:00
static void process_req ( struct work_struct * work )
2006-06-18 07:37:28 +04:00
{
struct addr_req * req , * temp_req ;
2008-12-24 21:16:37 +03:00
struct sockaddr * src_in , * dst_in ;
2006-06-18 07:37:28 +04:00
struct list_head done_list ;
INIT_LIST_HEAD ( & done_list ) ;
mutex_lock ( & lock ) ;
list_for_each_entry_safe ( req , temp_req , & req_list , list ) {
2006-11-24 13:32:34 +03:00
if ( req - > status = = - ENODATA ) {
2008-12-24 21:16:37 +03:00
src_in = ( struct sockaddr * ) & req - > src_addr ;
dst_in = ( struct sockaddr * ) & req - > dst_addr ;
2006-06-18 07:37:28 +04:00
req - > status = addr_resolve_remote ( src_in , dst_in ,
req - > addr ) ;
2006-11-24 13:32:34 +03:00
if ( req - > status & & time_after_eq ( jiffies , req - > timeout ) )
req - > status = - ETIMEDOUT ;
else if ( req - > status = = - ENODATA )
continue ;
2006-06-18 07:37:28 +04:00
}
2006-11-30 02:33:09 +03:00
list_move_tail ( & req - > list , & done_list ) ;
2006-06-18 07:37:28 +04:00
}
if ( ! list_empty ( & req_list ) ) {
req = list_entry ( req_list . next , struct addr_req , list ) ;
set_timeout ( req - > timeout ) ;
}
mutex_unlock ( & lock ) ;
list_for_each_entry_safe ( req , temp_req , & done_list , list ) {
list_del ( & req - > list ) ;
2008-12-24 21:16:37 +03:00
req - > callback ( req - > status , ( struct sockaddr * ) & req - > src_addr ,
req - > addr , req - > context ) ;
2006-10-31 22:12:59 +03:00
put_client ( req - > client ) ;
2006-06-18 07:37:28 +04:00
kfree ( req ) ;
}
}
2008-12-24 21:16:37 +03:00
static int addr_resolve_local ( struct sockaddr * src_in ,
struct sockaddr * dst_in ,
2006-06-18 07:37:28 +04:00
struct rdma_dev_addr * addr )
{
struct net_device * dev ;
int ret ;
2008-12-30 10:37:14 +03:00
switch ( dst_in - > sa_family ) {
case AF_INET :
{
2008-12-24 21:16:37 +03:00
__be32 src_ip = ( ( struct sockaddr_in * ) src_in ) - > sin_addr . s_addr ;
__be32 dst_ip = ( ( struct sockaddr_in * ) dst_in ) - > sin_addr . s_addr ;
dev = ip_dev_find ( & init_net , dst_ip ) ;
if ( ! dev )
return - EADDRNOTAVAIL ;
if ( ipv4_is_zeronet ( src_ip ) ) {
src_in - > sa_family = dst_in - > sa_family ;
( ( struct sockaddr_in * ) src_in ) - > sin_addr . s_addr = dst_ip ;
ret = rdma_copy_addr ( addr , dev , dev - > dev_addr ) ;
} else if ( ipv4_is_loopback ( src_ip ) ) {
ret = rdma_translate_ip ( dst_in , addr ) ;
if ( ! ret )
memcpy ( addr - > dst_dev_addr , dev - > dev_addr , MAX_ADDR_LEN ) ;
} else {
ret = rdma_translate_ip ( src_in , addr ) ;
if ( ! ret )
memcpy ( addr - > dst_dev_addr , dev - > dev_addr , MAX_ADDR_LEN ) ;
}
dev_put ( dev ) ;
2008-12-30 10:37:14 +03:00
break ;
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
case AF_INET6 :
{
2008-12-24 21:16:37 +03:00
struct in6_addr * a ;
for_each_netdev ( & init_net , dev )
if ( ipv6_chk_addr ( & init_net ,
& ( ( struct sockaddr_in6 * ) addr ) - > sin6_addr ,
dev , 1 ) )
break ;
if ( ! dev )
return - EADDRNOTAVAIL ;
a = & ( ( struct sockaddr_in6 * ) src_in ) - > sin6_addr ;
if ( ipv6_addr_any ( a ) ) {
src_in - > sa_family = dst_in - > sa_family ;
( ( struct sockaddr_in6 * ) src_in ) - > sin6_addr =
( ( struct sockaddr_in6 * ) dst_in ) - > sin6_addr ;
ret = rdma_copy_addr ( addr , dev , dev - > dev_addr ) ;
} else if ( ipv6_addr_loopback ( a ) ) {
ret = rdma_translate_ip ( dst_in , addr ) ;
if ( ! ret )
memcpy ( addr - > dst_dev_addr , dev - > dev_addr , MAX_ADDR_LEN ) ;
} else {
ret = rdma_translate_ip ( src_in , addr ) ;
if ( ! ret )
memcpy ( addr - > dst_dev_addr , dev - > dev_addr , MAX_ADDR_LEN ) ;
}
2008-12-30 10:37:14 +03:00
break ;
}
# endif
default :
ret = - EADDRNOTAVAIL ;
break ;
2006-06-18 07:37:28 +04:00
}
return ret ;
}
2006-10-31 22:12:59 +03:00
int rdma_resolve_ip ( struct rdma_addr_client * client ,
struct sockaddr * src_addr , struct sockaddr * dst_addr ,
2006-06-18 07:37:28 +04:00
struct rdma_dev_addr * addr , int timeout_ms ,
void ( * callback ) ( int status , struct sockaddr * src_addr ,
struct rdma_dev_addr * addr , void * context ) ,
void * context )
{
2008-12-24 21:16:37 +03:00
struct sockaddr * src_in , * dst_in ;
2006-06-18 07:37:28 +04:00
struct addr_req * req ;
int ret = 0 ;
some kmalloc/memset ->kzalloc (tree wide)
Transform some calls to kmalloc/memset to a single kzalloc (or kcalloc).
Here is a short excerpt of the semantic patch performing
this transformation:
@@
type T2;
expression x;
identifier f,fld;
expression E;
expression E1,E2;
expression e1,e2,e3,y;
statement S;
@@
x =
- kmalloc
+ kzalloc
(E1,E2)
... when != \(x->fld=E;\|y=f(...,x,...);\|f(...,x,...);\|x=E;\|while(...) S\|for(e1;e2;e3) S\)
- memset((T2)x,0,E1);
@@
expression E1,E2,E3;
@@
- kzalloc(E1 * E2,E3)
+ kcalloc(E1,E2,E3)
[akpm@linux-foundation.org: get kcalloc args the right way around]
Signed-off-by: Yoann Padioleau <padator@wanadoo.fr>
Cc: Richard Henderson <rth@twiddle.net>
Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
Acked-by: Russell King <rmk@arm.linux.org.uk>
Cc: Bryan Wu <bryan.wu@analog.com>
Acked-by: Jiri Slaby <jirislaby@gmail.com>
Cc: Dave Airlie <airlied@linux.ie>
Acked-by: Roland Dreier <rolandd@cisco.com>
Cc: Jiri Kosina <jkosina@suse.cz>
Acked-by: Dmitry Torokhov <dtor@mail.ru>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Acked-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Acked-by: Pierre Ossman <drzeus-list@drzeus.cx>
Cc: Jeff Garzik <jeff@garzik.org>
Cc: "David S. Miller" <davem@davemloft.net>
Acked-by: Greg KH <greg@kroah.com>
Cc: James Bottomley <James.Bottomley@steeleye.com>
Cc: "Antonino A. Daplas" <adaplas@pol.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-07-19 12:49:03 +04:00
req = kzalloc ( sizeof * req , GFP_KERNEL ) ;
2006-06-18 07:37:28 +04:00
if ( ! req )
return - ENOMEM ;
if ( src_addr )
memcpy ( & req - > src_addr , src_addr , ip_addr_size ( src_addr ) ) ;
memcpy ( & req - > dst_addr , dst_addr , ip_addr_size ( dst_addr ) ) ;
req - > addr = addr ;
req - > callback = callback ;
req - > context = context ;
2006-10-31 22:12:59 +03:00
req - > client = client ;
atomic_inc ( & client - > refcount ) ;
2006-06-18 07:37:28 +04:00
2008-12-24 21:16:37 +03:00
src_in = ( struct sockaddr * ) & req - > src_addr ;
dst_in = ( struct sockaddr * ) & req - > dst_addr ;
2006-06-18 07:37:28 +04:00
req - > status = addr_resolve_local ( src_in , dst_in , addr ) ;
if ( req - > status = = - EADDRNOTAVAIL )
req - > status = addr_resolve_remote ( src_in , dst_in , addr ) ;
switch ( req - > status ) {
case 0 :
req - > timeout = jiffies ;
queue_req ( req ) ;
break ;
case - ENODATA :
req - > timeout = msecs_to_jiffies ( timeout_ms ) + jiffies ;
queue_req ( req ) ;
addr_send_arp ( dst_in ) ;
break ;
default :
ret = req - > status ;
2006-10-31 22:12:59 +03:00
atomic_dec ( & client - > refcount ) ;
2006-06-18 07:37:28 +04:00
kfree ( req ) ;
break ;
}
return ret ;
}
EXPORT_SYMBOL ( rdma_resolve_ip ) ;
void rdma_addr_cancel ( struct rdma_dev_addr * addr )
{
struct addr_req * req , * temp_req ;
mutex_lock ( & lock ) ;
list_for_each_entry_safe ( req , temp_req , & req_list , list ) {
if ( req - > addr = = addr ) {
req - > status = - ECANCELED ;
req - > timeout = jiffies ;
2006-11-30 02:33:09 +03:00
list_move ( & req - > list , & req_list ) ;
2006-06-18 07:37:28 +04:00
set_timeout ( req - > timeout ) ;
break ;
}
}
mutex_unlock ( & lock ) ;
}
EXPORT_SYMBOL ( rdma_addr_cancel ) ;
2006-09-23 02:22:46 +04:00
static int netevent_callback ( struct notifier_block * self , unsigned long event ,
2006-07-31 07:44:19 +04:00
void * ctx )
2006-06-18 07:37:28 +04:00
{
2006-09-23 02:22:46 +04:00
if ( event = = NETEVENT_NEIGH_UPDATE ) {
2006-07-31 07:44:19 +04:00
struct neighbour * neigh = ctx ;
2006-06-18 07:37:28 +04:00
2007-01-24 04:03:17 +03:00
if ( neigh - > nud_state & NUD_VALID ) {
2006-07-31 07:44:19 +04:00
set_timeout ( jiffies ) ;
}
}
2006-06-18 07:37:28 +04:00
return 0 ;
}
2006-07-31 07:44:19 +04:00
static struct notifier_block nb = {
. notifier_call = netevent_callback
2006-06-18 07:37:28 +04:00
} ;
static int addr_init ( void )
{
2007-02-01 23:23:37 +03:00
addr_wq = create_singlethread_workqueue ( " ib_addr " ) ;
2006-06-18 07:37:28 +04:00
if ( ! addr_wq )
return - ENOMEM ;
2006-07-31 07:44:19 +04:00
register_netevent_notifier ( & nb ) ;
2006-06-18 07:37:28 +04:00
return 0 ;
}
static void addr_cleanup ( void )
{
2006-07-31 07:44:19 +04:00
unregister_netevent_notifier ( & nb ) ;
2006-06-18 07:37:28 +04:00
destroy_workqueue ( addr_wq ) ;
}
module_init ( addr_init ) ;
module_exit ( addr_cleanup ) ;