# # Copyright 2010, 2013 Red Hat, Inc. # Cole Robinson # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, # MA 02110-1301 USA. import re from virtinst.xmlbuilder import XMLBuilder, XMLProperty def get_phy_cpus(conn): """ Get number of physical CPUs. """ hostinfo = conn.getInfo() pcpus = hostinfo[4] * hostinfo[5] * hostinfo[6] * hostinfo[7] return pcpus class DomainNumatune(XMLBuilder): """ Class for generating XML """ @staticmethod def validate_cpuset(conn, val): if val is None or val == "": return if not isinstance(val, str) or len(val) == 0: raise ValueError(_("cpuset must be string")) if re.match("^[0-9,-^]*$", val) is None: raise ValueError(_("cpuset can only contain numeric, ',', '^', or " "'-' characters")) pcpus = get_phy_cpus(conn) for c in val.split(','): # Redundant commas if not c: continue if "-" in c: (x, y) = c.split('-', 1) x = int(x) y = int(y) if x > y: raise ValueError(_("cpuset contains invalid format.")) if x >= pcpus or y >= pcpus: raise ValueError(_("cpuset's pCPU numbers must be less " "than pCPUs.")) else: if c.startswith("^"): c = c[1:] c = int(c) if c >= pcpus: raise ValueError(_("cpuset's pCPU numbers must be less " "than pCPUs.")) @staticmethod def cpuset_str_to_tuple(conn, cpuset): DomainNumatune.validate_cpuset(conn, cpuset) pinlist = [False] * get_phy_cpus(conn) entries = cpuset.split(",") for e in entries: series = e.split("-", 1) if len(series) == 1: pinlist[int(series[0])] = True continue start = int(series[0]) end = int(series[1]) for i in range(start, end + 1): pinlist[i] = True return tuple(pinlist) @staticmethod def generate_cpuset(conn, mem): """ Generates a cpu pinning string based on host NUMA configuration. If host doesn't have a suitable NUMA configuration, a RuntimeError is thrown. """ if conn.caps.host.topology is None: raise RuntimeError(_("No topology section in capabilities xml.")) cells = conn.caps.host.topology.cells if len(cells) <= 1: raise RuntimeError(_("Capabilities only show <= 1 cell. " "Not NUMA capable")) # Capabilities tells us about the available memory 'cells' on the # system. Each 'cell' has associated 'cpu's. # # Use getCellsFreeMemory to determine which 'cell' has the smallest # amount of memory which fits the requested VM memory amount, then # pin the VM to that 'cell's associated 'cpu's cell_mem = conn.getCellsFreeMemory(0, len(cells)) cell_id = -1 for i in range(len(cells)): if cell_mem[i] < mem: # Cell doesn't have enough mem to fit, skip it continue if len(cells[i].cpus) == 0: # No cpus to use for the cell continue # Find smallest cell that fits if cell_id < 0 or cell_mem[i] < cell_mem[cell_id]: cell_id = i if cell_id < 0: raise RuntimeError(_("Could not find any usable NUMA " "cell/cpu combinations.")) # Build cpuset string cpustr = "" for cpu in cells[cell_id].cpus: if cpustr != "": cpustr += "," cpustr += str(cpu.id) return cpustr MEMORY_MODES = ["interleave", "strict", "preferred"] _XML_ROOT_NAME = "numatune" _XML_PROP_ORDER = ["memory_mode", "memory_nodeset"] memory_nodeset = XMLProperty("./memory/@nodeset") memory_mode = XMLProperty("./memory/@mode")