|  | // Copyright 2014 The Bazel Authors. All rights reserved. | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //    http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | package com.google.devtools.build.lib.unix; | 
|  |  | 
|  | import com.google.common.annotations.VisibleForTesting; | 
|  | import com.google.common.base.CharMatcher; | 
|  | import com.google.common.collect.ImmutableMap; | 
|  | import com.google.common.io.Files; | 
|  | import java.io.File; | 
|  | import java.io.IOException; | 
|  | import java.nio.charset.Charset; | 
|  | import java.util.HashMap; | 
|  | import java.util.List; | 
|  | import java.util.Map; | 
|  |  | 
|  | /** | 
|  | * Parse and return information from /proc/meminfo. In case of duplicate entries the first one is | 
|  | * used and other values are skipped. | 
|  | */ | 
|  | public class ProcMeminfoParser { | 
|  |  | 
|  | public static final String FILE = "/proc/meminfo"; | 
|  |  | 
|  | private final Map<String, Long> memInfo; | 
|  |  | 
|  | /** | 
|  | * Populates memory information by reading /proc/meminfo. | 
|  | * @throws IOException if reading the file failed. | 
|  | */ | 
|  | public ProcMeminfoParser() throws IOException { | 
|  | this(FILE); | 
|  | } | 
|  |  | 
|  | @VisibleForTesting | 
|  | public ProcMeminfoParser(String fileName) throws IOException { | 
|  | List<String> lines = Files.readLines(new File(fileName), Charset.defaultCharset()); | 
|  | Map<String, Long> newMemInfo = new HashMap<>(); | 
|  | for (String line : lines) { | 
|  | int colon = line.indexOf(':'); | 
|  | if (colon == -1) { | 
|  | continue; | 
|  | } | 
|  | String keyword = line.substring(0, colon); | 
|  | String valString = line.substring(colon + 1); | 
|  | try { | 
|  | long val =  Long.parseLong(CharMatcher.inRange('0', '9').retainFrom(valString)); | 
|  | newMemInfo.putIfAbsent(keyword, val); | 
|  | } catch (NumberFormatException e) { | 
|  | // Ignore: we'll fail later if somebody tries to capture this value. | 
|  | } | 
|  | } | 
|  | memInfo = ImmutableMap.copyOf(newMemInfo); | 
|  | } | 
|  |  | 
|  | /** Gets a named field in KB. */ | 
|  | long getRamKb(String keyword) throws KeywordNotFoundException { | 
|  | Long val = memInfo.get(keyword); | 
|  | if (val == null) { | 
|  | throw new KeywordNotFoundException(keyword); | 
|  | } | 
|  | return val; | 
|  | } | 
|  |  | 
|  | /** Return the total physical memory. */ | 
|  | public long getTotalKb() throws KeywordNotFoundException { | 
|  | return getRamKb("MemTotal"); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Convert KB to MB. | 
|  | */ | 
|  | public static double kbToMb(long kb) { | 
|  | return kb >> 10; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Reads the amount of *available* memory as reported by the kernel. See https://goo.gl/ABn283 for | 
|  | * why this is better than trying to figure it out ourselves. This corresponds to the MemAvailable | 
|  | * line in /proc/meminfo. | 
|  | */ | 
|  | public long getFreeRamKb() throws KeywordNotFoundException { | 
|  | if (memInfo.containsKey("MemAvailable")) { | 
|  | return getRamKb("MemAvailable"); | 
|  | } | 
|  | // We have no MemAvailable in /proc/meminfo; fall back to the previous estimation. | 
|  | return getRamKb("MemTotal") | 
|  | - getRamKb("Active") | 
|  | // Blaze doesn't want to use more than a third of inactive ram... | 
|  | - (long) (getRamKb("Inactive") * 0.3) | 
|  | // ...and doesn't want to assume more than 80% of the slab memory can be reallocated. | 
|  | - (long) (getRamKb("Slab") * 0.8); | 
|  | // That said, this estimate will be more inaccurate as it diverges from kernel internals. | 
|  | } | 
|  |  | 
|  | /** Exception thrown when /proc/meminfo does not have a requested key. Should be tolerated. */ | 
|  | public static class KeywordNotFoundException extends IOException { | 
|  | private KeywordNotFoundException(String keyword) { | 
|  | super("Can't locate " + keyword + " in the /proc/meminfo"); | 
|  | } | 
|  | } | 
|  | } |