2007-04-26 15:49:28 -07:00
/* AFS Cache Manager Service
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 2002 Red Hat , Inc . All Rights Reserved .
* Written by David Howells ( dhowells @ redhat . com )
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/sched.h>
2007-04-26 15:55:03 -07:00
# include <linux/ip.h>
2005-04-16 15:20:36 -07:00
# include "internal.h"
2007-04-26 15:55:03 -07:00
# include "afs_cm.h"
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
struct workqueue_struct * afs_cm_workqueue ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
static int afs_deliver_cb_init_call_back_state ( struct afs_call * ,
struct sk_buff * , bool ) ;
2007-04-26 15:58:49 -07:00
static int afs_deliver_cb_init_call_back_state3 ( struct afs_call * ,
struct sk_buff * , bool ) ;
2007-04-26 15:55:03 -07:00
static int afs_deliver_cb_probe ( struct afs_call * , struct sk_buff * , bool ) ;
static int afs_deliver_cb_callback ( struct afs_call * , struct sk_buff * , bool ) ;
2007-04-26 15:58:17 -07:00
static int afs_deliver_cb_get_capabilities ( struct afs_call * , struct sk_buff * ,
bool ) ;
2007-04-26 15:55:03 -07:00
static void afs_cm_destructor ( struct afs_call * ) ;
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* CB . CallBack operation type
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static const struct afs_call_type afs_SRXCBCallBack = {
2007-04-26 15:57:07 -07:00
. name = " CB.CallBack " ,
2007-04-26 15:55:03 -07:00
. deliver = afs_deliver_cb_callback ,
. abort_to_error = afs_abort_to_error ,
. destructor = afs_cm_destructor ,
} ;
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* CB . InitCallBackState operation type
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static const struct afs_call_type afs_SRXCBInitCallBackState = {
2007-04-26 15:57:07 -07:00
. name = " CB.InitCallBackState " ,
2007-04-26 15:55:03 -07:00
. deliver = afs_deliver_cb_init_call_back_state ,
. abort_to_error = afs_abort_to_error ,
. destructor = afs_cm_destructor ,
} ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:58:49 -07:00
/*
* CB . InitCallBackState3 operation type
*/
static const struct afs_call_type afs_SRXCBInitCallBackState3 = {
. name = " CB.InitCallBackState3 " ,
. deliver = afs_deliver_cb_init_call_back_state3 ,
. abort_to_error = afs_abort_to_error ,
. destructor = afs_cm_destructor ,
} ;
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* CB . Probe operation type
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static const struct afs_call_type afs_SRXCBProbe = {
2007-04-26 15:57:07 -07:00
. name = " CB.Probe " ,
2007-04-26 15:55:03 -07:00
. deliver = afs_deliver_cb_probe ,
. abort_to_error = afs_abort_to_error ,
. destructor = afs_cm_destructor ,
} ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:58:17 -07:00
/*
* CB . GetCapabilities operation type
*/
static const struct afs_call_type afs_SRXCBGetCapabilites = {
. name = " CB.GetCapabilities " ,
. deliver = afs_deliver_cb_get_capabilities ,
. abort_to_error = afs_abort_to_error ,
. destructor = afs_cm_destructor ,
} ;
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* route an incoming cache manager call
* - return T if supported , F if not
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
bool afs_cm_incoming_call ( struct afs_call * call )
2005-04-16 15:20:36 -07:00
{
2007-04-26 15:55:03 -07:00
u32 operation_id = ntohl ( call - > operation_ID ) ;
_enter ( " {CB.OP %u} " , operation_id ) ;
switch ( operation_id ) {
case CBCallBack :
call - > type = & afs_SRXCBCallBack ;
return true ;
case CBInitCallBackState :
call - > type = & afs_SRXCBInitCallBackState ;
return true ;
2007-04-26 15:58:49 -07:00
case CBInitCallBackState3 :
call - > type = & afs_SRXCBInitCallBackState3 ;
return true ;
2007-04-26 15:55:03 -07:00
case CBProbe :
call - > type = & afs_SRXCBProbe ;
return true ;
2007-04-26 15:58:17 -07:00
case CBGetCapabilities :
call - > type = & afs_SRXCBGetCapabilites ;
return true ;
2007-04-26 15:55:03 -07:00
default :
return false ;
2005-04-16 15:20:36 -07:00
}
2007-04-26 15:49:28 -07:00
}
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* clean up a cache manager call
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static void afs_cm_destructor ( struct afs_call * call )
2005-04-16 15:20:36 -07:00
{
2007-04-26 15:55:03 -07:00
_enter ( " " ) ;
afs_put_server ( call - > server ) ;
call - > server = NULL ;
kfree ( call - > buffer ) ;
call - > buffer = NULL ;
2007-04-26 15:49:28 -07:00
}
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* allow the fileserver to see if the cache manager is still alive
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static void SRXAFSCB_CallBack ( struct work_struct * work )
2005-04-16 15:20:36 -07:00
{
2007-04-26 15:55:03 -07:00
struct afs_call * call = container_of ( work , struct afs_call , work ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
_enter ( " " ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/* be sure to send the reply *before* attempting to spam the AFS server
* with FSFetchStatus requests on the vnodes with broken callbacks lest
* the AFS server get into a vicious cycle of trying to break further
* callbacks because it hadn ' t received completion of the CBCallBack op
* yet */
afs_send_empty_reply ( call ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
afs_break_callbacks ( call - > server , call - > count , call - > request ) ;
_leave ( " " ) ;
2007-04-26 15:49:28 -07:00
}
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* deliver request data to a CB . CallBack call
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static int afs_deliver_cb_callback ( struct afs_call * call , struct sk_buff * skb ,
bool last )
2005-04-16 15:20:36 -07:00
{
2007-04-26 15:55:03 -07:00
struct afs_callback * cb ;
struct afs_server * server ;
struct in_addr addr ;
__be32 * bp ;
u32 tmp ;
int ret , loop ;
_enter ( " {%u},{%u},%d " , call - > unmarshall , skb - > len , last ) ;
switch ( call - > unmarshall ) {
case 0 :
call - > offset = 0 ;
call - > unmarshall + + ;
/* extract the FID array and its count in two steps */
case 1 :
_debug ( " extract FID count " ) ;
ret = afs_extract_data ( call , skb , last , & call - > tmp , 4 ) ;
switch ( ret ) {
case 0 : break ;
case - EAGAIN : return 0 ;
default : return ret ;
2005-04-16 15:20:36 -07:00
}
2007-04-26 15:55:03 -07:00
call - > count = ntohl ( call - > tmp ) ;
_debug ( " FID count: %u " , call - > count ) ;
if ( call - > count > AFSCBMAX )
return - EBADMSG ;
call - > buffer = kmalloc ( call - > count * 3 * 4 , GFP_KERNEL ) ;
if ( ! call - > buffer )
return - ENOMEM ;
call - > offset = 0 ;
call - > unmarshall + + ;
case 2 :
_debug ( " extract FID array " ) ;
ret = afs_extract_data ( call , skb , last , call - > buffer ,
call - > count * 3 * 4 ) ;
switch ( ret ) {
case 0 : break ;
case - EAGAIN : return 0 ;
default : return ret ;
2005-04-16 15:20:36 -07:00
}
2007-04-26 15:55:03 -07:00
_debug ( " unmarshall FID array " ) ;
call - > request = kcalloc ( call - > count ,
sizeof ( struct afs_callback ) ,
GFP_KERNEL ) ;
if ( ! call - > request )
return - ENOMEM ;
cb = call - > request ;
bp = call - > buffer ;
for ( loop = call - > count ; loop > 0 ; loop - - , cb + + ) {
cb - > fid . vid = ntohl ( * bp + + ) ;
cb - > fid . vnode = ntohl ( * bp + + ) ;
cb - > fid . unique = ntohl ( * bp + + ) ;
cb - > type = AFSCM_CB_UNTYPED ;
2005-04-16 15:20:36 -07:00
}
2007-04-26 15:55:03 -07:00
call - > offset = 0 ;
call - > unmarshall + + ;
/* extract the callback array and its count in two steps */
case 3 :
_debug ( " extract CB count " ) ;
ret = afs_extract_data ( call , skb , last , & call - > tmp , 4 ) ;
switch ( ret ) {
case 0 : break ;
case - EAGAIN : return 0 ;
default : return ret ;
}
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
tmp = ntohl ( call - > tmp ) ;
_debug ( " CB count: %u " , tmp ) ;
if ( tmp ! = call - > count & & tmp ! = 0 )
return - EBADMSG ;
call - > offset = 0 ;
call - > unmarshall + + ;
if ( tmp = = 0 )
goto empty_cb_array ;
case 4 :
_debug ( " extract CB array " ) ;
ret = afs_extract_data ( call , skb , last , call - > request ,
call - > count * 3 * 4 ) ;
switch ( ret ) {
case 0 : break ;
case - EAGAIN : return 0 ;
default : return ret ;
2005-04-16 15:20:36 -07:00
}
2007-04-26 15:55:03 -07:00
_debug ( " unmarshall CB array " ) ;
cb = call - > request ;
bp = call - > buffer ;
for ( loop = call - > count ; loop > 0 ; loop - - , cb + + ) {
cb - > version = ntohl ( * bp + + ) ;
cb - > expiry = ntohl ( * bp + + ) ;
cb - > type = ntohl ( * bp + + ) ;
}
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
empty_cb_array :
call - > offset = 0 ;
call - > unmarshall + + ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
case 5 :
_debug ( " trailer " ) ;
if ( skb - > len ! = 0 )
return - EBADMSG ;
2005-04-16 15:20:36 -07:00
break ;
}
2007-04-26 15:55:03 -07:00
if ( ! last )
return 0 ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
call - > state = AFS_CALL_REPLYING ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/* we'll need the file server record as that tells us which set of
* vnodes to operate upon */
memcpy ( & addr , & ip_hdr ( skb ) - > saddr , 4 ) ;
server = afs_find_server ( & addr ) ;
if ( ! server )
return - ENOTCONN ;
call - > server = server ;
INIT_WORK ( & call - > work , SRXAFSCB_CallBack ) ;
schedule_work ( & call - > work ) ;
return 0 ;
2007-04-26 15:49:28 -07:00
}
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* allow the fileserver to request callback state ( re - ) initialisation
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static void SRXAFSCB_InitCallBackState ( struct work_struct * work )
2005-04-16 15:20:36 -07:00
{
2007-04-26 15:55:03 -07:00
struct afs_call * call = container_of ( work , struct afs_call , work ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
_enter ( " {%p} " , call - > server ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
afs_init_callback_state ( call - > server ) ;
afs_send_empty_reply ( call ) ;
_leave ( " " ) ;
2007-04-26 15:49:28 -07:00
}
2005-04-16 15:20:36 -07:00
/*
2007-04-26 15:55:03 -07:00
* deliver request data to a CB . InitCallBackState call
2005-04-16 15:20:36 -07:00
*/
2007-04-26 15:55:03 -07:00
static int afs_deliver_cb_init_call_back_state ( struct afs_call * call ,
struct sk_buff * skb ,
bool last )
2005-04-16 15:20:36 -07:00
{
struct afs_server * server ;
2007-04-26 15:55:03 -07:00
struct in_addr addr ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
_enter ( " ,{%u},%d " , skb - > len , last ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
if ( skb - > len > 0 )
return - EBADMSG ;
if ( ! last )
return 0 ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/* no unmarshalling required */
call - > state = AFS_CALL_REPLYING ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/* we'll need the file server record as that tells us which set of
* vnodes to operate upon */
memcpy ( & addr , & ip_hdr ( skb ) - > saddr , 4 ) ;
server = afs_find_server ( & addr ) ;
if ( ! server )
return - ENOTCONN ;
call - > server = server ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
INIT_WORK ( & call - > work , SRXAFSCB_InitCallBackState ) ;
schedule_work ( & call - > work ) ;
return 0 ;
}
2005-04-16 15:20:36 -07:00
2007-04-26 15:58:49 -07:00
/*
* deliver request data to a CB . InitCallBackState3 call
*/
static int afs_deliver_cb_init_call_back_state3 ( struct afs_call * call ,
struct sk_buff * skb ,
bool last )
{
struct afs_server * server ;
struct in_addr addr ;
_enter ( " ,{%u},%d " , skb - > len , last ) ;
if ( ! last )
return 0 ;
/* no unmarshalling required */
call - > state = AFS_CALL_REPLYING ;
/* we'll need the file server record as that tells us which set of
* vnodes to operate upon */
memcpy ( & addr , & ip_hdr ( skb ) - > saddr , 4 ) ;
server = afs_find_server ( & addr ) ;
if ( ! server )
return - ENOTCONN ;
call - > server = server ;
INIT_WORK ( & call - > work , SRXAFSCB_InitCallBackState ) ;
schedule_work ( & call - > work ) ;
return 0 ;
}
2007-04-26 15:55:03 -07:00
/*
* allow the fileserver to see if the cache manager is still alive
*/
static void SRXAFSCB_Probe ( struct work_struct * work )
{
struct afs_call * call = container_of ( work , struct afs_call , work ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
_enter ( " " ) ;
afs_send_empty_reply ( call ) ;
_leave ( " " ) ;
}
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/*
* deliver request data to a CB . Probe call
*/
static int afs_deliver_cb_probe ( struct afs_call * call , struct sk_buff * skb ,
bool last )
{
_enter ( " ,{%u},%d " , skb - > len , last ) ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
if ( skb - > len > 0 )
return - EBADMSG ;
if ( ! last )
return 0 ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
/* no unmarshalling required */
call - > state = AFS_CALL_REPLYING ;
2005-04-16 15:20:36 -07:00
2007-04-26 15:55:03 -07:00
INIT_WORK ( & call - > work , SRXAFSCB_Probe ) ;
schedule_work ( & call - > work ) ;
return 0 ;
2007-04-26 15:49:28 -07:00
}
2007-04-26 15:58:17 -07:00
/*
* allow the fileserver to ask about the cache manager ' s capabilities
*/
static void SRXAFSCB_GetCapabilities ( struct work_struct * work )
{
struct afs_interface * ifs ;
struct afs_call * call = container_of ( work , struct afs_call , work ) ;
int loop , nifs ;
struct {
struct /* InterfaceAddr */ {
__be32 nifs ;
__be32 uuid [ 11 ] ;
__be32 ifaddr [ 32 ] ;
__be32 netmask [ 32 ] ;
__be32 mtu [ 32 ] ;
} ia ;
struct /* Capabilities */ {
__be32 capcount ;
__be32 caps [ 1 ] ;
} cap ;
} reply ;
_enter ( " " ) ;
nifs = 0 ;
ifs = kcalloc ( 32 , sizeof ( * ifs ) , GFP_KERNEL ) ;
if ( ifs ) {
nifs = afs_get_ipv4_interfaces ( ifs , 32 , false ) ;
if ( nifs < 0 ) {
kfree ( ifs ) ;
ifs = NULL ;
nifs = 0 ;
}
}
memset ( & reply , 0 , sizeof ( reply ) ) ;
reply . ia . nifs = htonl ( nifs ) ;
reply . ia . uuid [ 0 ] = htonl ( afs_uuid . time_low ) ;
reply . ia . uuid [ 1 ] = htonl ( afs_uuid . time_mid ) ;
reply . ia . uuid [ 2 ] = htonl ( afs_uuid . time_hi_and_version ) ;
reply . ia . uuid [ 3 ] = htonl ( ( s8 ) afs_uuid . clock_seq_hi_and_reserved ) ;
reply . ia . uuid [ 4 ] = htonl ( ( s8 ) afs_uuid . clock_seq_low ) ;
for ( loop = 0 ; loop < 6 ; loop + + )
reply . ia . uuid [ loop + 5 ] = htonl ( ( s8 ) afs_uuid . node [ loop ] ) ;
if ( ifs ) {
for ( loop = 0 ; loop < nifs ; loop + + ) {
reply . ia . ifaddr [ loop ] = ifs [ loop ] . address . s_addr ;
reply . ia . netmask [ loop ] = ifs [ loop ] . netmask . s_addr ;
reply . ia . mtu [ loop ] = htonl ( ifs [ loop ] . mtu ) ;
}
2007-05-03 03:27:39 -07:00
kfree ( ifs ) ;
2007-04-26 15:58:17 -07:00
}
reply . cap . capcount = htonl ( 1 ) ;
reply . cap . caps [ 0 ] = htonl ( AFS_CAP_ERROR_TRANSLATION ) ;
afs_send_simple_reply ( call , & reply , sizeof ( reply ) ) ;
_leave ( " " ) ;
}
/*
* deliver request data to a CB . GetCapabilities call
*/
static int afs_deliver_cb_get_capabilities ( struct afs_call * call ,
struct sk_buff * skb , bool last )
{
_enter ( " ,{%u},%d " , skb - > len , last ) ;
if ( skb - > len > 0 )
return - EBADMSG ;
if ( ! last )
return 0 ;
/* no unmarshalling required */
call - > state = AFS_CALL_REPLYING ;
INIT_WORK ( & call - > work , SRXAFSCB_GetCapabilities ) ;
schedule_work ( & call - > work ) ;
return 0 ;
}