test_imperative_ptb_rnn.py 16.1 KB
Newer Older
J
JiabinYang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
#   Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import unittest
16 17 18 19 20

import numpy as np
from test_imperative_base import new_program_scope
from utils import DyGraphProgramDescTracerTestHelper, is_equal_program

L
Leo Chen 已提交
21
import paddle
J
JiabinYang 已提交
22
import paddle.fluid as fluid
23
import paddle.fluid.core as core
J
JiabinYang 已提交
24
import paddle.fluid.framework as framework
L
lujun 已提交
25
from paddle.fluid.dygraph.base import to_variable
26 27 28
from paddle.fluid.dygraph.nn import Embedding
from paddle.fluid.framework import _in_legacy_dygraph, _test_eager_guard
from paddle.fluid.optimizer import SGDOptimizer
29
from paddle.jit import TracedLayer
J
JiabinYang 已提交
30 31


32
class SimpleLSTMRNN(fluid.Layer):
33 34 35
    def __init__(
        self, hidden_size, num_steps, num_layers=2, init_scale=0.1, dropout=None
    ):
36
        super().__init__()
J
JiabinYang 已提交
37 38 39 40
        self._hidden_size = hidden_size
        self._num_layers = num_layers
        self._init_scale = init_scale
        self._dropout = dropout
41 42
        self._input = None
        self._num_steps = num_steps
43 44
        self.cell_array = []
        self.hidden_array = []
45
        self._create_parameter()
J
JiabinYang 已提交
46

47
    def _create_parameter(self):
J
JiabinYang 已提交
48 49 50 51 52 53
        self.weight_1_arr = []
        self.weight_2_arr = []
        self.bias_arr = []
        self.mask_array = []

        for i in range(self._num_layers):
54
            weight_1 = self.create_parameter(
55 56
                attr=fluid.ParamAttr(
                    initializer=fluid.initializer.UniformInitializer(
57 58 59
                        low=-self._init_scale, high=self._init_scale
                    )
                ),
J
JiabinYang 已提交
60 61 62
                shape=[self._hidden_size * 2, self._hidden_size * 4],
                dtype="float32",
                default_initializer=fluid.initializer.UniformInitializer(
63 64 65
                    low=-self._init_scale, high=self._init_scale
                ),
            )
66
            self.weight_1_arr.append(self.add_parameter('w_%d' % i, weight_1))
67
            bias_1 = self.create_parameter(
68 69
                attr=fluid.ParamAttr(
                    initializer=fluid.initializer.UniformInitializer(
70 71 72
                        low=-self._init_scale, high=self._init_scale
                    )
                ),
73
                shape=[self._hidden_size * 4],
J
JiabinYang 已提交
74
                dtype="float32",
75 76
                default_initializer=fluid.initializer.Constant(0.0),
            )
77
            self.bias_arr.append(self.add_parameter('b_%d' % i, bias_1))
J
JiabinYang 已提交
78

79 80 81 82 83
    def forward(self, input_embedding, init_hidden=None, init_cell=None):
        self.cell_array = []
        self.hidden_array = []

        for i in range(self._num_layers):
2
201716010711 已提交
84
            pre_hidden = paddle.slice(
85 86
                init_hidden, axes=[0], starts=[i], ends=[i + 1]
            )
2
201716010711 已提交
87
            pre_cell = paddle.slice(
88 89
                init_cell, axes=[0], starts=[i], ends=[i + 1]
            )
90
            pre_hidden = paddle.reshape(
91 92
                pre_hidden, shape=[-1, self._hidden_size]
            )
93
            pre_cell = paddle.reshape(pre_cell, shape=[-1, self._hidden_size])
J
JiabinYang 已提交
94 95 96 97
            self.hidden_array.append(pre_hidden)
            self.cell_array.append(pre_cell)

        res = []
98
        for index in range(self._num_steps):
2
201716010711 已提交
99
            self._input = paddle.slice(
100 101
                input_embedding, axes=[1], starts=[index], ends=[index + 1]
            )
102
            self._input = paddle.reshape(
103 104
                self._input, shape=[-1, self._hidden_size]
            )
J
JiabinYang 已提交
105 106 107 108 109 110
            for k in range(self._num_layers):
                pre_hidden = self.hidden_array[k]
                pre_cell = self.cell_array[k]
                weight_1 = self.weight_1_arr[k]
                bias = self.bias_arr[k]

111
                nn = fluid.layers.concat([self._input, pre_hidden], 1)
J
JiabinYang 已提交
112 113
                gate_input = fluid.layers.matmul(x=nn, y=weight_1)

114
                gate_input = paddle.add(gate_input, bias)
115 116 117
                i, j, f, o = fluid.layers.split(
                    gate_input, num_or_sections=4, dim=-1
                )
118 119 120 121
                c = pre_cell * paddle.nn.functional.sigmoid(
                    f
                ) + paddle.nn.functional.sigmoid(i) * paddle.tanh(j)
                m = paddle.tanh(c) * paddle.nn.functional.sigmoid(o)
122 123 124 125 126 127 128 129
                self.hidden_array[k] = m
                self.cell_array[k] = c
                self._input = m

                if self._dropout is not None and self._dropout > 0.0:
                    self._input = fluid.layers.dropout(
                        self._input,
                        dropout_prob=self._dropout,
130 131
                        dropout_implementation='upscale_in_train',
                    )
132
            res.append(
133
                paddle.reshape(self._input, shape=[1, -1, self._hidden_size])
134
            )
135
        real_res = fluid.layers.concat(res, 0)
136
        real_res = paddle.transpose(x=real_res, perm=[1, 0, 2])
137
        last_hidden = fluid.layers.concat(self.hidden_array, 1)
138
        last_hidden = paddle.reshape(
139 140
            last_hidden, shape=[-1, self._num_layers, self._hidden_size]
        )
141
        last_hidden = paddle.transpose(x=last_hidden, perm=[1, 0, 2])
142
        last_cell = fluid.layers.concat(self.cell_array, 1)
143
        last_cell = paddle.reshape(
144 145
            last_cell, shape=[-1, self._num_layers, self._hidden_size]
        )
146
        last_cell = paddle.transpose(x=last_cell, perm=[1, 0, 2])
147
        return real_res, last_hidden, last_cell
J
JiabinYang 已提交
148 149


150
class PtbModel(fluid.Layer):
151 152 153 154 155 156 157 158 159 160
    def __init__(
        self,
        hidden_size,
        vocab_size,
        num_layers=2,
        num_steps=20,
        init_scale=0.1,
        is_sparse=False,
        dropout=None,
    ):
161
        super().__init__()
J
JiabinYang 已提交
162 163 164 165 166 167
        self.hidden_size = hidden_size
        self.vocab_size = vocab_size
        self.init_scale = init_scale
        self.num_layers = num_layers
        self.num_steps = num_steps
        self.dropout = dropout
168 169 170 171 172 173 174
        self.simple_lstm_rnn = SimpleLSTMRNN(
            hidden_size,
            num_steps,
            num_layers=num_layers,
            init_scale=init_scale,
            dropout=dropout,
        )
175
        self.embedding = Embedding(
J
JiabinYang 已提交
176 177
            size=[vocab_size, hidden_size],
            dtype='float32',
178
            is_sparse=is_sparse,
J
JiabinYang 已提交
179 180 181
            param_attr=fluid.ParamAttr(
                name='embedding_para',
                initializer=fluid.initializer.UniformInitializer(
182 183 184 185
                    low=-init_scale, high=init_scale
                ),
            ),
        )
186
        self.softmax_weight = self.create_parameter(
187 188
            attr=fluid.ParamAttr(),
            shape=[self.hidden_size, self.vocab_size],
J
JiabinYang 已提交
189 190
            dtype="float32",
            default_initializer=fluid.initializer.UniformInitializer(
191 192 193
                low=-self.init_scale, high=self.init_scale
            ),
        )
194
        self.softmax_bias = self.create_parameter(
195 196
            attr=fluid.ParamAttr(),
            shape=[self.vocab_size],
J
JiabinYang 已提交
197 198
            dtype="float32",
            default_initializer=fluid.initializer.UniformInitializer(
199 200 201
                low=-self.init_scale, high=self.init_scale
            ),
        )
J
JiabinYang 已提交
202 203

    def forward(self, input, label, init_hidden, init_cell):
204
        init_h = paddle.reshape(
205 206
            init_hidden, shape=[self.num_layers, -1, self.hidden_size]
        )
J
JiabinYang 已提交
207

208
        init_c = paddle.reshape(
209 210
            init_cell, shape=[self.num_layers, -1, self.hidden_size]
        )
J
JiabinYang 已提交
211 212

        x_emb = self.embedding(input)
213
        x_emb = paddle.reshape(
214 215
            x_emb, shape=[-1, self.num_steps, self.hidden_size]
        )
J
JiabinYang 已提交
216 217 218 219
        if self.dropout is not None and self.dropout > 0.0:
            x_emb = fluid.layers.dropout(
                x_emb,
                dropout_prob=self.drop_out,
220 221
                dropout_implementation='upscale_in_train',
            )
222
        rnn_out, last_hidden, last_cell = self.simple_lstm_rnn(
223 224
            x_emb, init_h, init_c
        )
225
        rnn_out = paddle.reshape(
226 227
            rnn_out, shape=[-1, self.num_steps, self.hidden_size]
        )
228
        projection = fluid.layers.matmul(rnn_out, self.softmax_weight)
229
        projection = paddle.add(projection, self.softmax_bias)
230
        projection = paddle.reshape(projection, shape=[-1, self.vocab_size])
231
        loss = paddle.nn.functional.softmax_with_cross_entropy(
232 233
            logits=projection, label=label, soft_label=False
        )
234
        loss = paddle.reshape(loss, shape=[-1, self.num_steps])
235
        loss = paddle.mean(loss, axis=[0])
236
        loss = paddle.sum(loss)
J
JiabinYang 已提交
237 238 239 240

        return loss, last_hidden, last_cell


L
lujun 已提交
241
class TestDygraphPtbRnn(unittest.TestCase):
242
    def func_test_ptb_rnn(self):
243 244 245
        for is_sparse in [True, False]:
            self.ptb_rnn_cpu_float32(is_sparse)

246 247 248 249 250
    def test_ptb_rnn(self):
        with _test_eager_guard():
            self.func_test_ptb_rnn()
        self.func_test_ptb_rnn()

251
    def ptb_rnn_cpu_float32(self, is_sparse):
J
JiabinYang 已提交
252 253 254 255 256 257 258
        seed = 90
        hidden_size = 10
        vocab_size = 1000
        num_layers = 1
        num_steps = 3
        init_scale = 0.1
        batch_size = 4
259
        batch_num = 200
260 261
        traced_layer = None

L
lujun 已提交
262
        with fluid.dygraph.guard():
C
cnn 已提交
263
            paddle.seed(seed)
L
Leo Chen 已提交
264
            paddle.framework.random._manual_program_seed(seed)
J
JiabinYang 已提交
265
            # TODO: marsyang1993 Change seed to
266 267 268 269 270 271 272 273 274 275 276 277
            ptb_model = PtbModel(
                hidden_size=hidden_size,
                vocab_size=vocab_size,
                num_layers=num_layers,
                num_steps=num_steps,
                init_scale=init_scale,
                is_sparse=is_sparse,
            )

            sgd = SGDOptimizer(
                learning_rate=1e-3, parameter_list=ptb_model.parameters()
            )
278 279
            dy_param_updated = dict()
            dy_param_init = dict()
J
JiabinYang 已提交
280 281 282
            dy_loss = None
            last_hidden = None
            last_cell = None
283

284 285
            helper = DyGraphProgramDescTracerTestHelper(self)
            program = None
286

287
            for i in range(batch_num):
J
JiabinYang 已提交
288 289 290 291
                x_data = np.arange(12).reshape(4, 3).astype('int64')
                y_data = np.arange(1, 13).reshape(4, 3).astype('int64')
                y_data = y_data.reshape((-1, 1))
                init_hidden_data = np.zeros(
292 293 294 295 296
                    (num_layers, batch_size, hidden_size), dtype='float32'
                )
                init_cell_data = np.zeros(
                    (num_layers, batch_size, hidden_size), dtype='float32'
                )
J
JiabinYang 已提交
297 298 299 300
                x = to_variable(x_data)
                y = to_variable(y_data)
                init_hidden = to_variable(init_hidden_data)
                init_cell = to_variable(init_cell_data)
J
Jiabin Yang 已提交
301
                if i % 5 == 0 and _in_legacy_dygraph():
302
                    outs, traced_layer = TracedLayer.trace(
303 304
                        ptb_model, [x, y, init_hidden, init_cell]
                    )
305
                    outs_static = traced_layer([x, y, init_hidden, init_cell])
306
                    helper.assertEachVar(outs, outs_static)
307 308 309

                    if program is not None:
                        self.assertTrue(
310 311
                            is_equal_program(traced_layer.program, program)
                        )
312 313 314 315

                    program = traced_layer.program

                    traced_layer.save_inference_model(
316 317
                        './infe_imperative_ptb_rnn', feed=list(range(4))
                    )
318 319 320 321 322
                else:
                    outs = ptb_model(x, y, init_hidden, init_cell)

                dy_loss, last_hidden, last_cell = outs

J
JiabinYang 已提交
323
                if i == 0:
324
                    for param in ptb_model.parameters():
325
                        dy_param_init[param.name] = param.numpy()
L
lujun 已提交
326
                dy_loss.backward()
J
JiabinYang 已提交
327
                sgd.minimize(dy_loss)
328 329 330
                ptb_model.clear_gradients()
                if i == batch_num - 1:
                    for param in ptb_model.parameters():
331
                        dy_param_updated[param.name] = param.numpy()
332

333 334 335 336
            dy_loss_value = dy_loss.numpy()
            dy_last_cell_value = last_cell.numpy()
            dy_last_hidden_value = last_hidden.numpy()

337
        with new_program_scope():
C
cnn 已提交
338
            paddle.seed(seed)
L
Leo Chen 已提交
339
            paddle.framework.random._manual_program_seed(seed)
340 341 342 343 344 345 346 347 348 349 350 351 352 353
            ptb_model = PtbModel(
                hidden_size=hidden_size,
                vocab_size=vocab_size,
                num_layers=num_layers,
                num_steps=num_steps,
                init_scale=init_scale,
                is_sparse=is_sparse,
            )

            exe = fluid.Executor(
                fluid.CPUPlace()
                if not core.is_compiled_with_cuda()
                else fluid.CUDAPlace(0)
            )
354
            sgd = SGDOptimizer(learning_rate=1e-3)
355 356 357
            x = fluid.layers.data(
                name="x", shape=[-1, num_steps], dtype='int64'
            )
358
            y = fluid.layers.data(name="y", shape=[-1, 1], dtype='float32')
359 360 361 362 363 364
            init_hidden = fluid.layers.data(
                name="init_hidden", shape=[1], dtype='float32'
            )
            init_cell = fluid.layers.data(
                name="init_cell", shape=[1], dtype='float32'
            )
365 366

            static_loss, static_last_hidden, static_last_cell = ptb_model(
367 368
                x, y, init_hidden, init_cell
            )
369 370 371 372
            sgd.minimize(static_loss)
            static_param_updated = dict()
            static_param_init = dict()
            static_param_name_list = list()
373
            for param in ptb_model.parameters():
374 375
                static_param_name_list.append(param.name)

376 377 378 379
            out = exe.run(
                framework.default_startup_program(),
                fetch_list=static_param_name_list,
            )
380 381
            for i in range(len(static_param_name_list)):
                static_param_init[static_param_name_list[i]] = out[i]
J
JiabinYang 已提交
382 383 384
            static_loss_value = None
            static_last_cell_value = None
            static_last_hidden_value = None
385
            for i in range(batch_num):
386 387 388 389 390
                x_data = np.arange(12).reshape(4, 3).astype('int64')
                y_data = np.arange(1, 13).reshape(4, 3).astype('int64')
                x_data = x_data.reshape((-1, num_steps, 1))
                y_data = y_data.reshape((-1, 1))
                init_hidden_data = np.zeros(
391 392 393 394 395
                    (num_layers, batch_size, hidden_size), dtype='float32'
                )
                init_cell_data = np.zeros(
                    (num_layers, batch_size, hidden_size), dtype='float32'
                )
396 397
                fetch_list = [static_loss, static_last_hidden, static_last_cell]
                fetch_list.extend(static_param_name_list)
398 399 400 401 402 403 404 405 406 407
                out = exe.run(
                    fluid.default_main_program(),
                    feed={
                        "x": x_data,
                        "y": y_data,
                        "init_hidden": init_hidden_data,
                        "init_cell": init_cell_data,
                    },
                    fetch_list=fetch_list,
                )
408
                static_loss_value = out[0]
409 410
                static_last_hidden_value = out[1]
                static_last_cell_value = out[2]
J
JiabinYang 已提交
411

412 413
                if i == batch_num - 1:
                    for k in range(3, len(out)):
414 415 416
                        static_param_updated[
                            static_param_name_list[k - 3]
                        ] = out[k]
417

418
        np.testing.assert_array_equal(static_loss_value, dy_loss_value)
419 420 421 422 423 424
        np.testing.assert_array_equal(
            static_last_cell_value, dy_last_cell_value
        )
        np.testing.assert_array_equal(
            static_last_hidden_value, dy_last_hidden_value
        )
425
        for key, value in static_param_init.items():
426
            np.testing.assert_array_equal(value, dy_param_init[key])
427
        for key, value in static_param_updated.items():
428
            np.testing.assert_array_equal(value, dy_param_updated[key])
J
JiabinYang 已提交
429 430 431 432


if __name__ == '__main__':
    unittest.main()