nvitop/nvitop/callbacks/pytorch_lightning.py
Xuehai Pan 1287159fb3 refactor(core/libnvml): rename nvml to libnvml
Signed-off-by: Xuehai Pan <XuehaiPan@pku.edu.cn>
2022-07-17 16:11:05 +08:00

148 lines
6.7 KiB
Python

# This file is part of nvitop, the interactive NVIDIA-GPU process viewer.
# License: GNU GPL version 3.
# pylint: disable=missing-module-docstring,missing-function-docstring
# pylint: disable=unused-argument,attribute-defined-outside-init
import time
from typing import Dict
from pytorch_lightning.callbacks import Callback # pylint: disable=import-error
from pytorch_lightning.utilities import rank_zero_only # pylint: disable=import-error
from pytorch_lightning.utilities.exceptions import MisconfigurationException # pylint: disable=import-error
from nvitop.core import libnvml
from nvitop.callbacks.utils import get_devices_by_logical_ids, get_gpu_stats
# Modified from pytorch_lightning.callbacks.GPUStatsMonitor
class GpuStatsLogger(Callback): # pylint: disable=too-many-instance-attributes
r"""
Automatically log GPU stats during training stage. :class:`GpuStatsLogger` is a
callback and in order to use it you need to assign a logger in the ``Trainer``.
Args:
memory_utilization (bool):
Set to :data:`True` to log used, free and the percentage of memory
utilization at the start and end of each step. Default: :data:`True`.
gpu_utilization (bool):
Set to :data:`True` to log the percentage of GPU utilization
at the start and end of each step. Default: :data:`True`.
intra_step_time (bool):
Set to :data:`True` to log the time of each step. Default: :data:`False`.
inter_step_time (bool):
Set to :data:`True` to log the time between the end of one step
and the start of the next step. Default: :data:`False`.
fan_speed (bool):
Set to :data:`True` to log percentage of fan speed. Default: :data:`False`.
temperature (bool):
Set to :data:`True` to log the gpu temperature in degree Celsius.
Default: :data:`False`.
Raises:
MisconfigurationException:
If NVIDIA driver is not installed, not running on GPUs, or ``Trainer`` has no logger.
Example::
>>> from pytorch_lightning import Trainer
>>> from nvitop.callbacks.pytorch_lightning import GpuStatsLogger
>>> gpu_stats = GpuStatsLogger()
>>> trainer = Trainer(gpus=[..], logger=True, callbacks=[gpu_stats])
GPU stats are mainly based on NVML queries. The description of the queries is as follows:
- **fan.speed** - The fan speed value is the percent of maximum speed that the device's fan is currently
intended to run at. It ranges from 0 to 100 %. Note: The reported speed is the intended fan speed.
If the fan is physically blocked and unable to spin, this output will not match the actual fan speed.
Many parts do not report fan speeds because they rely on cooling via fans in the surrounding enclosure.
- **memory.used** - Total memory allocated by active contexts, in MiBs.
- **memory.free** - Total free memory, in MiBs.
- **utilization.gpu** - Percent of time over the past sample period during which one or more kernels was
executing on the GPU. The sample period may be between 1 second and 1/6 second depending on the product.
- **utilization.memory** - Percent of time over the past sample period during which global (device) memory was
being read or written. The sample period may be between 1 second and 1/6 second depending on the product.
- **temperature** - Core GPU temperature, in degrees C.
"""
def __init__( # pylint: disable=too-many-arguments
self,
memory_utilization: bool = True,
gpu_utilization: bool = True,
intra_step_time: bool = False,
inter_step_time: bool = False,
fan_speed: bool = False,
temperature: bool = False
) -> None:
super().__init__()
try:
libnvml.nvmlInit()
except libnvml.NVMLError as ex:
raise MisconfigurationException(
'Cannot use GpuStatsLogger callback because NVIDIA driver is not installed.'
) from ex
self._memory_utilization = memory_utilization
self._gpu_utilization = gpu_utilization
self._intra_step_time = intra_step_time
self._inter_step_time = inter_step_time
self._fan_speed = fan_speed
self._temperature = temperature
def on_train_start(self, trainer, pl_module) -> None:
if not trainer.logger:
raise MisconfigurationException('Cannot use GpuStatsLogger callback with Trainer that has no logger.')
if trainer.strategy.root_device.type != 'cuda':
raise MisconfigurationException(
'You are using GpuStatsLogger but are not running on GPU. '
'The root device type is {}.'.format(trainer.strategy.root_device.type)
)
device_ids = trainer.data_parallel_device_ids
try:
self._devices = get_devices_by_logical_ids(device_ids, unique=True)
except (libnvml.NVMLError, RuntimeError) as ex:
raise ValueError(
'Cannot use GpuStatsLogger callback because devices unavailable. '
'Received: `gpus={}`'.format(device_ids)
) from ex
def on_train_epoch_start(self, trainer, pl_module) -> None:
self._snap_intra_step_time = None
self._snap_inter_step_time = None
@rank_zero_only
def on_train_batch_start(self, trainer, **kwargs) -> None: # pylint: disable=arguments-differ
if self._intra_step_time:
self._snap_intra_step_time = time.monotonic()
logs = self._get_gpu_stats()
if self._inter_step_time and self._snap_inter_step_time:
# First log at beginning of second step
logs['batch_time/inter_step (ms)'] = (time.monotonic() - self._snap_inter_step_time) * 1000.0
trainer.logger.log_metrics(logs, step=trainer.global_step)
@rank_zero_only
def on_train_batch_end(self, trainer, **kwargs) -> None: # pylint: disable=arguments-differ
if self._inter_step_time:
self._snap_inter_step_time = time.monotonic()
logs = self._get_gpu_stats()
if self._intra_step_time and self._snap_intra_step_time:
logs['batch_time/intra_step (ms)'] = (time.monotonic() - self._snap_intra_step_time) * 1000.0
trainer.logger.log_metrics(logs, step=trainer.global_step)
def _get_gpu_stats(self) -> Dict[str, float]:
"""Get the gpu status from NVML queries"""
return get_gpu_stats(devices=self._devices,
memory_utilization=self._memory_utilization,
gpu_utilization=self._gpu_utilization,
fan_speed=self._fan_speed,
temperature=self._temperature)