123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364 |
- /*-
- * Copyright 2009 Colin Percival
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
- * This file was originally written by Colin Percival as part of the Tarsnap
- * online backup system.
- */
- /* We use non-POSIX functionality in this file. */
- #undef _POSIX_C_SOURCE
- #undef _XOPEN_SOURCE
- #include "platform.h"
- #include <sys/types.h>
- #include <sys/resource.h>
- #ifdef HAVE_SYS_PARAM_H
- #include <sys/param.h>
- #endif
- #ifdef HAVE_SYS_SYSCTL_H
- #include <sys/sysctl.h>
- #endif
- #ifdef HAVE_SYS_SYSINFO_H
- #include <sys/sysinfo.h>
- #endif
- #include <assert.h>
- #include <errno.h>
- #include <stddef.h>
- #include <stdint.h>
- #include <string.h>
- #include <unistd.h>
- #ifdef DEBUG
- #include <stdio.h>
- #endif
- #include "memlimit.h"
- /* If we don't have CTL_HW, we can't use HW_USERMEM. */
- #ifndef CTL_HW
- #undef HW_USERMEM
- #endif
- /* Is RLIMIT_DATA relevant on this platform? */
- #if !defined(HAVE_MMAP) || defined(__OpenBSD__)
- #define USE_RLIMIT_DATA
- #endif
- #ifdef CTL_HW
- static int
- memlimit_sysctl_hw(size_t * memlimit, int mibleaf)
- {
- int mib[2];
- uint8_t sysctlbuf[8];
- size_t sysctlbuflen = 8;
- uint64_t sysctlval;
- /* Ask the kernel how much RAM we have. */
- mib[0] = CTL_HW;
- mib[1] = mibleaf;
- if (sysctl(mib, 2, sysctlbuf, &sysctlbuflen, NULL, 0))
- return (1);
- /*
- * If we read 8 bytes out, assume this is a system-endian uint64_t.
- * If we only read 4 bytes out, the OS is trying to give us a
- * uint32_t answer -- but given how many systems now have 4GB+ of RAM,
- * it's probably truncating, and we really can't trust the value we
- * have returned to us.
- */
- if (sysctlbuflen == sizeof(uint64_t))
- memcpy(&sysctlval, sysctlbuf, sizeof(uint64_t));
- else if (sysctlbuflen == sizeof(uint32_t))
- sysctlval = SIZE_MAX;
- else
- return (1);
- /* Return the sysctl value, but clamp to SIZE_MAX if necessary. */
- #if UINT64_MAX > SIZE_MAX
- if (sysctlval > SIZE_MAX)
- *memlimit = SIZE_MAX;
- else
- *memlimit = (size_t)sysctlval;
- #else
- *memlimit = sysctlval;
- #endif
- /* Success! */
- return (0);
- }
- #endif
- /* If we don't HAVE_STRUCT_SYSINFO, we can't use sysinfo. */
- #ifndef HAVE_STRUCT_SYSINFO
- #undef HAVE_SYSINFO
- #endif
- /* If we don't HAVE_STRUCT_SYSINFO_TOTALRAM, we can't use sysinfo. */
- #ifndef HAVE_STRUCT_SYSINFO_TOTALRAM
- #undef HAVE_SYSINFO
- #endif
- #ifdef HAVE_SYSINFO
- static int
- memlimit_sysinfo(size_t * memlimit)
- {
- struct sysinfo info;
- uint64_t totalmem;
- /* Get information from the kernel. */
- if (sysinfo(&info))
- return (1);
- totalmem = info.totalram;
- /* If we're on a modern kernel, adjust based on mem_unit. */
- #ifdef HAVE_STRUCT_SYSINFO_MEM_UNIT
- totalmem = totalmem * info.mem_unit;
- #endif
- /* Return the value, but clamp to SIZE_MAX if necessary. */
- #if UINT64_MAX > SIZE_MAX
- if (totalmem > SIZE_MAX)
- *memlimit = SIZE_MAX;
- else
- *memlimit = (size_t)totalmem;
- #else
- *memlimit = totalmem;
- #endif
- /* Success! */
- return (0);
- }
- #endif /* HAVE_SYSINFO */
- static int
- memlimit_rlimit(size_t * memlimit)
- {
- struct rlimit rl;
- uint64_t memrlimit;
- /* Find the least of... */
- memrlimit = (uint64_t)(-1);
- /* ... RLIMIT_AS... */
- #ifdef RLIMIT_AS
- if (getrlimit(RLIMIT_AS, &rl))
- return (1);
- if ((rl.rlim_cur != RLIM_INFINITY) &&
- ((uint64_t)rl.rlim_cur < memrlimit))
- memrlimit = (uint64_t)rl.rlim_cur;
- #endif
- #ifdef USE_RLIMIT_DATA
- /* ... RLIMIT_DATA (if we're not using mmap)... */
- if (getrlimit(RLIMIT_DATA, &rl))
- return (1);
- if ((rl.rlim_cur != RLIM_INFINITY) &&
- ((uint64_t)rl.rlim_cur < memrlimit))
- memrlimit = (uint64_t)rl.rlim_cur;
- #endif
- /* ... and RLIMIT_RSS. */
- #ifdef RLIMIT_RSS
- if (getrlimit(RLIMIT_RSS, &rl))
- return (1);
- if ((rl.rlim_cur != RLIM_INFINITY) &&
- ((uint64_t)rl.rlim_cur < memrlimit))
- memrlimit = (uint64_t)rl.rlim_cur;
- #endif
- /* Return the value, but clamp to SIZE_MAX if necessary. */
- #if UINT64_MAX > SIZE_MAX
- if (memrlimit > SIZE_MAX)
- *memlimit = SIZE_MAX;
- else
- *memlimit = (size_t)memrlimit;
- #else
- *memlimit = memrlimit;
- #endif
- /* Success! */
- return (0);
- }
- #ifdef _SC_PHYS_PAGES
- /* Some systems define _SC_PAGESIZE instead of _SC_PAGE_SIZE. */
- #ifndef _SC_PAGE_SIZE
- #define _SC_PAGE_SIZE _SC_PAGESIZE
- #endif
- static int
- memlimit_sysconf(size_t * memlimit)
- {
- long pagesize;
- long physpages;
- uint64_t totalmem;
- /* Set errno to 0 in order to distinguish "no limit" from "error". */
- errno = 0;
- /* Read the two limits. */
- if (((pagesize = sysconf(_SC_PAGE_SIZE)) == -1) ||
- ((physpages = sysconf(_SC_PHYS_PAGES)) == -1)) {
- /*
- * Did an error occur? OS X may return EINVAL due to not
- * supporting _SC_PHYS_PAGES in spite of defining it.
- */
- if (errno != 0 && errno != EINVAL)
- return (1);
- /* If not, there is no limit. */
- totalmem = (uint64_t)(-1);
- } else {
- /* Compute the limit. */
- totalmem = (uint64_t)(pagesize) * (uint64_t)(physpages);
- }
- /* Return the value, but clamp to SIZE_MAX if necessary. */
- #if UINT64_MAX > SIZE_MAX
- if (totalmem > SIZE_MAX)
- *memlimit = SIZE_MAX;
- else
- *memlimit = (size_t)totalmem;
- #else
- *memlimit = totalmem;
- #endif
- /* Success! */
- return (0);
- }
- #endif
- /**
- * memtouse(maxmem, maxmemfrac, memlimit):
- * Examine the system and return the amount of RAM which should be
- * used in ${memlimit}. This value should be the specified
- * ${maxmemfrac} fraction of available RAM, but no more than
- * ${maxmem} and no less than 1 MiB. ${maxmemfrac} must be larger than 0
- * and <= 1.0.
- */
- int
- memtouse(size_t maxmem, double maxmemfrac, size_t * memlimit)
- {
- size_t usermem_memlimit, memsize_memlimit;
- size_t sysinfo_memlimit, rlimit_memlimit;
- size_t sysconf_memlimit;
- size_t memlimit_min;
- size_t memavail;
- /* Check limit. */
- assert((maxmemfrac > 0) && (maxmemfrac <= 1.0));
- /* Get memory limits. */
- #ifdef HW_USERMEM
- if (memlimit_sysctl_hw(&usermem_memlimit, HW_USERMEM))
- return (1);
- #else
- usermem_memlimit = SIZE_MAX;
- #endif
- #ifdef HW_MEMSIZE
- if (memlimit_sysctl_hw(&memsize_memlimit, HW_MEMSIZE))
- return (1);
- #else
- memsize_memlimit = SIZE_MAX;
- #endif
- #ifdef HAVE_SYSINFO
- if (memlimit_sysinfo(&sysinfo_memlimit))
- return (1);
- #else
- sysinfo_memlimit = SIZE_MAX;
- #endif
- if (memlimit_rlimit(&rlimit_memlimit))
- return (1);
- #ifdef _SC_PHYS_PAGES
- if (memlimit_sysconf(&sysconf_memlimit))
- return (1);
- #else
- sysconf_memlimit = SIZE_MAX;
- #endif
- #ifdef DEBUG
- /* rlimit has two '\t' so that they line up. */
- fprintf(stderr, "Memory limits are:\n\tusermem:\t%zu\n"
- "\tmemsize:\t%zu\n\tsysinfo:\t%zu\n\trlimit:\t\t%zu\n"
- "\tsysconf:\t%zu\n", usermem_memlimit, memsize_memlimit,
- sysinfo_memlimit, rlimit_memlimit, sysconf_memlimit);
- #endif
- /*
- * Some systems return bogus values for hw.usermem due to ZFS making
- * use of wired pages. Assume that at least 50% of physical pages
- * are available to userland on demand.
- */
- if (sysconf_memlimit != SIZE_MAX) {
- if (usermem_memlimit < sysconf_memlimit / 2)
- usermem_memlimit = sysconf_memlimit / 2;
- }
- /* Find the smallest of them. */
- memlimit_min = SIZE_MAX;
- if (memlimit_min > usermem_memlimit)
- memlimit_min = usermem_memlimit;
- if (memlimit_min > memsize_memlimit)
- memlimit_min = memsize_memlimit;
- if (memlimit_min > sysinfo_memlimit)
- memlimit_min = sysinfo_memlimit;
- if (memlimit_min > rlimit_memlimit)
- memlimit_min = rlimit_memlimit;
- if (memlimit_min > sysconf_memlimit)
- memlimit_min = sysconf_memlimit;
- #ifdef DEBUG
- /* This line is continued after the maxmemfrac calculation. */
- fprintf(stderr, "Minimum is %zu;", memlimit_min);
- #endif
- /* Only use the specified fraction of the available memory. */
- memavail = (size_t)(maxmemfrac * (double)memlimit_min);
- #ifdef DEBUG
- fprintf(stderr, " will apply maxmemfrac of %g\n", maxmemfrac);
- #endif
- /* Don't use more than the specified maximum. */
- if ((maxmem > 0) && (memavail > maxmem))
- memavail = maxmem;
- /* But always allow at least 1 MiB. */
- if (memavail < 1048576)
- memavail = 1048576;
- #ifdef DEBUG
- fprintf(stderr, "Memory allowed:\t\t%zu\n", memavail);
- #endif
- /* Return limit via the provided pointer. */
- *memlimit = memavail;
- return (0);
- }
|