| |
| /* |
| * 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) |
| { |
| u_char *pattern, *p; |
| ngx_err_t err; |
| |
| pattern = malloc(name->len + 3); |
| if (pattern == NULL) { |
| return NGX_ERROR; |
| } |
| |
| p = ngx_cpymem(pattern, name->data, name->len); |
| |
| *p++ = '/'; |
| *p++ = '*'; |
| *p = '\0'; |
| |
| dir->dir = FindFirstFile((const char *) pattern, &dir->finddata); |
| |
| if (dir->dir == INVALID_HANDLE_VALUE) { |
| err = ngx_errno; |
| ngx_free(pattern); |
| ngx_set_errno(err); |
| return NGX_ERROR; |
| } |
| |
| ngx_free(pattern); |
| |
| 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_close_dir(ngx_dir_t *dir) |
| { |
| if (FindClose(dir->dir) == 0) { |
| return NGX_ERROR; |
| } |
| |
| return NGX_OK; |
| } |
| |
| |
| 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; |
| } |
| |
| |
| off_t |
| ngx_fs_available(u_char *name) |
| { |
| ULARGE_INTEGER navail; |
| |
| if (GetDiskFreeSpaceEx((const char *) name, &navail, NULL, NULL) == 0) { |
| return NGX_MAX_OFF_T_VALUE; |
| } |
| |
| return (off_t) navail.QuadPart; |
| } |
| |
| |
| 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; |
| } |
| |
| 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 * |
| 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; |
| } |
| |
| if (u + 1 == last) { |
| *len = u - utf16; |
| break; |
| } |
| |
| n = ngx_utf8_decode(&p, 4); |
| |
| if (n > 0x10ffff) { |
| ngx_set_errno(NGX_EILSEQ); |
| return NULL; |
| } |
| |
| if (n > 0xffff) { |
| n -= 0x10000; |
| *u++ = (u_short) (0xd800 + (n >> 10)); |
| *u++ = (u_short) (0xdc00 + (n & 0x03ff)); |
| continue; |
| } |
| |
| *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 > 0x10ffff) { |
| ngx_free(utf16); |
| ngx_set_errno(NGX_EILSEQ); |
| return NULL; |
| } |
| |
| if (n > 0xffff) { |
| n -= 0x10000; |
| *u++ = (u_short) (0xd800 + (n >> 10)); |
| *u++ = (u_short) (0xdc00 + (n & 0x03ff)); |
| continue; |
| } |
| |
| *u++ = (u_short) n; |
| } |
| |
| /* unreachable */ |
| } |