Mercurial > pylearn
annotate nnet_ops.py @ 440:18dbc1c11647
Work on softmax operators
author | Pascal Lamblin <lamblinp@iro.umontreal.ca> |
---|---|
date | Thu, 21 Aug 2008 13:55:16 -0400 |
parents | 43d9aa93934e |
children | b3315b252824 |
rev | line source |
---|---|
419
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
1 ## This file contain ops that are not currently integrated in the core of threano. |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
2 ## Not all of those ops have been thoroughly tested. |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
3 |
24 | 4 import theano |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
5 from theano import tensor, scalar |
24 | 6 import numpy |
7 | |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
8 ############ |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
9 # |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
10 # SCALAR OPS |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
11 # |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
12 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
13 class ScalarSigmoid(scalar.UnaryScalarOp): |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
14 @staticmethod |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
15 def st_impl(x): |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
16 if x < -30.0: |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
17 return 0.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
18 if x > 30.0: |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
19 return 1.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
20 return 1.0 / (1.0 + numpy.exp(-x)) |
24 | 21 def impl(self, x): |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
22 return ScalarSigmoid.st_impl(x) |
24 | 23 def grad(self, (x,), (gz,)): |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
24 y = scalar_sigmoid(x) |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
25 return [gz * y * (1.0 - y)] |
181
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
26 def c_code(self, node, name, (x,), (z,), sub): |
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
27 if node.inputs[0].type in [scalar.float32, scalar.float64]: |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
28 return """%(z)s = |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
29 %(x)s < -30.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
30 ? 0.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
31 : %(x)s > 30.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
32 ? 1.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
33 : 1.0 /(1.0+exp(-%(x)s));""" % locals() |
181
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
34 raise NotImplementedError('only floatingpoint is implemented') |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
35 scalar_sigmoid = ScalarSigmoid(scalar.upgrade_to_float, name='scalar_sigmoid') |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
36 sigmoid = tensor.Elemwise(scalar_sigmoid, name='sigmoid') |
24 | 37 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
38 class ScalarSoftplus(scalar.UnaryScalarOp): |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
39 @staticmethod |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
40 def static_impl(x): |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
41 if x < -30.0: |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
42 return 0.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
43 if x > 30.0: |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
44 return x |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
45 return numpy.log1p(numpy.exp(x)) |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
46 def impl(self, x): |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
47 return ScalarSoftplus.static_impl(x) |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
48 def grad(self, (x,), (gz,)): |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
49 return [gz * scalar_sigmoid(x)] |
222
f6a7eb1b7970
redo what James had done, so invert node and name
Thierry Bertin-Mahieux <bertinmt@iro.umontreal.ca>
parents:
218
diff
changeset
|
50 def c_code(self, node, name, (x,), (z,), sub): |
181
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
51 if node.inputs[0].type in [scalar.float32, scalar.float64]: |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
52 return """%(z)s = |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
53 %(x)s < -30.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
54 ? 0.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
55 : %(x)s > 30.0 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
56 ? %(x)s |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
57 : log1p(exp(%(x)s));""" % locals() |
181
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
58 raise NotImplementedError('only floating point x is implemented') |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
59 scalar_softplus = ScalarSoftplus(scalar.upgrade_to_float, name='scalar_softplus') |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
60 softplus = tensor.Elemwise(scalar_softplus, name='softplus') |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
61 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
62 |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
63 ############ |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
64 # |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
65 # TENSOR OPS |
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
66 # |
24 | 67 |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
68 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
69 class SoftmaxWithBias(theano.Op): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
70 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
71 An L{Op} for the output of neural-net multiclass classifiers. |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
72 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
73 @type x: is a matrix of floats (32 or 64) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
74 @type b: is a [row] vector of floats (32 or 64), length is number of cols in x |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
75 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
76 This L{Op}'s output is softmax(x+b). |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
77 softmax(x[i]) is the i'th distribution over len(x[i]) options. |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
78 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
79 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
80 nin = 2 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
81 nout = 1 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
82 def __init__(self, **kwargs): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
83 theano.Op.__init__(self, **kwargs) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
84 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
85 def make_node(self, x, b): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
86 x = tensor.as_tensor(x) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
87 b = tensor.as_tensor(b) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
88 if x.type.ndim != 2 \ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
89 or x.type.dtype not in ['float32', 'float64']: |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
90 raise ValueError('x must be 2-d tensor of floats') |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
91 if b.type.ndim != 1 \ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
92 or x.type.dtype not in ['float32', 'float64']: |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
93 raise ValueError('b must be 1-d tensor of floats') |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
94 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
95 sm = x.type.make_result() |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
96 return theano.Apply(self, [x, b], [sm]) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
97 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
98 def perform(self, node, input_storage, output_storage): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
99 x, b = input_storage |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
100 if b.shape[0] != x.shape[1]: |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
101 raise ValueError('b must have same number of columns as x') |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
102 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
103 sm = nympy.zeros_like(x) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
104 for i in xrange(sm.shape[0]): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
105 row = x[i] + b |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
106 sm[i] = numpy.exp(row - numpy.max(row)) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
107 sm[i] *= 1.0 / numpy.sum(sm[i]) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
108 output_storage[0][0] = nll |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
109 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
110 def grad(self, (x, b), (g_sm,)): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
111 sm = softmax_with_bias(x, b) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
112 dx = SoftmaxWithBiasDx()(g_sm, sm) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
113 db = tensor.sum(dx, axis = 0) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
114 return dx, db |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
115 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
116 def c_headers(self): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
117 return ['<iostream>'] |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
118 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
119 @staticmethod |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
120 def c_code_template(): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
121 # this implementation was lifted from |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
122 # /u/bergstrj/cvs/bergstrj/src/feb07/nn.cxx |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
123 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
124 #TODO: put this into a templated function, in the support code |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
125 #TODO: declare the max of each row as an Op output |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
126 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
127 #TODO: set error messages for failures in this code |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
128 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
129 #TODO: use this to accept float32 and int32: node.inputs[0].type.dtype_specs()[1] |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
130 init_decl = """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
131 npy_intp* Nx = %(x)s->dimensions; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
132 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
133 if (%(x)s->nd != 2) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
134 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
135 PyErr_SetString(PyExc_ValueError, "a not 2d tensor"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
136 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
137 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
138 if (%(b)s->nd != 1) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
139 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
140 PyErr_SetString(PyExc_ValueError, "b not 1d tensor"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
141 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
142 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
143 if (%(x)s->descr->type_num != PyArray_DOUBLE) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
144 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
145 PyErr_SetString(PyExc_TypeError, "a not float64"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
146 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
147 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
148 if (%(b)s->descr->type_num != PyArray_DOUBLE) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
149 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
150 PyErr_SetString(PyExc_TypeError, "b not float64"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
151 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
152 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
153 if ((%(x)s->dimensions[1] != %(b)s->dimensions[0])) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
154 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
155 PyErr_SetString(PyExc_ValueError, "dimension mismatch in arguments"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
156 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
157 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
158 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
159 if ((NULL == %(sm)s) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
160 || (%(sm)s->dimensions[0] != %(x)s->dimensions[0]) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
161 || (%(sm)s->dimensions[1] != %(x)s->dimensions[1])) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
162 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
163 if (NULL != %(sm)s) Py_XDECREF(%(sm)s); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
164 %(sm)s = (PyArrayObject*)PyArray_SimpleNew(2, PyArray_DIMS(%(x)s), type_num_%(x)s); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
165 if(!%(sm)s) { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
166 // The normal cleanup code will take care of %(nll)s |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
167 // Py_XDECREF(%(nll)s); %(nll)s=NULL; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
168 PyErr_SetString(PyExc_MemoryError, "failed to alloc sm output"); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
169 %(fail)s |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
170 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
171 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
172 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
173 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
174 begin_row_loop = """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
175 for (size_t i = 0; i < Nx[0]; ++i) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
176 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
177 size_t j; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
178 double sum = 0.0; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
179 bool discount_max = false; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
180 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
181 const double* __restrict__ x_i = (double*)(%(x)s->data + %(x)s->strides[0] * i); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
182 const double* __restrict__ b_i = (double*)(%(b)s->data); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
183 double* __restrict__ sm_i = (double*)(%(sm)s->data + %(sm)s->strides[0] * i); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
184 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
185 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
186 inside_row_loop = """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
187 npy_intp Sx = %(x)s->strides[1]/sizeof(double); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
188 npy_intp Sb = %(b)s->strides[0]/sizeof(double); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
189 npy_intp Ssm = %(sm)s->strides[1]/sizeof(double); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
190 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
191 size_t row_max_j=0; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
192 double row_max = x_i[0] + b_i[0]; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
193 // Get the maximum value of the row |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
194 for (j = 0; j < Nx[1]; ++j) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
195 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
196 double row_ij = x_i[j * Sx] + b_i[j * Sb]; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
197 row_max_j = (row_ij > row_max) ? j : row_max_j; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
198 row_max = (row_ij > row_max) ? row_ij : row_max; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
199 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
200 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
201 for (j = 0; j < Nx[1]; ++j) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
202 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
203 double row_ij = x_i[j * Sx] + b_i[j * Sb]; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
204 double sm_ij = exp(row_ij - row_max); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
205 sum += sm_ij; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
206 sm_i[j * Ssm] = sm_ij; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
207 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
208 if ( (0.0 == sum) || (isinf(sum))) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
209 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
210 //that was our best... |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
211 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
212 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
213 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
214 //cblas_dscal(x.N, 1.0 / sum, &mat_at(s,i,0), s.n); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
215 double sum_inv = 1.0 / sum; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
216 for (j = 0; j < Nx[1]; ++j) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
217 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
218 sm_i[j * Ssm] *= sum_inv; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
219 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
220 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
221 if (y_i >= Nx[1]) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
222 { |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
223 %(fail)s; |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
224 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
225 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
226 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
227 end_row_loop = """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
228 } |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
229 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
230 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
231 return (init_decl, begin_row_loop, inside_row_loop, end_row_loop) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
232 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
233 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
234 def c_code(self, node, name, (x, b), (sm,), sub): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
235 code_template = ''.join(self.c_code_template()) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
236 return code_template % dict(locals(), **sub) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
237 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
238 softmax_with_bias = SoftmaxWithBias() |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
239 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
240 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
241 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
242 class CrossentropySoftmax1HotWithBias(theano.Op): |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
243 """A special compound L{Op} for the output of neural-net classifiers. |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
244 |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
245 @type x: is a matrix of floats (32 or 64) |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
246 @type b: is a [row] vector of floats (32 or 64), length is number of cols in x |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
247 @type y_idx: a [column] vector of int (32 or 64), length is number of rows in x |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
248 |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
249 @precondition: every entry in y_idx is a valid (non-negative) column index into x |
24 | 250 |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
251 This L{Op} has two outputs: |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
252 - KL(softmax(x+b), y) |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
253 - softmax(x+b) |
24 | 254 |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
255 |
24 | 256 softmax(x[i]) is the i'th distribution over len(x[i]) options |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
257 |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
258 y_idx[i] is an integer index, encoding a 1-hot distribution. |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
259 |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
260 In practice, when we're trying to do classification, we have one row in x |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
261 and y_idx per example, and y[i] is the index of the (correct) class of the |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
262 i'th example. |
24 | 263 |
264 """ | |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
265 nin=3 |
24 | 266 nout=2 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
267 def __init__(self, **kwargs): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
268 theano.Op.__init__(self, **kwargs) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
269 |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
270 def make_node(self, x, b, y_idx): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
271 x = tensor.as_tensor(x) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
272 b = tensor.as_tensor(b) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
273 y_idx = tensor.as_tensor(y_idx) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
274 if x.type.ndim != 2 \ |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
275 or x.type.dtype not in ['float32', 'float64']: |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
276 raise ValueError('x must be 2-d tensor of floats') |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
277 if b.type.ndim != 1 \ |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
278 or x.type.dtype not in ['float32', 'float64']: |
121 | 279 raise ValueError('b must be 1-d tensor of floats') |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
280 if y_idx.type.ndim != 1 \ |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
281 or y_idx.type.dtype not in ['int8', 'int16', 'int32', 'int64']: |
121 | 282 raise ValueError('y_idx must be 1-d tensor of ints') |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
283 |
24 | 284 # TODO: Is this correct? It used to be y, not y_idx |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
285 nll = tensor.Tensor(x.type.dtype, |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
286 y_idx.type.broadcastable).make_result() |
24 | 287 # nll = Tensor(x.dtype, y.broadcastable) |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
288 sm = x.type.make_result() |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
289 return theano.Apply(self, [x, b, y_idx], [nll, sm]) |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
290 def perform(self, node, input_storage, output_storage): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
291 x, b, y_idx = input_storage |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
292 if b.shape[0] != x.shape[1]: |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
293 raise ValueError('b must have same number of columns as x') |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
294 if y_idx.shape[0] != x.shape[0]: |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
295 raise ValueError('y_idx must have same number of rows as x') |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
296 |
24 | 297 sm = numpy.zeros_like(x) # softmax |
298 nll = numpy.zeros(x.shape[0]) #nll(y | softmax(x)) | |
299 for i in xrange(sm.shape[0]): | |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
300 row = x[i] + b |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
301 sm[i] = numpy.exp(row - numpy.max(row)) #softmax |
24 | 302 sm[i] *= 1.0 / numpy.sum(sm[i]) #vector scale |
303 nll[i] = -numpy.log( sm[i, y_idx[i]]) #cross-entropy | |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
304 output_storage[0][0] = nll |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
305 output_storage[1][0] = sm |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
306 def grad(self, (x, b, y_idx), (g_nll, g_sm)): |
24 | 307 if g_sm is not None: |
308 raise NotImplementedError() | |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
309 nll, sm = crossentropy_softmax_1hot_with_bias(x, b, y_idx) |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
310 dx = CrossentropySoftmax1HotWithBiasDx()(g_nll, sm, y_idx) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
311 db = tensor.sum(dx, axis = [0]) |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
312 return dx, db, None |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
313 |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
314 def c_headers(self): return ['<iostream>'] |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
315 |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
316 @staticmethod |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
317 def c_code_template(): |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
318 # this implementation was lifted from |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
319 # /u/bergstrj/cvs/bergstrj/src/feb07/nn.cxx |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
320 |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
321 #TODO: put this into a templated function, in the support code |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
322 #TODO: declare the max of each row as an Op output |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
323 |
32 | 324 #TODO: set error messages for failures in this code |
325 | |
184
9a2aecc57a79
added TODO to nnet_ops
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
181
diff
changeset
|
326 #TODO: use this to accept float32 and int32: node.inputs[0].type.dtype_specs()[1] |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
327 (init_decl, begin_row_loop, inside_row_loop, end_row_loop) = \ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
328 SoftmaxWithBias.c_code_template() |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
329 return (init_decl, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
330 """ |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
331 if (%(y_idx)s->nd != 1) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
332 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
333 PyErr_SetString(PyExc_ValueError, "y_idx not 1d tensor"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
334 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
335 } |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
336 if ((%(y_idx)s->descr->type_num != PyArray_INT64) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
337 && (%(y_idx)s->descr->type_num != PyArray_INT32) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
338 && (%(y_idx)s->descr->type_num != PyArray_INT16) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
339 && (%(y_idx)s->descr->type_num != PyArray_INT8)) |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
340 { |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
341 PyErr_SetString(PyExc_TypeError, "y_idx not int8, int16, int32, or int64"); |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
342 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
343 } |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
344 if (%(x)s->dimensions[0] != %(y_idx)s->dimensions[0]) |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
345 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
346 PyErr_SetString(PyExc_ValueError, "dimension mismatch in arguments"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
347 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
348 } |
34 | 349 |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
350 if ((NULL == %(nll)s) //initial condition |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
351 || (%(nll)s->dimensions[0] != %(y_idx)s->dimensions[0])) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
352 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
353 if (NULL != %(nll)s) Py_XDECREF(%(nll)s); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
354 %(nll)s = (PyArrayObject*)PyArray_SimpleNew(1, PyArray_DIMS(%(y_idx)s), type_num_%(x)s); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
355 if(!%(nll)s) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
356 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
357 PyErr_SetString(PyExc_MemoryError, "failed to alloc nll output"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
358 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
359 } |
34 | 360 } |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
361 """, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
362 begin_row_loop, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
363 """ |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
364 const %(y_idx_type)s y_i = ((%(y_idx_type)s*)(%(y_idx)s->data + %(y_idx)s->strides[0] * i))[0]; |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
365 double* __restrict__ nll_i = (double*)(%(nll)s->data + %(nll)s->strides[0] * i); |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
366 """, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
367 inside_row_loop, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
368 """ |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
369 nll_i[0] = - x_i[y_i*Sx] |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
370 - b_i[y_i*Sb] |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
371 + log(sum); |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
372 """, |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
373 end_row_loop) |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
374 |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
375 |
440
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
376 def c_code(self, node, name, (x, b, y_idx), (nll, sm), sub): |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
377 y_idx_type = node.inputs[2].type.dtype_specs()[1] |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
378 code_template = ''.join(self.c_code_template()) |
18dbc1c11647
Work on softmax operators
Pascal Lamblin <lamblinp@iro.umontreal.ca>
parents:
419
diff
changeset
|
379 return code_template % dict(locals(), **sub) |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
380 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
381 crossentropy_softmax_1hot_with_bias = CrossentropySoftmax1HotWithBias() |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
382 |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
383 class CrossentropySoftmax1HotWithBiasDx (theano.Op): |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
384 nin=3 |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
385 nout=1 |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
386 """Gradient wrt x of the CrossentropySoftmax1Hot Op""" |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
387 def __init__(self, **kwargs): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
388 theano.Op.__init__(self,**kwargs) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
389 def make_node(self, dy, sm, y_idx,**kwargs): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
390 dy = tensor.as_tensor(dy) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
391 sm = tensor.as_tensor(sm) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
392 y_idx = tensor.as_tensor(y_idx) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
393 return theano.Apply(self, [dy, sm, y_idx],[sm.type.make_result()]) |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
394 def perform(self, node, input_storage, output_storage): |
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
395 dy,sm,y_idx = input_storage |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
396 dx = numpy.zeros_like(sm) |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
397 for i in xrange(sm.shape[0]): |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
398 dx[i] = dy[i] * sm[i] #vector scale |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
399 dx[i, y_idx[i]] -= dy[i] #scalar decrement |
117
3ef569b92fba
ported nnet_ops to new theano
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
70
diff
changeset
|
400 output_storage[0][0] = dx |
30
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
401 def grad(self, *args): |
bf0145fa73e8
added c implementation for CrossentropySoftmax1Hot
bergstrj@iro.umontreal.ca
parents:
25
diff
changeset
|
402 raise NotImplementedError() |
181
1b06bc2c3ca9
fixed c_code for the ops in nnet_ops.py
Olivier Breuleux <breuleuo@iro.umontreal.ca>
parents:
121
diff
changeset
|
403 def c_code(self, node, name, (dnll, sm, y_idx), (dx,), sub): |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
404 y_idx_type = node.inputs[2].type.dtype_specs()[1] |
32 | 405 return """ |
406 | |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
407 if ((%(dnll)s->descr->type_num != PyArray_DOUBLE) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
408 || (%(sm)s->descr->type_num != PyArray_DOUBLE) |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
409 ) |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
410 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
411 PyErr_SetString(PyExc_TypeError, "types should be float64, float64, int64"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
412 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
413 } |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
414 if ((%(y_idx)s->descr->type_num != PyArray_INT64) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
415 && (%(y_idx)s->descr->type_num != PyArray_INT32) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
416 && (%(y_idx)s->descr->type_num != PyArray_INT16) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
417 && (%(y_idx)s->descr->type_num != PyArray_INT8)) |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
418 { |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
419 PyErr_SetString(PyExc_TypeError, "y_idx not int8, int16, int32, or int64"); |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
420 %(fail)s; |
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
421 } |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
422 if ((%(dnll)s->nd != 1) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
423 || (%(sm)s->nd != 2) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
424 || (%(y_idx)s->nd != 1)) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
425 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
426 PyErr_SetString(PyExc_ValueError, "rank error"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
427 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
428 } |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
429 if ((%(dnll)s->dimensions[0] != %(sm)s->dimensions[0]) |
68 | 430 || (%(dnll)s->dimensions[0] != %(y_idx)s->dimensions[0])) |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
431 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
432 PyErr_SetString(PyExc_ValueError, "dimension mismatch"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
433 %(fail)s; |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
434 } |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
435 if ((NULL == %(dx)s) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
436 || (%(dx)s->dimensions[0] != %(sm)s->dimensions[0]) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
437 || (%(dx)s->dimensions[1] != %(sm)s->dimensions[1])) |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
438 { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
439 if (NULL != %(dx)s) Py_XDECREF(%(dx)s); |
68 | 440 %(dx)s = (PyArrayObject*)PyArray_SimpleNew(2, PyArray_DIMS(%(sm)s), type_num_%(sm)s); |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
441 if(!%(dx)s) { |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
442 PyErr_SetString(PyExc_MemoryError, "failed to alloc dx output"); |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
443 %(fail)s |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
444 } |
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
445 } |
24 | 446 |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
447 for (size_t i = 0; i < %(dx)s->dimensions[0]; ++i) |
32 | 448 { |
449 const double dnll_i = ((double*)(%(dnll)s->data + %(dnll)s->strides[0] * i))[0]; | |
450 | |
185
3d953844abd3
support for more int types in crossentropysoftmax1hot
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
184
diff
changeset
|
451 const %(y_idx_type)s y_i = ((%(y_idx_type)s*)(%(y_idx)s->data + %(y_idx)s->strides[0] * i))[0]; |
32 | 452 |
453 const double* __restrict__ sm_i = (double*)(%(sm)s->data + %(sm)s->strides[0] * i); | |
454 npy_intp Ssm = %(sm)s->strides[1]/sizeof(double); | |
455 | |
456 double* __restrict__ dx_i = (double*)(%(dx)s->data + %(dx)s->strides[0] * i); | |
457 npy_intp Sdx = %(dx)s->strides[1]/sizeof(double); | |
458 | |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
459 for (size_t j = 0; j < %(dx)s->dimensions[1]; ++j) |
32 | 460 { |
461 dx_i[j * Sdx] = dnll_i * sm_i[j * Ssm]; | |
462 } | |
67
810a8e3c85e1
fixed horrible memory leak from crossentropy...
bergstra@is23.m
parents:
34
diff
changeset
|
463 if (y_i >= %(dx)s->dimensions[1]) |
32 | 464 { |
465 %(fail)s; | |
466 } | |
467 dx_i[y_i * Sdx] -= dnll_i; | |
468 } | |
469 """ % dict(locals(), **sub) | |
69
8c2607f387e6
added softplus, elaborated sigmoid
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
68
diff
changeset
|
470 |
70
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
471 def crossentropy_softmax_1hot(x, y_idx, **kwargs): |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
472 b = tensor.zeros_like(x[0,:]) |
76e5c0f37165
better docs & precondition testing for cross_entropy_softmax_1hot & friends
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
69
diff
changeset
|
473 return crossentropy_softmax_1hot_with_bias(x, b, y_idx, **kwargs) |
382
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
474 |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
475 def binary_crossentropy(output, target): |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
476 """ |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
477 Compute the crossentropy of binary output wrt binary target. |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
478 @note: We do not sum, crossentropy is computed by component. |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
479 @todo: Rewrite as a scalar, and then broadcast to tensor. |
b4efd192d880
Moved xent loss to nnet_ups
Joseph Turian <turian@gmail.com>
parents:
381
diff
changeset
|
480 """ |
383 | 481 return -(target * tensor.log(output) + (1 - target) * tensor.log(1 - output)) |
419
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
482 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
483 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
484 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
485 class Prepend_scalar_constant_to_each_row(theano.Op): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
486 def __init__(self, val = 0): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
487 if isinstance(val, float): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
488 val = scalar.constant(val) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
489 self.val = val |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
490 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
491 def make_node(self, mat): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
492 #check type of input |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
493 if not isinstance(mat,theano.Result) or not mat.type==tensor.matrix().type: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
494 raise TypeError("Expected a matrix as input") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
495 x = tensor.as_tensor(mat) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
496 y = tensor.as_tensor(self.val) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
497 if x.type.dtype != y.type.dtype: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
498 TypeError("the value to prepend don't have the same type as the matrix") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
499 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
500 node = theano.Apply(op=self, inputs=[mat], outputs=[tensor.matrix()]) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
501 return node |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
502 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
503 def perform(self, node, (mat, ), (output, )): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
504 new_shape=(mat.shape[0],mat.shape[1]+1) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
505 if output[0] == None: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
506 output[0]=numpy.empty(new_shape,dtype=mat.dtype) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
507 out=output[0] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
508 else: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
509 if output[0].shape!=new_shape: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
510 try: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
511 output[0].resize(new_shape) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
512 except: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
513 output[0]=numpy.empty(new_shape, dtype=mat.dtype) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
514 out=output[0] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
515 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
516 out[:,0].fill(self.val.data) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
517 out[:,1:]=mat |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
518 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
519 def grad(self, (mat,), (goutput,)): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
520 return goutput[:,1:] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
521 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
522 class Prepend_scalar_to_each_row(theano.Op): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
523 def make_node(self, val, mat): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
524 #check type of input |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
525 if isinstance(val, float): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
526 val = scalar.constant(val) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
527 if not isinstance(mat,theano.Result) or not mat.type==tensor.matrix().type: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
528 raise TypeError("Expected a matrix as input") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
529 x = tensor.as_tensor(mat) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
530 y = tensor.as_tensor(val) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
531 if x.type.dtype != y.type.dtype: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
532 TypeError("the value to prepend don't have the same type as the matrix") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
533 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
534 node = theano.Apply(op=self, inputs=[val,mat], outputs=[tensor.matrix()]) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
535 return node |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
536 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
537 def perform(self, node, (val,mat), (output, )): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
538 new_shape=(mat.shape[0],mat.shape[1]+1) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
539 if output[0] == None: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
540 output[0]=numpy.empty(new_shape,dtype=mat.dtype) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
541 out=output[0] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
542 else: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
543 if output[0].shape!=new_shape: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
544 try: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
545 output[0].resize(new_shape) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
546 except: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
547 output[0]=numpy.empty(new_shape, dtype=mat.dtype) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
548 out=output[0] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
549 out[:,0].fill(val) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
550 out[:,1:]=mat |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
551 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
552 def grad(self, (val, mat), (goutput,)): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
553 return goutput[:,0], goutput[:,1:] |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
554 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
555 prepend_scalar_to_each_row = Prepend_scalar_to_each_row() |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
556 prepend_0_to_each_row = Prepend_scalar_constant_to_each_row(0.) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
557 prepend_1_to_each_row = Prepend_scalar_constant_to_each_row(1.) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
558 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
559 class solve(theano.Op): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
560 """ |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
561 Find the solution to the linear equation Ax=b, |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
562 where A is a 2d matrix and b is a 1d or 2d matrix. |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
563 It use numpy.solve to find the solution. |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
564 """ |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
565 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
566 def make_node(self, A, b): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
567 if not isinstance(A, theano.Result) or not A.type==tensor.matrix().type: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
568 raise TypeError("We expected that A had a matrix type") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
569 if not isinstance(B, theano.Result) or not B.type==tensor.matrix().type: |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
570 raise TypeError("We expected that B had a matrix type") |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
571 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
572 node = theano.Apply(op=self, inputs=[A, B], outputs=[tensor.matrix()]) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
573 return node |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
574 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
575 def perform(self, node, (A, B), (output, )): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
576 ret=numpy.solve(A,B) |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
577 output[0]=ret |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
578 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
579 def grad(self, (theta, A, B), (gtheta,)): |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
580 raise NotImplementedError() |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
581 |
43d9aa93934e
added other_ops.py to nnet_ops; added basic tests, no docs.
James Bergstra <bergstrj@iro.umontreal.ca>
parents:
383
diff
changeset
|
582 |