mirror of
https://github.com/MariaDB/server.git
synced 2025-01-25 00:04:33 +01:00
280 lines
7.9 KiB
C
280 lines
7.9 KiB
C
/*****************************************************************************
|
|
|
|
Copyright (c) 1995, 2009, Innobase Oy. All Rights Reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify it under
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
|
|
Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file os/os0proc.c
|
|
The interface to the operating system
|
|
process control primitives
|
|
|
|
Created 9/30/1995 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
#include "os0proc.h"
|
|
#ifdef UNIV_NONINL
|
|
#include "os0proc.ic"
|
|
#endif
|
|
|
|
#include "ut0mem.h"
|
|
#include "ut0byte.h"
|
|
|
|
/* Linux release version */
|
|
#if defined(UNIV_LINUX) && defined(_GNU_SOURCE)
|
|
#include <string.h> /* strverscmp() */
|
|
#include <sys/utsname.h> /* uname() */
|
|
#endif
|
|
|
|
/* FreeBSD for example has only MAP_ANON, Linux has MAP_ANONYMOUS and
|
|
MAP_ANON but MAP_ANON is marked as deprecated */
|
|
#if defined(MAP_ANONYMOUS)
|
|
#define OS_MAP_ANON MAP_ANONYMOUS
|
|
#elif defined(MAP_ANON)
|
|
#define OS_MAP_ANON MAP_ANON
|
|
#endif
|
|
|
|
/* Linux's MAP_POPULATE */
|
|
#if defined(MAP_POPULATE)
|
|
#define OS_MAP_POPULATE MAP_POPULATE
|
|
#else
|
|
#define OS_MAP_POPULATE 0
|
|
#endif
|
|
|
|
UNIV_INTERN ibool os_use_large_pages;
|
|
/* Large page size. This may be a boot-time option on some platforms */
|
|
UNIV_INTERN ulint os_large_page_size;
|
|
|
|
/****************************************************************//**
|
|
Converts the current process id to a number. It is not guaranteed that the
|
|
number is unique. In Linux returns the 'process number' of the current
|
|
thread. That number is the same as one sees in 'top', for example. In Linux
|
|
the thread id is not the same as one sees in 'top'.
|
|
@return process id as a number */
|
|
UNIV_INTERN
|
|
ulint
|
|
os_proc_get_number(void)
|
|
/*====================*/
|
|
{
|
|
#ifdef __WIN__
|
|
return((ulint)GetCurrentProcessId());
|
|
#else
|
|
return((ulint)getpid());
|
|
#endif
|
|
}
|
|
|
|
/****************************************************************//**
|
|
Retrieve and compare operating system release.
|
|
@return TRUE if the OS release is equal to, or later than release. */
|
|
UNIV_INTERN
|
|
ibool
|
|
os_compare_release(
|
|
/*===============*/
|
|
const char* release /*!< in: OS release */
|
|
__attribute__((unused)))
|
|
{
|
|
#if defined(UNIV_LINUX) && defined(_GNU_SOURCE)
|
|
struct utsname name;
|
|
return uname(&name) == 0 && strverscmp(name.release, release) >= 0;
|
|
#else
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
/****************************************************************//**
|
|
Allocates large pages memory.
|
|
@return allocated memory */
|
|
UNIV_INTERN
|
|
void*
|
|
os_mem_alloc_large(
|
|
/*===============*/
|
|
ulint* n, /*!< in/out: number of bytes */
|
|
ibool populate) /*!< in: virtual page preallocation */
|
|
{
|
|
void* ptr;
|
|
ulint size;
|
|
#if defined HAVE_LARGE_PAGES && defined UNIV_LINUX
|
|
int shmid;
|
|
struct shmid_ds buf;
|
|
|
|
if (!os_use_large_pages || !os_large_page_size) {
|
|
goto skip;
|
|
}
|
|
|
|
/* Align block size to os_large_page_size */
|
|
ut_ad(ut_is_2pow(os_large_page_size));
|
|
size = ut_2pow_round(*n + (os_large_page_size - 1),
|
|
os_large_page_size);
|
|
|
|
shmid = shmget(IPC_PRIVATE, (size_t)size, SHM_HUGETLB | SHM_R | SHM_W);
|
|
if (shmid < 0) {
|
|
fprintf(stderr, "InnoDB: HugeTLB: Warning: Failed to allocate"
|
|
" %lu bytes. errno %d\n", size, errno);
|
|
ptr = NULL;
|
|
} else {
|
|
ptr = shmat(shmid, NULL, 0);
|
|
if (ptr == (void *)-1) {
|
|
fprintf(stderr, "InnoDB: HugeTLB: Warning: Failed to"
|
|
" attach shared memory segment, errno %d\n",
|
|
errno);
|
|
ptr = NULL;
|
|
}
|
|
|
|
/* Remove the shared memory segment so that it will be
|
|
automatically freed after memory is detached or
|
|
process exits */
|
|
shmctl(shmid, IPC_RMID, &buf);
|
|
}
|
|
|
|
if (ptr) {
|
|
*n = size;
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_total_allocated_memory += size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_ALLOC(ptr, size);
|
|
return(ptr);
|
|
}
|
|
|
|
fprintf(stderr, "InnoDB HugeTLB: Warning: Using conventional"
|
|
" memory pool\n");
|
|
skip:
|
|
#endif /* HAVE_LARGE_PAGES && UNIV_LINUX */
|
|
|
|
#ifdef __WIN__
|
|
SYSTEM_INFO system_info;
|
|
GetSystemInfo(&system_info);
|
|
|
|
/* Align block size to system page size */
|
|
ut_ad(ut_is_2pow(system_info.dwPageSize));
|
|
/* system_info.dwPageSize is only 32-bit. Casting to ulint is required
|
|
on 64-bit Windows. */
|
|
size = *n = ut_2pow_round(*n + (system_info.dwPageSize - 1),
|
|
(ulint) system_info.dwPageSize);
|
|
ptr = VirtualAlloc(NULL, size, MEM_COMMIT | MEM_RESERVE,
|
|
PAGE_READWRITE);
|
|
if (!ptr) {
|
|
fprintf(stderr, "InnoDB: VirtualAlloc(%lu bytes) failed;"
|
|
" Windows error %lu\n",
|
|
(ulong) size, (ulong) GetLastError());
|
|
} else {
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_total_allocated_memory += size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_ALLOC(ptr, size);
|
|
}
|
|
#elif !defined OS_MAP_ANON
|
|
size = *n;
|
|
ptr = ut_malloc_low(size, TRUE, FALSE);
|
|
#else
|
|
# ifdef HAVE_GETPAGESIZE
|
|
size = getpagesize();
|
|
# else
|
|
size = UNIV_PAGE_SIZE;
|
|
# endif
|
|
/* Align block size to system page size */
|
|
ut_ad(ut_is_2pow(size));
|
|
size = *n = ut_2pow_round(*n + (size - 1), size);
|
|
ptr = mmap(NULL, size, PROT_READ | PROT_WRITE,
|
|
MAP_PRIVATE | OS_MAP_ANON |
|
|
(populate ? OS_MAP_POPULATE : 0), -1, 0);
|
|
if (UNIV_UNLIKELY(ptr == (void*) -1)) {
|
|
fprintf(stderr, "InnoDB: mmap(%lu bytes) failed;"
|
|
" errno %lu\n",
|
|
(ulong) size, (ulong) errno);
|
|
return(NULL);
|
|
} else {
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_total_allocated_memory += size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_ALLOC(ptr, size);
|
|
}
|
|
#endif
|
|
|
|
#if OS_MAP_ANON && OS_MAP_POPULATE
|
|
/* MAP_POPULATE is only supported for private mappings
|
|
since Linux 2.6.23. */
|
|
populate = populate && !os_compare_release("2.6.23");
|
|
|
|
if (populate) {
|
|
fprintf(stderr, "InnoDB: Warning: mmap(MAP_POPULATE) "
|
|
"is not supported for private mappings. "
|
|
"Forcing preallocation by faulting in pages.\n");
|
|
}
|
|
#endif
|
|
|
|
/* Initialize the entire buffer to force the allocation
|
|
of physical memory page frames. */
|
|
if (populate) {
|
|
memset(ptr, '\0', size);
|
|
}
|
|
|
|
return(ptr);
|
|
}
|
|
|
|
/****************************************************************//**
|
|
Frees large pages memory. */
|
|
UNIV_INTERN
|
|
void
|
|
os_mem_free_large(
|
|
/*==============*/
|
|
void *ptr, /*!< in: pointer returned by
|
|
os_mem_alloc_large() */
|
|
ulint size) /*!< in: size returned by
|
|
os_mem_alloc_large() */
|
|
{
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_a(ut_total_allocated_memory >= size);
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
|
|
#if defined HAVE_LARGE_PAGES && defined UNIV_LINUX
|
|
if (os_use_large_pages && os_large_page_size && !shmdt(ptr)) {
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_a(ut_total_allocated_memory >= size);
|
|
ut_total_allocated_memory -= size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_FREE(ptr, size);
|
|
return;
|
|
}
|
|
#endif /* HAVE_LARGE_PAGES && UNIV_LINUX */
|
|
#ifdef __WIN__
|
|
/* When RELEASE memory, the size parameter must be 0.
|
|
Do not use MEM_RELEASE with MEM_DECOMMIT. */
|
|
if (!VirtualFree(ptr, 0, MEM_RELEASE)) {
|
|
fprintf(stderr, "InnoDB: VirtualFree(%p, %lu) failed;"
|
|
" Windows error %lu\n",
|
|
ptr, (ulong) size, (ulong) GetLastError());
|
|
} else {
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_a(ut_total_allocated_memory >= size);
|
|
ut_total_allocated_memory -= size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_FREE(ptr, size);
|
|
}
|
|
#elif !defined OS_MAP_ANON
|
|
ut_free(ptr);
|
|
#else
|
|
if (munmap(ptr, size)) {
|
|
fprintf(stderr, "InnoDB: munmap(%p, %lu) failed;"
|
|
" errno %lu\n",
|
|
ptr, (ulong) size, (ulong) errno);
|
|
} else {
|
|
os_fast_mutex_lock(&ut_list_mutex);
|
|
ut_a(ut_total_allocated_memory >= size);
|
|
ut_total_allocated_memory -= size;
|
|
os_fast_mutex_unlock(&ut_list_mutex);
|
|
UNIV_MEM_FREE(ptr, size);
|
|
}
|
|
#endif
|
|
}
|