1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "mozilla/MemUtils.h"
#if defined(XP_WIN)
# include <windows.h>
# include "mozilla/Maybe.h"
#else
# include <sys/mman.h>
#endif
#if defined(XP_WIN)
typedef BOOL(WINAPI* PrefetchVirtualMemoryFn)(HANDLE, ULONG_PTR, PVOID, ULONG);
static mozilla::Maybe<PrefetchVirtualMemoryFn> sPrefetchVirtualMemory;
void MaybeInitPrefetchVirtualMemory() {
if (sPrefetchVirtualMemory.isNothing()) {
sPrefetchVirtualMemory.emplace(
reinterpret_cast<PrefetchVirtualMemoryFn>(GetProcAddress(
GetModuleHandleW(L"kernel32.dll"), "PrefetchVirtualMemory")));
}
}
#endif
bool mozilla::CanPrefetchMemory() {
#if defined(XP_SOLARIS) || defined(XP_UNIX)
return true;
#elif defined(XP_WIN)
MaybeInitPrefetchVirtualMemory();
return *sPrefetchVirtualMemory;
#else
return false;
#endif
}
void mozilla::PrefetchMemory(uint8_t* aStart, size_t aNumBytes) {
if (aNumBytes == 0) {
return;
}
#if defined(XP_SOLARIS)
posix_madvise(aStart, aNumBytes, POSIX_MADV_WILLNEED);
#elif defined(XP_UNIX)
madvise(aStart, aNumBytes, MADV_WILLNEED);
#elif defined(XP_WIN)
MaybeInitPrefetchVirtualMemory();
if (*sPrefetchVirtualMemory) {
// Normally, we'd use WIN32_MEMORY_RANGE_ENTRY, but that requires
// a different _WIN32_WINNT value before including windows.h, but
// that causes complications with unified sources. It's a simple
// enough struct anyways.
struct {
PVOID VirtualAddress;
SIZE_T NumberOfBytes;
} entry;
entry.VirtualAddress = aStart;
entry.NumberOfBytes = aNumBytes;
(*sPrefetchVirtualMemory)(GetCurrentProcess(), 1, &entry, 0);
return;
}
#endif
}
|