-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathmnist_pixelcnn_train.py
129 lines (116 loc) · 5.46 KB
/
mnist_pixelcnn_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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
import torch
import torch.nn as nn
import torch.nn.functional as F
from torch.autograd import Variable
from data.mnist_dataset import MnistDataset
from data.data_affine import RandomAffineTransform
from models.mnist_pixelcnn import *
from models.components.shared import *
from train_utils import *
def quantisize(images, levels):
return (np.digitize(images, np.arange(levels) / levels) - 1).astype('i')
class Evaluator:
def __init__(self, args, model, ds=None, batch_size=100):
self.model = model
self.args = args
if ds is None:
ds = MnistDataset(transform=[], train=False)
self.ds = ds
self.batch_size = batch_size
self.loader = torch.utils.data.DataLoader(ds, batch_size=batch_size,
shuffle=False, num_workers=args.workers, drop_last=False)
def evaluate(self, show_prog=False):
nll_total = 0
Ntotal = 0
if show_prog:
print("Evaluating...")
prog = tqdm.tqdm(total=len(self.ds))
for i, (labels, x, _) in enumerate(self.loader, 0):
N, labels, x, x_quant = preprocess(self.args, labels, x)
logits = self.model(x, labels=labels)
nll = negative_log_likelihood(logits, x_quant)
nll_total = nll_total + nll.data * N
Ntotal = Ntotal + N
if show_prog:
prog.update(N)
if show_prog:
prog.close()
nll = nll_total / Ntotal
return nll
def preprocess(args, labels, x):
x_quant = torch.from_numpy(quantisize(x.numpy(),
args.levels)).type(torch.LongTensor)
x = x_quant.float() / (args.levels - 1)
if args.gpu:
x = x.cuda()
x_quant = x_quant.cuda()
labels = labels.cuda()
x = Variable(x, requires_grad=False)
x_quant = Variable(x_quant, requires_grad=False)
N = x.size(0)
labels = Variable(labels, requires_grad=False)
return N, labels, x, x_quant
def negative_log_likelihood(logits, x_quant, dim=1):
log_probs = F.log_softmax(logits, dim=dim)
flatten = lambda x, shape: x.transpose(1, -1).contiguous().view(*shape)
size_factor = float(np.prod([x for i, x in enumerate(logits.size()) if i not in {0, dim}]).astype(np.float32))
nll = torch.nn.functional.nll_loss(
flatten(log_probs, (-1, args.levels)),
flatten(x_quant, (-1,))) * size_factor
return nll
if __name__ == "__main__":
parser = ArgParser(description='Train pixelcnn on MNIST')
parser.add_argument('--conditional', action='store_true',
help='conditioned on digit labels')
parser.add_argument('--levels', type=int, default=8,
help='levels for quantisization')
parser.add_argument('--layers', type=int, default=5,
help='layers')
args = parser.parse_args()
model = MNIST_PixelCNNNew(levels=args.levels,
layers=args.layers,
conditional=args.conditional) # MNIST_PixelCNN(levels=args.levels, conditional=args.conditional)
if args.gpu:
model = model.cuda()
ds = MnistDataset(transform=[])
loader = torch.utils.data.DataLoader(ds, batch_size=args.batch_size,
shuffle=True, num_workers=args.workers, pin_memory=args.gpu,
drop_last=False)
evaluator = Evaluator(args, model)
#optimizer = torch.optim.Adam(model.parameters(), lr=args.lr)
optimizer = torch.optim.RMSprop(model.parameters(), lr=args.lr)
trainer = Trainer(args, model, optimizer)
for epoch in range(trainer.start_epoch, args.epochs):
trainer.update_lr(epoch)
print_now('epoch=%d --------------------------------' % epoch)
evaluation_nll = evaluator.evaluate(show_prog=True)
print("Epoch %d evaluation NLL: %.f" % (epoch, evaluation_nll))
if trainer.writer is not None:
trainer.writer.add_scalar('loss/eval-nll', evaluation_nll, trainer.step)
for i, (labels, x, _) in enumerate(loader, 0):
N, labels, x, x_quant = preprocess(args, labels, x)
logits = model(x, labels=labels)
nll = negative_log_likelihood(logits, x_quant)
loss = nll
trainer.loss_update(loss.data.cpu().numpy())
if i % args.print_every == 0:
print(" epoch %d, step %d. Loss ema: %.3f" % (epoch, i, trainer.loss))
if trainer.writer is not None:
#trainer.writer.add_scalar('loss/log_prob_x', log_prob_x, trainer.step)
trainer.writer.add_scalar('loss/loss', loss, trainer.step)
trainer.writer.add_scalar('loss/loss-ema', trainer.loss, trainer.step)
trainer.minimize(loss)
if trainer.writer is not None:
new_labels = torch.LongTensor([0, 1, 2])
if labels.data.is_cuda:
new_labels = new_labels.cuda()
new_labels = Variable(new_labels, requires_grad=False)
x_samp = model.generate_samples(28, 28, 1, 3, labels=new_labels)
if x_samp is not None:
for ex in range(3):
trainer.writer.add_image('x-samples/samp%d' % ex, torch_bw_img_to_np_img(x_samp[ex]), trainer.step)
mlp = logits.max(dim=1, keepdim=False)[1].float() / (args.levels - 1)
for ex in range(3):
trainer.writer.add_image('x-train/mlp-%d' % ex, torch_bw_img_to_np_img(mlp[ex]), trainer.step)
trainer.writer.add_image('x-train/input-%d' % ex, torch_bw_img_to_np_img(x[ex]), trainer.step)
trainer.save_checkpoint(epoch + 1)