|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | * Copyright (C) Nginx, Inc. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include <ngx_config.h> | 
|  | #include <ngx_core.h> | 
|  |  | 
|  |  | 
|  | #define NGX_UTF16_BUFLEN  256 | 
|  |  | 
|  | static ngx_int_t ngx_win32_check_filename(u_char *name, u_short *u, | 
|  | size_t len); | 
|  | 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) | 
|  | { | 
|  | size_t      len; | 
|  | u_short    *u; | 
|  | ngx_fd_t    fd; | 
|  | ngx_err_t   err; | 
|  | u_short     utf16[NGX_UTF16_BUFLEN]; | 
|  |  | 
|  | len = NGX_UTF16_BUFLEN; | 
|  | u = ngx_utf8_to_utf16(utf16, name, &len); | 
|  |  | 
|  | if (u == NULL) { | 
|  | return INVALID_HANDLE_VALUE; | 
|  | } | 
|  |  | 
|  | fd = INVALID_HANDLE_VALUE; | 
|  |  | 
|  | if (create == NGX_FILE_OPEN | 
|  | && ngx_win32_check_filename(name, u, len) != NGX_OK) | 
|  | { | 
|  | goto failed; | 
|  | } | 
|  |  | 
|  | fd = CreateFileW(u, mode, | 
|  | FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, | 
|  | NULL, create, FILE_FLAG_BACKUP_SEMANTICS, NULL); | 
|  |  | 
|  | 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) | 
|  | { | 
|  | u_long      n; | 
|  | ngx_err_t   err; | 
|  | OVERLAPPED  ovlp, *povlp; | 
|  |  | 
|  | ovlp.Internal = 0; | 
|  | ovlp.InternalHigh = 0; | 
|  | ovlp.Offset = (u_long) offset; | 
|  | ovlp.OffsetHigh = (u_long) (offset >> 32); | 
|  | ovlp.hEvent = NULL; | 
|  |  | 
|  | povlp = &ovlp; | 
|  |  | 
|  | if (ReadFile(file->fd, buf, size, &n, povlp) == 0) { | 
|  | err = ngx_errno; | 
|  |  | 
|  | if (err == ERROR_HANDLE_EOF) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | ngx_log_error(NGX_LOG_ERR, file->log, err, | 
|  | "ReadFile() \"%s\" failed", file->name.data); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | file->offset += n; | 
|  |  | 
|  | return n; | 
|  | } | 
|  |  | 
|  |  | 
|  | ssize_t | 
|  | ngx_write_file(ngx_file_t *file, u_char *buf, size_t size, off_t offset) | 
|  | { | 
|  | u_long      n; | 
|  | OVERLAPPED  ovlp, *povlp; | 
|  |  | 
|  | ovlp.Internal = 0; | 
|  | ovlp.InternalHigh = 0; | 
|  | ovlp.Offset = (u_long) offset; | 
|  | ovlp.OffsetHigh = (u_long) (offset >> 32); | 
|  | ovlp.hEvent = NULL; | 
|  |  | 
|  | povlp = &ovlp; | 
|  |  | 
|  | if (WriteFile(file->fd, buf, size, &n, povlp) == 0) { | 
|  | ngx_log_error(NGX_LOG_ERR, file->log, ngx_errno, | 
|  | "WriteFile() \"%s\" failed", file->name.data); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | 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) | 
|  | { | 
|  | u_char   *buf, *prev; | 
|  | size_t    size; | 
|  | ssize_t   total, n; | 
|  |  | 
|  | total = 0; | 
|  |  | 
|  | while (cl) { | 
|  | buf = cl->buf->pos; | 
|  | prev = buf; | 
|  | size = 0; | 
|  |  | 
|  | /* coalesce the neighbouring bufs */ | 
|  |  | 
|  | while (cl && prev == cl->buf->pos) { | 
|  | size += cl->buf->last - cl->buf->pos; | 
|  | prev = cl->buf->last; | 
|  | 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) | 
|  | { | 
|  | u_char             *name; | 
|  | ngx_err_t           err; | 
|  | ngx_uint_t          collision; | 
|  | ngx_atomic_uint_t   num; | 
|  |  | 
|  | name = ngx_alloc(to->len + 1 + NGX_ATOMIC_T_LEN + 1 + sizeof("DELETE"), | 
|  | log); | 
|  | if (name == NULL) { | 
|  | return NGX_ENOMEM; | 
|  | } | 
|  |  | 
|  | ngx_memcpy(name, to->data, to->len); | 
|  |  | 
|  | collision = 0; | 
|  |  | 
|  | /* mutex_lock() (per cache or single ?) */ | 
|  |  | 
|  | for ( ;; ) { | 
|  | num = ngx_next_temp_number(collision); | 
|  |  | 
|  | ngx_sprintf(name + to->len, ".%0muA.DELETE%Z", num); | 
|  |  | 
|  | if (MoveFile((const char *) to->data, (const char *) name) != 0) { | 
|  | break; | 
|  | } | 
|  |  | 
|  | collision = 1; | 
|  |  | 
|  | ngx_log_error(NGX_LOG_CRIT, log, ngx_errno, | 
|  | "MoveFile() \"%s\" to \"%s\" failed", to->data, name); | 
|  | } | 
|  |  | 
|  | if (MoveFile((const char *) from->data, (const char *) to->data) == 0) { | 
|  | err = ngx_errno; | 
|  |  | 
|  | } else { | 
|  | err = 0; | 
|  | } | 
|  |  | 
|  | if (DeleteFile((const char *) name) == 0) { | 
|  | ngx_log_error(NGX_LOG_CRIT, log, ngx_errno, | 
|  | "DeleteFile() \"%s\" failed", name); | 
|  | } | 
|  |  | 
|  | /* mutex_unlock() */ | 
|  |  | 
|  | ngx_free(name); | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_file_info(u_char *file, ngx_file_info_t *sb) | 
|  | { | 
|  | size_t                      len; | 
|  | long                        rc; | 
|  | u_short                    *u; | 
|  | ngx_err_t                   err; | 
|  | WIN32_FILE_ATTRIBUTE_DATA   fa; | 
|  | u_short                     utf16[NGX_UTF16_BUFLEN]; | 
|  |  | 
|  | len = NGX_UTF16_BUFLEN; | 
|  |  | 
|  | u = ngx_utf8_to_utf16(utf16, file, &len); | 
|  |  | 
|  | if (u == NULL) { | 
|  | return NGX_FILE_ERROR; | 
|  | } | 
|  |  | 
|  | rc = NGX_FILE_ERROR; | 
|  |  | 
|  | if (ngx_win32_check_filename(file, u, len) != NGX_OK) { | 
|  | goto failed; | 
|  | } | 
|  |  | 
|  | rc = GetFileAttributesExW(u, GetFileExInfoStandard, &fa); | 
|  |  | 
|  | 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); | 
|  | } | 
|  |  | 
|  | return rc; | 
|  | } | 
|  |  | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  |  | 
|  | 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 * | 
|  | ngx_realpath(u_char *path, u_char *resolved) | 
|  | { | 
|  | /* STUB */ | 
|  | return path; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_open_dir(ngx_str_t *name, ngx_dir_t *dir) | 
|  | { | 
|  | ngx_cpystrn(name->data + name->len, NGX_DIR_MASK, NGX_DIR_MASK_LEN + 1); | 
|  |  | 
|  | dir->dir = FindFirstFile((const char *) name->data, &dir->finddata); | 
|  |  | 
|  | if (dir->dir == INVALID_HANDLE_VALUE) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | dir->valid_info = 1; | 
|  | dir->ready = 1; | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_read_dir(ngx_dir_t *dir) | 
|  | { | 
|  | if (dir->ready) { | 
|  | dir->ready = 0; | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | if (FindNextFile(dir->dir, &dir->finddata) != 0) { | 
|  | dir->type = 1; | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_open_glob(ngx_glob_t *gl) | 
|  | { | 
|  | u_char     *p; | 
|  | size_t      len; | 
|  | ngx_err_t   err; | 
|  |  | 
|  | gl->dir = FindFirstFile((const char *) gl->pattern, &gl->finddata); | 
|  |  | 
|  | if (gl->dir == INVALID_HANDLE_VALUE) { | 
|  |  | 
|  | err = ngx_errno; | 
|  |  | 
|  | if ((err == ERROR_FILE_NOT_FOUND || err == ERROR_PATH_NOT_FOUND) | 
|  | && gl->test) | 
|  | { | 
|  | gl->no_match = 1; | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | 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); | 
|  |  | 
|  | gl->ready = 1; | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_read_glob(ngx_glob_t *gl, ngx_str_t *name) | 
|  | { | 
|  | size_t     len; | 
|  | ngx_err_t  err; | 
|  |  | 
|  | if (gl->no_match) { | 
|  | return NGX_DONE; | 
|  | } | 
|  |  | 
|  | if (gl->ready) { | 
|  | *name = gl->name; | 
|  |  | 
|  | gl->ready = 0; | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ngx_free(gl->name.data); | 
|  | gl->name.data = NULL; | 
|  |  | 
|  | if (FindNextFile(gl->dir, &gl->finddata) != 0) { | 
|  |  | 
|  | 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; | 
|  |  | 
|  | 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) | 
|  | { | 
|  | if (gl->name.data) { | 
|  | ngx_free(gl->name.data); | 
|  | } | 
|  |  | 
|  | if (gl->dir == INVALID_HANDLE_VALUE) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (FindClose(gl->dir) == 0) { | 
|  | 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; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_read_ahead(ngx_fd_t fd, size_t n) | 
|  | { | 
|  | return ~NGX_FILE_ERROR; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_directio_on(ngx_fd_t fd) | 
|  | { | 
|  | return ~NGX_FILE_ERROR; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_directio_off(ngx_fd_t fd) | 
|  | { | 
|  | return ~NGX_FILE_ERROR; | 
|  | } | 
|  |  | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | 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 * | 
|  | 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; | 
|  | last = utf16 + *len; | 
|  |  | 
|  | while (u < last) { | 
|  |  | 
|  | if (*p < 0x80) { | 
|  | *u++ = (u_short) *p; | 
|  |  | 
|  | if (*p == 0) { | 
|  | *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; | 
|  | } | 
|  |  | 
|  | ngx_memcpy(u, utf16, *len * 2); | 
|  |  | 
|  | utf16 = u; | 
|  | u += *len; | 
|  |  | 
|  | for ( ;; ) { | 
|  |  | 
|  | if (*p < 0x80) { | 
|  | *u++ = (u_short) *p; | 
|  |  | 
|  | if (*p == 0) { | 
|  | *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 */ | 
|  | } |