/* NFSv4.1 client for Windows * Copyright © 2012 The Regents of the University of Michigan * * Olga Kornievskaia * Casey Bodley * * This library is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2.1 of the License, or (at * your option) any later version. * * This library is distributed in the hope that it will be useful, but * without any warranty; without even the implied warranty of merchantability * or fitness for a particular purpose. See the GNU Lesser General Public * License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this library; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA */ #include #include #include #include "daemon_debug.h" #include "nfs41_ops.h" #include "upcall.h" #include "util.h" /* NFS41_MOUNT */ static int parse_mount(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall) { int status; mount_upcall_args *args = &upcall->args.mount; status = get_name(&buffer, &length, &args->hostname); if(status) goto out; status = get_name(&buffer, &length, &args->path); if(status) goto out; status = safe_read(&buffer, &length, &args->sec_flavor, sizeof(DWORD)); if (status) goto out; status = safe_read(&buffer, &length, &args->rsize, sizeof(DWORD)); if (status) goto out; status = safe_read(&buffer, &length, &args->wsize, sizeof(DWORD)); if (status) goto out; dprintf(1, "parsing NFS14_MOUNT: srv_name=%s root=%s sec_flavor=%s " "rsize=%d wsize=%d\n", args->hostname, args->path, secflavorop2name(args->sec_flavor), args->rsize, args->wsize); out: return status; } static int handle_mount(nfs41_upcall *upcall) { int status; mount_upcall_args *args = &upcall->args.mount; nfs41_abs_path path; multi_addr4 addrs; nfs41_root *root; nfs41_client *client; nfs41_path_fh file; // resolve hostname,port status = nfs41_server_resolve(args->hostname, 2049, &addrs); if (status) { eprintf("nfs41_server_resolve() failed with %d\n", status); goto out; } if (upcall->root_ref != INVALID_HANDLE_VALUE) { /* use an existing root from a previous mount, but don't take an * extra reference; we'll only get one UNMOUNT upcall for each root */ root = upcall->root_ref; } else { // create root status = nfs41_root_create(args->hostname, args->sec_flavor, args->wsize + WRITE_OVERHEAD, args->rsize + READ_OVERHEAD, &root); if (status) { eprintf("nfs41_root_create() failed %d\n", status); goto out; } root->uid = upcall->uid; root->gid = upcall->gid; } // find or create the client/session status = nfs41_root_mount_addrs(root, &addrs, 0, 0, &client); if (status) { eprintf("nfs41_root_mount_addrs() failed with %d\n", status); goto out_err; } // make a copy of the path for nfs41_lookup() InitializeSRWLock(&path.lock); if (FAILED(StringCchCopyA(path.path, NFS41_MAX_PATH_LEN, args->path))) { status = ERROR_FILENAME_EXCED_RANGE; goto out_err; } path.len = (unsigned short)strlen(path.path); // look up the mount path, and fail if it doesn't exist status = nfs41_lookup(root, client->session, &path, NULL, &file, NULL, NULL); if (status) { eprintf("nfs41_lookup('%s') failed with %d\n", path.path, status); status = ERROR_BAD_NETPATH; goto out_err; } nfs41_superblock_fs_attributes(file.fh.superblock, &args->FsAttrs); if (upcall->root_ref == INVALID_HANDLE_VALUE) nfs41_root_ref(root); upcall->root_ref = root; args->lease_time = client->session->lease_time; out: return status; out_err: if (upcall->root_ref == INVALID_HANDLE_VALUE) nfs41_root_deref(root); goto out; } static int marshall_mount(unsigned char *buffer, uint32_t *length, nfs41_upcall *upcall) { mount_upcall_args *args = &upcall->args.mount; int status; dprintf(2, "NFS41_MOUNT: writing pointer to nfs41_root %p, version %d, " "lease_time %d\n", upcall->root_ref, NFS41D_VERSION, args->lease_time); status = safe_write(&buffer, length, &upcall->root_ref, sizeof(HANDLE)); if (status) goto out; status = safe_write(&buffer, length, &NFS41D_VERSION, sizeof(DWORD)); if (status) goto out; status = safe_write(&buffer, length, &args->lease_time, sizeof(DWORD)); if (status) goto out; status = safe_write(&buffer, length, &args->FsAttrs, sizeof(args->FsAttrs)); out: return status; } static void cancel_mount(IN nfs41_upcall *upcall) { if (upcall->root_ref != INVALID_HANDLE_VALUE) nfs41_root_deref(upcall->root_ref); } const nfs41_upcall_op nfs41_op_mount = { parse_mount, handle_mount, marshall_mount, cancel_mount }; /* NFS41_UNMOUNT */ static int parse_unmount(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall) { dprintf(1, "parsing NFS41_UNMOUNT: root=%p\n", upcall->root_ref); return ERROR_SUCCESS; } static int handle_unmount(nfs41_upcall *upcall) { /* release the original reference from nfs41_root_create() */ nfs41_root_deref(upcall->root_ref); return ERROR_SUCCESS; } const nfs41_upcall_op nfs41_op_unmount = { parse_unmount, handle_unmount };