2011-08-12 13:20:12 -04:00
|
|
|
/* Copyright (c) 2010, 2011
|
2010-10-11 14:59:26 -04:00
|
|
|
* The Regents of the University of Michigan
|
|
|
|
|
* All Rights Reserved
|
2011-08-12 13:20:12 -04:00
|
|
|
*
|
|
|
|
|
* Olga Kornievskaia <aglo@umich.edu>
|
|
|
|
|
* Casey Bodley <cbodley@umich.edu>
|
2010-10-11 14:59:26 -04:00
|
|
|
*
|
|
|
|
|
* Permission is granted to use, copy and redistribute this software
|
|
|
|
|
* for noncommercial education and research purposes, so long as no
|
|
|
|
|
* fee is charged, and so long as the name of the University of Michigan
|
|
|
|
|
* is not used in any advertising or publicity pertaining to the use
|
|
|
|
|
* or distribution of this software without specific, written prior
|
|
|
|
|
* authorization. Permission to modify or otherwise create derivative
|
|
|
|
|
* works of this software is not granted.
|
|
|
|
|
*
|
|
|
|
|
* This software is provided as is, without representation or warranty
|
|
|
|
|
* of any kind either express or implied, including without limitation
|
|
|
|
|
* the implied warranties of merchantability, fitness for a particular
|
|
|
|
|
* purpose, or noninfringement. The Regents of the University of
|
|
|
|
|
* Michigan shall not be liable for any damages, including special,
|
|
|
|
|
* indirect, incidental, or consequential damages, with respect to any
|
|
|
|
|
* claim arising out of or in connection with the use of the software,
|
|
|
|
|
* even if it has been or is hereafter advised of the possibility of
|
|
|
|
|
* such damages.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <Windows.h>
|
|
|
|
|
#include <stdio.h>
|
2011-09-22 14:56:11 -04:00
|
|
|
#include <strsafe.h>
|
2010-10-11 14:59:26 -04:00
|
|
|
#include "from_kernel.h"
|
2011-09-22 14:56:11 -04:00
|
|
|
#include "delegation.h"
|
2010-10-11 14:59:26 -04:00
|
|
|
#include "daemon_debug.h"
|
|
|
|
|
#include "nfs41_ops.h"
|
|
|
|
|
#include "name_cache.h"
|
|
|
|
|
#include "upcall.h"
|
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int nfs41_cached_getattr(
|
|
|
|
|
IN nfs41_session *session,
|
|
|
|
|
IN nfs41_path_fh *file,
|
|
|
|
|
OUT nfs41_file_info *info)
|
|
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
|
|
|
|
|
/* first look for cached attributes */
|
|
|
|
|
status = nfs41_attr_cache_lookup(session_name_cache(session),
|
|
|
|
|
file->fh.fileid, info);
|
|
|
|
|
|
|
|
|
|
if (status) {
|
|
|
|
|
/* fetch attributes from the server */
|
|
|
|
|
bitmap4 attr_request;
|
|
|
|
|
init_getattr_request(&attr_request);
|
|
|
|
|
|
|
|
|
|
status = nfs41_getattr(session, file, &attr_request, info);
|
|
|
|
|
if (status) {
|
|
|
|
|
eprintf("nfs41_getattr() failed with %s\n",
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_BAD_NET_RESP);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-27 09:22:20 -04:00
|
|
|
/* NFS41_FILE_QUERY */
|
|
|
|
|
static int parse_getattr(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall)
|
2010-10-11 14:59:26 -04:00
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
getattr_upcall_args *args = &upcall->args.getattr;
|
|
|
|
|
|
|
|
|
|
status = safe_read(&buffer, &length, &args->query_class, sizeof(args->query_class));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_read(&buffer, &length, &args->buf_len, sizeof(args->buf_len));
|
|
|
|
|
if (status) goto out;
|
2010-10-12 10:03:03 -04:00
|
|
|
|
2011-11-07 10:50:00 -05:00
|
|
|
dprintf(1, "parsing NFS41_FILE_QUERY: info_class=%d buf_len=%d file=%.*s\n",
|
|
|
|
|
args->query_class, args->buf_len, upcall->state_ref->path.len,
|
|
|
|
|
upcall->state_ref->path.path);
|
2010-10-11 14:59:26 -04:00
|
|
|
out:
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-27 09:22:20 -04:00
|
|
|
static int handle_getattr(nfs41_upcall *upcall)
|
2010-10-11 14:59:26 -04:00
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
getattr_upcall_args *args = &upcall->args.getattr;
|
2011-04-13 20:07:37 -04:00
|
|
|
nfs41_open_state *state = upcall->state_ref;
|
2011-08-24 17:43:20 -04:00
|
|
|
nfs41_file_info info = { 0 };
|
2010-10-11 14:59:26 -04:00
|
|
|
|
|
|
|
|
status = nfs41_cached_getattr(state->session, &state->file, &info);
|
|
|
|
|
if (status) {
|
|
|
|
|
eprintf("nfs41_cached_getattr() failed with %d\n", status);
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-14 15:20:45 -04:00
|
|
|
if (info.type == NF4LNK) {
|
|
|
|
|
nfs41_file_info target_info;
|
2011-04-13 20:07:37 -04:00
|
|
|
int target_status = nfs41_symlink_follow(upcall->root_ref,
|
2010-10-14 15:20:45 -04:00
|
|
|
state->session, &state->file, &target_info);
|
|
|
|
|
if (target_status == NO_ERROR && target_info.type == NF4DIR)
|
|
|
|
|
info.symlink_dir = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-11 14:59:26 -04:00
|
|
|
switch (args->query_class) {
|
|
|
|
|
case FileBasicInformation:
|
|
|
|
|
nfs_to_basic_info(&info, &args->basic_info);
|
2012-02-06 14:23:47 -05:00
|
|
|
args->ctime = info.change;
|
2010-10-11 14:59:26 -04:00
|
|
|
break;
|
|
|
|
|
case FileStandardInformation:
|
|
|
|
|
nfs_to_standard_info(&info, &args->std_info);
|
|
|
|
|
break;
|
|
|
|
|
case FileAttributeTagInformation:
|
|
|
|
|
args->tag_info.FileAttributes = nfs_file_info_to_attributes(&info);
|
2010-09-23 12:17:04 -04:00
|
|
|
args->tag_info.ReparseTag = info.type == NF4LNK ?
|
|
|
|
|
IO_REPARSE_TAG_SYMLINK : 0;
|
2010-10-11 14:59:26 -04:00
|
|
|
break;
|
2011-03-30 14:21:40 -04:00
|
|
|
case FileInternalInformation:
|
|
|
|
|
args->intr_info.IndexNumber.QuadPart = info.fileid;
|
|
|
|
|
break;
|
2010-10-11 14:59:26 -04:00
|
|
|
default:
|
|
|
|
|
eprintf("unhandled file query class %d\n", args->query_class);
|
|
|
|
|
status = ERROR_INVALID_PARAMETER;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
out:
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-27 09:22:20 -04:00
|
|
|
static int marshall_getattr(unsigned char *buffer, uint32_t *length, nfs41_upcall *upcall)
|
2010-10-11 14:59:26 -04:00
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
getattr_upcall_args *args = &upcall->args.getattr;
|
|
|
|
|
uint32_t info_len;
|
|
|
|
|
|
|
|
|
|
switch (args->query_class) {
|
|
|
|
|
case FileBasicInformation:
|
|
|
|
|
info_len = sizeof(args->basic_info);
|
|
|
|
|
status = safe_write(&buffer, length, &info_len, sizeof(info_len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_write(&buffer, length, &args->basic_info, info_len);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
break;
|
|
|
|
|
case FileStandardInformation:
|
|
|
|
|
info_len = sizeof(args->std_info);
|
|
|
|
|
status = safe_write(&buffer, length, &info_len, sizeof(info_len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_write(&buffer, length, &args->std_info, info_len);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
break;
|
|
|
|
|
case FileAttributeTagInformation:
|
|
|
|
|
info_len = sizeof(args->tag_info);
|
|
|
|
|
status = safe_write(&buffer, length, &info_len, sizeof(info_len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_write(&buffer, length, &args->tag_info, info_len);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
break;
|
2011-03-30 14:21:40 -04:00
|
|
|
case FileInternalInformation:
|
|
|
|
|
info_len = sizeof(args->intr_info);
|
|
|
|
|
status = safe_write(&buffer, length, &info_len, sizeof(info_len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_write(&buffer, length, &args->intr_info, info_len);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
break;
|
2010-10-11 14:59:26 -04:00
|
|
|
default:
|
|
|
|
|
eprintf("unknown file query class %d\n", args->query_class);
|
|
|
|
|
status = 103;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
2012-02-06 14:23:47 -05:00
|
|
|
status = safe_write(&buffer, length, &args->ctime, sizeof(args->ctime));
|
|
|
|
|
if (status) goto out;
|
2012-02-13 16:08:21 -05:00
|
|
|
dprintf(1, "NFS41_FILE_QUERY: downcall changattr=%llu\n", args->ctime);
|
2010-10-11 14:59:26 -04:00
|
|
|
out:
|
|
|
|
|
return status;
|
|
|
|
|
}
|
2010-10-27 09:22:20 -04:00
|
|
|
|
|
|
|
|
|
2011-09-22 14:56:11 -04:00
|
|
|
static int parse_getexattr(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall)
|
|
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
getexattr_upcall_args *args = &upcall->args.getexattr;
|
|
|
|
|
|
|
|
|
|
status = get_name(&buffer, &length, &args->path);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_read(&buffer, &length, &args->eaindex, sizeof(args->eaindex));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_read(&buffer, &length, &args->restart, sizeof(args->restart));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_read(&buffer, &length, &args->single, sizeof(args->single));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_read(&buffer, &length, &args->ealist_len, sizeof(args->ealist_len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
args->ealist = buffer;
|
|
|
|
|
|
|
|
|
|
dprintf(1, "parsing NFS41_EA_QUERY: buf_len=%d Initial %d Restart %d "
|
|
|
|
|
"Single %d\n", args->buf_len,args->eaindex, args->restart, args->single);
|
|
|
|
|
out:
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int marshall_getexattr(unsigned char *buffer, uint32_t *length, nfs41_upcall *upcall)
|
|
|
|
|
{
|
|
|
|
|
int status = NO_ERROR;
|
|
|
|
|
getexattr_upcall_args *args = &upcall->args.getexattr;
|
|
|
|
|
uint32_t len = args->buf_len;
|
|
|
|
|
|
|
|
|
|
status = safe_write(&buffer, length, &len, sizeof(len));
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
status = safe_write(&buffer, length, args->buf, len);
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
out:
|
|
|
|
|
free(args->buf);
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int handle_getexattr(nfs41_upcall *upcall)
|
|
|
|
|
{
|
|
|
|
|
int status = 0;
|
|
|
|
|
getexattr_upcall_args *args = &upcall->args.getexattr;
|
|
|
|
|
PFILE_GET_EA_INFORMATION gea =
|
|
|
|
|
(PFILE_GET_EA_INFORMATION)args->ealist, prev = NULL;
|
|
|
|
|
PFILE_FULL_EA_INFORMATION eainfo, entry_pos;
|
|
|
|
|
unsigned char *entry_buf, buf[NFS4_EASIZE] = { 0 };
|
|
|
|
|
nfs41_open_state *state = upcall->state_ref;
|
|
|
|
|
nfs41_path_fh parent, file;
|
|
|
|
|
open_claim4 claim;
|
|
|
|
|
stateid4 open_stateid;
|
|
|
|
|
stateid_arg stateid;
|
|
|
|
|
nfs41_component dst_name;
|
|
|
|
|
open_delegation4 delegation = { 0 };
|
|
|
|
|
bool_t eof;
|
|
|
|
|
uint32_t bytes_read = 0;
|
|
|
|
|
ULONG buflen = 0, needed = 0;
|
|
|
|
|
|
|
|
|
|
status = nfs41_rpc_openattr(state->session, &state->file, FALSE, &parent.fh);
|
|
|
|
|
if (status){
|
|
|
|
|
dprintf(1, "nfs41_rpc_openattr() failed with error %s.\n",
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_NOT_SUPPORTED);
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
entry_buf = malloc(UPCALL_BUF_SIZE);
|
|
|
|
|
if (entry_buf == NULL) {
|
|
|
|
|
status = GetLastError();
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
entry_pos = eainfo = (PFILE_FULL_EA_INFORMATION)entry_buf;
|
|
|
|
|
|
|
|
|
|
while (gea != prev) {
|
|
|
|
|
dst_name.name = gea->EaName;
|
|
|
|
|
dst_name.len = gea->EaNameLength;
|
|
|
|
|
claim.claim = CLAIM_NULL;
|
|
|
|
|
claim.u.null.filename = &dst_name;
|
|
|
|
|
status = nfs41_open(state->session, &parent, &file, &state->owner,
|
|
|
|
|
&claim, OPEN4_SHARE_ACCESS_READ, OPEN4_SHARE_DENY_BOTH,
|
|
|
|
|
OPEN4_NOCREATE, UNCHECKED4, 0, TRUE, &open_stateid,
|
|
|
|
|
&delegation, NULL);
|
|
|
|
|
if (status) {
|
|
|
|
|
dprintf(1, "nfs41_open() failed with error %s.\n",
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_NOT_SUPPORTED);
|
|
|
|
|
goto out_free;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
stateid.stateid = open_stateid;
|
|
|
|
|
stateid.stateid.seqid = 0;
|
|
|
|
|
status = nfs41_read(state->session, &file, &stateid, 0, NFS4_EASIZE,
|
|
|
|
|
buf, &bytes_read, &eof);
|
|
|
|
|
if (status) {
|
|
|
|
|
dprintf(2, "nfs41_rpc_read EA attribute failed\n");
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_NET_WRITE_FAULT);
|
|
|
|
|
nfs41_close(state->session, &file, &stateid);
|
|
|
|
|
goto out_free;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (eof) {
|
|
|
|
|
dprintf(1, "read thread reached eof: bytes_read %d\n", bytes_read);
|
|
|
|
|
eainfo->EaNameLength = gea->EaNameLength;
|
|
|
|
|
if (FAILED(StringCchCopy((LPSTR)eainfo->EaName, gea->EaNameLength + 1,
|
|
|
|
|
(LPCSTR)gea->EaName))) {
|
|
|
|
|
status = ERROR_BUFFER_OVERFLOW;
|
|
|
|
|
nfs41_close(state->session, &file, &stateid);
|
|
|
|
|
goto out_free;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (FAILED(StringCchCopy((LPSTR)eainfo->EaName +
|
|
|
|
|
eainfo->EaNameLength + 1, bytes_read + 1, (LPCSTR)buf))) {
|
|
|
|
|
status = ERROR_BUFFER_OVERFLOW;
|
|
|
|
|
nfs41_close(state->session, &file, &stateid);
|
|
|
|
|
goto out_free;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memset(buf, 0, NFS4_EASIZE);
|
|
|
|
|
eainfo->EaValueLength = (USHORT) bytes_read;
|
|
|
|
|
needed = (eainfo->EaNameLength + eainfo->EaValueLength) +
|
|
|
|
|
FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName);
|
|
|
|
|
|
|
|
|
|
if (needed % 4)
|
|
|
|
|
needed = needed + (4 - (needed % 4));
|
|
|
|
|
|
|
|
|
|
eainfo->NextEntryOffset = needed;
|
|
|
|
|
eainfo->Flags = 0;
|
|
|
|
|
|
|
|
|
|
buflen = buflen + needed;
|
|
|
|
|
prev = gea;
|
|
|
|
|
|
|
|
|
|
if (gea->NextEntryOffset != 0) {
|
|
|
|
|
gea = (PFILE_GET_EA_INFORMATION)
|
|
|
|
|
((PBYTE) gea + gea->NextEntryOffset);
|
|
|
|
|
eainfo = (PFILE_FULL_EA_INFORMATION)
|
|
|
|
|
((PBYTE) eainfo + eainfo->NextEntryOffset);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
status = nfs41_close(state->session, &file, &stateid);
|
|
|
|
|
if (status) {
|
|
|
|
|
dprintf(1, "nfs41_close() failed with error %s.\n",
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_NOT_SUPPORTED);
|
|
|
|
|
goto out_free;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
dprintf(2, "Size of the EA value is greater than %d\n", NFS4_EASIZE);
|
|
|
|
|
status = nfs41_close(state->session, &file, &stateid);
|
|
|
|
|
if (status) {
|
|
|
|
|
dprintf(1, "nfs41_rpc_openattr() failed with error %s.\n",
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
status = nfs_to_windows_error(status, ERROR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
/* treating extended attribute values larger than NFS4_EASIZE as failure */
|
|
|
|
|
status = ERROR_INVALID_DATA;
|
|
|
|
|
goto out_free;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
eainfo->NextEntryOffset = 0;
|
|
|
|
|
args->buf = (unsigned char *)entry_pos;
|
|
|
|
|
args->buf_len = buflen;
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
out_free:
|
|
|
|
|
free(entry_buf);
|
|
|
|
|
out:
|
|
|
|
|
return status;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-10-27 09:22:20 -04:00
|
|
|
const nfs41_upcall_op nfs41_op_getattr = {
|
|
|
|
|
parse_getattr,
|
|
|
|
|
handle_getattr,
|
|
|
|
|
marshall_getattr
|
|
|
|
|
};
|
2011-09-22 14:56:11 -04:00
|
|
|
|
|
|
|
|
const nfs41_upcall_op nfs41_op_getexattr = {
|
|
|
|
|
parse_getexattr,
|
|
|
|
|
handle_getexattr,
|
|
|
|
|
marshall_getexattr
|
|
|
|
|
};
|