test_fit_a_line.py 7.4 KB
Newer Older
1
#   Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved.
D
dzhwinter 已提交
2
#
D
dzhwinter 已提交
3 4 5
# 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
D
dzhwinter 已提交
6
#
D
dzhwinter 已提交
7
#     http://www.apache.org/licenses/LICENSE-2.0
D
dzhwinter 已提交
8
#
D
dzhwinter 已提交
9 10 11 12 13 14
# 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.

15 16
from __future__ import print_function

17
import paddle
18
import paddle.fluid as fluid
A
arlesniak 已提交
19 20
import paddle.static.amp as amp

Y
Yang Yu 已提交
21
import contextlib
22
import numpy
Y
Yang Yu 已提交
23
import unittest
24 25
import math
import sys
武毅 已提交
26
import os
Q
QI JUN 已提交
27

P
pangyoki 已提交
28 29
paddle.enable_static()

Q
QI JUN 已提交
30

A
arlesniak 已提交
31
def train(use_cuda, save_dirname, is_local, use_bf16, pure_bf16):
Y
Yang Yu 已提交
32 33
    x = fluid.layers.data(name='x', shape=[13], dtype='float32')
    y = fluid.layers.data(name='y', shape=[1], dtype='float32')
Q
QI JUN 已提交
34

A
arlesniak 已提交
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
    if use_bf16:
        if not pure_bf16:
            with amp.bf16.bf16_guard():
                y_predict = fluid.layers.fc(input=x, size=1, act=None)
            cost = fluid.layers.square_error_cost(input=y_predict, label=y)
            avg_cost = fluid.layers.mean(cost)
        else:
            y_predict = fluid.layers.fc(input=x, size=1, act=None)
            with amp.bf16.bf16_guard():
                cost = fluid.layers.square_error_cost(input=y_predict, label=y)
                avg_cost = fluid.layers.mean(cost)
    else:
        y_predict = fluid.layers.fc(input=x, size=1, act=None)
        cost = fluid.layers.square_error_cost(input=y_predict, label=y)
        avg_cost = fluid.layers.mean(cost)
Q
QI JUN 已提交
50

Y
Yang Yu 已提交
51
    sgd_optimizer = fluid.optimizer.SGD(learning_rate=0.001)
A
arlesniak 已提交
52

53
    if use_bf16:
A
arlesniak 已提交
54 55 56 57 58
        sgd_optimizer = amp.bf16.decorate_bf16(
            sgd_optimizer,
            amp_lists=amp.bf16.AutoMixedPrecisionListsBF16(),
            use_bf16_guard=False,
            use_pure_bf16=pure_bf16)
W
Wu Yi 已提交
59
    sgd_optimizer.minimize(avg_cost)
Q
QI JUN 已提交
60

Y
Yang Yu 已提交
61
    BATCH_SIZE = 20
Q
QI JUN 已提交
62

Y
Yang Yu 已提交
63 64 65 66
    train_reader = paddle.batch(
        paddle.reader.shuffle(
            paddle.dataset.uci_housing.train(), buf_size=500),
        batch_size=BATCH_SIZE)
Q
QI JUN 已提交
67

Y
Yang Yu 已提交
68 69 70
    place = fluid.CUDAPlace(0) if use_cuda else fluid.CPUPlace()
    exe = fluid.Executor(place)

武毅 已提交
71 72 73
    def train_loop(main_program):
        feeder = fluid.DataFeeder(place=place, feed_list=[x, y])
        exe.run(fluid.default_startup_program())
A
arlesniak 已提交
74 75 76 77
        test_prog = main_program.clone(for_test=True)
        if pure_bf16:
            sgd_optimizer.amp_init(
                exe.place, test_program=test_prog, use_bf16_test=True)
武毅 已提交
78 79 80 81 82 83 84

        PASS_NUM = 100
        for pass_id in range(PASS_NUM):
            for data in train_reader():
                avg_loss_value, = exe.run(main_program,
                                          feed=feeder.feed(data),
                                          fetch_list=[avg_cost])
A
arlesniak 已提交
85 86
                if avg_loss_value[0] < 10.0 or pure_bf16:
                    if save_dirname is not None and not pure_bf16:
武毅 已提交
87 88 89 90 91 92 93 94 95 96 97
                        fluid.io.save_inference_model(save_dirname, ['x'],
                                                      [y_predict], exe)
                    return
                if math.isnan(float(avg_loss_value)):
                    sys.exit("got NaN loss, training failed.")
        raise AssertionError("Fit a line cost is too large, {0:2.2}".format(
            avg_loss_value[0]))

    if is_local:
        train_loop(fluid.default_main_program())
    else:
G
gongweibao 已提交
98 99
        port = os.getenv("PADDLE_PSERVER_PORT", "6174")
        pserver_ips = os.getenv("PADDLE_PSERVER_IPS")  # ip,ip...
武毅 已提交
100 101 102 103
        eplist = []
        for ip in pserver_ips.split(","):
            eplist.append(':'.join([ip, port]))
        pserver_endpoints = ",".join(eplist)  # ip:port,ip:port...
G
gongweibao 已提交
104
        trainers = int(os.getenv("PADDLE_TRAINERS"))
武毅 已提交
105
        current_endpoint = os.getenv("POD_IP") + ":" + port
G
gongweibao 已提交
106 107
        trainer_id = int(os.getenv("PADDLE_TRAINER_ID"))
        training_role = os.getenv("PADDLE_TRAINING_ROLE", "TRAINER")
武毅 已提交
108
        t = fluid.DistributeTranspiler()
Y
Yancey1989 已提交
109
        t.transpile(trainer_id, pservers=pserver_endpoints, trainers=trainers)
武毅 已提交
110 111 112 113 114 115 116 117
        if training_role == "PSERVER":
            pserver_prog = t.get_pserver_program(current_endpoint)
            pserver_startup = t.get_startup_program(current_endpoint,
                                                    pserver_prog)
            exe.run(pserver_startup)
            exe.run(pserver_prog)
        elif training_role == "TRAINER":
            train_loop(t.get_trainer_program())
Y
Yang Yu 已提交
118 119


A
arlesniak 已提交
120
def infer(use_cuda, save_dirname=None, use_bf16=False):
121 122 123 124 125 126
    if save_dirname is None:
        return

    place = fluid.CUDAPlace(0) if use_cuda else fluid.CPUPlace()
    exe = fluid.Executor(place)

127 128 129
    inference_scope = fluid.core.Scope()
    with fluid.scope_guard(inference_scope):
        # Use fluid.io.load_inference_model to obtain the inference program desc,
T
tianshuo78520a 已提交
130
        # the feed_target_names (the names of variables that will be fed
131 132 133 134 135 136 137 138
        # data using feed operators), and the fetch_targets (variables that
        # we want to obtain data from using fetch operators).
        [inference_program, feed_target_names,
         fetch_targets] = fluid.io.load_inference_model(save_dirname, exe)

        # The input's dimension should be 2-D and the second dim is 13
        # The input data should be >= 0
        batch_size = 10
139 140 141 142

        test_reader = paddle.batch(
            paddle.dataset.uci_housing.test(), batch_size=batch_size)

143
        test_data = next(test_reader())
144 145 146 147 148
        test_feat = numpy.array(
            [data[0] for data in test_data]).astype("float32")
        test_label = numpy.array(
            [data[1] for data in test_data]).astype("float32")

149 150
        assert feed_target_names[0] == 'x'
        results = exe.run(inference_program,
151
                          feed={feed_target_names[0]: numpy.array(test_feat)},
152
                          fetch_list=fetch_targets)
153 154 155
        print("infer shape: ", results[0].shape)
        print("infer results: ", results[0])
        print("ground truth: ", test_label)
156 157


A
arlesniak 已提交
158
def main(use_cuda, is_local=True, use_bf16=False, pure_bf16=False):
159 160 161
    if use_cuda and not fluid.core.is_compiled_with_cuda():
        return

162 163 164
    if use_bf16 and not fluid.core.is_compiled_with_mkldnn():
        return

165 166 167
    # Directory for saving the trained model
    save_dirname = "fit_a_line.inference.model"

A
arlesniak 已提交
168 169 170 171 172 173 174 175 176 177 178 179 180
    train(use_cuda, save_dirname, is_local, use_bf16, pure_bf16)
    infer(use_cuda, save_dirname, use_bf16)


class TestFitALineBase(unittest.TestCase):
    @contextlib.contextmanager
    def program_scope_guard(self):
        prog = fluid.Program()
        startup_prog = fluid.Program()
        scope = fluid.core.Scope()
        with fluid.scope_guard(scope):
            with fluid.program_guard(prog, startup_prog):
                yield
181 182


A
arlesniak 已提交
183
class TestFitALine(TestFitALineBase):
Y
Yang Yu 已提交
184 185 186 187 188 189 190 191
    def test_cpu(self):
        with self.program_scope_guard():
            main(use_cuda=False)

    def test_cuda(self):
        with self.program_scope_guard():
            main(use_cuda=True)

A
arlesniak 已提交
192 193 194 195

@unittest.skipIf(not fluid.core.supports_bfloat16(),
                 "place does not support BF16 evaluation")
class TestFitALineBF16(TestFitALineBase):
196 197 198 199
    def test_bf16(self):
        with self.program_scope_guard():
            main(use_cuda=False, use_bf16=True)

A
arlesniak 已提交
200 201 202
    def test_pure_bf16(self):
        with self.program_scope_guard():
            main(use_cuda=False, use_bf16=True, pure_bf16=True)
Y
Yang Yu 已提交
203 204 205 206


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