1
/*****************************************************************************
3
Copyright (c) 1995, 2009, Innobase Oy. All Rights Reserved.
5
This program is free software; you can redistribute it and/or modify it under
6
the terms of the GNU General Public License as published by the Free Software
7
Foundation; version 2 of the License.
9
This program is distributed in the hope that it will be useful, but WITHOUT
10
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
13
You should have received a copy of the GNU General Public License along with
14
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
15
Place, Suite 330, Boston, MA 02111-1307 USA
17
*****************************************************************************/
19
/**************************************************//**
21
The interface to the operating system
22
process control primitives
24
Created 9/30/1995 Heikki Tuuri
25
*******************************************************/
37
/* FreeBSD for example has only MAP_ANON, Linux has MAP_ANONYMOUS and
38
MAP_ANON but MAP_ANON is marked as deprecated */
39
#if defined(MAP_ANONYMOUS)
40
#define OS_MAP_ANON MAP_ANONYMOUS
41
#elif defined(MAP_ANON)
42
#define OS_MAP_ANON MAP_ANON
45
UNIV_INTERN ibool os_use_large_pages;
46
/* Large page size. This may be a boot-time option on some platforms */
47
UNIV_INTERN ulint os_large_page_size;
49
/****************************************************************//**
50
Converts the current process id to a number. It is not guaranteed that the
51
number is unique. In Linux returns the 'process number' of the current
52
thread. That number is the same as one sees in 'top', for example. In Linux
53
the thread id is not the same as one sees in 'top'.
54
@return process id as a number */
57
os_proc_get_number(void)
58
/*====================*/
61
return((ulint)GetCurrentProcessId());
63
return((ulint)getpid());
67
/****************************************************************//**
68
Allocates large pages memory.
69
@return allocated memory */
74
ulint* n) /*!< in/out: number of bytes */
78
#if defined HAVE_LARGE_PAGES && defined UNIV_LINUX
82
if (!os_use_large_pages || !os_large_page_size) {
86
/* Align block size to os_large_page_size */
87
ut_ad(ut_is_2pow(os_large_page_size));
88
size = ut_2pow_round(*n + (os_large_page_size - 1),
91
shmid = shmget(IPC_PRIVATE, (size_t)size, SHM_HUGETLB | SHM_R | SHM_W);
93
fprintf(stderr, "InnoDB: HugeTLB: Warning: Failed to allocate"
94
" %lu bytes. errno %d\n", size, errno);
97
ptr = shmat(shmid, NULL, 0);
98
if (ptr == (void *)-1) {
99
fprintf(stderr, "InnoDB: HugeTLB: Warning: Failed to"
100
" attach shared memory segment, errno %d\n",
104
/* Remove the shared memory segment so that it will be
105
automatically freed after memory is detached or
107
shmctl(shmid, IPC_RMID, &buf);
112
os_fast_mutex_lock(&ut_list_mutex);
113
ut_total_allocated_memory += size;
114
os_fast_mutex_unlock(&ut_list_mutex);
115
# ifdef UNIV_SET_MEM_TO_ZERO
116
memset(ptr, '\0', size);
118
UNIV_MEM_ALLOC(ptr, size);
122
fprintf(stderr, "InnoDB HugeTLB: Warning: Using conventional"
125
#endif /* HAVE_LARGE_PAGES && UNIV_LINUX */
128
SYSTEM_INFO system_info;
129
GetSystemInfo(&system_info);
131
/* Align block size to system page size */
132
ut_ad(ut_is_2pow(system_info.dwPageSize));
133
/* system_info.dwPageSize is only 32-bit. Casting to ulint is required
134
on 64-bit Windows. */
135
size = *n = ut_2pow_round(*n + (system_info.dwPageSize - 1),
136
(ulint) system_info.dwPageSize);
137
ptr = VirtualAlloc(NULL, size, MEM_COMMIT | MEM_RESERVE,
140
fprintf(stderr, "InnoDB: VirtualAlloc(%lu bytes) failed;"
141
" Windows error %lu\n",
142
(ulong) size, (ulong) GetLastError());
144
os_fast_mutex_lock(&ut_list_mutex);
145
ut_total_allocated_memory += size;
146
os_fast_mutex_unlock(&ut_list_mutex);
147
UNIV_MEM_ALLOC(ptr, size);
149
#elif defined __NETWARE__ || !defined OS_MAP_ANON
151
ptr = ut_malloc_low(size, TRUE, FALSE);
153
# ifdef HAVE_GETPAGESIZE
154
size = getpagesize();
156
size = UNIV_PAGE_SIZE;
158
/* Align block size to system page size */
159
ut_ad(ut_is_2pow(size));
160
size = *n = ut_2pow_round(*n + (size - 1), size);
161
ptr = mmap(NULL, size, PROT_READ | PROT_WRITE,
162
MAP_PRIVATE | OS_MAP_ANON, -1, 0);
163
if (UNIV_UNLIKELY(ptr == (void*) -1)) {
164
fprintf(stderr, "InnoDB: mmap(%lu bytes) failed;"
166
(ulong) size, (ulong) errno);
169
os_fast_mutex_lock(&ut_list_mutex);
170
ut_total_allocated_memory += size;
171
os_fast_mutex_unlock(&ut_list_mutex);
172
UNIV_MEM_ALLOC(ptr, size);
178
/****************************************************************//**
179
Frees large pages memory. */
184
void *ptr, /*!< in: pointer returned by
185
os_mem_alloc_large() */
186
ulint size) /*!< in: size returned by
187
os_mem_alloc_large() */
189
os_fast_mutex_lock(&ut_list_mutex);
190
ut_a(ut_total_allocated_memory >= size);
191
os_fast_mutex_unlock(&ut_list_mutex);
193
#if defined HAVE_LARGE_PAGES && defined UNIV_LINUX
194
if (os_use_large_pages && os_large_page_size && !shmdt(ptr)) {
195
os_fast_mutex_lock(&ut_list_mutex);
196
ut_a(ut_total_allocated_memory >= size);
197
ut_total_allocated_memory -= size;
198
os_fast_mutex_unlock(&ut_list_mutex);
199
UNIV_MEM_FREE(ptr, size);
202
#endif /* HAVE_LARGE_PAGES && UNIV_LINUX */
204
/* When RELEASE memory, the size parameter must be 0.
205
Do not use MEM_RELEASE with MEM_DECOMMIT. */
206
if (!VirtualFree(ptr, 0, MEM_RELEASE)) {
207
fprintf(stderr, "InnoDB: VirtualFree(%p, %lu) failed;"
208
" Windows error %lu\n",
209
ptr, (ulong) size, (ulong) GetLastError());
211
os_fast_mutex_lock(&ut_list_mutex);
212
ut_a(ut_total_allocated_memory >= size);
213
ut_total_allocated_memory -= size;
214
os_fast_mutex_unlock(&ut_list_mutex);
215
UNIV_MEM_FREE(ptr, size);
217
#elif defined __NETWARE__ || !defined OS_MAP_ANON
220
if (munmap(ptr, size)) {
221
fprintf(stderr, "InnoDB: munmap(%p, %lu) failed;"
223
ptr, (ulong) size, (ulong) errno);
225
os_fast_mutex_lock(&ut_list_mutex);
226
ut_a(ut_total_allocated_memory >= size);
227
ut_total_allocated_memory -= size;
228
os_fast_mutex_unlock(&ut_list_mutex);
229
UNIV_MEM_FREE(ptr, size);