-
Notifications
You must be signed in to change notification settings - Fork 8
/
train.py
110 lines (92 loc) · 3.6 KB
/
train.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
"""
Main training script for NERSC PyTorch examples
"""
# System
import os
import argparse
import logging
# Externals
import yaml
import numpy as np
# Locals
from datasets import get_data_loaders
from trainers import get_trainer
from utils.logging import config_logging
from utils.distributed import init_workers, try_barrier
def parse_args():
"""Parse command line arguments."""
parser = argparse.ArgumentParser()
add_arg = parser.add_argument
add_arg('config', nargs='?', default='configs/hello.yaml',
help='YAML configuration file')
add_arg('-d', '--distributed-backend', choices=['mpi', 'nccl', 'nccl-lsf', 'gloo'],
help='Specify the distributed backend to use')
add_arg('--gpu', type=int,
help='Choose a specific GPU by ID')
add_arg('--ranks-per-node', type=int, default=8,
help='Specifying number of ranks per node')
add_arg('--rank-gpu', action='store_true',
help='Choose GPU according to local rank')
add_arg('--resume', action='store_true',
help='Resume training from last checkpoint')
add_arg('-v', '--verbose', action='store_true',
help='Enable verbose logging')
return parser.parse_args()
def load_config(config_file):
with open(config_file) as f:
config = yaml.load(f, Loader=yaml.FullLoader)
return config
def main():
"""Main function"""
# Initialization
args = parse_args()
rank, n_ranks = init_workers(args.distributed_backend)
# Load configuration
config = load_config(args.config)
# Prepare output directory
output_dir = config.get('output_dir', None)
if output_dir is not None:
output_dir = os.path.expandvars(output_dir)
os.makedirs(output_dir, exist_ok=True)
# Setup logging
log_file = (os.path.join(output_dir, 'out_%i.log' % rank)
if output_dir is not None else None)
config_logging(verbose=args.verbose, log_file=log_file, append=args.resume)
logging.info('Initialized rank %i out of %i', rank, n_ranks)
try_barrier()
if rank == 0:
logging.info('Configuration: %s' % config)
# Load the datasets
distributed = args.distributed_backend is not None
train_data_loader, valid_data_loader = get_data_loaders(
distributed=distributed, **config['data'])
# Load the trainer
gpu = (rank % args.ranks_per_node) if args.rank_gpu else args.gpu
if gpu is not None:
logging.info('Using GPU %i', gpu)
trainer = get_trainer(name=config['trainer'], distributed=distributed,
rank=rank, output_dir=output_dir, gpu=gpu)
# Build the model and optimizer
trainer.build(config)
# Resume from checkpoint
if args.resume:
trainer.load_checkpoint()
# Run the training
summary = trainer.train(train_data_loader=train_data_loader,
valid_data_loader=valid_data_loader,
**config['train'])
# Print some conclusions
try_barrier()
n_train_samples = len(train_data_loader.sampler)
logging.info('Finished training')
train_time = np.mean(summary['train_time'])
logging.info('Train samples %g time %g s rate %g samples/s',
n_train_samples, train_time, n_train_samples / train_time)
if valid_data_loader is not None:
n_valid_samples = len(valid_data_loader.sampler)
valid_time = np.mean(summary['valid_time'])
logging.info('Valid samples %g time %g s rate %g samples/s',
n_valid_samples, valid_time, n_valid_samples / valid_time)
logging.info('All done!')
if __name__ == '__main__':
main()