-
Notifications
You must be signed in to change notification settings - Fork 60
/
mixed_precision.py
89 lines (65 loc) · 2.25 KB
/
mixed_precision.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
""" Handles the mixed precision logic """
import torch
from apex.fp16_utils import FP16_Optimizer
MP_CONFIG = {
'enabled': False,
'optimization_level': 'O2'
}
def enable_mixed_precision():
MP_CONFIG['enabled'] = True
def is_mixed_precision():
return MP_CONFIG['enabled']
def get_optim_level():
return MP_CONFIG['optimization_level']
def get_optimizer(obj):
'''
Apex introduces the FP16_optimizer object.
However this isn't really an optimizer, but only a wrapper around one.
This function returns the actual optimizer.
'''
if type(obj) == FP16_Optimizer:
return obj.optimizer
# If obj is not an FP16_Optimizer then we are not running in mixed precision
# and the passed object is already an actual optimizer
return obj
def set_optim_level(opt_level):
"""Defines the optimization level that will be used by AMP
See: https://nvidia.github.io/apex/amp.html#opt-levels
Arguments:
opt_level {string} -- The optimization level to use, should be O1 or O2.
"""
MP_CONFIG['optimization_level'] = opt_level
def maybe_half(tensor):
"""Convert a tensor to half precision if mixed precision is on.
Arguments:
tensor {torch.Tensor} -- The tensor to convert
Returns:
torch.Tensor -- returns the converted input tensor
"""
return tensor.half() if is_mixed_precision() else tensor
def initialize(model, optimizers):
"""Initialize mixed precision
Arguments:
model {nn.Module} -- The model to convert
optimizers -- The model
Returns:
[nn.Module, Optimizer] -- Converted model and optimizer
"""
if is_mixed_precision():
from apex import amp
if optimizers is not None:
model, optimizers = \
amp.initialize(model, optimizers, opt_level=get_optim_level())
else:
model = amp.initialize(model, opt_level=get_optim_level())
return model, optimizers
def backward(loss, optimizer):
"""Calls backward on the loss. If mixed precision is on, will
scale the loss.
"""
if is_mixed_precision():
from apex import amp
with amp.scale_loss(loss, optimizer) as scaled_loss:
scaled_loss.backward()
else:
loss.backward()