# Copyright 2021-2024 VyOS maintainers and contributors # # 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 . import re from json import loads from vyos.utils.network import interface_exists from vyos.utils.process import popen # These drivers do not support using ethtool to change the speed, duplex, or # flow control settings _drivers_without_speed_duplex_flow = ['vmxnet3', 'virtio_net', 'xen_netfront', 'iavf', 'ice', 'i40e', 'hv_netvsc', 'veth', 'ixgbevf', 'tun', 'vif'] class Ethtool: """ Class is used to retrive and cache information about an ethernet adapter """ # dictionary containing driver featurs, it will be populated on demand and # the content will look like: # [{'esp-hw-offload': {'active': False, 'fixed': True, 'requested': False}, # 'esp-tx-csum-hw-offload': {'active': False, # 'fixed': True, # 'requested': False}, # 'fcoe-mtu': {'active': False, 'fixed': True, 'requested': False}, # 'generic-receive-offload': {'active': True, # 'fixed': False, # 'requested': True}, # 'generic-segmentation-offload': {'active': True, # 'fixed': False, # 'requested': True}, # 'highdma': {'active': True, 'fixed': False, 'requested': True}, # 'ifname': 'eth0', # 'l2-fwd-offload': {'active': False, 'fixed': True, 'requested': False}, # 'large-receive-offload': {'active': False, # 'fixed': False, # 'requested': False}, # ... _features = { } # dictionary containing available interface speed and duplex settings # { # '10' : {'full': '', 'half': ''}, # '100' : {'full': '', 'half': ''}, # '1000': {'full': ''} # } _ring_buffer = None _driver_name = None _flow_control = None def __init__(self, ifname): # Get driver used for interface if not interface_exists(ifname): raise ValueError(f'Interface "{ifname}" does not exist!') out, _ = popen(f'ethtool --driver {ifname}') driver = re.search(r'driver:\s(\w+)', out) if driver: self._driver_name = driver.group(1) # Build a dictinary of supported link-speed and dupley settings. # [ { # "ifname": "eth0", # "supported-ports": [ "TP" ], # "supported-link-modes": [ "10baseT/Half","10baseT/Full","100baseT/Half","100baseT/Full","1000baseT/Full" ], # "supported-pause-frame-use": "Symmetric", # "supports-auto-negotiation": true, # "supported-fec-modes": [ ], # "advertised-link-modes": [ "10baseT/Half","10baseT/Full","100baseT/Half","100baseT/Full","1000baseT/Full" ], # "advertised-pause-frame-use": "Symmetric", # "advertised-auto-negotiation": true, # "advertised-fec-modes": [ ], # "speed": 1000, # "duplex": "Full", # "auto-negotiation": false, # "port": "Twisted Pair", # "phyad": 1, # "transceiver": "internal", # "supports-wake-on": "pumbg", # "wake-on": "g", # "current-message-level": 7, # "link-detected": true # } ] out, _ = popen(f'ethtool --json {ifname}') self._base_settings = loads(out)[0] # Now populate driver features out, _ = popen(f'ethtool --json --show-features {ifname}') self._features = loads(out)[0] # Get information about NIC ring buffers out, err = popen(f'ethtool --json --show-ring {ifname}') if not bool(err): self._ring_buffer = loads(out)[0] # Get current flow control settings, but this is not supported by # all NICs (e.g. vmxnet3 does not support is) out, err = popen(f'ethtool --json --show-pause {ifname}') if not bool(err): self._flow_control = loads(out)[0] def check_auto_negotiation_supported(self): """ Check if the NIC supports changing auto-negotiation """ return self._base_settings['supports-auto-negotiation'] def get_auto_negotiation(self): return self._base_settings['supports-auto-negotiation'] and self._base_settings['auto-negotiation'] def get_driver_name(self): return self._driver_name def _get_generic(self, feature): """ Generic method to read self._features and return a tuple for feature enabled and feature is fixed. In case of a missing key, return "fixed = True and enabled = False" """ active = False fixed = True if feature in self._features: active = bool(self._features[feature]['active']) fixed = bool(self._features[feature]['fixed']) return active, fixed def get_generic_receive_offload(self): return self._get_generic('generic-receive-offload') def get_generic_segmentation_offload(self): return self._get_generic('generic-segmentation-offload') def get_hw_tc_offload(self): return self._get_generic('hw-tc-offload') def get_large_receive_offload(self): return self._get_generic('large-receive-offload') def get_scatter_gather(self): return self._get_generic('scatter-gather') def get_tcp_segmentation_offload(self): return self._get_generic('tcp-segmentation-offload') def get_ring_buffer_max(self, rx_tx): # Configuration of RX/TX ring-buffers is not supported on every device, # thus when it's impossible return None if rx_tx not in ['rx', 'tx']: ValueError('Ring-buffer type must be either "rx" or "tx"') return str(self._ring_buffer.get(f'{rx_tx}-max', None)) def get_ring_buffer(self, rx_tx): # Configuration of RX/TX ring-buffers is not supported on every device, # thus when it's impossible return None if rx_tx not in ['rx', 'tx']: ValueError('Ring-buffer type must be either "rx" or "tx"') return str(self._ring_buffer.get(rx_tx, None)) def check_speed_duplex(self, speed, duplex): """ Check if the passed speed and duplex combination is supported by the underlaying network adapter. """ if isinstance(speed, int): speed = str(speed) if speed != 'auto' and not speed.isdigit(): raise ValueError(f'Value "{speed}" for speed is invalid!') if duplex not in ['auto', 'full', 'half']: raise ValueError(f'Value "{duplex}" for duplex is invalid!') if speed == 'auto' and duplex == 'auto': return True if self.get_driver_name() in _drivers_without_speed_duplex_flow: return False # ['10baset/half', '10baset/full', '100baset/half', '100baset/full', '1000baset/full'] tmp = [x.lower() for x in self._base_settings['supported-link-modes']] if f'{speed}baset/{duplex}' in tmp: return True return False def check_flow_control(self): """ Check if the NIC supports flow-control """ return bool(self._flow_control) def get_flow_control(self): if self._flow_control == None: raise ValueError('Interface does not support changing '\ 'flow-control settings!') return 'on' if bool(self._flow_control['autonegotiate']) else 'off'