mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-31 20:42:39 +00:00 
			
		
		
		
	 1da177e4c3
			
		
	
	
		1da177e4c3
		
	
	
	
	
		
			
			Initial git repository build. I'm not bothering with the full history, even though we have it. We can create a separate "historical" git archive of that later if we want to, and in the meantime it's about 3.2GB when imported into git - space that would just make the early git days unnecessarily complicated, when we don't have a lot of good infrastructure for it. Let it rip!
		
			
				
	
	
		
			696 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			696 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* vlclient.c: AFS Volume Location Service client
 | |
|  *
 | |
|  * 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/init.h>
 | |
| #include <linux/sched.h>
 | |
| #include <rxrpc/rxrpc.h>
 | |
| #include <rxrpc/transport.h>
 | |
| #include <rxrpc/connection.h>
 | |
| #include <rxrpc/call.h>
 | |
| #include "server.h"
 | |
| #include "volume.h"
 | |
| #include "vlclient.h"
 | |
| #include "kafsasyncd.h"
 | |
| #include "kafstimod.h"
 | |
| #include "errors.h"
 | |
| #include "internal.h"
 | |
| 
 | |
| #define VLGETENTRYBYID		503	/* AFS Get Cache Entry By ID operation ID */
 | |
| #define VLGETENTRYBYNAME	504	/* AFS Get Cache Entry By Name operation ID */
 | |
| #define VLPROBE			514	/* AFS Probe Volume Location Service operation ID */
 | |
| 
 | |
| static void afs_rxvl_get_entry_by_id_attn(struct rxrpc_call *call);
 | |
| static void afs_rxvl_get_entry_by_id_error(struct rxrpc_call *call);
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * map afs VL abort codes to/from Linux error codes
 | |
|  * - called with call->lock held
 | |
|  */
 | |
| static void afs_rxvl_aemap(struct rxrpc_call *call)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	_enter("{%u,%u,%d}",
 | |
| 	       call->app_err_state, call->app_abort_code, call->app_errno);
 | |
| 
 | |
| 	switch (call->app_err_state) {
 | |
| 	case RXRPC_ESTATE_LOCAL_ABORT:
 | |
| 		call->app_abort_code = -call->app_errno;
 | |
| 		return;
 | |
| 
 | |
| 	case RXRPC_ESTATE_PEER_ABORT:
 | |
| 		switch (call->app_abort_code) {
 | |
| 		case AFSVL_IDEXIST:		err = -EEXIST;		break;
 | |
| 		case AFSVL_IO:			err = -EREMOTEIO;	break;
 | |
| 		case AFSVL_NAMEEXIST:		err = -EEXIST;		break;
 | |
| 		case AFSVL_CREATEFAIL:		err = -EREMOTEIO;	break;
 | |
| 		case AFSVL_NOENT:		err = -ENOMEDIUM;	break;
 | |
| 		case AFSVL_EMPTY:		err = -ENOMEDIUM;	break;
 | |
| 		case AFSVL_ENTDELETED:		err = -ENOMEDIUM;	break;
 | |
| 		case AFSVL_BADNAME:		err = -EINVAL;		break;
 | |
| 		case AFSVL_BADINDEX:		err = -EINVAL;		break;
 | |
| 		case AFSVL_BADVOLTYPE:		err = -EINVAL;		break;
 | |
| 		case AFSVL_BADSERVER:		err = -EINVAL;		break;
 | |
| 		case AFSVL_BADPARTITION:	err = -EINVAL;		break;
 | |
| 		case AFSVL_REPSFULL:		err = -EFBIG;		break;
 | |
| 		case AFSVL_NOREPSERVER:		err = -ENOENT;		break;
 | |
| 		case AFSVL_DUPREPSERVER:	err = -EEXIST;		break;
 | |
| 		case AFSVL_RWNOTFOUND:		err = -ENOENT;		break;
 | |
| 		case AFSVL_BADREFCOUNT:		err = -EINVAL;		break;
 | |
| 		case AFSVL_SIZEEXCEEDED:	err = -EINVAL;		break;
 | |
| 		case AFSVL_BADENTRY:		err = -EINVAL;		break;
 | |
| 		case AFSVL_BADVOLIDBUMP:	err = -EINVAL;		break;
 | |
| 		case AFSVL_IDALREADYHASHED:	err = -EINVAL;		break;
 | |
| 		case AFSVL_ENTRYLOCKED:		err = -EBUSY;		break;
 | |
| 		case AFSVL_BADVOLOPER:		err = -EBADRQC;		break;
 | |
| 		case AFSVL_BADRELLOCKTYPE:	err = -EINVAL;		break;
 | |
| 		case AFSVL_RERELEASE:		err = -EREMOTEIO;	break;
 | |
| 		case AFSVL_BADSERVERFLAG:	err = -EINVAL;		break;
 | |
| 		case AFSVL_PERM:		err = -EACCES;		break;
 | |
| 		case AFSVL_NOMEM:		err = -EREMOTEIO;	break;
 | |
| 		default:
 | |
| 			err = afs_abort_to_error(call->app_abort_code);
 | |
| 			break;
 | |
| 		}
 | |
| 		call->app_errno = err;
 | |
| 		return;
 | |
| 
 | |
| 	default:
 | |
| 		return;
 | |
| 	}
 | |
| } /* end afs_rxvl_aemap() */
 | |
| 
 | |
| #if 0
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * probe a volume location server to see if it is still alive -- unused
 | |
|  */
 | |
| static int afs_rxvl_probe(struct afs_server *server, int alloc_flags)
 | |
| {
 | |
| 	struct rxrpc_connection *conn;
 | |
| 	struct rxrpc_call *call;
 | |
| 	struct kvec piov[1];
 | |
| 	size_t sent;
 | |
| 	int ret;
 | |
| 	__be32 param[1];
 | |
| 
 | |
| 	DECLARE_WAITQUEUE(myself, current);
 | |
| 
 | |
| 	/* get hold of the vlserver connection */
 | |
| 	ret = afs_server_get_vlconn(server, &conn);
 | |
| 	if (ret < 0)
 | |
| 		goto out;
 | |
| 
 | |
| 	/* create a call through that connection */
 | |
| 	ret = rxrpc_create_call(conn, NULL, NULL, afs_rxvl_aemap, &call);
 | |
| 	if (ret < 0) {
 | |
| 		printk("kAFS: Unable to create call: %d\n", ret);
 | |
| 		goto out_put_conn;
 | |
| 	}
 | |
| 	call->app_opcode = VLPROBE;
 | |
| 
 | |
| 	/* we want to get event notifications from the call */
 | |
| 	add_wait_queue(&call->waitq, &myself);
 | |
| 
 | |
| 	/* marshall the parameters */
 | |
| 	param[0] = htonl(VLPROBE);
 | |
| 	piov[0].iov_len = sizeof(param);
 | |
| 	piov[0].iov_base = param;
 | |
| 
 | |
| 	/* send the parameters to the server */
 | |
| 	ret = rxrpc_call_write_data(call, 1, piov, RXRPC_LAST_PACKET,
 | |
| 				    alloc_flags, 0, &sent);
 | |
| 	if (ret < 0)
 | |
| 		goto abort;
 | |
| 
 | |
| 	/* wait for the reply to completely arrive */
 | |
| 	for (;;) {
 | |
| 		set_current_state(TASK_INTERRUPTIBLE);
 | |
| 		if (call->app_call_state != RXRPC_CSTATE_CLNT_RCV_REPLY ||
 | |
| 		    signal_pending(current))
 | |
| 			break;
 | |
| 		schedule();
 | |
| 	}
 | |
| 	set_current_state(TASK_RUNNING);
 | |
| 
 | |
| 	ret = -EINTR;
 | |
| 	if (signal_pending(current))
 | |
| 		goto abort;
 | |
| 
 | |
| 	switch (call->app_call_state) {
 | |
| 	case RXRPC_CSTATE_ERROR:
 | |
| 		ret = call->app_errno;
 | |
| 		goto out_unwait;
 | |
| 
 | |
| 	case RXRPC_CSTATE_CLNT_GOT_REPLY:
 | |
| 		ret = 0;
 | |
| 		goto out_unwait;
 | |
| 
 | |
| 	default:
 | |
| 		BUG();
 | |
| 	}
 | |
| 
 | |
|  abort:
 | |
| 	set_current_state(TASK_UNINTERRUPTIBLE);
 | |
| 	rxrpc_call_abort(call, ret);
 | |
| 	schedule();
 | |
| 
 | |
|  out_unwait:
 | |
| 	set_current_state(TASK_RUNNING);
 | |
| 	remove_wait_queue(&call->waitq, &myself);
 | |
| 	rxrpc_put_call(call);
 | |
|  out_put_conn:
 | |
| 	rxrpc_put_connection(conn);
 | |
|  out:
 | |
| 	return ret;
 | |
| 
 | |
| } /* end afs_rxvl_probe() */
 | |
| #endif
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * look up a volume location database entry by name
 | |
|  */
 | |
| int afs_rxvl_get_entry_by_name(struct afs_server *server,
 | |
| 			       const char *volname,
 | |
| 			       unsigned volnamesz,
 | |
| 			       struct afs_cache_vlocation *entry)
 | |
| {
 | |
| 	DECLARE_WAITQUEUE(myself, current);
 | |
| 
 | |
| 	struct rxrpc_connection *conn;
 | |
| 	struct rxrpc_call *call;
 | |
| 	struct kvec piov[3];
 | |
| 	unsigned tmp;
 | |
| 	size_t sent;
 | |
| 	int ret, loop;
 | |
| 	__be32 *bp, param[2], zero;
 | |
| 
 | |
| 	_enter(",%*.*s,%u,", volnamesz, volnamesz, volname, volnamesz);
 | |
| 
 | |
| 	memset(entry, 0, sizeof(*entry));
 | |
| 
 | |
| 	/* get hold of the vlserver connection */
 | |
| 	ret = afs_server_get_vlconn(server, &conn);
 | |
| 	if (ret < 0)
 | |
| 		goto out;
 | |
| 
 | |
| 	/* create a call through that connection */
 | |
| 	ret = rxrpc_create_call(conn, NULL, NULL, afs_rxvl_aemap, &call);
 | |
| 	if (ret < 0) {
 | |
| 		printk("kAFS: Unable to create call: %d\n", ret);
 | |
| 		goto out_put_conn;
 | |
| 	}
 | |
| 	call->app_opcode = VLGETENTRYBYNAME;
 | |
| 
 | |
| 	/* we want to get event notifications from the call */
 | |
| 	add_wait_queue(&call->waitq, &myself);
 | |
| 
 | |
| 	/* marshall the parameters */
 | |
| 	piov[1].iov_len = volnamesz;
 | |
| 	piov[1].iov_base = (char *) volname;
 | |
| 
 | |
| 	zero = 0;
 | |
| 	piov[2].iov_len = (4 - (piov[1].iov_len & 3)) & 3;
 | |
| 	piov[2].iov_base = &zero;
 | |
| 
 | |
| 	param[0] = htonl(VLGETENTRYBYNAME);
 | |
| 	param[1] = htonl(piov[1].iov_len);
 | |
| 
 | |
| 	piov[0].iov_len = sizeof(param);
 | |
| 	piov[0].iov_base = param;
 | |
| 
 | |
| 	/* send the parameters to the server */
 | |
| 	ret = rxrpc_call_write_data(call, 3, piov, RXRPC_LAST_PACKET, GFP_NOFS,
 | |
| 				    0, &sent);
 | |
| 	if (ret < 0)
 | |
| 		goto abort;
 | |
| 
 | |
| 	/* wait for the reply to completely arrive */
 | |
| 	bp = rxrpc_call_alloc_scratch(call, 384);
 | |
| 
 | |
| 	ret = rxrpc_call_read_data(call, bp, 384,
 | |
| 				   RXRPC_CALL_READ_BLOCK |
 | |
| 				   RXRPC_CALL_READ_ALL);
 | |
| 	if (ret < 0) {
 | |
| 		if (ret == -ECONNABORTED) {
 | |
| 			ret = call->app_errno;
 | |
| 			goto out_unwait;
 | |
| 		}
 | |
| 		goto abort;
 | |
| 	}
 | |
| 
 | |
| 	/* unmarshall the reply */
 | |
| 	for (loop = 0; loop < 64; loop++)
 | |
| 		entry->name[loop] = ntohl(*bp++);
 | |
| 	bp++; /* final NUL */
 | |
| 
 | |
| 	bp++; /* type */
 | |
| 	entry->nservers = ntohl(*bp++);
 | |
| 
 | |
| 	for (loop = 0; loop < 8; loop++)
 | |
| 		entry->servers[loop].s_addr = *bp++;
 | |
| 
 | |
| 	bp += 8; /* partition IDs */
 | |
| 
 | |
| 	for (loop = 0; loop < 8; loop++) {
 | |
| 		tmp = ntohl(*bp++);
 | |
| 		if (tmp & AFS_VLSF_RWVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_RW;
 | |
| 		if (tmp & AFS_VLSF_ROVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_RO;
 | |
| 		if (tmp & AFS_VLSF_BACKVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_BAK;
 | |
| 	}
 | |
| 
 | |
| 	entry->vid[0] = ntohl(*bp++);
 | |
| 	entry->vid[1] = ntohl(*bp++);
 | |
| 	entry->vid[2] = ntohl(*bp++);
 | |
| 
 | |
| 	bp++; /* clone ID */
 | |
| 
 | |
| 	tmp = ntohl(*bp++); /* flags */
 | |
| 	if (tmp & AFS_VLF_RWEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_RW;
 | |
| 	if (tmp & AFS_VLF_ROEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_RO;
 | |
| 	if (tmp & AFS_VLF_BACKEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_BAK;
 | |
| 
 | |
| 	ret = -ENOMEDIUM;
 | |
| 	if (!entry->vidmask)
 | |
| 		goto abort;
 | |
| 
 | |
| 	/* success */
 | |
| 	entry->rtime = get_seconds();
 | |
| 	ret = 0;
 | |
| 
 | |
|  out_unwait:
 | |
| 	set_current_state(TASK_RUNNING);
 | |
| 	remove_wait_queue(&call->waitq, &myself);
 | |
| 	rxrpc_put_call(call);
 | |
|  out_put_conn:
 | |
| 	rxrpc_put_connection(conn);
 | |
|  out:
 | |
| 	_leave(" = %d", ret);
 | |
| 	return ret;
 | |
| 
 | |
|  abort:
 | |
| 	set_current_state(TASK_UNINTERRUPTIBLE);
 | |
| 	rxrpc_call_abort(call, ret);
 | |
| 	schedule();
 | |
| 	goto out_unwait;
 | |
| } /* end afs_rxvl_get_entry_by_name() */
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * look up a volume location database entry by ID
 | |
|  */
 | |
| int afs_rxvl_get_entry_by_id(struct afs_server *server,
 | |
| 			     afs_volid_t volid,
 | |
| 			     afs_voltype_t voltype,
 | |
| 			     struct afs_cache_vlocation *entry)
 | |
| {
 | |
| 	DECLARE_WAITQUEUE(myself, current);
 | |
| 
 | |
| 	struct rxrpc_connection *conn;
 | |
| 	struct rxrpc_call *call;
 | |
| 	struct kvec piov[1];
 | |
| 	unsigned tmp;
 | |
| 	size_t sent;
 | |
| 	int ret, loop;
 | |
| 	__be32 *bp, param[3];
 | |
| 
 | |
| 	_enter(",%x,%d,", volid, voltype);
 | |
| 
 | |
| 	memset(entry, 0, sizeof(*entry));
 | |
| 
 | |
| 	/* get hold of the vlserver connection */
 | |
| 	ret = afs_server_get_vlconn(server, &conn);
 | |
| 	if (ret < 0)
 | |
| 		goto out;
 | |
| 
 | |
| 	/* create a call through that connection */
 | |
| 	ret = rxrpc_create_call(conn, NULL, NULL, afs_rxvl_aemap, &call);
 | |
| 	if (ret < 0) {
 | |
| 		printk("kAFS: Unable to create call: %d\n", ret);
 | |
| 		goto out_put_conn;
 | |
| 	}
 | |
| 	call->app_opcode = VLGETENTRYBYID;
 | |
| 
 | |
| 	/* we want to get event notifications from the call */
 | |
| 	add_wait_queue(&call->waitq, &myself);
 | |
| 
 | |
| 	/* marshall the parameters */
 | |
| 	param[0] = htonl(VLGETENTRYBYID);
 | |
| 	param[1] = htonl(volid);
 | |
| 	param[2] = htonl(voltype);
 | |
| 
 | |
| 	piov[0].iov_len = sizeof(param);
 | |
| 	piov[0].iov_base = param;
 | |
| 
 | |
| 	/* send the parameters to the server */
 | |
| 	ret = rxrpc_call_write_data(call, 1, piov, RXRPC_LAST_PACKET, GFP_NOFS,
 | |
| 				    0, &sent);
 | |
| 	if (ret < 0)
 | |
| 		goto abort;
 | |
| 
 | |
| 	/* wait for the reply to completely arrive */
 | |
| 	bp = rxrpc_call_alloc_scratch(call, 384);
 | |
| 
 | |
| 	ret = rxrpc_call_read_data(call, bp, 384,
 | |
| 				   RXRPC_CALL_READ_BLOCK |
 | |
| 				   RXRPC_CALL_READ_ALL);
 | |
| 	if (ret < 0) {
 | |
| 		if (ret == -ECONNABORTED) {
 | |
| 			ret = call->app_errno;
 | |
| 			goto out_unwait;
 | |
| 		}
 | |
| 		goto abort;
 | |
| 	}
 | |
| 
 | |
| 	/* unmarshall the reply */
 | |
| 	for (loop = 0; loop < 64; loop++)
 | |
| 		entry->name[loop] = ntohl(*bp++);
 | |
| 	bp++; /* final NUL */
 | |
| 
 | |
| 	bp++; /* type */
 | |
| 	entry->nservers = ntohl(*bp++);
 | |
| 
 | |
| 	for (loop = 0; loop < 8; loop++)
 | |
| 		entry->servers[loop].s_addr = *bp++;
 | |
| 
 | |
| 	bp += 8; /* partition IDs */
 | |
| 
 | |
| 	for (loop = 0; loop < 8; loop++) {
 | |
| 		tmp = ntohl(*bp++);
 | |
| 		if (tmp & AFS_VLSF_RWVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_RW;
 | |
| 		if (tmp & AFS_VLSF_ROVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_RO;
 | |
| 		if (tmp & AFS_VLSF_BACKVOL)
 | |
| 			entry->srvtmask[loop] |= AFS_VOL_VTM_BAK;
 | |
| 	}
 | |
| 
 | |
| 	entry->vid[0] = ntohl(*bp++);
 | |
| 	entry->vid[1] = ntohl(*bp++);
 | |
| 	entry->vid[2] = ntohl(*bp++);
 | |
| 
 | |
| 	bp++; /* clone ID */
 | |
| 
 | |
| 	tmp = ntohl(*bp++); /* flags */
 | |
| 	if (tmp & AFS_VLF_RWEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_RW;
 | |
| 	if (tmp & AFS_VLF_ROEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_RO;
 | |
| 	if (tmp & AFS_VLF_BACKEXISTS)
 | |
| 		entry->vidmask |= AFS_VOL_VTM_BAK;
 | |
| 
 | |
| 	ret = -ENOMEDIUM;
 | |
| 	if (!entry->vidmask)
 | |
| 		goto abort;
 | |
| 
 | |
| #if 0 /* TODO: remove */
 | |
| 	entry->nservers = 3;
 | |
| 	entry->servers[0].s_addr = htonl(0xac101249);
 | |
| 	entry->servers[1].s_addr = htonl(0xac101243);
 | |
| 	entry->servers[2].s_addr = htonl(0xac10125b /*0xac10125b*/);
 | |
| 
 | |
| 	entry->srvtmask[0] = AFS_VOL_VTM_RO;
 | |
| 	entry->srvtmask[1] = AFS_VOL_VTM_RO;
 | |
| 	entry->srvtmask[2] = AFS_VOL_VTM_RO | AFS_VOL_VTM_RW;
 | |
| #endif
 | |
| 
 | |
| 	/* success */
 | |
| 	entry->rtime = get_seconds();
 | |
| 	ret = 0;
 | |
| 
 | |
|  out_unwait:
 | |
| 	set_current_state(TASK_RUNNING);
 | |
| 	remove_wait_queue(&call->waitq, &myself);
 | |
| 	rxrpc_put_call(call);
 | |
|  out_put_conn:
 | |
| 	rxrpc_put_connection(conn);
 | |
|  out:
 | |
| 	_leave(" = %d", ret);
 | |
| 	return ret;
 | |
| 
 | |
|  abort:
 | |
| 	set_current_state(TASK_UNINTERRUPTIBLE);
 | |
| 	rxrpc_call_abort(call, ret);
 | |
| 	schedule();
 | |
| 	goto out_unwait;
 | |
| } /* end afs_rxvl_get_entry_by_id() */
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * look up a volume location database entry by ID asynchronously
 | |
|  */
 | |
| int afs_rxvl_get_entry_by_id_async(struct afs_async_op *op,
 | |
| 				   afs_volid_t volid,
 | |
| 				   afs_voltype_t voltype)
 | |
| {
 | |
| 	struct rxrpc_connection *conn;
 | |
| 	struct rxrpc_call *call;
 | |
| 	struct kvec piov[1];
 | |
| 	size_t sent;
 | |
| 	int ret;
 | |
| 	__be32 param[3];
 | |
| 
 | |
| 	_enter(",%x,%d,", volid, voltype);
 | |
| 
 | |
| 	/* get hold of the vlserver connection */
 | |
| 	ret = afs_server_get_vlconn(op->server, &conn);
 | |
| 	if (ret < 0) {
 | |
| 		_leave(" = %d", ret);
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	/* create a call through that connection */
 | |
| 	ret = rxrpc_create_call(conn,
 | |
| 				afs_rxvl_get_entry_by_id_attn,
 | |
| 				afs_rxvl_get_entry_by_id_error,
 | |
| 				afs_rxvl_aemap,
 | |
| 				&op->call);
 | |
| 	rxrpc_put_connection(conn);
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		printk("kAFS: Unable to create call: %d\n", ret);
 | |
| 		_leave(" = %d", ret);
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	op->call->app_opcode = VLGETENTRYBYID;
 | |
| 	op->call->app_user = op;
 | |
| 
 | |
| 	call = op->call;
 | |
| 	rxrpc_get_call(call);
 | |
| 
 | |
| 	/* send event notifications from the call to kafsasyncd */
 | |
| 	afs_kafsasyncd_begin_op(op);
 | |
| 
 | |
| 	/* marshall the parameters */
 | |
| 	param[0] = htonl(VLGETENTRYBYID);
 | |
| 	param[1] = htonl(volid);
 | |
| 	param[2] = htonl(voltype);
 | |
| 
 | |
| 	piov[0].iov_len = sizeof(param);
 | |
| 	piov[0].iov_base = param;
 | |
| 
 | |
| 	/* allocate result read buffer in scratch space */
 | |
| 	call->app_scr_ptr = rxrpc_call_alloc_scratch(op->call, 384);
 | |
| 
 | |
| 	/* send the parameters to the server */
 | |
| 	ret = rxrpc_call_write_data(call, 1, piov, RXRPC_LAST_PACKET, GFP_NOFS,
 | |
| 				    0, &sent);
 | |
| 	if (ret < 0) {
 | |
| 		rxrpc_call_abort(call, ret); /* handle from kafsasyncd */
 | |
| 		ret = 0;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	/* wait for the reply to completely arrive */
 | |
| 	ret = rxrpc_call_read_data(call, call->app_scr_ptr, 384, 0);
 | |
| 	switch (ret) {
 | |
| 	case 0:
 | |
| 	case -EAGAIN:
 | |
| 	case -ECONNABORTED:
 | |
| 		ret = 0;
 | |
| 		break;	/* all handled by kafsasyncd */
 | |
| 
 | |
| 	default:
 | |
| 		rxrpc_call_abort(call, ret); /* make kafsasyncd handle it */
 | |
| 		ret = 0;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
|  out:
 | |
| 	rxrpc_put_call(call);
 | |
| 	_leave(" = %d", ret);
 | |
| 	return ret;
 | |
| 
 | |
| } /* end afs_rxvl_get_entry_by_id_async() */
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * attend to the asynchronous get VLDB entry by ID
 | |
|  */
 | |
| int afs_rxvl_get_entry_by_id_async2(struct afs_async_op *op,
 | |
| 				    struct afs_cache_vlocation *entry)
 | |
| {
 | |
| 	__be32 *bp;
 | |
| 	__u32 tmp;
 | |
| 	int loop, ret;
 | |
| 
 | |
| 	_enter("{op=%p cst=%u}", op, op->call->app_call_state);
 | |
| 
 | |
| 	memset(entry, 0, sizeof(*entry));
 | |
| 
 | |
| 	if (op->call->app_call_state == RXRPC_CSTATE_COMPLETE) {
 | |
| 		/* operation finished */
 | |
| 		afs_kafsasyncd_terminate_op(op);
 | |
| 
 | |
| 		bp = op->call->app_scr_ptr;
 | |
| 
 | |
| 		/* unmarshall the reply */
 | |
| 		for (loop = 0; loop < 64; loop++)
 | |
| 			entry->name[loop] = ntohl(*bp++);
 | |
| 		bp++; /* final NUL */
 | |
| 
 | |
| 		bp++; /* type */
 | |
| 		entry->nservers = ntohl(*bp++);
 | |
| 
 | |
| 		for (loop = 0; loop < 8; loop++)
 | |
| 			entry->servers[loop].s_addr = *bp++;
 | |
| 
 | |
| 		bp += 8; /* partition IDs */
 | |
| 
 | |
| 		for (loop = 0; loop < 8; loop++) {
 | |
| 			tmp = ntohl(*bp++);
 | |
| 			if (tmp & AFS_VLSF_RWVOL)
 | |
| 				entry->srvtmask[loop] |= AFS_VOL_VTM_RW;
 | |
| 			if (tmp & AFS_VLSF_ROVOL)
 | |
| 				entry->srvtmask[loop] |= AFS_VOL_VTM_RO;
 | |
| 			if (tmp & AFS_VLSF_BACKVOL)
 | |
| 				entry->srvtmask[loop] |= AFS_VOL_VTM_BAK;
 | |
| 		}
 | |
| 
 | |
| 		entry->vid[0] = ntohl(*bp++);
 | |
| 		entry->vid[1] = ntohl(*bp++);
 | |
| 		entry->vid[2] = ntohl(*bp++);
 | |
| 
 | |
| 		bp++; /* clone ID */
 | |
| 
 | |
| 		tmp = ntohl(*bp++); /* flags */
 | |
| 		if (tmp & AFS_VLF_RWEXISTS)
 | |
| 			entry->vidmask |= AFS_VOL_VTM_RW;
 | |
| 		if (tmp & AFS_VLF_ROEXISTS)
 | |
| 			entry->vidmask |= AFS_VOL_VTM_RO;
 | |
| 		if (tmp & AFS_VLF_BACKEXISTS)
 | |
| 			entry->vidmask |= AFS_VOL_VTM_BAK;
 | |
| 
 | |
| 		ret = -ENOMEDIUM;
 | |
| 		if (!entry->vidmask) {
 | |
| 			rxrpc_call_abort(op->call, ret);
 | |
| 			goto done;
 | |
| 		}
 | |
| 
 | |
| #if 0 /* TODO: remove */
 | |
| 		entry->nservers = 3;
 | |
| 		entry->servers[0].s_addr = htonl(0xac101249);
 | |
| 		entry->servers[1].s_addr = htonl(0xac101243);
 | |
| 		entry->servers[2].s_addr = htonl(0xac10125b /*0xac10125b*/);
 | |
| 
 | |
| 		entry->srvtmask[0] = AFS_VOL_VTM_RO;
 | |
| 		entry->srvtmask[1] = AFS_VOL_VTM_RO;
 | |
| 		entry->srvtmask[2] = AFS_VOL_VTM_RO | AFS_VOL_VTM_RW;
 | |
| #endif
 | |
| 
 | |
| 		/* success */
 | |
| 		entry->rtime = get_seconds();
 | |
| 		ret = 0;
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	if (op->call->app_call_state == RXRPC_CSTATE_ERROR) {
 | |
| 		/* operation error */
 | |
| 		ret = op->call->app_errno;
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	_leave(" = -EAGAIN");
 | |
| 	return -EAGAIN;
 | |
| 
 | |
|  done:
 | |
| 	rxrpc_put_call(op->call);
 | |
| 	op->call = NULL;
 | |
| 	_leave(" = %d", ret);
 | |
| 	return ret;
 | |
| } /* end afs_rxvl_get_entry_by_id_async2() */
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * handle attention events on an async get-entry-by-ID op
 | |
|  * - called from krxiod
 | |
|  */
 | |
| static void afs_rxvl_get_entry_by_id_attn(struct rxrpc_call *call)
 | |
| {
 | |
| 	struct afs_async_op *op = call->app_user;
 | |
| 
 | |
| 	_enter("{op=%p cst=%u}", op, call->app_call_state);
 | |
| 
 | |
| 	switch (call->app_call_state) {
 | |
| 	case RXRPC_CSTATE_COMPLETE:
 | |
| 		afs_kafsasyncd_attend_op(op);
 | |
| 		break;
 | |
| 	case RXRPC_CSTATE_CLNT_RCV_REPLY:
 | |
| 		if (call->app_async_read)
 | |
| 			break;
 | |
| 	case RXRPC_CSTATE_CLNT_GOT_REPLY:
 | |
| 		if (call->app_read_count == 0)
 | |
| 			break;
 | |
| 		printk("kAFS: Reply bigger than expected"
 | |
| 		       " {cst=%u asyn=%d mark=%Zu rdy=%Zu pr=%u%s}",
 | |
| 		       call->app_call_state,
 | |
| 		       call->app_async_read,
 | |
| 		       call->app_mark,
 | |
| 		       call->app_ready_qty,
 | |
| 		       call->pkt_rcv_count,
 | |
| 		       call->app_last_rcv ? " last" : "");
 | |
| 
 | |
| 		rxrpc_call_abort(call, -EBADMSG);
 | |
| 		break;
 | |
| 	default:
 | |
| 		BUG();
 | |
| 	}
 | |
| 
 | |
| 	_leave("");
 | |
| 
 | |
| } /* end afs_rxvl_get_entry_by_id_attn() */
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
|  * handle error events on an async get-entry-by-ID op
 | |
|  * - called from krxiod
 | |
|  */
 | |
| static void afs_rxvl_get_entry_by_id_error(struct rxrpc_call *call)
 | |
| {
 | |
| 	struct afs_async_op *op = call->app_user;
 | |
| 
 | |
| 	_enter("{op=%p cst=%u}", op, call->app_call_state);
 | |
| 
 | |
| 	afs_kafsasyncd_attend_op(op);
 | |
| 
 | |
| 	_leave("");
 | |
| 
 | |
| } /* end afs_rxvl_get_entry_by_id_error() */
 |