| /* 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/gfp.h> |
| #include <linux/init.h> |
| #include <linux/sched.h> |
| #include "afs_fs.h" |
| #include "internal.h" |
| |
| /* |
| * Deliver reply data to a VL.GetEntryByNameU call. |
| */ |
| static int afs_deliver_vl_get_entry_by_name_u(struct afs_call *call) |
| { |
| struct afs_uvldbentry__xdr *uvldb; |
| struct afs_vldb_entry *entry; |
| bool new_only = false; |
| u32 tmp, nr_servers; |
| int i, ret; |
| |
| _enter(""); |
| |
| ret = afs_transfer_reply(call); |
| if (ret < 0) |
| return ret; |
| |
| /* unmarshall the reply once we've received all of it */ |
| uvldb = call->buffer; |
| entry = call->reply[0]; |
| |
| nr_servers = ntohl(uvldb->nServers); |
| if (nr_servers > AFS_NMAXNSERVERS) |
| nr_servers = AFS_NMAXNSERVERS; |
| |
| for (i = 0; i < ARRAY_SIZE(uvldb->name) - 1; i++) |
| entry->name[i] = (u8)ntohl(uvldb->name[i]); |
| entry->name[i] = 0; |
| entry->name_len = strlen(entry->name); |
| |
| /* If there is a new replication site that we can use, ignore all the |
| * sites that aren't marked as new. |
| */ |
| for (i = 0; i < nr_servers; i++) { |
| tmp = ntohl(uvldb->serverFlags[i]); |
| if (!(tmp & AFS_VLSF_DONTUSE) && |
| (tmp & AFS_VLSF_NEWREPSITE)) |
| new_only = true; |
| } |
| |
| for (i = 0; i < nr_servers; i++) { |
| struct afs_uuid__xdr *xdr; |
| struct afs_uuid *uuid; |
| int j; |
| |
| tmp = ntohl(uvldb->serverFlags[i]); |
| if (tmp & AFS_VLSF_DONTUSE || |
| (new_only && !(tmp & AFS_VLSF_NEWREPSITE))) |
| continue; |
| if (tmp & AFS_VLSF_RWVOL) |
| entry->fs_mask[i] |= AFS_VOL_VTM_RW; |
| if (tmp & AFS_VLSF_ROVOL) |
| entry->fs_mask[i] |= AFS_VOL_VTM_RO; |
| if (tmp & AFS_VLSF_BACKVOL) |
| entry->fs_mask[i] |= AFS_VOL_VTM_BAK; |
| if (!entry->fs_mask[i]) |
| continue; |
| |
| xdr = &uvldb->serverNumber[i]; |
| uuid = (struct afs_uuid *)&entry->fs_server[i]; |
| uuid->time_low = xdr->time_low; |
| uuid->time_mid = htons(ntohl(xdr->time_mid)); |
| uuid->time_hi_and_version = htons(ntohl(xdr->time_hi_and_version)); |
| uuid->clock_seq_hi_and_reserved = (u8)ntohl(xdr->clock_seq_hi_and_reserved); |
| uuid->clock_seq_low = (u8)ntohl(xdr->clock_seq_low); |
| for (j = 0; j < 6; j++) |
| uuid->node[j] = (u8)ntohl(xdr->node[j]); |
| |
| entry->nr_servers++; |
| } |
| |
| for (i = 0; i < AFS_MAXTYPES; i++) |
| entry->vid[i] = ntohl(uvldb->volumeId[i]); |
| |
| tmp = ntohl(uvldb->flags); |
| if (tmp & AFS_VLF_RWEXISTS) |
| __set_bit(AFS_VLDB_HAS_RW, &entry->flags); |
| if (tmp & AFS_VLF_ROEXISTS) |
| __set_bit(AFS_VLDB_HAS_RO, &entry->flags); |
| if (tmp & AFS_VLF_BACKEXISTS) |
| __set_bit(AFS_VLDB_HAS_BAK, &entry->flags); |
| |
| if (!(tmp & (AFS_VLF_RWEXISTS | AFS_VLF_ROEXISTS | AFS_VLF_BACKEXISTS))) { |
| entry->error = -ENOMEDIUM; |
| __set_bit(AFS_VLDB_QUERY_ERROR, &entry->flags); |
| } |
| |
| __set_bit(AFS_VLDB_QUERY_VALID, &entry->flags); |
| _leave(" = 0 [done]"); |
| return 0; |
| } |
| |
| static void afs_destroy_vl_get_entry_by_name_u(struct afs_call *call) |
| { |
| kfree(call->reply[0]); |
| afs_flat_call_destructor(call); |
| } |
| |
| /* |
| * VL.GetEntryByNameU operation type. |
| */ |
| static const struct afs_call_type afs_RXVLGetEntryByNameU = { |
| .name = "VL.GetEntryByNameU", |
| .op = afs_VL_GetEntryByNameU, |
| .deliver = afs_deliver_vl_get_entry_by_name_u, |
| .destructor = afs_destroy_vl_get_entry_by_name_u, |
| }; |
| |
| /* |
| * Dispatch a get volume entry by name or ID operation (uuid variant). If the |
| * volname is a decimal number then it's a volume ID not a volume name. |
| */ |
| struct afs_vldb_entry *afs_vl_get_entry_by_name_u(struct afs_net *net, |
| struct afs_addr_cursor *ac, |
| struct key *key, |
| const char *volname, |
| int volnamesz) |
| { |
| struct afs_vldb_entry *entry; |
| struct afs_call *call; |
| size_t reqsz, padsz; |
| __be32 *bp; |
| |
| _enter(""); |
| |
| padsz = (4 - (volnamesz & 3)) & 3; |
| reqsz = 8 + volnamesz + padsz; |
| |
| entry = kzalloc(sizeof(struct afs_vldb_entry), GFP_KERNEL); |
| if (!entry) |
| return ERR_PTR(-ENOMEM); |
| |
| call = afs_alloc_flat_call(net, &afs_RXVLGetEntryByNameU, reqsz, |
| sizeof(struct afs_uvldbentry__xdr)); |
| if (!call) { |
| kfree(entry); |
| return ERR_PTR(-ENOMEM); |
| } |
| |
| call->key = key; |
| call->reply[0] = entry; |
| call->ret_reply0 = true; |
| |
| /* Marshall the parameters */ |
| bp = call->request; |
| *bp++ = htonl(VLGETENTRYBYNAMEU); |
| *bp++ = htonl(volnamesz); |
| memcpy(bp, volname, volnamesz); |
| if (padsz > 0) |
| memset((void *)bp + volnamesz, 0, padsz); |
| |
| trace_afs_make_vl_call(call); |
| return (struct afs_vldb_entry *)afs_make_call(ac, call, GFP_KERNEL, false); |
| } |
| |
| /* |
| * Deliver reply data to a VL.GetAddrsU call. |
| * |
| * GetAddrsU(IN ListAddrByAttributes *inaddr, |
| * OUT afsUUID *uuidp1, |
| * OUT uint32_t *uniquifier, |
| * OUT uint32_t *nentries, |
| * OUT bulkaddrs *blkaddrs); |
| */ |
| static int afs_deliver_vl_get_addrs_u(struct afs_call *call) |
| { |
| struct afs_addr_list *alist; |
| __be32 *bp; |
| u32 uniquifier, nentries, count; |
| int i, ret; |
| |
| _enter("{%u,%zu/%u}", call->unmarshall, call->offset, call->count); |
| |
| again: |
| switch (call->unmarshall) { |
| case 0: |
| call->offset = 0; |
| call->unmarshall++; |
| |
| /* Extract the returned uuid, uniquifier, nentries and blkaddrs size */ |
| case 1: |
| ret = afs_extract_data(call, call->buffer, |
| sizeof(struct afs_uuid__xdr) + 3 * sizeof(__be32), |
| true); |
| if (ret < 0) |
| return ret; |
| |
| bp = call->buffer + sizeof(struct afs_uuid__xdr); |
| uniquifier = ntohl(*bp++); |
| nentries = ntohl(*bp++); |
| count = ntohl(*bp); |
| |
| nentries = min(nentries, count); |
| alist = afs_alloc_addrlist(nentries, FS_SERVICE, AFS_FS_PORT); |
| if (!alist) |
| return -ENOMEM; |
| alist->version = uniquifier; |
| call->reply[0] = alist; |
| call->count = count; |
| call->count2 = nentries; |
| call->offset = 0; |
| call->unmarshall++; |
| |
| /* Extract entries */ |
| case 2: |
| count = min(call->count, 4U); |
| ret = afs_extract_data(call, call->buffer, |
| count * sizeof(__be32), |
| call->count > 4); |
| if (ret < 0) |
| return ret; |
| |
| alist = call->reply[0]; |
| bp = call->buffer; |
| for (i = 0; i < count; i++) |
| if (alist->nr_addrs < call->count2) |
| afs_merge_fs_addr4(alist, *bp++, AFS_FS_PORT); |
| |
| call->count -= count; |
| if (call->count > 0) |
| goto again; |
| call->offset = 0; |
| call->unmarshall++; |
| break; |
| } |
| |
| _leave(" = 0 [done]"); |
| return 0; |
| } |
| |
| static void afs_vl_get_addrs_u_destructor(struct afs_call *call) |
| { |
| afs_put_server(call->net, (struct afs_server *)call->reply[0]); |
| kfree(call->reply[1]); |
| return afs_flat_call_destructor(call); |
| } |
| |
| /* |
| * VL.GetAddrsU operation type. |
| */ |
| static const struct afs_call_type afs_RXVLGetAddrsU = { |
| .name = "VL.GetAddrsU", |
| .op = afs_VL_GetAddrsU, |
| .deliver = afs_deliver_vl_get_addrs_u, |
| .destructor = afs_vl_get_addrs_u_destructor, |
| }; |
| |
| /* |
| * Dispatch an operation to get the addresses for a server, where the server is |
| * nominated by UUID. |
| */ |
| struct afs_addr_list *afs_vl_get_addrs_u(struct afs_net *net, |
| struct afs_addr_cursor *ac, |
| struct key *key, |
| const uuid_t *uuid) |
| { |
| struct afs_ListAddrByAttributes__xdr *r; |
| const struct afs_uuid *u = (const struct afs_uuid *)uuid; |
| struct afs_call *call; |
| __be32 *bp; |
| int i; |
| |
| _enter(""); |
| |
| call = afs_alloc_flat_call(net, &afs_RXVLGetAddrsU, |
| sizeof(__be32) + sizeof(struct afs_ListAddrByAttributes__xdr), |
| sizeof(struct afs_uuid__xdr) + 3 * sizeof(__be32)); |
| if (!call) |
| return ERR_PTR(-ENOMEM); |
| |
| call->key = key; |
| call->reply[0] = NULL; |
| call->ret_reply0 = true; |
| |
| /* Marshall the parameters */ |
| bp = call->request; |
| *bp++ = htonl(VLGETADDRSU); |
| r = (struct afs_ListAddrByAttributes__xdr *)bp; |
| r->Mask = htonl(AFS_VLADDR_UUID); |
| r->ipaddr = 0; |
| r->index = 0; |
| r->spare = 0; |
| r->uuid.time_low = u->time_low; |
| r->uuid.time_mid = htonl(ntohs(u->time_mid)); |
| r->uuid.time_hi_and_version = htonl(ntohs(u->time_hi_and_version)); |
| r->uuid.clock_seq_hi_and_reserved = htonl(u->clock_seq_hi_and_reserved); |
| r->uuid.clock_seq_low = htonl(u->clock_seq_low); |
| for (i = 0; i < 6; i++) |
| r->uuid.node[i] = ntohl(u->node[i]); |
| |
| trace_afs_make_vl_call(call); |
| return (struct afs_addr_list *)afs_make_call(ac, call, GFP_KERNEL, false); |
| } |
| |
| /* |
| * Deliver reply data to an VL.GetCapabilities operation. |
| */ |
| static int afs_deliver_vl_get_capabilities(struct afs_call *call) |
| { |
| u32 count; |
| int ret; |
| |
| _enter("{%u,%zu/%u}", call->unmarshall, call->offset, call->count); |
| |
| again: |
| switch (call->unmarshall) { |
| case 0: |
| call->offset = 0; |
| call->unmarshall++; |
| |
| /* Extract the capabilities word count */ |
| case 1: |
| ret = afs_extract_data(call, &call->tmp, |
| 1 * sizeof(__be32), |
| true); |
| if (ret < 0) |
| return ret; |
| |
| count = ntohl(call->tmp); |
| |
| call->count = count; |
| call->count2 = count; |
| call->offset = 0; |
| call->unmarshall++; |
| |
| /* Extract capabilities words */ |
| case 2: |
| count = min(call->count, 16U); |
| ret = afs_extract_data(call, call->buffer, |
| count * sizeof(__be32), |
| call->count > 16); |
| if (ret < 0) |
| return ret; |
| |
| /* TODO: Examine capabilities */ |
| |
| call->count -= count; |
| if (call->count > 0) |
| goto again; |
| call->offset = 0; |
| call->unmarshall++; |
| break; |
| } |
| |
| call->reply[0] = (void *)(unsigned long)call->service_id; |
| |
| _leave(" = 0 [done]"); |
| return 0; |
| } |
| |
| /* |
| * VL.GetCapabilities operation type |
| */ |
| static const struct afs_call_type afs_RXVLGetCapabilities = { |
| .name = "VL.GetCapabilities", |
| .op = afs_VL_GetCapabilities, |
| .deliver = afs_deliver_vl_get_capabilities, |
| .destructor = afs_flat_call_destructor, |
| }; |
| |
| /* |
| * Probe a fileserver for the capabilities that it supports. This can |
| * return up to 196 words. |
| * |
| * We use this to probe for service upgrade to determine what the server at the |
| * other end supports. |
| */ |
| int afs_vl_get_capabilities(struct afs_net *net, |
| struct afs_addr_cursor *ac, |
| struct key *key) |
| { |
| struct afs_call *call; |
| __be32 *bp; |
| |
| _enter(""); |
| |
| call = afs_alloc_flat_call(net, &afs_RXVLGetCapabilities, 1 * 4, 16 * 4); |
| if (!call) |
| return -ENOMEM; |
| |
| call->key = key; |
| call->upgrade = true; /* Let's see if this is a YFS server */ |
| call->reply[0] = (void *)VLGETCAPABILITIES; |
| call->ret_reply0 = true; |
| |
| /* marshall the parameters */ |
| bp = call->request; |
| *bp++ = htonl(VLGETCAPABILITIES); |
| |
| /* Can't take a ref on server */ |
| trace_afs_make_vl_call(call); |
| return afs_make_call(ac, call, GFP_KERNEL, false); |
| } |
| |
| /* |
| * Deliver reply data to a YFSVL.GetEndpoints call. |
| * |
| * GetEndpoints(IN yfsServerAttributes *attr, |
| * OUT opr_uuid *uuid, |
| * OUT afs_int32 *uniquifier, |
| * OUT endpoints *fsEndpoints, |
| * OUT endpoints *volEndpoints) |
| */ |
| static int afs_deliver_yfsvl_get_endpoints(struct afs_call *call) |
| { |
| struct afs_addr_list *alist; |
| __be32 *bp; |
| u32 uniquifier, size; |
| int ret; |
| |
| _enter("{%u,%zu/%u,%u}", call->unmarshall, call->offset, call->count, call->count2); |
| |
| again: |
| switch (call->unmarshall) { |
| case 0: |
| call->offset = 0; |
| call->unmarshall = 1; |
| |
| /* Extract the returned uuid, uniquifier, fsEndpoints count and |
| * either the first fsEndpoint type or the volEndpoints |
| * count if there are no fsEndpoints. */ |
| case 1: |
| ret = afs_extract_data(call, call->buffer, |
| sizeof(uuid_t) + |
| 3 * sizeof(__be32), |
| true); |
| if (ret < 0) |
| return ret; |
| |
| bp = call->buffer + sizeof(uuid_t); |
| uniquifier = ntohl(*bp++); |
| call->count = ntohl(*bp++); |
| call->count2 = ntohl(*bp); /* Type or next count */ |
| |
| if (call->count > YFS_MAXENDPOINTS) |
| return -EBADMSG; |
| |
| alist = afs_alloc_addrlist(call->count, FS_SERVICE, AFS_FS_PORT); |
| if (!alist) |
| return -ENOMEM; |
| alist->version = uniquifier; |
| call->reply[0] = alist; |
| call->offset = 0; |
| |
| if (call->count == 0) |
| goto extract_volendpoints; |
| |
| call->unmarshall = 2; |
| |
| /* Extract fsEndpoints[] entries */ |
| case 2: |
| switch (call->count2) { |
| case YFS_ENDPOINT_IPV4: |
| size = sizeof(__be32) * (1 + 1 + 1); |
| break; |
| case YFS_ENDPOINT_IPV6: |
| size = sizeof(__be32) * (1 + 4 + 1); |
| break; |
| default: |
| return -EBADMSG; |
| } |
| |
| size += sizeof(__be32); |
| ret = afs_extract_data(call, call->buffer, size, true); |
| if (ret < 0) |
| return ret; |
| |
| alist = call->reply[0]; |
| bp = call->buffer; |
| switch (call->count2) { |
| case YFS_ENDPOINT_IPV4: |
| if (ntohl(bp[0]) != sizeof(__be32) * 2) |
| return -EBADMSG; |
| afs_merge_fs_addr4(alist, bp[1], ntohl(bp[2])); |
| bp += 3; |
| break; |
| case YFS_ENDPOINT_IPV6: |
| if (ntohl(bp[0]) != sizeof(__be32) * 5) |
| return -EBADMSG; |
| afs_merge_fs_addr6(alist, bp + 1, ntohl(bp[5])); |
| bp += 6; |
| break; |
| default: |
| return -EBADMSG; |
| } |
| |
| /* Got either the type of the next entry or the count of |
| * volEndpoints if no more fsEndpoints. |
| */ |
| call->count2 = htonl(*bp++); |
| |
| call->offset = 0; |
| call->count--; |
| if (call->count > 0) |
| goto again; |
| |
| extract_volendpoints: |
| /* Extract the list of volEndpoints. */ |
| call->count = call->count2; |
| if (!call->count) |
| goto end; |
| if (call->count > YFS_MAXENDPOINTS) |
| return -EBADMSG; |
| |
| call->unmarshall = 3; |
| |
| /* Extract the type of volEndpoints[0]. Normally we would |
| * extract the type of the next endpoint when we extract the |
| * data of the current one, but this is the first... |
| */ |
| case 3: |
| ret = afs_extract_data(call, call->buffer, sizeof(__be32), true); |
| if (ret < 0) |
| return ret; |
| |
| bp = call->buffer; |
| call->count2 = htonl(*bp++); |
| call->offset = 0; |
| call->unmarshall = 4; |
| |
| /* Extract volEndpoints[] entries */ |
| case 4: |
| switch (call->count2) { |
| case YFS_ENDPOINT_IPV4: |
| size = sizeof(__be32) * (1 + 1 + 1); |
| break; |
| case YFS_ENDPOINT_IPV6: |
| size = sizeof(__be32) * (1 + 4 + 1); |
| break; |
| default: |
| return -EBADMSG; |
| } |
| |
| if (call->count > 1) |
| size += sizeof(__be32); |
| ret = afs_extract_data(call, call->buffer, size, true); |
| if (ret < 0) |
| return ret; |
| |
| bp = call->buffer; |
| switch (call->count2) { |
| case YFS_ENDPOINT_IPV4: |
| if (ntohl(bp[0]) != sizeof(__be32) * 2) |
| return -EBADMSG; |
| bp += 3; |
| break; |
| case YFS_ENDPOINT_IPV6: |
| if (ntohl(bp[0]) != sizeof(__be32) * 5) |
| return -EBADMSG; |
| bp += 6; |
| break; |
| default: |
| return -EBADMSG; |
| } |
| |
| /* Got either the type of the next entry or the count of |
| * volEndpoints if no more fsEndpoints. |
| */ |
| call->offset = 0; |
| call->count--; |
| if (call->count > 0) { |
| call->count2 = htonl(*bp++); |
| goto again; |
| } |
| |
| end: |
| call->unmarshall = 5; |
| |
| /* Done */ |
| case 5: |
| ret = afs_extract_data(call, call->buffer, 0, false); |
| if (ret < 0) |
| return ret; |
| call->unmarshall = 6; |
| |
| case 6: |
| break; |
| } |
| |
| alist = call->reply[0]; |
| |
| /* Start with IPv6 if available. */ |
| if (alist->nr_ipv4 < alist->nr_addrs) |
| alist->index = alist->nr_ipv4; |
| |
| _leave(" = 0 [done]"); |
| return 0; |
| } |
| |
| /* |
| * YFSVL.GetEndpoints operation type. |
| */ |
| static const struct afs_call_type afs_YFSVLGetEndpoints = { |
| .name = "YFSVL.GetEndpoints", |
| .op = afs_YFSVL_GetEndpoints, |
| .deliver = afs_deliver_yfsvl_get_endpoints, |
| .destructor = afs_vl_get_addrs_u_destructor, |
| }; |
| |
| /* |
| * Dispatch an operation to get the addresses for a server, where the server is |
| * nominated by UUID. |
| */ |
| struct afs_addr_list *afs_yfsvl_get_endpoints(struct afs_net *net, |
| struct afs_addr_cursor *ac, |
| struct key *key, |
| const uuid_t *uuid) |
| { |
| struct afs_call *call; |
| __be32 *bp; |
| |
| _enter(""); |
| |
| call = afs_alloc_flat_call(net, &afs_YFSVLGetEndpoints, |
| sizeof(__be32) * 2 + sizeof(*uuid), |
| sizeof(struct in6_addr) + sizeof(__be32) * 3); |
| if (!call) |
| return ERR_PTR(-ENOMEM); |
| |
| call->key = key; |
| call->reply[0] = NULL; |
| call->ret_reply0 = true; |
| |
| /* Marshall the parameters */ |
| bp = call->request; |
| *bp++ = htonl(YVLGETENDPOINTS); |
| *bp++ = htonl(YFS_SERVER_UUID); |
| memcpy(bp, uuid, sizeof(*uuid)); /* Type opr_uuid */ |
| |
| trace_afs_make_vl_call(call); |
| return (struct afs_addr_list *)afs_make_call(ac, call, GFP_KERNEL, false); |
| } |