diff options
| -rw-r--r-- | op-mode-definitions/show-system.xml.in | 4 | ||||
| -rw-r--r-- | python/vyos/cpu.py | 102 | ||||
| -rwxr-xr-x | src/op_mode/show_uptime.py | 17 | 
3 files changed, 114 insertions, 9 deletions
| diff --git a/op-mode-definitions/show-system.xml.in b/op-mode-definitions/show-system.xml.in index 0f852164e..68b473bc1 100644 --- a/op-mode-definitions/show-system.xml.in +++ b/op-mode-definitions/show-system.xml.in @@ -166,9 +166,9 @@            </leafNode>            <leafNode name="uptime">              <properties> -              <help>Show how long the system has been up</help> +              <help>Show system uptime and load averages</help>              </properties> -            <command>uptime</command> +            <command>${vyos_op_scripts_dir}/show_uptime.py</command>            </leafNode>          </children>        </node> diff --git a/python/vyos/cpu.py b/python/vyos/cpu.py new file mode 100644 index 000000000..a0ef864be --- /dev/null +++ b/python/vyos/cpu.py @@ -0,0 +1,102 @@ +#!/usr/bin/env python3 +# Copyright 2022 VyOS maintainers and contributors <maintainers@vyos.io> +# +# This library is free software; you can redistribute it and/or +# modify it under the terms of the GNU Lesser General Public +# License as published by the Free Software Foundation; either +# version 2.1 of the License, or (at your option) any later version. +# +# This library 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 +# Lesser General Public License for more details. +# +# You should have received a copy of the GNU Lesser General Public +# License along with this library.  If not, see <http://www.gnu.org/licenses/>. + +""" +Retrieves (or at least attempts to retrieve) the total number of real CPU cores +installed in a Linux system. + +The issue of core count is complicated by existence of SMT, e.g. Intel's Hyper Threading. +GNU nproc returns the number of LOGICAL cores, +which is 2x of the real cores if SMT is enabled. + +The idea is to find all physical CPUs and add up their core counts. +It has special cases for x86_64 and MAY work correctly on other architectures, +but nothing is certain. +""" + +import re + + +def _read_cpuinfo(): +    with open('/proc/cpuinfo', 'r') as f: +        return f.readlines() + +def _split_line(l): +    l = l.strip() +    parts = re.split(r'\s*:\s*', l) +    return (parts[0], ":".join(parts[1:])) + +def _find_cpus(cpuinfo_lines): +    # Make a dict because it's more convenient to work with later, +    # when we need to find physicall distinct CPUs there. +    cpus = {} + +    cpu_number = 0 + +    for l in cpuinfo_lines: +        key, value = _split_line(l) +        if key == 'processor': +            cpu_number = value +            cpus[cpu_number] = {} +        else: +            cpus[cpu_number][key] = value + +    return cpus + +def _find_physical_cpus(): +    cpus = _find_cpus(_read_cpuinfo()) + +    phys_cpus = {} + +    for num in cpus: +        if 'physical id' in cpus[num]: +            # On at least some architectures, CPUs in different sockets +            # have different 'physical id' field, e.g. on x86_64. +            phys_id = cpus[num]['physical id'] +            if phys_id not in phys_cpus: +                phys_cpus[phys_id] = cpus[num] +        else: +            # On other architectures, e.g. on ARM, there's no such field. +            # We just assume they are different CPUs, +            # whether single core ones or cores of physical CPUs. +            phys_cpus[num] = cpu[num] + +    return phys_cpus + +def get_cpus(): +    """ Returns a list of /proc/cpuinfo entries that belong to different CPUs. +    """ +    cpus_dict = _find_physical_cpus() +    return list(cpus_dict.values()) + +def get_core_count(): +    """ Returns the total number of physical CPU cores +        (even if Hyper-Threading or another SMT is enabled and has inflated +        the number of cores in /proc/cpuinfo) +    """ +    physical_cpus = _find_physical_cpus() + +    core_count = 0 + +    for num in physical_cpus: +        # Some architectures, e.g. x86_64, include a field for core count. +        # Since we found unique physical CPU entries, we can sum their core counts. +        if 'cpu cores' in physical_cpus[num]: +            core_count += int(physical_cpus[num]['cpu cores']) +        else: +            core_count += 1 + +    return core_count diff --git a/src/op_mode/show_uptime.py b/src/op_mode/show_uptime.py index 1b5e33fa9..b70c60cf8 100755 --- a/src/op_mode/show_uptime.py +++ b/src/op_mode/show_uptime.py @@ -1,6 +1,6 @@  #!/usr/bin/env python3  # -# Copyright (C) 2021 VyOS maintainers and contributors +# Copyright (C) 2021-2022 VyOS maintainers and contributors  #  # This program is free software; you can redistribute it and/or modify  # it under the terms of the GNU General Public License version 2 as @@ -26,14 +26,17 @@ def get_uptime_seconds():  def get_load_averages():      from re import search      from vyos.util import cmd +    from vyos.cpu import get_core_count      data = cmd("uptime")      matches = search(r"load average:\s*(?P<one>[0-9\.]+)\s*,\s*(?P<five>[0-9\.]+)\s*,\s*(?P<fifteen>[0-9\.]+)\s*", data) +    core_count = get_core_count() +      res = {} -    res[1]  = float(matches["one"]) -    res[5]  = float(matches["five"]) -    res[15] = float(matches["fifteen"]) +    res[1]  = float(matches["one"]) / core_count +    res[5]  = float(matches["five"]) / core_count +    res[15] = float(matches["fifteen"]) / core_count      return res @@ -53,9 +56,9 @@ def get_formatted_output():      out = "Uptime: {}\n\n".format(data["uptime"])      avgs = data["load_average"]      out += "Load averages:\n" -    out += "1  minute:   {:.02f}%\n".format(avgs[1]*100) -    out += "5  minutes:  {:.02f}%\n".format(avgs[5]*100) -    out += "15 minutes:  {:.02f}%\n".format(avgs[15]*100) +    out += "1  minute:   {:.01f}%\n".format(avgs[1]*100) +    out += "5  minutes:  {:.01f}%\n".format(avgs[5]*100) +    out += "15 minutes:  {:.01f}%\n".format(avgs[15]*100)      return out | 
