mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 05:20:54 +00:00
176 lines
5.5 KiB
C
176 lines
5.5 KiB
C
/* NFSv4.1 client for Windows
|
|
* Copyright © 2012 The Regents of the University of Michigan
|
|
*
|
|
* Olga Kornievskaia <aglo@umich.edu>
|
|
* Casey Bodley <cbodley@umich.edu>
|
|
*
|
|
* 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 <windows.h>
|
|
#include <strsafe.h>
|
|
#include <stdio.h>
|
|
|
|
#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
|
|
};
|