/*
* Copyright (C) Igor Sysoev
* Copyright (C) Nginx, Inc.
*/
#include <ngx_config.h>
#include <ngx_core.h>
/*
* Base addresses selected by system for shared memory mappings are likely
* to be different on Windows Vista and later versions due to address space
* layout randomization. This is however incompatible with storing absolute
* addresses within the shared memory.
*
* To make it possible to store absolute addresses we create mappings
* at the same address in all processes by starting mappings at predefined
* addresses. The addresses were selected somewhat randomly in order to
* minimize the probability that some other library doing something similar
* conflicts with us. The addresses are from the following typically free
* blocks:
*
* - 0x10000000 .. 0x70000000 (about 1.5 GB in total) on 32-bit platforms
* - 0x000000007fff0000 .. 0x000007f68e8b0000 (about 8 TB) on 64-bit platforms
*
* Additionally, we allow to change the mapping address once it was detected
* to be different from one originally used. This is needed to support
* reconfiguration.
*/
#ifdef _WIN64
#define NGX_SHMEM_BASE 0x0000047047e00000
#else
#define NGX_SHMEM_BASE 0x2efe0000
#endif
ngx_uint_t ngx_allocation_granularity;
ngx_int_t
ngx_shm_alloc(ngx_shm_t *shm)
{
u_char *name;
uint64_t size;
static u_char *base = (u_char *) NGX_SHMEM_BASE;
name = ngx_alloc(shm->name.len + 2 + NGX_INT32_LEN, shm->log);
if (name == NULL) {
return NGX_ERROR;
}
(void) ngx_sprintf(name, "%V_%s%Z", &shm->name, ngx_unique);
ngx_set_errno(0);
size = shm->size;
shm->handle = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE,
(u_long) (size >> 32),
(u_long) (size & 0xffffffff),
(char *) name);
if (shm->handle == NULL) {
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"CreateFileMapping(%uz, %s) failed",
shm->size, name);
ngx_free(name);
return NGX_ERROR;
}
ngx_free(name);
if (ngx_errno == ERROR_ALREADY_EXISTS) {
shm->exists = 1;
}
shm->addr = MapViewOfFileEx(shm->handle, FILE_MAP_WRITE, 0, 0, 0, base);
if (shm->addr != NULL) {
base += ngx_align(size, ngx_allocation_granularity);
return NGX_OK;
}
ngx_log_debug3(NGX_LOG_DEBUG_CORE, shm->log, ngx_errno,
"MapViewOfFileEx(%uz, %p) of file mapping \"%V\" failed, "
"retry without a base address",
shm->size, base, &shm->name);
/*
* Order of shared memory zones may be different in the master process
* and worker processes after reconfiguration. As a result, the above
* may fail due to a conflict with a previously created mapping remapped
* to a different address. Additionally, there may be a conflict with
* some other uses of the memory. In this case we retry without a base
* address to let the system assign the address itself.
*/
shm->addr = MapViewOfFile(shm->handle, FILE_MAP_WRITE, 0, 0, 0);
if (shm->addr != NULL) {
return NGX_OK;
}
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"MapViewOfFile(%uz) of file mapping \"%V\" failed",
shm->size, &shm->name);
if (CloseHandle(shm->handle) == 0) {
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"CloseHandle() of file mapping \"%V\" failed",
&shm->name);
}
return NGX_ERROR;
}
ngx_int_t
ngx_shm_remap(ngx_shm_t *shm, u_char *addr)
{
if (UnmapViewOfFile(shm->addr) == 0) {
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"UnmapViewOfFile(%p) of file mapping \"%V\" failed",
shm->addr, &shm->name);
return NGX_ERROR;
}
shm->addr = MapViewOfFileEx(shm->handle, FILE_MAP_WRITE, 0, 0, 0, addr);
if (shm->addr != NULL) {
return NGX_OK;
}
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"MapViewOfFileEx(%uz, %p) of file mapping \"%V\" failed",
shm->size, addr, &shm->name);
return NGX_ERROR;
}
void
ngx_shm_free(ngx_shm_t *shm)
{
if (UnmapViewOfFile(shm->addr) == 0) {
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"UnmapViewOfFile(%p) of file mapping \"%V\" failed",
shm->addr, &shm->name);
}
if (CloseHandle(shm->handle) == 0) {
ngx_log_error(NGX_LOG_ALERT, shm->log, ngx_errno,
"CloseHandle() of file mapping \"%V\" failed",
&shm->name);
}
}