xref: /openbsd-src/gnu/llvm/compiler-rt/lib/profile/InstrProfilingUtil.c (revision 810390e339a5425391477d5d41c78d7cab2424ac)
13cab2bb3Spatrick /*===- InstrProfilingUtil.c - Support library for PGO instrumentation -----===*\
23cab2bb3Spatrick |*
33cab2bb3Spatrick |* Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick |* See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick |* SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick |*
73cab2bb3Spatrick \*===----------------------------------------------------------------------===*/
83cab2bb3Spatrick 
93cab2bb3Spatrick #ifdef _WIN32
103cab2bb3Spatrick #include <direct.h>
113cab2bb3Spatrick #include <process.h>
123cab2bb3Spatrick #include <windows.h>
133cab2bb3Spatrick #include "WindowsMMap.h"
143cab2bb3Spatrick #else
15d89ec533Spatrick #include <errno.h>
16d89ec533Spatrick #include <fcntl.h>
173cab2bb3Spatrick #include <sys/file.h>
18d89ec533Spatrick #include <sys/mman.h>
193cab2bb3Spatrick #include <sys/stat.h>
203cab2bb3Spatrick #include <sys/types.h>
213cab2bb3Spatrick #include <unistd.h>
223cab2bb3Spatrick #endif
233cab2bb3Spatrick 
243cab2bb3Spatrick #ifdef COMPILER_RT_HAS_UNAME
253cab2bb3Spatrick #include <sys/utsname.h>
263cab2bb3Spatrick #endif
273cab2bb3Spatrick 
283cab2bb3Spatrick #include <stdlib.h>
293cab2bb3Spatrick #include <string.h>
303cab2bb3Spatrick 
313cab2bb3Spatrick #if defined(__linux__)
323cab2bb3Spatrick #include <signal.h>
333cab2bb3Spatrick #include <sys/prctl.h>
343cab2bb3Spatrick #endif
353cab2bb3Spatrick 
36d89ec533Spatrick #if defined(__Fuchsia__)
37*810390e3Srobert #include <zircon/process.h>
38d89ec533Spatrick #include <zircon/syscalls.h>
39d89ec533Spatrick #endif
40d89ec533Spatrick 
41*810390e3Srobert #if defined(__FreeBSD__)
42*810390e3Srobert #include <signal.h>
43*810390e3Srobert #include <sys/procctl.h>
44*810390e3Srobert #endif
45*810390e3Srobert 
463cab2bb3Spatrick #include "InstrProfiling.h"
473cab2bb3Spatrick #include "InstrProfilingUtil.h"
483cab2bb3Spatrick 
49*810390e3Srobert COMPILER_RT_VISIBILITY unsigned lprofDirMode = 0755;
503cab2bb3Spatrick 
513cab2bb3Spatrick COMPILER_RT_VISIBILITY
__llvm_profile_recursive_mkdir(char * path)523cab2bb3Spatrick void __llvm_profile_recursive_mkdir(char *path) {
533cab2bb3Spatrick   int i;
543cab2bb3Spatrick   int start = 1;
553cab2bb3Spatrick 
563cab2bb3Spatrick #if defined(__ANDROID__) && defined(__ANDROID_API__) &&                        \
573cab2bb3Spatrick     defined(__ANDROID_API_FUTURE__) &&                                         \
583cab2bb3Spatrick     __ANDROID_API__ == __ANDROID_API_FUTURE__
593cab2bb3Spatrick   // Avoid spammy selinux denial messages in Android by not attempting to
603cab2bb3Spatrick   // create directories in GCOV_PREFIX.  These denials occur when creating (or
613cab2bb3Spatrick   // even attempting to stat()) top-level directories like "/data".
623cab2bb3Spatrick   //
633cab2bb3Spatrick   // Do so by ignoring ${GCOV_PREFIX} when invoking mkdir().
643cab2bb3Spatrick   const char *gcov_prefix = getenv("GCOV_PREFIX");
653cab2bb3Spatrick   if (gcov_prefix != NULL) {
663cab2bb3Spatrick     const int gcov_prefix_len = strlen(gcov_prefix);
673cab2bb3Spatrick     if (strncmp(path, gcov_prefix, gcov_prefix_len) == 0)
683cab2bb3Spatrick       start = gcov_prefix_len;
693cab2bb3Spatrick   }
703cab2bb3Spatrick #endif
713cab2bb3Spatrick 
723cab2bb3Spatrick   for (i = start; path[i] != '\0'; ++i) {
733cab2bb3Spatrick     char save = path[i];
743cab2bb3Spatrick     if (!IS_DIR_SEPARATOR(path[i]))
753cab2bb3Spatrick       continue;
763cab2bb3Spatrick     path[i] = '\0';
773cab2bb3Spatrick #ifdef _WIN32
783cab2bb3Spatrick     _mkdir(path);
793cab2bb3Spatrick #else
803cab2bb3Spatrick     /* Some of these will fail, ignore it. */
813cab2bb3Spatrick     mkdir(path, __llvm_profile_get_dir_mode());
823cab2bb3Spatrick #endif
833cab2bb3Spatrick     path[i] = save;
843cab2bb3Spatrick   }
853cab2bb3Spatrick }
863cab2bb3Spatrick 
873cab2bb3Spatrick COMPILER_RT_VISIBILITY
__llvm_profile_set_dir_mode(unsigned Mode)883cab2bb3Spatrick void __llvm_profile_set_dir_mode(unsigned Mode) { lprofDirMode = Mode; }
893cab2bb3Spatrick 
903cab2bb3Spatrick COMPILER_RT_VISIBILITY
__llvm_profile_get_dir_mode(void)913cab2bb3Spatrick unsigned __llvm_profile_get_dir_mode(void) { return lprofDirMode; }
923cab2bb3Spatrick 
933cab2bb3Spatrick #if COMPILER_RT_HAS_ATOMICS != 1
943cab2bb3Spatrick COMPILER_RT_VISIBILITY
lprofBoolCmpXchg(void ** Ptr,void * OldV,void * NewV)953cab2bb3Spatrick uint32_t lprofBoolCmpXchg(void **Ptr, void *OldV, void *NewV) {
963cab2bb3Spatrick   void *R = *Ptr;
973cab2bb3Spatrick   if (R == OldV) {
983cab2bb3Spatrick     *Ptr = NewV;
993cab2bb3Spatrick     return 1;
1003cab2bb3Spatrick   }
1013cab2bb3Spatrick   return 0;
1023cab2bb3Spatrick }
1033cab2bb3Spatrick COMPILER_RT_VISIBILITY
lprofPtrFetchAdd(void ** Mem,long ByteIncr)1043cab2bb3Spatrick void *lprofPtrFetchAdd(void **Mem, long ByteIncr) {
1053cab2bb3Spatrick   void *Old = *Mem;
1063cab2bb3Spatrick   *((char **)Mem) += ByteIncr;
1073cab2bb3Spatrick   return Old;
1083cab2bb3Spatrick }
1093cab2bb3Spatrick 
1103cab2bb3Spatrick #endif
1113cab2bb3Spatrick 
1123cab2bb3Spatrick #ifdef _WIN32
lprofGetHostName(char * Name,int Len)1133cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofGetHostName(char *Name, int Len) {
1143cab2bb3Spatrick   WCHAR Buffer[COMPILER_RT_MAX_HOSTLEN];
1153cab2bb3Spatrick   DWORD BufferSize = sizeof(Buffer);
1163cab2bb3Spatrick   BOOL Result =
1173cab2bb3Spatrick       GetComputerNameExW(ComputerNameDnsFullyQualified, Buffer, &BufferSize);
1183cab2bb3Spatrick   if (!Result)
1193cab2bb3Spatrick     return -1;
1203cab2bb3Spatrick   if (WideCharToMultiByte(CP_UTF8, 0, Buffer, -1, Name, Len, NULL, NULL) == 0)
1213cab2bb3Spatrick     return -1;
1223cab2bb3Spatrick   return 0;
1233cab2bb3Spatrick }
1243cab2bb3Spatrick #elif defined(COMPILER_RT_HAS_UNAME)
lprofGetHostName(char * Name,int Len)1253cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofGetHostName(char *Name, int Len) {
1263cab2bb3Spatrick   struct utsname N;
1273cab2bb3Spatrick   int R = uname(&N);
1283cab2bb3Spatrick   if (R >= 0) {
1293cab2bb3Spatrick     strncpy(Name, N.nodename, Len);
1303cab2bb3Spatrick     return 0;
1313cab2bb3Spatrick   }
1323cab2bb3Spatrick   return R;
1333cab2bb3Spatrick }
1343cab2bb3Spatrick #endif
1353cab2bb3Spatrick 
lprofLockFd(int fd)1363cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofLockFd(int fd) {
1373cab2bb3Spatrick #ifdef COMPILER_RT_HAS_FCNTL_LCK
1383cab2bb3Spatrick   struct flock s_flock;
1393cab2bb3Spatrick 
1403cab2bb3Spatrick   s_flock.l_whence = SEEK_SET;
1413cab2bb3Spatrick   s_flock.l_start = 0;
1423cab2bb3Spatrick   s_flock.l_len = 0; /* Until EOF.  */
1433cab2bb3Spatrick   s_flock.l_pid = getpid();
1443cab2bb3Spatrick   s_flock.l_type = F_WRLCK;
1453cab2bb3Spatrick 
1463cab2bb3Spatrick   while (fcntl(fd, F_SETLKW, &s_flock) == -1) {
1473cab2bb3Spatrick     if (errno != EINTR) {
1483cab2bb3Spatrick       if (errno == ENOLCK) {
1493cab2bb3Spatrick         return -1;
1503cab2bb3Spatrick       }
1513cab2bb3Spatrick       break;
1523cab2bb3Spatrick     }
1533cab2bb3Spatrick   }
1543cab2bb3Spatrick   return 0;
1553cab2bb3Spatrick #else
1563cab2bb3Spatrick   flock(fd, LOCK_EX);
1573cab2bb3Spatrick   return 0;
1583cab2bb3Spatrick #endif
1593cab2bb3Spatrick }
1603cab2bb3Spatrick 
lprofUnlockFd(int fd)1613cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofUnlockFd(int fd) {
1623cab2bb3Spatrick #ifdef COMPILER_RT_HAS_FCNTL_LCK
1633cab2bb3Spatrick   struct flock s_flock;
1643cab2bb3Spatrick 
1653cab2bb3Spatrick   s_flock.l_whence = SEEK_SET;
1663cab2bb3Spatrick   s_flock.l_start = 0;
1673cab2bb3Spatrick   s_flock.l_len = 0; /* Until EOF.  */
1683cab2bb3Spatrick   s_flock.l_pid = getpid();
1693cab2bb3Spatrick   s_flock.l_type = F_UNLCK;
1703cab2bb3Spatrick 
1713cab2bb3Spatrick   while (fcntl(fd, F_SETLKW, &s_flock) == -1) {
1723cab2bb3Spatrick     if (errno != EINTR) {
1733cab2bb3Spatrick       if (errno == ENOLCK) {
1743cab2bb3Spatrick         return -1;
1753cab2bb3Spatrick       }
1763cab2bb3Spatrick       break;
1773cab2bb3Spatrick     }
1783cab2bb3Spatrick   }
1793cab2bb3Spatrick   return 0;
1803cab2bb3Spatrick #else
1813cab2bb3Spatrick   flock(fd, LOCK_UN);
1823cab2bb3Spatrick   return 0;
1833cab2bb3Spatrick #endif
1843cab2bb3Spatrick }
1853cab2bb3Spatrick 
lprofLockFileHandle(FILE * F)1863cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofLockFileHandle(FILE *F) {
1873cab2bb3Spatrick   int fd;
1883cab2bb3Spatrick #if defined(_WIN32)
1893cab2bb3Spatrick   fd = _fileno(F);
1903cab2bb3Spatrick #else
1913cab2bb3Spatrick   fd = fileno(F);
1923cab2bb3Spatrick #endif
1933cab2bb3Spatrick   return lprofLockFd(fd);
1943cab2bb3Spatrick }
1953cab2bb3Spatrick 
lprofUnlockFileHandle(FILE * F)1963cab2bb3Spatrick COMPILER_RT_VISIBILITY int lprofUnlockFileHandle(FILE *F) {
1973cab2bb3Spatrick   int fd;
1983cab2bb3Spatrick #if defined(_WIN32)
1993cab2bb3Spatrick   fd = _fileno(F);
2003cab2bb3Spatrick #else
2013cab2bb3Spatrick   fd = fileno(F);
2023cab2bb3Spatrick #endif
2033cab2bb3Spatrick   return lprofUnlockFd(fd);
2043cab2bb3Spatrick }
2053cab2bb3Spatrick 
lprofOpenFileEx(const char * ProfileName)2063cab2bb3Spatrick COMPILER_RT_VISIBILITY FILE *lprofOpenFileEx(const char *ProfileName) {
2073cab2bb3Spatrick   FILE *f;
2083cab2bb3Spatrick   int fd;
2093cab2bb3Spatrick #ifdef COMPILER_RT_HAS_FCNTL_LCK
2103cab2bb3Spatrick   fd = open(ProfileName, O_RDWR | O_CREAT, 0666);
2113cab2bb3Spatrick   if (fd < 0)
2123cab2bb3Spatrick     return NULL;
2133cab2bb3Spatrick 
2143cab2bb3Spatrick   if (lprofLockFd(fd) != 0)
2153cab2bb3Spatrick     PROF_WARN("Data may be corrupted during profile merging : %s\n",
2163cab2bb3Spatrick               "Fail to obtain file lock due to system limit.");
2173cab2bb3Spatrick 
2183cab2bb3Spatrick   f = fdopen(fd, "r+b");
2193cab2bb3Spatrick #elif defined(_WIN32)
2203cab2bb3Spatrick   // FIXME: Use the wide variants to handle Unicode filenames.
2213cab2bb3Spatrick   HANDLE h = CreateFileA(ProfileName, GENERIC_READ | GENERIC_WRITE,
2223cab2bb3Spatrick                          FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_ALWAYS,
2233cab2bb3Spatrick                          FILE_ATTRIBUTE_NORMAL, 0);
2243cab2bb3Spatrick   if (h == INVALID_HANDLE_VALUE)
2253cab2bb3Spatrick     return NULL;
2263cab2bb3Spatrick 
2273cab2bb3Spatrick   fd = _open_osfhandle((intptr_t)h, 0);
2283cab2bb3Spatrick   if (fd == -1) {
2293cab2bb3Spatrick     CloseHandle(h);
2303cab2bb3Spatrick     return NULL;
2313cab2bb3Spatrick   }
2323cab2bb3Spatrick 
2333cab2bb3Spatrick   if (lprofLockFd(fd) != 0)
2343cab2bb3Spatrick     PROF_WARN("Data may be corrupted during profile merging : %s\n",
2353cab2bb3Spatrick               "Fail to obtain file lock due to system limit.");
2363cab2bb3Spatrick 
2373cab2bb3Spatrick   f = _fdopen(fd, "r+b");
2383cab2bb3Spatrick   if (f == 0) {
2393cab2bb3Spatrick     CloseHandle(h);
2403cab2bb3Spatrick     return NULL;
2413cab2bb3Spatrick   }
2423cab2bb3Spatrick #else
2433cab2bb3Spatrick   /* Worst case no locking applied.  */
2443cab2bb3Spatrick   PROF_WARN("Concurrent file access is not supported : %s\n",
2453cab2bb3Spatrick             "lack file locking");
2463cab2bb3Spatrick   fd = open(ProfileName, O_RDWR | O_CREAT, 0666);
2473cab2bb3Spatrick   if (fd < 0)
2483cab2bb3Spatrick     return NULL;
2493cab2bb3Spatrick   f = fdopen(fd, "r+b");
2503cab2bb3Spatrick #endif
2513cab2bb3Spatrick 
2523cab2bb3Spatrick   return f;
2533cab2bb3Spatrick }
2543cab2bb3Spatrick 
lprofGetPathPrefix(int * PrefixStrip,size_t * PrefixLen)2553cab2bb3Spatrick COMPILER_RT_VISIBILITY const char *lprofGetPathPrefix(int *PrefixStrip,
2563cab2bb3Spatrick                                                       size_t *PrefixLen) {
2573cab2bb3Spatrick   const char *Prefix = getenv("GCOV_PREFIX");
2583cab2bb3Spatrick   const char *PrefixStripStr = getenv("GCOV_PREFIX_STRIP");
2593cab2bb3Spatrick 
2603cab2bb3Spatrick   *PrefixLen = 0;
2613cab2bb3Spatrick   *PrefixStrip = 0;
2623cab2bb3Spatrick   if (Prefix == NULL || Prefix[0] == '\0')
2633cab2bb3Spatrick     return NULL;
2643cab2bb3Spatrick 
2653cab2bb3Spatrick   if (PrefixStripStr) {
2663cab2bb3Spatrick     *PrefixStrip = atoi(PrefixStripStr);
2673cab2bb3Spatrick 
2683cab2bb3Spatrick     /* Negative GCOV_PREFIX_STRIP values are ignored */
2693cab2bb3Spatrick     if (*PrefixStrip < 0)
2703cab2bb3Spatrick       *PrefixStrip = 0;
2713cab2bb3Spatrick   } else {
2723cab2bb3Spatrick     *PrefixStrip = 0;
2733cab2bb3Spatrick   }
2743cab2bb3Spatrick   *PrefixLen = strlen(Prefix);
2753cab2bb3Spatrick 
2763cab2bb3Spatrick   return Prefix;
2773cab2bb3Spatrick }
2783cab2bb3Spatrick 
2793cab2bb3Spatrick COMPILER_RT_VISIBILITY void
lprofApplyPathPrefix(char * Dest,const char * PathStr,const char * Prefix,size_t PrefixLen,int PrefixStrip)2803cab2bb3Spatrick lprofApplyPathPrefix(char *Dest, const char *PathStr, const char *Prefix,
2813cab2bb3Spatrick                      size_t PrefixLen, int PrefixStrip) {
2823cab2bb3Spatrick 
2833cab2bb3Spatrick   const char *Ptr;
2843cab2bb3Spatrick   int Level;
2853cab2bb3Spatrick   const char *StrippedPathStr = PathStr;
2863cab2bb3Spatrick 
2873cab2bb3Spatrick   for (Level = 0, Ptr = PathStr + 1; Level < PrefixStrip; ++Ptr) {
2883cab2bb3Spatrick     if (*Ptr == '\0')
2893cab2bb3Spatrick       break;
2903cab2bb3Spatrick 
2913cab2bb3Spatrick     if (!IS_DIR_SEPARATOR(*Ptr))
2923cab2bb3Spatrick       continue;
2933cab2bb3Spatrick 
2943cab2bb3Spatrick     StrippedPathStr = Ptr;
2953cab2bb3Spatrick     ++Level;
2963cab2bb3Spatrick   }
2973cab2bb3Spatrick 
2983cab2bb3Spatrick   memcpy(Dest, Prefix, PrefixLen);
2993cab2bb3Spatrick 
3003cab2bb3Spatrick   if (!IS_DIR_SEPARATOR(Prefix[PrefixLen - 1]))
3013cab2bb3Spatrick     Dest[PrefixLen++] = DIR_SEPARATOR;
3023cab2bb3Spatrick 
3033cab2bb3Spatrick   memcpy(Dest + PrefixLen, StrippedPathStr, strlen(StrippedPathStr) + 1);
3043cab2bb3Spatrick }
3053cab2bb3Spatrick 
3063cab2bb3Spatrick COMPILER_RT_VISIBILITY const char *
lprofFindFirstDirSeparator(const char * Path)3073cab2bb3Spatrick lprofFindFirstDirSeparator(const char *Path) {
3083cab2bb3Spatrick   const char *Sep = strchr(Path, DIR_SEPARATOR);
3093cab2bb3Spatrick #if defined(DIR_SEPARATOR_2)
3103cab2bb3Spatrick   const char *Sep2 = strchr(Path, DIR_SEPARATOR_2);
3113cab2bb3Spatrick   if (Sep2 && (!Sep || Sep2 < Sep))
3123cab2bb3Spatrick     Sep = Sep2;
3133cab2bb3Spatrick #endif
3143cab2bb3Spatrick   return Sep;
3153cab2bb3Spatrick }
3163cab2bb3Spatrick 
lprofFindLastDirSeparator(const char * Path)3173cab2bb3Spatrick COMPILER_RT_VISIBILITY const char *lprofFindLastDirSeparator(const char *Path) {
3183cab2bb3Spatrick   const char *Sep = strrchr(Path, DIR_SEPARATOR);
3193cab2bb3Spatrick #if defined(DIR_SEPARATOR_2)
3203cab2bb3Spatrick   const char *Sep2 = strrchr(Path, DIR_SEPARATOR_2);
3213cab2bb3Spatrick   if (Sep2 && (!Sep || Sep2 > Sep))
3223cab2bb3Spatrick     Sep = Sep2;
3233cab2bb3Spatrick #endif
3243cab2bb3Spatrick   return Sep;
3253cab2bb3Spatrick }
3263cab2bb3Spatrick 
lprofSuspendSigKill(void)327*810390e3Srobert COMPILER_RT_VISIBILITY int lprofSuspendSigKill(void) {
3283cab2bb3Spatrick #if defined(__linux__)
3293cab2bb3Spatrick   int PDeachSig = 0;
3303cab2bb3Spatrick   /* Temporarily suspend getting SIGKILL upon exit of the parent process. */
3313cab2bb3Spatrick   if (prctl(PR_GET_PDEATHSIG, &PDeachSig) == 0 && PDeachSig == SIGKILL)
3323cab2bb3Spatrick     prctl(PR_SET_PDEATHSIG, 0);
3333cab2bb3Spatrick   return (PDeachSig == SIGKILL);
334*810390e3Srobert #elif defined(__FreeBSD__)
335*810390e3Srobert   int PDeachSig = 0, PDisableSig = 0;
336*810390e3Srobert   if (procctl(P_PID, 0, PROC_PDEATHSIG_STATUS, &PDeachSig) == 0 &&
337*810390e3Srobert       PDeachSig == SIGKILL)
338*810390e3Srobert     procctl(P_PID, 0, PROC_PDEATHSIG_CTL, &PDisableSig);
339*810390e3Srobert   return (PDeachSig == SIGKILL);
3403cab2bb3Spatrick #else
3413cab2bb3Spatrick   return 0;
3423cab2bb3Spatrick #endif
3433cab2bb3Spatrick }
3443cab2bb3Spatrick 
lprofRestoreSigKill(void)345*810390e3Srobert COMPILER_RT_VISIBILITY void lprofRestoreSigKill(void) {
3463cab2bb3Spatrick #if defined(__linux__)
3473cab2bb3Spatrick   prctl(PR_SET_PDEATHSIG, SIGKILL);
348*810390e3Srobert #elif defined(__FreeBSD__)
349*810390e3Srobert   int PEnableSig = SIGKILL;
350*810390e3Srobert   procctl(P_PID, 0, PROC_PDEATHSIG_CTL, &PEnableSig);
3513cab2bb3Spatrick #endif
3523cab2bb3Spatrick }
353d89ec533Spatrick 
lprofReleaseMemoryPagesToOS(uintptr_t Begin,uintptr_t End)354d89ec533Spatrick COMPILER_RT_VISIBILITY int lprofReleaseMemoryPagesToOS(uintptr_t Begin,
355d89ec533Spatrick                                                        uintptr_t End) {
356*810390e3Srobert #if defined(__ve__)
357*810390e3Srobert   // VE doesn't support madvise.
358*810390e3Srobert   return 0;
359*810390e3Srobert #else
360d89ec533Spatrick   size_t PageSize = getpagesize();
361d89ec533Spatrick   uintptr_t BeginAligned = lprofRoundUpTo((uintptr_t)Begin, PageSize);
362d89ec533Spatrick   uintptr_t EndAligned = lprofRoundDownTo((uintptr_t)End, PageSize);
363d89ec533Spatrick   if (BeginAligned < EndAligned) {
364d89ec533Spatrick #if defined(__Fuchsia__)
365d89ec533Spatrick     return _zx_vmar_op_range(_zx_vmar_root_self(), ZX_VMAR_OP_DECOMMIT,
366d89ec533Spatrick                              (zx_vaddr_t)BeginAligned,
367d89ec533Spatrick                              EndAligned - BeginAligned, NULL, 0);
368d89ec533Spatrick #else
369d89ec533Spatrick     return madvise((void *)BeginAligned, EndAligned - BeginAligned,
370d89ec533Spatrick                    MADV_DONTNEED);
371d89ec533Spatrick #endif
372d89ec533Spatrick   }
373d89ec533Spatrick   return 0;
374*810390e3Srobert #endif
375d89ec533Spatrick }
376