test_model_base.py 14.2 KB
Newer Older
1
2
3
4
5
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT License.

"""Tests for BenchmarkRegistry module."""

6
from superbench.benchmarks import Platform, Framework, Precision, BenchmarkRegistry, BenchmarkType, ReturnCode
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
from superbench.benchmarks.model_benchmarks import ModelBenchmark


class FakeModelBenchmark(ModelBenchmark):
    """Fake benchmark inherit from ModelBenchmark."""
    def __init__(self, name, parameters=''):
        """Constructor.

        Args:
            name: benchmark name.
            parameters: benchmark parameters.
        """
        super().__init__(name, parameters)
        self._supported_precision = [Precision.FLOAT32, Precision.FLOAT16]

    def add_parser_arguments(self):
        """Add the specified arguments."""
        super().add_parser_arguments()
        self._parser.add_argument(
            '--hidden_size',
            type=int,
            default=1024,
            required=False,
30
            help='Hidden size.',
31
32
33
34
35
36
37
        )

        self._parser.add_argument(
            '--seq_len',
            type=int,
            default=512,
            required=False,
38
            help='Sequence length.',
39
40
        )

41
42
43
44
    def _judge_gpu_availability(self):
        """Judge GPUs' availability according to arguments and running environment."""
        self._gpu_available = False

45
46
    def _init_distributed_setting(self):
        """Initialize the distributed library and bind the worker to GPU."""
47
        return True
48
49
50

    def _generate_dataset(self):
        """Generate dataset for benchmarking according to shape info."""
51
        return True
52
53
54

    def _init_dataloader(self):
        """Initialize the distributed dataloader."""
55
        return True
56
57
58

    def _create_optimizer(self):
        """Create the optimzier instance used for training."""
59
        return True
60
61
62

    def _create_model(self, precision):
        """Construct the model for benchmarking."""
63
        return True
64
65
66
67
68
69
70
71
72
73
74
75
76

    def _train_step(self, precision):
        """Define the training process.

        Args:
            precision (str): precision of model and input data,
              such as float, half.

        Return:
            The step-time list of every training step.
        """
        duration = []
        for i in range(self._args.num_steps):
77
            duration.append(2.0)
78
79
80
81
82
83
84
85
86
87
88
89
90
91
        return duration

    def _inference_step(self, precision):
        """Define the inference process.

        Args:
            precision (str): precision of model and input data,
              such as float, half.

        Return:
            The latency list of every inference operation.
        """
        duration = []
        for i in range(self._args.num_steps):
92
            duration.append(4.0)
93
94
        return duration

95
    def _cal_params_count(self):
96
97
98
99
100
101
102
103
        """Calculate the parameters scale of the model.

        Return:
            The count of trainable parameters.
        """
        return 200


104
def create_benchmark(params='--num_steps 8'):
105
106
107
108
109
    """Register and create benchmark."""
    # Register the FakeModelBenchmark benchmark.
    BenchmarkRegistry.register_benchmark(
        'pytorch-fake-model',
        FakeModelBenchmark,
110
        parameters='--hidden_size 2',
111
112
        platform=Platform.CUDA,
    )
113
114
115
    context = BenchmarkRegistry.create_benchmark_context(
        'fake-model', platform=Platform.CUDA, parameters=params, framework=Framework.PYTORCH
    )
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
    name = BenchmarkRegistry._BenchmarkRegistry__get_benchmark_name(context)
    assert (name)
    (benchmark_class, predefine_params) = BenchmarkRegistry._BenchmarkRegistry__select_benchmark(name, context.platform)
    assert (benchmark_class)
    return benchmark_class(name, predefine_params + ' ' + context.parameters)


def test_arguments_related_interfaces():
    """Test arguments related interfaces.

    Benchmark.add_parser_arguments(),
    Benchmark.parse_args(),
    Benchmark.get_configurable_settings()
    """
    # Positive case for parse_args().
131
    benchmark = create_benchmark('--num_steps 9')
132
133
134
135
136
    benchmark.add_parser_arguments()
    (ret, args, unknown) = benchmark.parse_args()
    assert (ret and args.num_steps == 9)

    # Negative case for parse_args() - invalid precision.
137
    benchmark = create_benchmark('--num_steps 8 --precision fp32')
138
139
140
141
142
143
144
145
146
147
    benchmark.add_parser_arguments()
    (ret, args, unknown) = benchmark.parse_args()
    assert (ret is False)

    # Test get_configurable_settings().
    settings = benchmark.get_configurable_settings()
    expected_settings = (
        """optional arguments:
  --run_count int       The run count of benchmark.
  --duration int        The elapsed time of benchmark in seconds.
148
149
  --num_warmup int      The number of warmup step.
  --num_steps int       The number of test step.
150
  --sample_count int    The number of data samples in dataset.
151
  --batch_size int      The number of batch size.
152
153
154
155
156
157
158
  --precision Precision [Precision ...]
                        Model precision. E.g. float16 float32 float64 bfloat16
                        uint8 int8 int16 int32 int64.
  --model_action ModelAction [ModelAction ...]
                        Benchmark model process. E.g. train inference.
  --distributed_impl DistributedImpl
                        Distributed implementations. E.g. ddp mirrored
159
                        multiworkermirrored parameterserver horovod.
160
  --distributed_backend DistributedBackend
161
162
                        Distributed backends. E.g. nccl mpi gloo.
  --no_gpu              Disable GPU training.
163
  --pin_memory          Enable option to pin memory in data loader.
164
165
  --hidden_size int     Hidden size.
  --seq_len int         Sequence length."""
166
167
168
169
170
171
172
    )
    assert (settings == expected_settings)


def test_preprocess():
    """Test interface Benchmark._preprocess()."""
    # Positive case for _preprocess().
173
    benchmark = create_benchmark('--num_steps 8')
174
175
176
177
178
179
180
    assert (benchmark._preprocess())
    assert (benchmark.return_code == ReturnCode.SUCCESS)
    settings = benchmark.get_configurable_settings()
    expected_settings = (
        """optional arguments:
  --run_count int       The run count of benchmark.
  --duration int        The elapsed time of benchmark in seconds.
181
182
  --num_warmup int      The number of warmup step.
  --num_steps int       The number of test step.
183
  --sample_count int    The number of data samples in dataset.
184
  --batch_size int      The number of batch size.
185
186
187
188
189
190
191
  --precision Precision [Precision ...]
                        Model precision. E.g. float16 float32 float64 bfloat16
                        uint8 int8 int16 int32 int64.
  --model_action ModelAction [ModelAction ...]
                        Benchmark model process. E.g. train inference.
  --distributed_impl DistributedImpl
                        Distributed implementations. E.g. ddp mirrored
192
                        multiworkermirrored parameterserver horovod.
193
  --distributed_backend DistributedBackend
194
195
                        Distributed backends. E.g. nccl mpi gloo.
  --no_gpu              Disable GPU training.
196
  --pin_memory          Enable option to pin memory in data loader.
197
198
  --hidden_size int     Hidden size.
  --seq_len int         Sequence length."""
199
200
201
202
    )
    assert (settings == expected_settings)

    # Negative case for _preprocess() - invalid precision.
203
    benchmark = create_benchmark('--num_steps 8 --precision fp32')
204
205
206
207
    assert (benchmark._preprocess() is False)
    assert (benchmark.return_code == ReturnCode.INVALID_ARGUMENT)

    # Negative case for _preprocess() - invalid benchmark type.
208
    benchmark = create_benchmark('--num_steps 8 --precision float32')
209
210
211
212
213
214
215
216
217
218
    benchmark._benchmark_type = Platform.CUDA
    assert (benchmark._preprocess() is False)
    assert (benchmark.return_code == ReturnCode.INVALID_BENCHMARK_TYPE)


def test_train():
    """Test interface Benchmark.__train()."""
    benchmark = create_benchmark()
    expected_result = (
        '{"name": "pytorch-fake-model", "type": "model", "run_count": 1, "return_code": 0, '
219
220
        '"start_time": null, "end_time": null, "raw_data": {'
        '"steptime_train_float32": [[2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0]], '
221
        '"throughput_train_float32": [[16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0]]}, '
222
        '"result": {"steptime_train_float32": [2.0], "throughput_train_float32": [16000.0]}, '
223
        '"reduce_op": {"steptime_train_float32": "max", "throughput_train_float32": "min"}}'
224
225
226
227
228
229
    )
    assert (benchmark._preprocess())
    assert (benchmark._ModelBenchmark__train(Precision.FLOAT32))
    assert (benchmark.serialized_result == expected_result)

    # Step time list is empty (simulate training failure).
230
    benchmark = create_benchmark('--num_steps 0')
231
    expected_result = (
232
        '{"name": "pytorch-fake-model", "type": "model", "run_count": 1, "return_code": 3, '
233
        '"start_time": null, "end_time": null, "raw_data": {}, "result": {}, "reduce_op": {}}'
234
235
236
237
238
239
240
241
242
243
244
    )
    assert (benchmark._preprocess())
    assert (benchmark._ModelBenchmark__train(Precision.FLOAT32) is False)
    assert (benchmark.serialized_result == expected_result)


def test_inference():
    """Test interface Benchmark.__inference()."""
    benchmark = create_benchmark()
    expected_result = (
        '{"name": "pytorch-fake-model", "type": "model", "run_count": 1, "return_code": 0, '
245
246
        '"start_time": null, "end_time": null, "raw_data": {'
        '"steptime_inference_float16": [[4.0, 4.0, 4.0, 4.0, 4.0, 4.0, 4.0, 4.0]], '
247
        '"throughput_inference_float16": [[8000.0, 8000.0, 8000.0, 8000.0, 8000.0, 8000.0, 8000.0, 8000.0]]}, '
248
        '"result": {"steptime_inference_float16": [4.0], "throughput_inference_float16": [8000.0]}, '
249
        '"reduce_op": {"steptime_inference_float16": null, "throughput_inference_float16": null}}'
250
251
252
253
254
255
    )
    assert (benchmark._preprocess())
    assert (benchmark._ModelBenchmark__inference(Precision.FLOAT16))
    assert (benchmark.serialized_result == expected_result)

    # Step time list is empty (simulate inference failure).
256
    benchmark = create_benchmark('--num_steps 0')
257
    expected_result = (
258
        '{"name": "pytorch-fake-model", "type": "model", "run_count": 1, "return_code": 3, '
259
        '"start_time": null, "end_time": null, "raw_data": {}, "result": {}, "reduce_op": {}}'
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
    )
    assert (benchmark._preprocess())
    assert (benchmark._ModelBenchmark__inference(Precision.FLOAT16) is False)
    assert (benchmark.serialized_result == expected_result)


def test_benchmark():
    """Test interface Benchmark._benchmark()."""
    # Positive case for _benchmark().
    benchmark = create_benchmark()
    benchmark._preprocess()
    assert (benchmark._benchmark())
    assert (benchmark.name == 'pytorch-fake-model')
    assert (benchmark.type == BenchmarkType.MODEL)
    assert (benchmark.run_count == 1)
    assert (benchmark.return_code == ReturnCode.SUCCESS)
    expected_raw_data = {
277
        'steptime_train_float32': [[2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0]],
278
        'throughput_train_float32': [[16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0]],
279
        'steptime_train_float16': [[2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0]],
280
281
282
283
284
285
286
287
288
289
290
291
292
        'throughput_train_float16': [[16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0]]
    }
    assert (benchmark.raw_data == expected_raw_data)
    expected_result = {
        'steptime_train_float32': [2.0],
        'throughput_train_float32': [16000.0],
        'steptime_train_float16': [2.0],
        'throughput_train_float16': [16000.0]
    }
    assert (benchmark.result == expected_result)

    expected_serialized_result = (
        '{"name": "pytorch-fake-model", "type": "model", "run_count": 1, "return_code": 0, "start_time": null, '
293
        '"end_time": null, "raw_data": {"steptime_train_float32": [[2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0]], '
294
        '"throughput_train_float32": [[16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0]], '
295
        '"steptime_train_float16": [[2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0, 2.0]], '
296
297
        '"throughput_train_float16": [[16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0, 16000.0]]}, '
        '"result": {"steptime_train_float32": [2.0], "throughput_train_float32": [16000.0], '
298
        '"steptime_train_float16": [2.0], "throughput_train_float16": [16000.0]}, '
299
        '"reduce_op": {"steptime_train_float32": "max", "throughput_train_float32": "min", '
300
        '"steptime_train_float16": "max", "throughput_train_float16": "min"}}'
301
302
303
304
    )
    assert (benchmark.serialized_result == expected_serialized_result)

    # Negative case for _benchmark() - no supported precision found.
305
    benchmark = create_benchmark('--precision int16')
306
307
308
309
310
    assert (benchmark._preprocess())
    assert (benchmark._benchmark() is False)
    assert (benchmark.return_code == ReturnCode.NO_SUPPORTED_PRECISION)

    # Negative case for _benchmark() - model train failure, step time list is empty.
311
    benchmark = create_benchmark('--num_steps 0')
312
313
    assert (benchmark._preprocess())
    assert (benchmark._benchmark() is False)
314
    assert (benchmark.return_code == ReturnCode.INVALID_BENCHMARK_RESULT)
315
316

    # Negative case for _benchmark() - model inference failure, step time list is empty.
317
    benchmark = create_benchmark('--model_action inference --num_steps 0')
318
319
    assert (benchmark._preprocess())
    assert (benchmark._benchmark() is False)
320
    assert (benchmark.return_code == ReturnCode.INVALID_BENCHMARK_RESULT)
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343


def test_check_result_format():
    """Test interface Benchmark.__check_result_format()."""
    # Positive case for __check_result_format().
    benchmark = create_benchmark()
    benchmark._preprocess()
    assert (benchmark._benchmark())
    assert (benchmark._Benchmark__check_result_type())
    assert (benchmark._Benchmark__check_summarized_result())
    assert (benchmark._Benchmark__check_raw_data())

    # Negative case for __check_result_format() - change List[int] to List[str].
    benchmark._result._BenchmarkResult__result = {'metric1': ['2.0']}
    assert (benchmark._Benchmark__check_summarized_result() is False)

    # Negative case for __check_raw_data() - change List[List[int]] to List[List[str]].
    benchmark._result._BenchmarkResult__raw_data = {'metric1': [['2.0']]}
    assert (benchmark._Benchmark__check_raw_data() is False)

    # Negative case for __check_raw_data() - invalid benchmark result.
    assert (benchmark._Benchmark__check_result_format() is False)
    assert (benchmark.return_code == ReturnCode.INVALID_BENCHMARK_RESULT)