forked from aleju/face-generator
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrain.lua
212 lines (187 loc) · 8.55 KB
/
train.lua
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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
require 'torch'
require 'optim'
require 'image'
--require 'datasets'
require 'pl'
require 'paths'
ok, DISP = pcall(require, 'display')
if not ok then print('display not found. unable to plot') end
ADVERSARIAL = require 'adversarial'
DATASET = require 'dataset'
NN_UTILS = require 'utils.nn_utils'
MODELS = require 'models'
----------------------------------------------------------------------
-- parse command-line options
OPT = lapp[[
--batchSize (default 32) batch size (must be even, must be >= 4)
--save (default "logs") subdirectory to save logs
--saveFreq (default 30) save every saveFreq epochs
--network (default "") reload pretrained network
--noplot Whether to not plot while training
--N_epoch (default 1000) Number of examples per epoch (-1 means all)
--G_SGD_lr (default 0.02) SGD learning rate for G
--G_SGD_momentum (default 0) SGD momentum for G
--D_SGD_lr (default 0.02) SGD learning rate for D
--D_SGD_momentum (default 0) SGD momentum for D
--G_adam_lr (default -1) Adam learning rate for G (-1 is automatic)
--D_adam_lr (default -1) Adam learning rate for D (-1 is automatic)
--G_L1 (default 0e-6) L1 penalty on the weights of G
--G_L2 (default 0e-6) L2 penalty on the weights of G
--D_L1 (default 0e-7) L1 penalty on the weights of D
--D_L2 (default 1e-4) L2 penalty on the weights of D
--D_iterations (default 1) How often to optimize D per batch
--G_iterations (default 1) How often to optimize G per batch
--D_maxAcc (default 1.01) stop training of D roughly around that accuracy level
--D_clamp (default 1) To which value to clamp D's gradients (e.g. 5 means -5 to +5, 0 is off)
--G_clamp (default 5) To which value to clamp G's gradients (e.g. 5 means -5 to +5, 0 is off)
--D_optmethod (default "adam") Optimizer to use for D, either "sgd" or "adam" or "adagrad"
--G_optmethod (default "adam") Optimizer to use for D, either "sgd" or "adam" or "adagrad"
--threads (default 8) number of threads
--gpu (default 0) gpu to run on (0-4 or -1 for cpu)
--noiseDim (default 100) dimensionality of noise vector
--window (default 3) ID of the first plotting window, will also use some window-ids beyond that
--scale (default 32) scale of images to train on (height, width)
--seed (default 1) Seed to use for the RNG
--weightsVisFreq (default 0) How often to update the windows showing the weights (only if >0; implies starting with qlua if >0)
--grayscale Whether to activate grayscale mode on the images
--denoise Whether to apply the denoiser trained with train_denoiser to the generated images
--aws Activate AWS settings
]]
if OPT.scale ~= 16 and OPT.scale ~= 32 then
print("[Warning] models are not optimized for chosen scale")
end
-- check is batch size is valid (x >= 4 and an even number)
if OPT.batchSize % 2 ~= 0 or OPT.batchSize < 4 then
print("[ERROR] batch size must be a multiple of 2 and higher or equal to 4")
sys.exit()
end
-- GPU, seed, threads
if OPT.gpu < 0 or OPT.gpu > 3 then OPT.gpu = false end
math.randomseed(OPT.seed)
torch.manualSeed(OPT.seed)
torch.setnumthreads(OPT.threads)
print(OPT)
print('<torch> set nb of threads to ' .. torch.getnumthreads())
-- run on gpu if chosen
require 'nn'
require 'cutorch'
require 'cunn'
require 'layers.cudnnSpatialConvolutionUpsample'
if OPT.gpu then
print("<trainer> starting gpu support...")
require 'cutorch'
require 'cunn'
cutorch.setDevice(OPT.gpu + 1)
cutorch.manualSeed(OPT.seed)
print(string.format("<trainer> using gpu device %d", OPT.gpu))
else
require 'nn'
end
require 'dpnn'
require 'LeakyReLU'
torch.setdefaulttensortype('torch.FloatTensor')
CLASSES = {"0", "1"} -- possible output of disciminator, used for confusion matrix
Y_GENERATOR = 0 -- Y=Image was created by generator
Y_NOT_GENERATOR = 1 -- Y=Image was from training dataset
IMG_DIMENSIONS = {3, OPT.scale, OPT.scale} -- axis of images: 1 or 3 channels, <scale> px height, <scale> px width
if OPT.grayscale then IMG_DIMENSIONS[1] = 1 end
INPUT_SZ = IMG_DIMENSIONS[1] * IMG_DIMENSIONS[2] * IMG_DIMENSIONS[3] -- size in values/pixels per input image (channels*height*width)
-- Main function, creates/loads networks, loads dataset, starts training
function main()
----------------------------------------------------------------------
-- Load / Define network
----------------------------------------------------------------------
if OPT.denoise then
-- load denoiser
local filename = paths.concat(OPT.save, string.format('denoiser_%dx%dx%d.net', IMG_DIMENSIONS[1], IMG_DIMENSIONS[2], IMG_DIMENSIONS[3]))
local tmp = torch.load(filename)
DENOISER = nn.Sequential()
DENOISER:add(tmp.AE1_DECODER)
--DENOISER:add(tmp.AE2_DECODER)
DENOISER:float()
DENOISER:evaluate()
end
-- load previous networks (D and G)
-- or initialize them new
if OPT.network ~= "" then
print(string.format("<trainer> reloading previously trained network: %s", OPT.network))
require 'cutorch'
require 'cunn'
local tmp = torch.load(OPT.network)
MODEL_D = tmp.D
MODEL_G = tmp.G
--OPTSTATE = tmp.optstate
EPOCH = tmp.epoch
if EPOCH ~= nil then EPOCH = EPOCH + 1 end
if OPT.gpu == false then
MODEL_D:float()
MODEL_G:float()
end
else
--------------
-- D, G
--------------
MODEL_D = MODELS.create_D(IMG_DIMENSIONS)
MODEL_G = MODELS.create_G(IMG_DIMENSIONS, OPT.noiseDim)
NN_UTILS.initializeWeights(MODEL_D)
NN_UTILS.initializeWeights(MODEL_G)
end
-- Activate GPU mode
if OPT.gpu then
MODEL_D = NN_UTILS.activateCuda(MODEL_D)
MODEL_G = NN_UTILS.activateCuda(MODEL_G)
end
-- loss function: negative log-likelihood
CRITERION = nn.BCECriterion()
-- retrieve parameters and gradients
PARAMETERS_D, GRAD_PARAMETERS_D = MODEL_D:getParameters()
PARAMETERS_G, GRAD_PARAMETERS_G = MODEL_G:getParameters()
-- print networks
print(MODEL_G)
print(MODEL_D)
-- count free parameters in D/G
print(string.format('Number of free parameters in D: %d', NN_UTILS.getNumberOfParameters(MODEL_D)))
print(string.format('Number of free parameters in G: %d', NN_UTILS.getNumberOfParameters(MODEL_G)))
----------------------------------------------------------------------
-- get/create dataset
----------------------------------------------------------------------
-- adjust dataset
if OPT.aws then
DATASET.setDirs({"/mnt/datasets/out_aug_64x64"})
else
DATASET.setDirs({"dataset/out_aug_64x64"})
end
DATASET.setFileExtension("jpg")
DATASET.setScale(OPT.scale)
DATASET.setNbChannels(IMG_DIMENSIONS[1])
----------------------------------------------------------------------
-- this matrix records the current confusion across classes
CONFUSION = optim.ConfusionMatrix(CLASSES)
-- Set optimizer state
OPTSTATE = {
adagrad = {D = {}, G = {}},
adam = {D = {}, G = {}},
rmsprop = {D = {}, G = {}},
sgd = {
D = {learningRate = OPT.D_SGD_lr, momentum = OPT.D_SGD_momentum},
G = {learningRate = OPT.G_SGD_lr, momentum = OPT.G_SGD_momentum}
}
}
if OPT.D_adam_lr ~= -1 then OPTSTATE.adam.D["learningRate"] = OPT.D_adam_lr end
if OPT.G_adam_lr ~= -1 then OPTSTATE.adam.G["learningRate"] = OPT.G_adam_lr end
-- Visualize initially, before the first epoch (usually just white noise, more if
-- the networks were loaded from file)
VIS_NOISE_INPUTS = NN_UTILS.createNoiseInputs(100)
-- training loop
EPOCH = 1
while true do
print('Loading new training data...')
TRAIN_DATA = DATASET.loadRandomImages(OPT.N_epoch)
if not OPT.noplot then
NN_UTILS.visualizeProgress(VIS_NOISE_INPUTS)
end
ADVERSARIAL.train(TRAIN_DATA, OPT.D_maxAcc, math.max(20, math.min(1000/OPT.batchSize, 250)))
end
end
--------------------------------------
main()