2020-10-13 11:18:07 +00:00
|
|
|
# Copyright The PyTorch Lightning team.
|
|
|
|
#
|
|
|
|
# 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.
|
2020-08-27 17:50:32 +00:00
|
|
|
import os
|
2021-01-14 18:15:34 +00:00
|
|
|
|
|
|
|
import numpy as np
|
2020-08-27 17:50:32 +00:00
|
|
|
import pytest
|
|
|
|
import torch
|
|
|
|
|
|
|
|
from pytorch_lightning import Trainer
|
|
|
|
from pytorch_lightning.callbacks import GPUStatsMonitor
|
|
|
|
from pytorch_lightning.loggers import CSVLogger
|
|
|
|
from pytorch_lightning.loggers.csv_logs import ExperimentWriter
|
2021-01-12 00:36:48 +00:00
|
|
|
from pytorch_lightning.trainer.states import TrainerState
|
2020-08-27 17:50:32 +00:00
|
|
|
from pytorch_lightning.utilities.exceptions import MisconfigurationException
|
2021-02-16 19:59:57 +00:00
|
|
|
from tests.helpers import BoringModel
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(not torch.cuda.is_available(), reason="test requires GPU machine")
|
|
|
|
def test_gpu_stats_monitor(tmpdir):
|
|
|
|
"""
|
|
|
|
Test GPU stats are logged using a logger.
|
|
|
|
"""
|
2021-02-16 19:59:57 +00:00
|
|
|
model = BoringModel()
|
2020-10-22 11:08:03 +00:00
|
|
|
gpu_stats = GPUStatsMonitor(intra_step_time=True)
|
2020-08-27 17:50:32 +00:00
|
|
|
logger = CSVLogger(tmpdir)
|
2020-10-22 11:08:03 +00:00
|
|
|
log_every_n_steps = 2
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
trainer = Trainer(
|
|
|
|
default_root_dir=tmpdir,
|
2020-10-22 11:08:03 +00:00
|
|
|
max_epochs=2,
|
|
|
|
limit_train_batches=7,
|
|
|
|
log_every_n_steps=log_every_n_steps,
|
2020-08-27 17:50:32 +00:00
|
|
|
gpus=1,
|
|
|
|
callbacks=[gpu_stats],
|
|
|
|
logger=logger
|
|
|
|
)
|
|
|
|
|
2021-01-12 00:36:48 +00:00
|
|
|
trainer.fit(model)
|
|
|
|
assert trainer.state == TrainerState.FINISHED, f"Training failed with {trainer.state}"
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
path_csv = os.path.join(logger.log_dir, ExperimentWriter.NAME_METRICS_FILE)
|
2020-10-22 11:08:03 +00:00
|
|
|
met_data = np.genfromtxt(path_csv, delimiter=',', names=True, deletechars='', replace_space=' ')
|
2020-08-27 17:50:32 +00:00
|
|
|
|
2020-10-22 11:08:03 +00:00
|
|
|
batch_time_data = met_data['batch_time/intra_step (ms)']
|
|
|
|
batch_time_data = batch_time_data[~np.isnan(batch_time_data)]
|
|
|
|
assert batch_time_data.shape[0] == trainer.global_step // log_every_n_steps
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
fields = [
|
2020-09-04 10:02:16 +00:00
|
|
|
'utilization.gpu',
|
|
|
|
'memory.used',
|
|
|
|
'memory.free',
|
2021-02-06 12:28:26 +00:00
|
|
|
'utilization.memory',
|
2020-08-27 17:50:32 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
for f in fields:
|
2020-10-22 11:08:03 +00:00
|
|
|
assert any([f in h for h in met_data.dtype.names])
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(torch.cuda.is_available(), reason="test requires CPU machine")
|
|
|
|
def test_gpu_stats_monitor_cpu_machine(tmpdir):
|
|
|
|
"""
|
|
|
|
Test GPUStatsMonitor on CPU machine.
|
|
|
|
"""
|
|
|
|
with pytest.raises(MisconfigurationException, match='NVIDIA driver is not installed'):
|
2020-12-21 05:40:55 +00:00
|
|
|
GPUStatsMonitor()
|
2020-08-27 17:50:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(not torch.cuda.is_available(), reason="test requires GPU machine")
|
|
|
|
def test_gpu_stats_monitor_no_logger(tmpdir):
|
|
|
|
"""
|
|
|
|
Test GPUStatsMonitor with no logger in Trainer.
|
|
|
|
"""
|
2021-02-16 19:59:57 +00:00
|
|
|
model = BoringModel()
|
2020-08-27 17:50:32 +00:00
|
|
|
gpu_stats = GPUStatsMonitor()
|
|
|
|
|
|
|
|
trainer = Trainer(
|
|
|
|
default_root_dir=tmpdir,
|
|
|
|
callbacks=[gpu_stats],
|
|
|
|
max_epochs=1,
|
|
|
|
gpus=1,
|
2021-02-06 12:28:26 +00:00
|
|
|
logger=False,
|
2020-08-27 17:50:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
with pytest.raises(MisconfigurationException, match='Trainer that has no logger.'):
|
|
|
|
trainer.fit(model)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(not torch.cuda.is_available(), reason="test requires GPU machine")
|
|
|
|
def test_gpu_stats_monitor_no_gpu_warning(tmpdir):
|
|
|
|
"""
|
|
|
|
Test GPUStatsMonitor raises a warning when not training on GPU device.
|
|
|
|
"""
|
2021-02-16 19:59:57 +00:00
|
|
|
model = BoringModel()
|
2020-08-27 17:50:32 +00:00
|
|
|
gpu_stats = GPUStatsMonitor()
|
|
|
|
|
|
|
|
trainer = Trainer(
|
|
|
|
default_root_dir=tmpdir,
|
|
|
|
callbacks=[gpu_stats],
|
|
|
|
max_steps=1,
|
2021-02-06 12:28:26 +00:00
|
|
|
gpus=None,
|
2020-08-27 17:50:32 +00:00
|
|
|
)
|
|
|
|
|
2020-09-04 10:02:16 +00:00
|
|
|
with pytest.raises(MisconfigurationException, match='not running on GPU'):
|
2020-08-27 17:50:32 +00:00
|
|
|
trainer.fit(model)
|
2020-09-25 05:30:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_gpu_stats_monitor_parse_gpu_stats():
|
|
|
|
logs = GPUStatsMonitor._parse_gpu_stats('1,2', [[3, 4, 5], [6, 7]], [('gpu', 'a'), ('memory', 'b')])
|
|
|
|
expected = {'gpu_id: 1/gpu (a)': 3, 'gpu_id: 1/memory (b)': 4, 'gpu_id: 2/gpu (a)': 6, 'gpu_id: 2/memory (b)': 7}
|
|
|
|
assert logs == expected
|