-
Notifications
You must be signed in to change notification settings - Fork 232
/
Copy pathTest.lua
96 lines (77 loc) · 2.75 KB
/
Test.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
eps = 1e-3
function testJacobian(module, input, x, dx)
module:forward(input:clone())
x = x or input
local sx = torch.CudaTensor(x:storage())
local gradInput = torch.CudaTensor():resizeAs(module.output)
local sgradInput = torch.CudaTensor(gradInput:storage())
local jacobian = torch.Tensor(sx:nElement(), gradInput:nElement())
local jacobian_hat = torch.Tensor(sx:nElement(), gradInput:nElement())
-- Build Jacobian from module's updateGradInput
for i = 1,gradInput:nElement() do
sgradInput:zero()
sgradInput[i] = 1
module:updateGradInput(input, gradInput)
if dx then
dx:zero()
module:accGradParameters(input, gradInput)
jacobian:select(2, i):copy(dx)
else
jacobian:select(2, i):copy(module.gradInput)
end
end
-- Numerically estimate the Jacobian
for i = 1,sx:nElement() do
orig = sx[i]
sx[i] = orig + eps
module:forward(input:clone())
local f1 = module.output:clone()
sx[i] = orig - eps
module:forward(input:clone())
local f2 = module.output:clone()
jacobian_hat:select(1, i):copy(f1:add(-1, f2):div(2 * eps))
sx[i] = orig
end
return jacobian:add(-1, jacobian_hat):abs():max()
end
function testJacobianParameters(module, input)
x, dx = module:getParameters()
return testJacobian(module, input, x, dx)
end
function testCriterion(module, input, target)
local sinput = torch.CudaTensor(input:storage())
local grad_hat = torch.Tensor(sinput:nElement())
for i = 1,sinput:nElement() do
local orig = sinput[i]
sinput[i] = orig + eps
local f1 = module:forward(input, target)
sinput[i] = orig - eps
local f2 = module:forward(input, target)
grad_hat[i] = (f1 - f2) / (2 * eps)
sinput[i] = orig
end
module:forward(input, target)
module:backward(input, target)
return module.gradInput:double():add(-1, grad_hat):abs():max()
end
function testNetworkParameters(network, criterion, input, target)
local parameters, grad_parameters = network:getParameters()
local grad_hat = torch.Tensor(parameters:nElement())
for i = 1,parameters:nElement() do
local orig = parameters[i]
parameters[i] = orig + eps
network:forward(input)
local f1 = criterion:forward(network.output, target)
parameters[i] = orig - eps
network:forward(input)
local f2 = criterion:forward(network.output, target)
grad_hat[i] = (f1 - f2) / (2 * eps)
parameters[i] = orig
end
network:forward(input)
criterion:forward(network.output, target)
criterion:backward(network.output, target)
network:zeroGradParameters()
network:backward(input, criterion.gradInput)
return grad_parameters:double():add(-1, grad_hat):abs():max()
end