/* Copyright (c) 2010 * The Regents of the University of Michigan * All Rights Reserved * * 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 #include #include #include #include "daemon_debug.h" #include "util.h" #include "nfs41_ops.h" int safe_read(unsigned char **pos, uint32_t *remaining, void *dest, uint32_t dest_len) { if (*remaining < dest_len) return ERROR_BUFFER_OVERFLOW; CopyMemory(dest, *pos, dest_len); *pos += dest_len; *remaining -= dest_len; return 0; } int safe_write(unsigned char **pos, uint32_t *remaining, void *src, uint32_t src_len) { if (*remaining < src_len) return ERROR_BUFFER_OVERFLOW; CopyMemory(*pos, src, src_len); *pos += src_len; *remaining -= src_len; return 0; } int get_name(unsigned char **pos, uint32_t *remaining, const char **out_name) { int status; USHORT len; status = safe_read(pos, remaining, &len, sizeof(USHORT)); if (status) goto out; if (*remaining < len) { status = ERROR_BUFFER_OVERFLOW; goto out; } *out_name = (const char*)*pos; *pos += len; *remaining -= len; out: return status; } const char* strip_path( IN const char *path, OUT uint32_t *len_out) { const char *name = strrchr(path, '\\'); name = name ? name + 1 : path; if (len_out) *len_out = (uint32_t)strlen(name); return name; } uint32_t max_read_size( IN const nfs41_session *session, IN const nfs41_fh *fh) { const uint32_t maxresponse = session->fore_chan_attrs.ca_maxresponsesize; return (uint32_t)min(fh->superblock->maxread, maxresponse - READ_OVERHEAD); } uint32_t max_write_size( IN const nfs41_session *session, IN const nfs41_fh *fh) { const uint32_t maxrequest = session->fore_chan_attrs.ca_maxrequestsize; return (uint32_t)min(fh->superblock->maxwrite, maxrequest - WRITE_OVERHEAD); } bool_t verify_write( IN nfs41_write_verf *verf, IN OUT enum stable_how4 *stable) { if (verf->committed != UNSTABLE4) { *stable = verf->committed; dprintf(3, "verify_write: committed to stable storage\n"); return 1; } if (*stable != UNSTABLE4) { memcpy(verf->expected, verf->verf, NFS4_VERIFIER_SIZE); *stable = UNSTABLE4; dprintf(3, "verify_write: first unstable write, saving verifier\n"); return 1; } if (memcmp(verf->expected, verf->verf, NFS4_VERIFIER_SIZE) == 0) { dprintf(3, "verify_write: verifier matches expected\n"); return 1; } dprintf(2, "verify_write: verifier changed; writes have been lost!\n"); return 0; } ULONG nfs_file_info_to_attributes( IN const nfs41_file_info *info) { ULONG attrs = 0; if (info->type == NF4DIR) attrs |= FILE_ATTRIBUTE_DIRECTORY; else if (info->type == NF4LNK) attrs |= FILE_ATTRIBUTE_REPARSE_POINT; else if (info->type != NF4REG) dprintf(1, "unhandled file type %d, defaulting to NF4REG\n", info->type); if (info->mode == 0444) /* XXX: 0444 for READONLY */ attrs |= FILE_ATTRIBUTE_READONLY; /* TODO: FILE_ATTRIBUTE_HIDDEN */ // FILE_ATTRIBUTE_NORMAL attribute is only set if no other attributes are present. // all other override this value. return attrs ? attrs : FILE_ATTRIBUTE_NORMAL; } void nfs_to_basic_info( IN const nfs41_file_info *info, OUT PFILE_BASIC_INFO basic_out) { nfs_time_to_file_time(&info->time_create, &basic_out->CreationTime); nfs_time_to_file_time(&info->time_access, &basic_out->LastAccessTime); nfs_time_to_file_time(&info->time_modify, &basic_out->LastWriteTime); /* XXX: was using 'change' attr, but that wasn't giving a time */ nfs_time_to_file_time(&info->time_modify, &basic_out->ChangeTime); basic_out->FileAttributes = nfs_file_info_to_attributes(info); } void nfs_to_standard_info( IN const nfs41_file_info *info, OUT PFILE_STANDARD_INFO std_out) { std_out->AllocationSize.QuadPart = std_out->EndOfFile.QuadPart = (LONGLONG)info->size; std_out->NumberOfLinks = info->numlinks; std_out->DeletePending = FALSE; std_out->Directory = info->type == NF4DIR ? TRUE : FALSE; } /* http://msdn.microsoft.com/en-us/library/ms724290%28VS.85%29.aspx: * A file time is a 64-bit value that represents the number of * 100-nanosecond intervals that have elapsed since 12:00 A.M. * January 1, 1601 Coordinated Universal Time (UTC). */ static __inline void get_file_epoch( OUT PLARGE_INTEGER time_out) { static const SYSTEMTIME jan_1_1970 = {1970, 1, 4, 1, 0, 0, 0, 0}; SystemTimeToFileTime(&jan_1_1970, (LPFILETIME)time_out); } void file_time_to_nfs_time( IN const PLARGE_INTEGER file_time, OUT nfstime4 *nfs_time) { LARGE_INTEGER diff; get_file_epoch(&diff); diff.QuadPart = file_time->QuadPart - diff.QuadPart; nfs_time->seconds = diff.QuadPart / 10000000; nfs_time->nseconds = (uint32_t)((diff.QuadPart % 10000000)*100); } void nfs_time_to_file_time( IN const nfstime4 *nfs_time, OUT PLARGE_INTEGER file_time) { LARGE_INTEGER diff; get_file_epoch(&diff); file_time->QuadPart = diff.QuadPart + nfs_time->seconds * 10000000 + nfs_time->nseconds / 100; } void get_file_time( OUT PLARGE_INTEGER file_time) { GetSystemTimeAsFileTime((LPFILETIME)file_time); } void get_nfs_time( OUT nfstime4 *nfs_time) { LARGE_INTEGER file_time; get_file_time(&file_time); file_time_to_nfs_time(&file_time, nfs_time); } void map_access_2_allowdeny(ULONG access_mask, ULONG access_mode, uint32_t *allow, uint32_t *deny) { if ((access_mask & FILE_WRITE_DATA) && ((access_mask & FILE_READ_DATA) || (access_mask & FILE_EXECUTE))) *allow = OPEN4_SHARE_ACCESS_BOTH; else if ((access_mask & FILE_READ_DATA) || (access_mask & FILE_EXECUTE)) *allow = OPEN4_SHARE_ACCESS_READ; else if (access_mask & FILE_WRITE_DATA || (access_mask & FILE_APPEND_DATA) || (access_mask & FILE_WRITE_ATTRIBUTES)) *allow = OPEN4_SHARE_ACCESS_WRITE; #define FIX_ALLOW_DENY_WIN2NFS_CONVERSION #ifdef FIX_ALLOW_DENY_WIN2NFS_CONVERSION if ((access_mode & FILE_SHARE_READ) && (access_mode & FILE_SHARE_WRITE)) *deny = OPEN4_SHARE_DENY_NONE; else if (access_mode & FILE_SHARE_READ) *deny = OPEN4_SHARE_DENY_WRITE; else if (access_mode & FILE_SHARE_WRITE) *deny = OPEN4_SHARE_DENY_READ; else *deny = OPEN4_SHARE_DENY_BOTH; #else // AGLO: 11/13/2009. // readonly file that is being opened for reading with a // share read mode given above logic translates into deny // write and linux server does not allow it. *deny = OPEN4_SHARE_DENY_NONE; #endif } bool_t multi_addr_find( IN const multi_addr4 *addrs, IN const netaddr4 *addr, OUT OPTIONAL uint32_t *index_out) { uint32_t i; for (i = 0; i < addrs->count; i++) { const netaddr4 *saddr = &addrs->arr[i]; if (!strncmp(saddr->netid, addr->netid, NFS41_NETWORK_ID_LEN) && !strncmp(saddr->uaddr, addr->uaddr, NFS41_UNIVERSAL_ADDR_LEN)) { if (index_out) *index_out = i; return 1; } } return 0; } int nfs_to_windows_error(int status, int default_error) { /* make sure this is actually an nfs error */ if (status < 0 || (status > 70 && status < 10001) || status > 10087) { eprintf("nfs_to_windows_error called with non-nfs " "error code %d; returning the error as is\n", status); return status; } switch (status) { case NFS4_OK: return NO_ERROR; case NFS4ERR_PERM: return ERROR_ACCESS_DENIED; case NFS4ERR_NOENT: return ERROR_FILE_NOT_FOUND; case NFS4ERR_IO: return ERROR_NET_WRITE_FAULT; case NFS4ERR_ACCESS: return ERROR_ACCESS_DENIED; case NFS4ERR_EXIST: return ERROR_FILE_EXISTS; case NFS4ERR_XDEV: return ERROR_NOT_SAME_DEVICE; case NFS4ERR_INVAL: return ERROR_INVALID_PARAMETER; case NFS4ERR_FBIG: return ERROR_FILE_TOO_LARGE; case NFS4ERR_NOSPC: return ERROR_DISK_FULL; case NFS4ERR_ROFS: return ERROR_NETWORK_ACCESS_DENIED; case NFS4ERR_MLINK: return ERROR_TOO_MANY_LINKS; case NFS4ERR_NAMETOOLONG: return ERROR_FILENAME_EXCED_RANGE; case NFS4ERR_STALE: return ERROR_NETNAME_DELETED; case NFS4ERR_NOTEMPTY: return ERROR_NOT_EMPTY; case NFS4ERR_DENIED: return ERROR_LOCK_FAILED; case NFS4ERR_TOOSMALL: return ERROR_BUFFER_OVERFLOW; case NFS4ERR_LOCKED: return ERROR_LOCK_VIOLATION; case NFS4ERR_SHARE_DENIED: return ERROR_SHARING_VIOLATION; case NFS4ERR_LOCK_RANGE: return ERROR_NOT_LOCKED; case NFS4ERR_ATTRNOTSUPP: return ERROR_NOT_SUPPORTED; case NFS4ERR_OPENMODE: return ERROR_ACCESS_DENIED; case NFS4ERR_LOCK_NOTSUPP: return ERROR_ATOMIC_LOCKS_NOT_SUPPORTED; case NFS4ERR_BADCHAR: case NFS4ERR_BADNAME: return ERROR_INVALID_NAME; case NFS4ERR_NOTDIR: case NFS4ERR_ISDIR: case NFS4ERR_SYMLINK: case NFS4ERR_WRONG_TYPE: return ERROR_INVALID_PARAMETER; case NFS4ERR_OLD_STATEID: case NFS4ERR_BAD_STATEID: case NFS4ERR_ADMIN_REVOKED: return ERROR_FILE_INVALID; default: dprintf(1, "nfs error %s not mapped to windows error; " "returning default error %d\n", nfs_error_string(status), default_error); return default_error; } } bool_t next_component( IN const char *path, IN const char *path_end, OUT nfs41_component *component) { const char *component_end; component->name = next_non_delimiter(path, path_end); component_end = next_delimiter(component->name, path_end); component->len = (unsigned short)(component_end - component->name); return component->len > 0; } bool_t last_component( IN const char *path, IN const char *path_end, OUT nfs41_component *component) { const char *component_end = prev_delimiter(path_end, path); component->name = prev_non_delimiter(component_end, path); component->name = prev_delimiter(component->name, path); component->name = next_non_delimiter(component->name, component_end); component->len = (unsigned short)(component_end - component->name); return component->len > 0; } bool_t is_last_component( IN const char *path, IN const char *path_end) { path = next_delimiter(path, path_end); return next_non_delimiter(path, path_end) == path_end; } void abs_path_copy( OUT nfs41_abs_path *dst, IN const nfs41_abs_path *src) { dst->len = src->len; StringCchCopyNA(dst->path, NFS41_MAX_PATH_LEN, src->path, dst->len); } void path_fh_init( OUT nfs41_path_fh *file, IN nfs41_abs_path *path) { file->path = path; last_component(path->path, path->path + path->len, &file->name); } void fh_copy( OUT nfs41_fh *dst, IN const nfs41_fh *src) { dst->fileid = src->fileid; dst->superblock = src->superblock; dst->len = src->len; memcpy(dst->fh, src->fh, dst->len); } void path_fh_copy( OUT nfs41_path_fh *dst, IN const nfs41_path_fh *src) { dst->path = src->path; if (dst->path) { const size_t name_start = src->name.name - src->path->path; dst->name.name = dst->path->path + name_start; dst->name.len = src->name.len; } else { dst->name.name = NULL; dst->name.len = 0; } fh_copy(&dst->fh, &src->fh); } int create_silly_rename( IN nfs41_abs_path *path, IN const nfs41_fh *fh, OUT nfs41_component *silly) { const char *end = path->path + NFS41_MAX_PATH_LEN; const unsigned short extra_len = 2 + 2*(unsigned short)fh->len; char name[NFS41_MAX_COMPONENT_LEN+1]; char *tmp; uint32_t i; int status = NO_ERROR; if (path->len + extra_len >= NFS41_MAX_PATH_LEN) { status = ERROR_BUFFER_OVERFLOW; goto out; } last_component(path->path, path->path + path->len, silly); StringCchCopyNA(name, NFS41_MAX_COMPONENT_LEN+1, silly->name, silly->len); tmp = (char*)silly->name; StringCchPrintf(tmp, end - tmp, ".%s.", name); tmp += silly->len + 2; for (i = 0; i < fh->len; i++, tmp += 2) StringCchPrintf(tmp, end - tmp, "%02x", fh->fh[i]); path->len += extra_len; silly->len += extra_len; out: return status; }