nginx-quic/src/os/win32/ngx_files.c

852 lines
17 KiB
C
Raw Normal View History

2002-09-02 10:48:24 -04:00
/*
* Copyright (C) Igor Sysoev
2012-01-18 12:07:43 -03:00
* Copyright (C) Nginx, Inc.
*/
2002-12-27 04:27:47 -03:00
#include <ngx_config.h>
2002-09-02 10:48:24 -04:00
#include <ngx_core.h>
2003-06-03 11:42:58 -04:00
2002-09-02 10:48:24 -04:00
#define NGX_UTF16_BUFLEN 256
static ngx_int_t ngx_win32_check_filename(u_char *name, u_short *u,
size_t len);
2010-01-28 11:17:51 -03:00
static u_short *ngx_utf8_to_utf16(u_short *utf16, u_char *utf8, size_t *len);
/* FILE_FLAG_BACKUP_SEMANTICS allows to obtain a handle to a directory */
ngx_fd_t
ngx_open_file(u_char *name, u_long mode, u_long create, u_long access)
{
2010-01-28 11:17:51 -03:00
size_t len;
u_short *u;
2009-09-16 09:28:20 -04:00
ngx_fd_t fd;
ngx_err_t err;
u_short utf16[NGX_UTF16_BUFLEN];
2010-01-28 11:17:51 -03:00
len = NGX_UTF16_BUFLEN;
u = ngx_utf8_to_utf16(utf16, name, &len);
if (u == NULL) {
return INVALID_HANDLE_VALUE;
}
2010-01-28 11:17:51 -03:00
fd = INVALID_HANDLE_VALUE;
if (create == NGX_FILE_OPEN
&& ngx_win32_check_filename(name, u, len) != NGX_OK)
{
goto failed;
2010-01-28 11:17:51 -03:00
}
fd = CreateFileW(u, mode,
FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
NULL, create, FILE_FLAG_BACKUP_SEMANTICS, NULL);
2010-01-28 11:17:51 -03:00
failed:
if (u != utf16) {
err = ngx_errno;
ngx_free(u);
ngx_set_errno(err);
}
return fd;
}
ssize_t
ngx_read_file(ngx_file_t *file, u_char *buf, size_t size, off_t offset)
2002-09-02 10:48:24 -04:00
{
2004-03-16 03:10:12 -04:00
u_long n;
2003-11-14 04:20:34 -03:00
ngx_err_t err;
2003-11-13 03:14:05 -03:00
OVERLAPPED ovlp, *povlp;
2009-09-24 15:55:35 -04:00
ovlp.Internal = 0;
ovlp.InternalHigh = 0;
ovlp.Offset = (u_long) offset;
ovlp.OffsetHigh = (u_long) (offset >> 32);
ovlp.hEvent = NULL;
2003-11-13 03:14:05 -03:00
2009-09-24 15:55:35 -04:00
povlp = &ovlp;
2003-11-13 03:14:05 -03:00
if (ReadFile(file->fd, buf, size, &n, povlp) == 0) {
err = ngx_errno;
if (err == ERROR_HANDLE_EOF) {
return 0;
}
2009-09-24 16:04:10 -04:00
ngx_log_error(NGX_LOG_ERR, file->log, err,
"ReadFile() \"%s\" failed", file->name.data);
2002-09-02 10:48:24 -04:00
return NGX_ERROR;
}
2002-12-21 14:14:50 -03:00
file->offset += n;
2002-09-02 10:48:24 -04:00
return n;
}
2003-06-03 11:42:58 -04:00
ssize_t
ngx_write_file(ngx_file_t *file, u_char *buf, size_t size, off_t offset)
2003-11-13 03:14:05 -03:00
{
2004-03-16 03:10:12 -04:00
u_long n;
2003-11-13 13:16:33 -03:00
OVERLAPPED ovlp, *povlp;
2009-09-24 15:55:35 -04:00
ovlp.Internal = 0;
ovlp.InternalHigh = 0;
ovlp.Offset = (u_long) offset;
ovlp.OffsetHigh = (u_long) (offset >> 32);
ovlp.hEvent = NULL;
2003-11-13 13:16:33 -03:00
2009-09-24 15:55:35 -04:00
povlp = &ovlp;
2003-11-13 13:16:33 -03:00
if (WriteFile(file->fd, buf, size, &n, povlp) == 0) {
2009-09-24 16:04:10 -04:00
ngx_log_error(NGX_LOG_ERR, file->log, ngx_errno,
"WriteFile() \"%s\" failed", file->name.data);
2003-11-13 03:14:05 -03:00
return NGX_ERROR;
}
2009-09-24 16:05:21 -04:00
if (n != size) {
ngx_log_error(NGX_LOG_CRIT, file->log, 0,
"WriteFile() \"%s\" has written only %ul of %uz",
file->name.data, n, size);
return NGX_ERROR;
}
2003-11-13 03:14:05 -03:00
file->offset += n;
return n;
}
ssize_t
ngx_write_chain_to_file(ngx_file_t *file, ngx_chain_t *cl, off_t offset,
ngx_pool_t *pool)
2003-06-03 11:42:58 -04:00
{
2004-03-16 17:26:01 -04:00
u_char *buf, *prev;
2003-11-14 04:20:34 -03:00
size_t size;
ssize_t total, n;
total = 0;
while (cl) {
buf = cl->buf->pos;
2003-11-14 04:20:34 -03:00
prev = buf;
size = 0;
2004-06-01 02:04:46 -04:00
/* coalesce the neighbouring bufs */
2003-11-14 04:20:34 -03:00
while (cl && prev == cl->buf->pos) {
size += cl->buf->last - cl->buf->pos;
prev = cl->buf->last;
2003-11-14 04:20:34 -03:00
cl = cl->next;
}
n = ngx_write_file(file, buf, size, offset);
if (n == NGX_ERROR) {
return NGX_ERROR;
}
total += n;
offset += n;
}
return total;
}
ssize_t
ngx_read_fd(ngx_fd_t fd, void *buf, size_t size)
{
u_long n;
if (ReadFile(fd, buf, size, &n, NULL) != 0) {
return (size_t) n;
}
return -1;
}
ssize_t
ngx_write_fd(ngx_fd_t fd, void *buf, size_t size)
{
u_long n;
if (WriteFile(fd, buf, size, &n, NULL) != 0) {
return (size_t) n;
}
return -1;
}
ssize_t
ngx_write_console(ngx_fd_t fd, void *buf, size_t size)
{
u_long n;
(void) CharToOemBuff(buf, buf, size);
if (WriteFile(fd, buf, size, &n, NULL) != 0) {
return (size_t) n;
}
return -1;
}
ngx_err_t
ngx_win32_rename_file(ngx_str_t *from, ngx_str_t *to, ngx_log_t *log)
2003-11-14 04:20:34 -03:00
{
u_char *name;
ngx_err_t err;
ngx_uint_t collision;
ngx_atomic_uint_t num;
2003-11-14 04:20:34 -03:00
2011-08-22 07:07:27 -03:00
name = ngx_alloc(to->len + 1 + NGX_ATOMIC_T_LEN + 1 + sizeof("DELETE"),
log);
if (name == NULL) {
return NGX_ENOMEM;
2003-11-16 18:49:42 -03:00
}
2003-11-14 04:20:34 -03:00
ngx_memcpy(name, to->data, to->len);
2003-11-13 13:16:33 -03:00
2003-11-14 04:20:34 -03:00
collision = 0;
/* mutex_lock() (per cache or single ?) */
for ( ;; ) {
2003-11-14 04:20:34 -03:00
num = ngx_next_temp_number(collision);
2009-05-29 05:31:48 -04:00
ngx_sprintf(name + to->len, ".%0muA.DELETE%Z", num);
2003-11-14 04:20:34 -03:00
if (MoveFile((const char *) to->data, (const char *) name) != 0) {
break;
2003-11-14 04:20:34 -03:00
}
collision = 1;
2009-05-29 05:32:52 -04:00
ngx_log_error(NGX_LOG_CRIT, log, ngx_errno,
"MoveFile() \"%s\" to \"%s\" failed", to->data, name);
}
2003-11-14 04:20:34 -03:00
2004-03-16 17:26:01 -04:00
if (MoveFile((const char *) from->data, (const char *) to->data) == 0) {
err = ngx_errno;
2003-11-16 18:49:42 -03:00
} else {
err = 0;
2003-11-16 18:49:42 -03:00
}
2007-01-18 17:24:30 -03:00
if (DeleteFile((const char *) name) == 0) {
2009-05-29 05:32:52 -04:00
ngx_log_error(NGX_LOG_CRIT, log, ngx_errno,
"DeleteFile() \"%s\" failed", name);
2003-11-14 04:20:34 -03:00
}
/* mutex_unlock() */
ngx_free(name);
return err;
2003-11-14 04:20:34 -03:00
}
ngx_int_t
ngx_file_info(u_char *file, ngx_file_info_t *sb)
2003-11-14 13:52:04 -03:00
{
2010-01-28 11:17:51 -03:00
size_t len;
2009-09-16 09:30:13 -04:00
long rc;
u_short *u;
ngx_err_t err;
WIN32_FILE_ATTRIBUTE_DATA fa;
u_short utf16[NGX_UTF16_BUFLEN];
2003-11-14 13:52:04 -03:00
2010-01-28 11:17:51 -03:00
len = NGX_UTF16_BUFLEN;
u = ngx_utf8_to_utf16(utf16, file, &len);
2003-11-14 13:52:04 -03:00
2009-09-16 09:30:13 -04:00
if (u == NULL) {
return NGX_FILE_ERROR;
2003-11-14 13:52:04 -03:00
}
rc = NGX_FILE_ERROR;
2009-09-16 09:30:13 -04:00
if (ngx_win32_check_filename(file, u, len) != NGX_OK) {
goto failed;
2009-09-16 09:30:13 -04:00
}
rc = GetFileAttributesExW(u, GetFileExInfoStandard, &fa);
2003-11-14 13:52:04 -03:00
sb->dwFileAttributes = fa.dwFileAttributes;
sb->ftCreationTime = fa.ftCreationTime;
sb->ftLastAccessTime = fa.ftLastAccessTime;
sb->ftLastWriteTime = fa.ftLastWriteTime;
sb->nFileSizeHigh = fa.nFileSizeHigh;
sb->nFileSizeLow = fa.nFileSizeLow;
failed:
if (u != utf16) {
err = ngx_errno;
ngx_free(u);
ngx_set_errno(err);
}
2009-09-16 09:30:13 -04:00
return rc;
2003-11-14 13:52:04 -03:00
}
ngx_int_t
ngx_set_file_time(u_char *name, ngx_fd_t fd, time_t s)
{
uint64_t intervals;
FILETIME ft;
/* 116444736000000000 is commented in src/os/win32/ngx_time.c */
intervals = s * 10000000 + 116444736000000000;
ft.dwLowDateTime = (DWORD) intervals;
ft.dwHighDateTime = (DWORD) (intervals >> 32);
if (SetFileTime(fd, NULL, NULL, &ft) != 0) {
return NGX_OK;
}
return NGX_ERROR;
}
2010-06-29 11:18:50 -04:00
ngx_int_t
ngx_create_file_mapping(ngx_file_mapping_t *fm)
{
LARGE_INTEGER size;
fm->fd = ngx_open_file(fm->name, NGX_FILE_RDWR, NGX_FILE_TRUNCATE,
NGX_FILE_DEFAULT_ACCESS);
if (fm->fd == NGX_INVALID_FILE) {
ngx_log_error(NGX_LOG_CRIT, fm->log, ngx_errno,
ngx_open_file_n " \"%s\" failed", fm->name);
return NGX_ERROR;
}
fm->handle = NULL;
size.QuadPart = fm->size;
if (SetFilePointerEx(fm->fd, size, NULL, FILE_BEGIN) == 0) {
ngx_log_error(NGX_LOG_CRIT, fm->log, ngx_errno,
"SetFilePointerEx(\"%s\", %uz) failed",
fm->name, fm->size);
goto failed;
}
if (SetEndOfFile(fm->fd) == 0) {
ngx_log_error(NGX_LOG_CRIT, fm->log, ngx_errno,
"SetEndOfFile() \"%s\" failed", fm->name);
goto failed;
}
fm->handle = CreateFileMapping(fm->fd, NULL, PAGE_READWRITE,
(u_long) ((off_t) fm->size >> 32),
(u_long) ((off_t) fm->size & 0xffffffff),
NULL);
if (fm->handle == NULL) {
ngx_log_error(NGX_LOG_CRIT, fm->log, ngx_errno,
"CreateFileMapping(%s, %uz) failed",
fm->name, fm->size);
goto failed;
}
fm->addr = MapViewOfFile(fm->handle, FILE_MAP_WRITE, 0, 0, 0);
if (fm->addr != NULL) {
return NGX_OK;
}
ngx_log_error(NGX_LOG_CRIT, fm->log, ngx_errno,
"MapViewOfFile(%uz) of file mapping \"%s\" failed",
fm->size, fm->name);
failed:
if (fm->handle) {
if (CloseHandle(fm->handle) == 0) {
ngx_log_error(NGX_LOG_ALERT, fm->log, ngx_errno,
"CloseHandle() of file mapping \"%s\" failed",
fm->name);
}
}
if (ngx_close_file(fm->fd) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_ALERT, fm->log, ngx_errno,
ngx_close_file_n " \"%s\" failed", fm->name);
}
return NGX_ERROR;
}
void
ngx_close_file_mapping(ngx_file_mapping_t *fm)
{
if (UnmapViewOfFile(fm->addr) == 0) {
ngx_log_error(NGX_LOG_ALERT, fm->log, ngx_errno,
"UnmapViewOfFile(%p) of file mapping \"%s\" failed",
fm->addr, &fm->name);
}
if (CloseHandle(fm->handle) == 0) {
ngx_log_error(NGX_LOG_ALERT, fm->log, ngx_errno,
"CloseHandle() of file mapping \"%s\" failed",
&fm->name);
}
if (ngx_close_file(fm->fd) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_ALERT, fm->log, ngx_errno,
ngx_close_file_n " \"%s\" failed", fm->name);
}
}
u_char *
2008-09-27 07:53:41 -04:00
ngx_realpath(u_char *path, u_char *resolved)
{
/* STUB */
return path;
2008-09-27 07:53:41 -04:00
}
ngx_int_t
ngx_open_dir(ngx_str_t *name, ngx_dir_t *dir)
2003-11-16 18:49:42 -03:00
{
ngx_cpystrn(name->data + name->len, NGX_DIR_MASK, NGX_DIR_MASK_LEN + 1);
2006-10-02 04:46:45 -04:00
dir->dir = FindFirstFile((const char *) name->data, &dir->finddata);
2003-11-16 18:49:42 -03:00
if (dir->dir == INVALID_HANDLE_VALUE) {
return NGX_ERROR;
2003-11-16 18:49:42 -03:00
}
dir->valid_info = 1;
2003-11-16 18:49:42 -03:00
dir->ready = 1;
return NGX_OK;
}
ngx_int_t
ngx_read_dir(ngx_dir_t *dir)
2003-11-16 18:49:42 -03:00
{
if (dir->ready) {
dir->ready = 0;
return NGX_OK;
}
2006-10-02 04:46:45 -04:00
if (FindNextFile(dir->dir, &dir->finddata) != 0) {
dir->type = 1;
return NGX_OK;
2003-11-16 18:49:42 -03:00
}
return NGX_ERROR;
2003-11-16 18:49:42 -03:00
}
2003-11-14 13:52:04 -03:00
2006-10-02 04:46:45 -04:00
ngx_int_t
ngx_open_glob(ngx_glob_t *gl)
{
2009-01-21 11:17:57 -03:00
u_char *p;
size_t len;
ngx_err_t err;
2007-11-11 14:50:47 -03:00
2006-10-02 04:46:45 -04:00
gl->dir = FindFirstFile((const char *) gl->pattern, &gl->finddata);
if (gl->dir == INVALID_HANDLE_VALUE) {
2008-04-29 05:28:42 -04:00
2009-01-21 11:17:57 -03:00
err = ngx_errno;
if ((err == ERROR_FILE_NOT_FOUND || err == ERROR_PATH_NOT_FOUND)
&& gl->test)
{
2008-04-29 05:28:42 -04:00
gl->no_match = 1;
return NGX_OK;
}
2006-10-02 04:46:45 -04:00
return NGX_ERROR;
}
2007-11-11 14:50:47 -03:00
for (p = gl->pattern; *p; p++) {
if (*p == '/') {
gl->last = p + 1 - gl->pattern;
}
}
len = ngx_strlen(gl->finddata.cFileName);
gl->name.len = gl->last + len;
gl->name.data = ngx_alloc(gl->name.len + 1, gl->log);
if (gl->name.data == NULL) {
return NGX_ERROR;
}
ngx_memcpy(gl->name.data, gl->pattern, gl->last);
ngx_cpystrn(gl->name.data + gl->last, (u_char *) gl->finddata.cFileName,
len + 1);
2006-10-02 04:46:45 -04:00
gl->ready = 1;
return NGX_OK;
}
ngx_int_t
ngx_read_glob(ngx_glob_t *gl, ngx_str_t *name)
{
2007-11-11 14:50:47 -03:00
size_t len;
2006-10-02 04:46:45 -04:00
ngx_err_t err;
2008-04-29 05:28:42 -04:00
if (gl->no_match) {
return NGX_DONE;
}
2006-10-02 04:46:45 -04:00
if (gl->ready) {
2007-11-11 14:50:47 -03:00
*name = gl->name;
2006-10-02 04:46:45 -04:00
gl->ready = 0;
return NGX_OK;
}
2007-11-11 14:50:47 -03:00
ngx_free(gl->name.data);
gl->name.data = NULL;
2006-10-02 04:46:45 -04:00
if (FindNextFile(gl->dir, &gl->finddata) != 0) {
2007-11-11 14:50:47 -03:00
len = ngx_strlen(gl->finddata.cFileName);
gl->name.len = gl->last + len;
gl->name.data = ngx_alloc(gl->name.len + 1, gl->log);
if (gl->name.data == NULL) {
return NGX_ERROR;
}
ngx_memcpy(gl->name.data, gl->pattern, gl->last);
ngx_cpystrn(gl->name.data + gl->last, (u_char *) gl->finddata.cFileName,
len + 1);
*name = gl->name;
2006-10-02 04:46:45 -04:00
return NGX_OK;
}
err = ngx_errno;
if (err == NGX_ENOMOREFILES) {
return NGX_DONE;
}
ngx_log_error(NGX_LOG_ALERT, gl->log, err,
"FindNextFile(%s) failed", gl->pattern);
return NGX_ERROR;
}
void
ngx_close_glob(ngx_glob_t *gl)
{
2007-11-11 14:50:47 -03:00
if (gl->name.data) {
ngx_free(gl->name.data);
}
2008-04-29 05:28:42 -04:00
if (gl->dir == INVALID_HANDLE_VALUE) {
return;
}
2007-11-11 14:50:47 -03:00
if (FindClose(gl->dir) == 0) {
2006-10-02 04:46:45 -04:00
ngx_log_error(NGX_LOG_ALERT, gl->log, ngx_errno,
"FindClose(%s) failed", gl->pattern);
}
}
ngx_int_t
ngx_de_info(u_char *name, ngx_dir_t *dir)
{
return NGX_OK;
}
ngx_int_t
ngx_de_link_info(u_char *name, ngx_dir_t *dir)
{
return NGX_OK;
}
2009-09-30 09:21:52 -04:00
ngx_int_t
ngx_read_ahead(ngx_fd_t fd, size_t n)
{
return ~NGX_FILE_ERROR;
}
2008-08-04 05:14:30 -04:00
ngx_int_t
ngx_directio_on(ngx_fd_t fd)
{
return ~NGX_FILE_ERROR;
}
2009-03-30 03:43:06 -04:00
ngx_int_t
ngx_directio_off(ngx_fd_t fd)
2008-08-04 05:14:30 -04:00
{
return ~NGX_FILE_ERROR;
2008-08-04 05:14:30 -04:00
}
2009-03-30 03:43:06 -04:00
size_t
ngx_fs_bsize(u_char *name)
{
u_char root[4];
u_long sc, bs, nfree, ncl;
if (name[2] == ':') {
ngx_cpystrn(root, name, 4);
name = root;
}
if (GetDiskFreeSpace((const char *) name, &sc, &bs, &nfree, &ncl) == 0) {
return 512;
}
return sc * bs;
}
static ngx_int_t
ngx_win32_check_filename(u_char *name, u_short *u, size_t len)
{
u_char *p, ch;
u_long n;
u_short *lu;
ngx_err_t err;
enum {
sw_start = 0,
sw_normal,
sw_after_slash,
sw_after_colon,
sw_after_dot
} state;
/* check for NTFS streams (":"), trailing dots and spaces */
lu = NULL;
state = sw_start;
for (p = name; *p; p++) {
ch = *p;
switch (state) {
case sw_start:
/*
* skip till first "/" to allow paths starting with drive and
* relative path, like "c:html/"
*/
if (ch == '/' || ch == '\\') {
state = sw_after_slash;
}
break;
case sw_normal:
if (ch == ':') {
state = sw_after_colon;
break;
}
if (ch == '.' || ch == ' ') {
state = sw_after_dot;
break;
}
if (ch == '/' || ch == '\\') {
state = sw_after_slash;
break;
}
break;
case sw_after_slash:
if (ch == '/' || ch == '\\') {
break;
}
if (ch == '.') {
break;
}
if (ch == ':') {
state = sw_after_colon;
break;
}
state = sw_normal;
break;
case sw_after_colon:
if (ch == '/' || ch == '\\') {
state = sw_after_slash;
break;
}
goto invalid;
case sw_after_dot:
if (ch == '/' || ch == '\\') {
goto invalid;
}
if (ch == ':') {
goto invalid;
}
2012-06-25 14:09:38 -04:00
if (ch == '.' || ch == ' ') {
break;
}
state = sw_normal;
break;
}
}
if (state == sw_after_dot) {
goto invalid;
}
/* check if long name match */
lu = malloc(len * 2);
if (lu == NULL) {
return NGX_ERROR;
}
n = GetLongPathNameW(u, lu, len);
if (n == 0) {
goto failed;
}
if (n != len - 1 || _wcsicmp(u, lu) != 0) {
goto invalid;
}
2013-10-31 11:23:49 -03:00
ngx_free(lu);
return NGX_OK;
invalid:
ngx_set_errno(NGX_ENOENT);
failed:
if (lu) {
err = ngx_errno;
ngx_free(lu);
ngx_set_errno(err);
}
return NGX_ERROR;
}
static u_short *
2010-01-28 11:17:51 -03:00
ngx_utf8_to_utf16(u_short *utf16, u_char *utf8, size_t *len)
{
u_char *p;
u_short *u, *last;
uint32_t n;
p = utf8;
u = utf16;
2010-01-28 11:17:51 -03:00
last = utf16 + *len;
while (u < last) {
if (*p < 0x80) {
2010-01-28 11:17:51 -03:00
*u++ = (u_short) *p;
if (*p == 0) {
2010-01-28 11:17:51 -03:00
*len = u - utf16;
return utf16;
}
p++;
continue;
}
n = ngx_utf8_decode(&p, 4);
if (n > 0xffff) {
ngx_set_errno(NGX_EILSEQ);
return NULL;
}
*u++ = (u_short) n;
}
/* the given buffer is not enough, allocate a new one */
u = malloc(((p - utf8) + ngx_strlen(p) + 1) * sizeof(u_short));
if (u == NULL) {
return NULL;
}
2010-01-28 11:17:51 -03:00
ngx_memcpy(u, utf16, *len * 2);
utf16 = u;
2010-01-28 11:17:51 -03:00
u += *len;
for ( ;; ) {
if (*p < 0x80) {
2010-01-28 11:17:51 -03:00
*u++ = (u_short) *p;
if (*p == 0) {
2010-01-28 11:17:51 -03:00
*len = u - utf16;
return utf16;
}
p++;
continue;
}
n = ngx_utf8_decode(&p, 4);
if (n > 0xffff) {
free(utf16);
ngx_set_errno(NGX_EILSEQ);
return NULL;
}
*u++ = (u_short) n;
}
/* unreachable */
}