# Copyright (c) 2016 Cisco and/or its affiliates. # 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. from __future__ import print_function """VPP Huge Page Utilities""" import re from vpplib.VPPUtil import VPPUtil # VPP Huge page File DEFAULT_VPP_HUGE_PAGE_CONFIG_FILENAME = "/etc/vpp/80-vpp.conf" VPP_HUGEPAGE_CONFIG = """ vm.nr_hugepages={nr_hugepages} vm.max_map_count={max_map_count} vm.hugetlb_shm_group=0 kernel.shmmax={shmmax} """ class VppHugePageUtil(object): """ Huge Page Utilities """ def hugepages_dryrun_apply(self): """ Apply the huge page configuration """ node = self._node hugepages = node["hugepages"] vpp_hugepage_config = VPP_HUGEPAGE_CONFIG.format( nr_hugepages=hugepages["total"], max_map_count=hugepages["max_map_count"], shmmax=hugepages["shmax"], ) rootdir = node["rootdir"] filename = rootdir + node["hugepages"]["hugepage_config_file"] cmd = 'echo "{0}" | sudo tee {1}'.format(vpp_hugepage_config, filename) (ret, stdout, stderr) = VPPUtil.exec_command(cmd) if ret != 0: raise RuntimeError( "{} failed on node {} {} {}".format(cmd, node["host"], stdout, stderr) ) def get_actual_huge_pages(self): """ Get the current huge page configuration :returns the hugepage total, hugepage free, hugepage size, total memory, and total memory free :rtype: tuple """ # Get the memory information using /proc/meminfo cmd = "sudo cat /proc/meminfo" (ret, stdout, stderr) = VPPUtil.exec_command(cmd) if ret != 0: raise RuntimeError( "{} failed on node {} {} {}".format( cmd, self._node["host"], stdout, stderr ) ) total = re.findall(r"HugePages_Total:\s+\w+", stdout) free = re.findall(r"HugePages_Free:\s+\w+", stdout) size = re.findall(r"Hugepagesize:\s+\w+\s+\w+", stdout) memtotal = re.findall(r"MemTotal:\s+\w+\s+\w+", stdout) memfree = re.findall(r"MemFree:\s+\w+\s+\w+", stdout) total = total[0].split(":")[1].lstrip() free = free[0].split(":")[1].lstrip() size = size[0].split(":")[1].lstrip() memtotal = memtotal[0].split(":")[1].lstrip() memfree = memfree[0].split(":")[1].lstrip() return total, free, size, memtotal, memfree def show_huge_pages(self): """ Print the current huge page configuration """ node = self._node hugepages = node["hugepages"] print(" {:30}: {}".format("Total System Memory", hugepages["memtotal"])) print(" {:30}: {}".format("Total Free Memory", hugepages["memfree"])) print(" {:30}: {}".format("Actual Huge Page Total", hugepages["actual_total"])) print(" {:30}: {}".format("Configured Huge Page Total", hugepages["total"])) print(" {:30}: {}".format("Huge Pages Free", hugepages["free"])) print(" {:30}: {}".format("Huge Page Size", hugepages["size"])) def get_huge_page_config(self): """ Returns the huge page config. :returns: The map max count and shmmax """ total = self._node["hugepages"]["total"] max_map_count = int(total) * 2 + 1024 shmmax = int(total) * 2 * 1024 * 1024 return max_map_count, shmmax def __init__(self, node): self._node = node