2012-04-06 14:26:13 -04:00
|
|
|
|
/* NFSv4.1 client for Windows
|
|
|
|
|
|
* Copyright <EFBFBD> 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 <stdio.h>
|
|
|
|
|
|
#include <strsafe.h>
|
|
|
|
|
|
|
|
|
|
|
|
#include "from_kernel.h"
|
|
|
|
|
|
#include "nfs41_ops.h"
|
|
|
|
|
|
#include "delegation.h"
|
|
|
|
|
|
#include "upcall.h"
|
|
|
|
|
|
#include "daemon_debug.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-04-06 16:15:24 -04:00
|
|
|
|
static int set_ea_value(
|
|
|
|
|
|
IN nfs41_session *session,
|
|
|
|
|
|
IN nfs41_path_fh *parent,
|
|
|
|
|
|
IN state_owner4 *owner,
|
|
|
|
|
|
IN PFILE_FULL_EA_INFORMATION ea)
|
|
|
|
|
|
{
|
|
|
|
|
|
nfs41_path_fh file = { 0 };
|
|
|
|
|
|
nfs41_file_info createattrs;
|
|
|
|
|
|
open_claim4 claim;
|
|
|
|
|
|
stateid_arg stateid;
|
|
|
|
|
|
open_delegation4 delegation = { 0 };
|
|
|
|
|
|
nfs41_write_verf verf;
|
|
|
|
|
|
uint32_t bytes_written;
|
|
|
|
|
|
int status;
|
|
|
|
|
|
|
|
|
|
|
|
/* don't allow values larger than NFS4_EASIZE */
|
|
|
|
|
|
if (ea->EaValueLength > NFS4_EASIZE) {
|
|
|
|
|
|
eprintf("trying to write extended attribute value of size %d, "
|
|
|
|
|
|
"max allowed %d\n", ea->EaValueLength, NFS4_EASIZE);
|
|
|
|
|
|
status = NFS4ERR_FBIG;
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
/* remove the file on empty value */
|
|
|
|
|
|
if (ea->EaValueLength == 0) {
|
|
|
|
|
|
nfs41_component name;
|
|
|
|
|
|
name.name = ea->EaName;
|
|
|
|
|
|
name.len = ea->EaNameLength;
|
|
|
|
|
|
nfs41_remove(session, parent, &name, 0);
|
|
|
|
|
|
status = NFS4_OK;
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
claim.claim = CLAIM_NULL;
|
|
|
|
|
|
claim.u.null.filename = &file.name;
|
|
|
|
|
|
file.name.name = ea->EaName;
|
|
|
|
|
|
file.name.len = ea->EaNameLength;
|
|
|
|
|
|
|
|
|
|
|
|
createattrs.attrmask.count = 2;
|
|
|
|
|
|
createattrs.attrmask.arr[0] = FATTR4_WORD0_SIZE;
|
|
|
|
|
|
createattrs.attrmask.arr[1] = FATTR4_WORD1_MODE;
|
|
|
|
|
|
createattrs.size = ea->EaValueLength;
|
|
|
|
|
|
createattrs.mode = 0664;
|
|
|
|
|
|
|
|
|
|
|
|
status = nfs41_open(session, parent, &file, owner, &claim,
|
|
|
|
|
|
OPEN4_SHARE_ACCESS_WRITE | OPEN4_SHARE_ACCESS_WANT_NO_DELEG,
|
|
|
|
|
|
OPEN4_SHARE_DENY_BOTH, OPEN4_CREATE, UNCHECKED4,
|
|
|
|
|
|
&createattrs, TRUE, &stateid.stateid, &delegation, NULL);
|
|
|
|
|
|
if (status) {
|
|
|
|
|
|
dprintf(1, "nfs41_open() failed with %s\n", nfs_error_string(status));
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status = nfs41_write(session, &file, &stateid,
|
|
|
|
|
|
(unsigned char*)ea->EaName + ea->EaNameLength + 1,
|
|
|
|
|
|
ea->EaValueLength, 0, FILE_SYNC4, &bytes_written,
|
|
|
|
|
|
&verf, NULL);
|
|
|
|
|
|
if (status) {
|
|
|
|
|
|
eprintf("nfs41_write() failed with %s\n", nfs_error_string(status));
|
|
|
|
|
|
goto out_close;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
out_close:
|
|
|
|
|
|
nfs41_close(session, &file, &stateid);
|
|
|
|
|
|
out:
|
|
|
|
|
|
return status;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int is_cygwin_ea(
|
|
|
|
|
|
PFILE_FULL_EA_INFORMATION ea)
|
|
|
|
|
|
{
|
|
|
|
|
|
return (strncmp("NfsV3Attributes", ea->EaName, ea->EaNameLength) == 0
|
|
|
|
|
|
&& sizeof("NfsV3Attributes")-1 == ea->EaNameLength)
|
|
|
|
|
|
|| (strncmp("NfsActOnLink", ea->EaName, ea->EaNameLength) == 0
|
|
|
|
|
|
&& sizeof("NfsActOnLink")-1 == ea->EaNameLength)
|
|
|
|
|
|
|| (strncmp("NfsSymlinkTargetName", ea->EaName, ea->EaNameLength) == 0
|
|
|
|
|
|
&& sizeof("NfsSymlinkTargetName")-1 == ea->EaNameLength);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2012-05-03 10:38:43 -04:00
|
|
|
|
#define NEXT_ENTRY(ea) ((PBYTE)(ea) + (ea)->NextEntryOffset)
|
|
|
|
|
|
|
2012-04-06 16:15:24 -04:00
|
|
|
|
int nfs41_ea_set(
|
|
|
|
|
|
IN nfs41_open_state *state,
|
|
|
|
|
|
IN PFILE_FULL_EA_INFORMATION ea)
|
|
|
|
|
|
{
|
|
|
|
|
|
nfs41_path_fh attrdir = { 0 };
|
|
|
|
|
|
int status;
|
|
|
|
|
|
|
|
|
|
|
|
status = nfs41_rpc_openattr(state->session, &state->file, TRUE, &attrdir.fh);
|
|
|
|
|
|
if (status) {
|
|
|
|
|
|
eprintf("nfs41_rpc_openattr() failed with error %s\n",
|
|
|
|
|
|
nfs_error_string(status));
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
while (status == NFS4_OK) {
|
|
|
|
|
|
if (!is_cygwin_ea(ea))
|
|
|
|
|
|
status = set_ea_value(state->session, &attrdir, &state->owner, ea);
|
|
|
|
|
|
|
|
|
|
|
|
if (ea->NextEntryOffset == 0)
|
|
|
|
|
|
break;
|
2012-05-03 10:38:43 -04:00
|
|
|
|
ea = (PFILE_FULL_EA_INFORMATION)NEXT_ENTRY(ea);
|
2012-04-06 16:15:24 -04:00
|
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
|
|
|
return status;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-04-06 14:26:13 -04:00
|
|
|
|
/* NFS41_EA_SET */
|
|
|
|
|
|
static int parse_setexattr(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall)
|
|
|
|
|
|
{
|
|
|
|
|
|
int status;
|
|
|
|
|
|
setexattr_upcall_args *args = &upcall->args.setexattr;
|
|
|
|
|
|
|
|
|
|
|
|
status = get_name(&buffer, &length, &args->path);
|
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
|
status = safe_read(&buffer, &length, &args->mode, sizeof(args->mode));
|
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
|
status = safe_read(&buffer, &length, &args->buf_len, sizeof(args->buf_len));
|
|
|
|
|
|
if (status) goto out;
|
|
|
|
|
|
args->buf = buffer;
|
|
|
|
|
|
|
|
|
|
|
|
dprintf(1, "parsing NFS41_EA_SET: mode=%o\n", args->mode);
|
|
|
|
|
|
out:
|
|
|
|
|
|
return status;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int handle_setexattr(nfs41_upcall *upcall)
|
|
|
|
|
|
{
|
|
|
|
|
|
int status;
|
|
|
|
|
|
setexattr_upcall_args *args = &upcall->args.setexattr;
|
|
|
|
|
|
nfs41_open_state *state = upcall->state_ref;
|
2012-04-06 16:15:24 -04:00
|
|
|
|
PFILE_FULL_EA_INFORMATION ea =
|
|
|
|
|
|
(PFILE_FULL_EA_INFORMATION)args->buf;
|
2012-04-06 14:26:13 -04:00
|
|
|
|
|
|
|
|
|
|
/* break read delegations before SETATTR */
|
|
|
|
|
|
nfs41_delegation_return(state->session, &state->file,
|
|
|
|
|
|
OPEN_DELEGATE_READ, FALSE);
|
|
|
|
|
|
|
2012-04-06 16:15:24 -04:00
|
|
|
|
if (strncmp("NfsV3Attributes", ea->EaName, ea->EaNameLength) == 0
|
|
|
|
|
|
&& sizeof("NfsV3Attributes")-1 == ea->EaNameLength) {
|
|
|
|
|
|
nfs41_file_info info;
|
|
|
|
|
|
stateid_arg stateid;
|
|
|
|
|
|
|
|
|
|
|
|
nfs41_open_stateid_arg(state, &stateid);
|
|
|
|
|
|
|
2012-04-06 14:26:13 -04:00
|
|
|
|
info.mode = args->mode;
|
2012-04-06 16:15:24 -04:00
|
|
|
|
info.attrmask.arr[0] = 0;
|
|
|
|
|
|
info.attrmask.arr[1] = FATTR4_WORD1_MODE;
|
2012-04-06 14:26:13 -04:00
|
|
|
|
info.attrmask.count = 2;
|
2012-04-06 16:15:24 -04:00
|
|
|
|
|
2012-04-06 14:26:13 -04:00
|
|
|
|
status = nfs41_setattr(state->session, &state->file, &stateid, &info);
|
|
|
|
|
|
if (status) {
|
|
|
|
|
|
dprintf(1, "nfs41_setattr() failed with error %s.\n",
|
|
|
|
|
|
nfs_error_string(status));
|
2012-04-06 16:15:24 -04:00
|
|
|
|
goto out;
|
2012-04-06 14:26:13 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2012-04-06 16:15:24 -04:00
|
|
|
|
args->ctime = info.change;
|
|
|
|
|
|
goto out;
|
2012-04-06 14:26:13 -04:00
|
|
|
|
}
|
2012-04-06 16:15:24 -04:00
|
|
|
|
|
|
|
|
|
|
status = nfs41_ea_set(state, ea);
|
2012-04-06 14:26:13 -04:00
|
|
|
|
out:
|
2012-04-06 16:15:24 -04:00
|
|
|
|
return nfs_to_windows_error(status, ERROR_NOT_SUPPORTED);
|
2012-04-06 14:26:13 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int marshall_setexattr(unsigned char *buffer, uint32_t *length, nfs41_upcall *upcall)
|
|
|
|
|
|
{
|
|
|
|
|
|
setexattr_upcall_args *args = &upcall->args.setexattr;
|
|
|
|
|
|
return safe_write(&buffer, length, &args->ctime, sizeof(args->ctime));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* NFS41_EA_GET */
|
|
|
|
|
|
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;
|
2012-04-16 12:44:16 -04:00
|
|
|
|
args->ealist = args->ealist_len ? buffer : NULL;
|
2012-04-06 14:26:13 -04:00
|
|
|
|
|
|
|
|
|
|
dprintf(1, "parsing NFS41_EA_GET: buf_len=%d Initial %d Restart %d "
|
|
|
|
|
|
"Single %d\n", args->buf_len,args->eaindex, args->restart, args->single);
|
|
|
|
|
|
out:
|
|
|
|
|
|
return status;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2012-05-03 10:38:43 -04:00
|
|
|
|
#define READDIR_LEN_INITIAL 8192
|
|
|
|
|
|
#define READDIR_LEN_MIN 2048
|
|
|
|
|
|
|
|
|
|
|
|
/* call readdir repeatedly to get a complete list of entries */
|
|
|
|
|
|
static int read_entire_dir(
|
|
|
|
|
|
IN nfs41_session *session,
|
|
|
|
|
|
IN nfs41_path_fh *eadir,
|
|
|
|
|
|
OUT unsigned char **buffer_out,
|
|
|
|
|
|
OUT uint32_t *length_out)
|
|
|
|
|
|
{
|
|
|
|
|
|
nfs41_readdir_cookie cookie = { 0 };
|
|
|
|
|
|
bitmap4 attr_request;
|
|
|
|
|
|
nfs41_readdir_entry *last_entry;
|
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
|
uint32_t buffer_len, len, total_len;
|
|
|
|
|
|
bool_t eof;
|
|
|
|
|
|
int status = NO_ERROR;
|
|
|
|
|
|
|
|
|
|
|
|
attr_request.count = 0; /* don't request attributes */
|
|
|
|
|
|
|
|
|
|
|
|
/* allocate the buffer for readdir entries */
|
|
|
|
|
|
buffer_len = READDIR_LEN_INITIAL;
|
|
|
|
|
|
buffer = calloc(1, buffer_len);
|
|
|
|
|
|
if (buffer == NULL) {
|
|
|
|
|
|
status = GetLastError();
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
last_entry = NULL;
|
|
|
|
|
|
total_len = 0;
|
|
|
|
|
|
eof = FALSE;
|
|
|
|
|
|
|
|
|
|
|
|
while (!eof) {
|
|
|
|
|
|
len = buffer_len - total_len;
|
|
|
|
|
|
if (len < READDIR_LEN_MIN) {
|
|
|
|
|
|
const ptrdiff_t diff = (unsigned char*)last_entry - buffer;
|
|
|
|
|
|
/* realloc the buffer to fit more entries */
|
|
|
|
|
|
unsigned char *tmp = realloc(buffer, buffer_len * 2);
|
|
|
|
|
|
if (tmp == NULL) {
|
|
|
|
|
|
status = GetLastError();
|
|
|
|
|
|
goto out_free;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (last_entry) /* fix last_entry pointer */
|
|
|
|
|
|
last_entry = (nfs41_readdir_entry*)(tmp + diff);
|
|
|
|
|
|
buffer = tmp;
|
|
|
|
|
|
buffer_len *= 2;
|
|
|
|
|
|
len = buffer_len - total_len;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* fetch the next group of entries */
|
|
|
|
|
|
status = nfs41_readdir(session, eadir, &attr_request,
|
|
|
|
|
|
&cookie, buffer + total_len, &len, &eof);
|
|
|
|
|
|
if (status)
|
|
|
|
|
|
goto out_free;
|
|
|
|
|
|
|
|
|
|
|
|
if (last_entry == NULL) {
|
|
|
|
|
|
/* initialize last_entry to the front of the list */
|
|
|
|
|
|
last_entry = (nfs41_readdir_entry*)(buffer + total_len);
|
|
|
|
|
|
} else if (len) {
|
|
|
|
|
|
/* link the previous list to the new one */
|
|
|
|
|
|
last_entry->next_entry_offset = (uint32_t)FIELD_OFFSET(
|
|
|
|
|
|
nfs41_readdir_entry, name) + last_entry->name_len;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* find the new last entry */
|
|
|
|
|
|
while (last_entry->next_entry_offset) {
|
|
|
|
|
|
last_entry = (nfs41_readdir_entry*)((char*)last_entry +
|
|
|
|
|
|
last_entry->next_entry_offset);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cookie.cookie = last_entry->cookie;
|
|
|
|
|
|
total_len += len;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
*buffer_out = buffer;
|
|
|
|
|
|
*length_out = total_len;
|
|
|
|
|
|
out:
|
|
|
|
|
|
return status;
|
|
|
|
|
|
|
|
|
|
|
|
out_free:
|
|
|
|
|
|
free(buffer);
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define ALIGNED_EASIZE(len) (align4(sizeof(FILE_GET_EA_INFORMATION) + len))
|
|
|
|
|
|
|
|
|
|
|
|
static uint32_t calculate_ea_list_length(
|
|
|
|
|
|
IN const unsigned char *position,
|
|
|
|
|
|
IN uint32_t remaining)
|
|
|
|
|
|
{
|
|
|
|
|
|
const nfs41_readdir_entry *entry;
|
|
|
|
|
|
uint32_t length = 0;
|
|
|
|
|
|
|
|
|
|
|
|
while (remaining) {
|
|
|
|
|
|
entry = (const nfs41_readdir_entry*)position;
|
|
|
|
|
|
length += ALIGNED_EASIZE(entry->name_len);
|
|
|
|
|
|
|
|
|
|
|
|
if (!entry->next_entry_offset)
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
position += entry->next_entry_offset;
|
|
|
|
|
|
remaining -= entry->next_entry_offset;
|
|
|
|
|
|
}
|
|
|
|
|
|
return length;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void populate_ea_list(
|
|
|
|
|
|
IN const unsigned char *position,
|
|
|
|
|
|
OUT PFILE_GET_EA_INFORMATION ea_list)
|
|
|
|
|
|
{
|
|
|
|
|
|
const nfs41_readdir_entry *entry;
|
|
|
|
|
|
PFILE_GET_EA_INFORMATION ea = ea_list, prev = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
|
|
entry = (const nfs41_readdir_entry*)position;
|
|
|
|
|
|
StringCchCopyA(ea->EaName, entry->name_len, entry->name);
|
|
|
|
|
|
ea->EaNameLength = (UCHAR)entry->name_len - 1;
|
|
|
|
|
|
|
|
|
|
|
|
if (!entry->next_entry_offset) {
|
|
|
|
|
|
ea->NextEntryOffset = 0;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
prev = ea;
|
|
|
|
|
|
ea->NextEntryOffset = ALIGNED_EASIZE(ea->EaNameLength);
|
|
|
|
|
|
ea = (PFILE_GET_EA_INFORMATION)NEXT_ENTRY(ea);
|
|
|
|
|
|
position += entry->next_entry_offset;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int get_ea_list(
|
|
|
|
|
|
IN OUT nfs41_open_state *state,
|
|
|
|
|
|
IN nfs41_path_fh *eadir,
|
|
|
|
|
|
OUT PFILE_GET_EA_INFORMATION *ealist_out)
|
|
|
|
|
|
{
|
|
|
|
|
|
unsigned char *entry_list;
|
|
|
|
|
|
PFILE_GET_EA_INFORMATION ea_list;
|
|
|
|
|
|
uint32_t entry_len, ea_size;
|
|
|
|
|
|
int status = NO_ERROR;
|
|
|
|
|
|
|
|
|
|
|
|
EnterCriticalSection(&state->ea.lock);
|
|
|
|
|
|
|
|
|
|
|
|
if (state->ea.list != INVALID_HANDLE_VALUE) {
|
|
|
|
|
|
/* use cached ea names */
|
|
|
|
|
|
*ealist_out = state->ea.list;
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* read the entire directory into a nfs41_readdir_entry buffer */
|
|
|
|
|
|
status = read_entire_dir(state->session, eadir, &entry_list, &entry_len);
|
|
|
|
|
|
if (status)
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
|
|
ea_size = calculate_ea_list_length(entry_list, entry_len);
|
|
|
|
|
|
if (ea_size == 0) {
|
|
|
|
|
|
*ealist_out = state->ea.list = NULL;
|
|
|
|
|
|
goto out_free;
|
|
|
|
|
|
}
|
|
|
|
|
|
ea_list = calloc(1, ea_size);
|
|
|
|
|
|
if (ea_list == NULL) {
|
|
|
|
|
|
status = GetLastError();
|
|
|
|
|
|
goto out_free;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
populate_ea_list(entry_list, ea_list);
|
|
|
|
|
|
|
|
|
|
|
|
*ealist_out = state->ea.list = ea_list;
|
|
|
|
|
|
|
|
|
|
|
|
out_free:
|
|
|
|
|
|
free(entry_list); /* allocated by read_entire_dir() */
|
|
|
|
|
|
out:
|
|
|
|
|
|
LeaveCriticalSection(&state->ea.lock);
|
|
|
|
|
|
return status;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2012-04-06 14:26:13 -04:00
|
|
|
|
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 = { 0 }, file = { 0 };
|
|
|
|
|
|
open_claim4 claim;
|
|
|
|
|
|
stateid4 open_stateid;
|
|
|
|
|
|
stateid_arg stateid;
|
|
|
|
|
|
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;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2012-05-03 10:38:43 -04:00
|
|
|
|
if (gea == NULL) {
|
|
|
|
|
|
/* if no names are queried, use READDIR to list them all */
|
|
|
|
|
|
status = get_ea_list(state, &parent, &gea);
|
|
|
|
|
|
if (status)
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2012-04-06 14:26:13 -04:00
|
|
|
|
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) {
|
|
|
|
|
|
file.name.name = gea->EaName;
|
|
|
|
|
|
file.name.len = gea->EaNameLength;
|
|
|
|
|
|
claim.claim = CLAIM_NULL;
|
|
|
|
|
|
claim.u.null.filename = &file.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;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const nfs41_upcall_op nfs41_op_setexattr = {
|
|
|
|
|
|
parse_setexattr,
|
|
|
|
|
|
handle_setexattr,
|
|
|
|
|
|
marshall_setexattr
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const nfs41_upcall_op nfs41_op_getexattr = {
|
|
|
|
|
|
parse_getexattr,
|
|
|
|
|
|
handle_getexattr,
|
|
|
|
|
|
marshall_getexattr
|
|
|
|
|
|
};
|