xref: /openbmc/linux/fs/afs/vlclient.c (revision 5f5bac82)
1 /* AFS Volume Location Service client
2  *
3  * Copyright (C) 2002 Red Hat, Inc. All Rights Reserved.
4  * Written by David Howells (dhowells@redhat.com)
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version
9  * 2 of the License, or (at your option) any later version.
10  */
11 
12 #include <linux/init.h>
13 #include <linux/sched.h>
14 #include "internal.h"
15 
16 /*
17  * map volume locator abort codes to error codes
18  */
19 static int afs_vl_abort_to_error(u32 abort_code)
20 {
21 	_enter("%u", abort_code);
22 
23 	switch (abort_code) {
24 	case AFSVL_IDEXIST:		return -EEXIST;
25 	case AFSVL_IO:			return -EREMOTEIO;
26 	case AFSVL_NAMEEXIST:		return -EEXIST;
27 	case AFSVL_CREATEFAIL:		return -EREMOTEIO;
28 	case AFSVL_NOENT:		return -ENOMEDIUM;
29 	case AFSVL_EMPTY:		return -ENOMEDIUM;
30 	case AFSVL_ENTDELETED:		return -ENOMEDIUM;
31 	case AFSVL_BADNAME:		return -EINVAL;
32 	case AFSVL_BADINDEX:		return -EINVAL;
33 	case AFSVL_BADVOLTYPE:		return -EINVAL;
34 	case AFSVL_BADSERVER:		return -EINVAL;
35 	case AFSVL_BADPARTITION:	return -EINVAL;
36 	case AFSVL_REPSFULL:		return -EFBIG;
37 	case AFSVL_NOREPSERVER:		return -ENOENT;
38 	case AFSVL_DUPREPSERVER:	return -EEXIST;
39 	case AFSVL_RWNOTFOUND:		return -ENOENT;
40 	case AFSVL_BADREFCOUNT:		return -EINVAL;
41 	case AFSVL_SIZEEXCEEDED:	return -EINVAL;
42 	case AFSVL_BADENTRY:		return -EINVAL;
43 	case AFSVL_BADVOLIDBUMP:	return -EINVAL;
44 	case AFSVL_IDALREADYHASHED:	return -EINVAL;
45 	case AFSVL_ENTRYLOCKED:		return -EBUSY;
46 	case AFSVL_BADVOLOPER:		return -EBADRQC;
47 	case AFSVL_BADRELLOCKTYPE:	return -EINVAL;
48 	case AFSVL_RERELEASE:		return -EREMOTEIO;
49 	case AFSVL_BADSERVERFLAG:	return -EINVAL;
50 	case AFSVL_PERM:		return -EACCES;
51 	case AFSVL_NOMEM:		return -EREMOTEIO;
52 	default:
53 		return afs_abort_to_error(abort_code);
54 	}
55 }
56 
57 /*
58  * deliver reply data to a VL.GetEntryByXXX call
59  */
60 static int afs_deliver_vl_get_entry_by_xxx(struct afs_call *call,
61 					   struct sk_buff *skb, bool last)
62 {
63 	struct afs_cache_vlocation *entry;
64 	__be32 *bp;
65 	u32 tmp;
66 	int loop;
67 
68 	_enter(",,%u", last);
69 
70 	afs_transfer_reply(call, skb);
71 	if (!last)
72 		return 0;
73 
74 	if (call->reply_size != call->reply_max)
75 		return -EBADMSG;
76 
77 	/* unmarshall the reply once we've received all of it */
78 	entry = call->reply;
79 	bp = call->buffer;
80 
81 	for (loop = 0; loop < 64; loop++)
82 		entry->name[loop] = ntohl(*bp++);
83 	entry->name[loop] = 0;
84 	bp++; /* final NUL */
85 
86 	bp++; /* type */
87 	entry->nservers = ntohl(*bp++);
88 
89 	for (loop = 0; loop < 8; loop++)
90 		entry->servers[loop].s_addr = *bp++;
91 
92 	bp += 8; /* partition IDs */
93 
94 	for (loop = 0; loop < 8; loop++) {
95 		tmp = ntohl(*bp++);
96 		entry->srvtmask[loop] = 0;
97 		if (tmp & AFS_VLSF_RWVOL)
98 			entry->srvtmask[loop] |= AFS_VOL_VTM_RW;
99 		if (tmp & AFS_VLSF_ROVOL)
100 			entry->srvtmask[loop] |= AFS_VOL_VTM_RO;
101 		if (tmp & AFS_VLSF_BACKVOL)
102 			entry->srvtmask[loop] |= AFS_VOL_VTM_BAK;
103 	}
104 
105 	entry->vid[0] = ntohl(*bp++);
106 	entry->vid[1] = ntohl(*bp++);
107 	entry->vid[2] = ntohl(*bp++);
108 
109 	bp++; /* clone ID */
110 
111 	tmp = ntohl(*bp++); /* flags */
112 	entry->vidmask = 0;
113 	if (tmp & AFS_VLF_RWEXISTS)
114 		entry->vidmask |= AFS_VOL_VTM_RW;
115 	if (tmp & AFS_VLF_ROEXISTS)
116 		entry->vidmask |= AFS_VOL_VTM_RO;
117 	if (tmp & AFS_VLF_BACKEXISTS)
118 		entry->vidmask |= AFS_VOL_VTM_BAK;
119 	if (!entry->vidmask)
120 		return -EBADMSG;
121 
122 	_leave(" = 0 [done]");
123 	return 0;
124 }
125 
126 /*
127  * VL.GetEntryByName operation type
128  */
129 static const struct afs_call_type afs_RXVLGetEntryByName = {
130 	.name		= "VL.GetEntryByName",
131 	.deliver	= afs_deliver_vl_get_entry_by_xxx,
132 	.abort_to_error	= afs_vl_abort_to_error,
133 	.destructor	= afs_flat_call_destructor,
134 };
135 
136 /*
137  * VL.GetEntryById operation type
138  */
139 static const struct afs_call_type afs_RXVLGetEntryById = {
140 	.name		= "VL.GetEntryById",
141 	.deliver	= afs_deliver_vl_get_entry_by_xxx,
142 	.abort_to_error	= afs_vl_abort_to_error,
143 	.destructor	= afs_flat_call_destructor,
144 };
145 
146 /*
147  * dispatch a get volume entry by name operation
148  */
149 int afs_vl_get_entry_by_name(struct in_addr *addr,
150 			     struct key *key,
151 			     const char *volname,
152 			     struct afs_cache_vlocation *entry,
153 			     const struct afs_wait_mode *wait_mode)
154 {
155 	struct afs_call *call;
156 	size_t volnamesz, reqsz, padsz;
157 	__be32 *bp;
158 
159 	_enter("");
160 
161 	volnamesz = strlen(volname);
162 	padsz = (4 - (volnamesz & 3)) & 3;
163 	reqsz = 8 + volnamesz + padsz;
164 
165 	call = afs_alloc_flat_call(&afs_RXVLGetEntryByName, reqsz, 384);
166 	if (!call)
167 		return -ENOMEM;
168 
169 	call->key = key;
170 	call->reply = entry;
171 	call->service_id = VL_SERVICE;
172 	call->port = htons(AFS_VL_PORT);
173 
174 	/* marshall the parameters */
175 	bp = call->request;
176 	*bp++ = htonl(VLGETENTRYBYNAME);
177 	*bp++ = htonl(volnamesz);
178 	memcpy(bp, volname, volnamesz);
179 	if (padsz > 0)
180 		memset((void *) bp + volnamesz, 0, padsz);
181 
182 	/* initiate the call */
183 	return afs_make_call(addr, call, GFP_KERNEL, wait_mode);
184 }
185 
186 /*
187  * dispatch a get volume entry by ID operation
188  */
189 int afs_vl_get_entry_by_id(struct in_addr *addr,
190 			   struct key *key,
191 			   afs_volid_t volid,
192 			   afs_voltype_t voltype,
193 			   struct afs_cache_vlocation *entry,
194 			   const struct afs_wait_mode *wait_mode)
195 {
196 	struct afs_call *call;
197 	__be32 *bp;
198 
199 	_enter("");
200 
201 	call = afs_alloc_flat_call(&afs_RXVLGetEntryById, 12, 384);
202 	if (!call)
203 		return -ENOMEM;
204 
205 	call->key = key;
206 	call->reply = entry;
207 	call->service_id = VL_SERVICE;
208 	call->port = htons(AFS_VL_PORT);
209 
210 	/* marshall the parameters */
211 	bp = call->request;
212 	*bp++ = htonl(VLGETENTRYBYID);
213 	*bp++ = htonl(volid);
214 	*bp   = htonl(voltype);
215 
216 	/* initiate the call */
217 	return afs_make_call(addr, call, GFP_KERNEL, wait_mode);
218 }
219