UNPKG

159 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright 2017 Google LLC. All Rights Reserved.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 * =============================================================================
16 */
17import * as conv_util from './conv_util';
18describe('conv_util computeConv2DInfo', () => {
19 it('1x1 conv over 1x1 array with same pad', () => {
20 const inShape = [1, 1, 1, 1];
21 const stride = 1;
22 const dilation = 1;
23 const convInfo = conv_util.computeConv2DInfo(inShape, [1, 1, 1, 1], stride, dilation, 'same');
24 expect(convInfo.batchSize).toEqual(1);
25 expect(convInfo.outHeight).toEqual(1);
26 expect(convInfo.outWidth).toEqual(1);
27 expect(convInfo.outChannels).toEqual(1);
28 expect(convInfo.effectiveFilterWidth).toEqual(1);
29 expect(convInfo.effectiveFilterHeight).toEqual(1);
30 });
31 it('2x2 conv over 3x3 array with same pad', () => {
32 const inShape = [1, 3, 3, 1];
33 const stride = 1;
34 const dilation = 1;
35 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilation, 'same');
36 expect(convInfo.batchSize).toEqual(1);
37 expect(convInfo.outHeight).toEqual(3);
38 expect(convInfo.outWidth).toEqual(3);
39 expect(convInfo.outChannels).toEqual(1);
40 expect(convInfo.effectiveFilterWidth).toEqual(2);
41 expect(convInfo.effectiveFilterHeight).toEqual(2);
42 // Should produce non-even padding with extra pixel at the right/bottom.
43 expect(convInfo.padInfo.left).toBe(0);
44 expect(convInfo.padInfo.right).toBe(1);
45 expect(convInfo.padInfo.top).toBe(0);
46 expect(convInfo.padInfo.bottom).toBe(1);
47 });
48 it('2x2 conv over 3x3 array with same pad', () => {
49 const inShape = [1, 3, 3, 1];
50 const stride = 1;
51 const dilation = 1;
52 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilation, 'same');
53 expect(convInfo.batchSize).toEqual(1);
54 expect(convInfo.outHeight).toEqual(3);
55 expect(convInfo.outWidth).toEqual(3);
56 expect(convInfo.outChannels).toEqual(1);
57 expect(convInfo.effectiveFilterWidth).toEqual(2);
58 expect(convInfo.effectiveFilterHeight).toEqual(2);
59 });
60 it('2x2 conv over 3x3 array with valid pad', () => {
61 const inShape = [1, 3, 3, 1];
62 const stride = 1;
63 const dilation = 1;
64 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilation, 'valid');
65 expect(convInfo.batchSize).toEqual(1);
66 expect(convInfo.outHeight).toEqual(2);
67 expect(convInfo.outWidth).toEqual(2);
68 expect(convInfo.outChannels).toEqual(1);
69 expect(convInfo.effectiveFilterWidth).toEqual(2);
70 expect(convInfo.effectiveFilterHeight).toEqual(2);
71 });
72 it('3x3 conv over 5x5 array with same pad with stride 2', () => {
73 const inShape = [1, 5, 5, 1];
74 const stride = 2;
75 const dilation = 1;
76 const convInfo = conv_util.computeConv2DInfo(inShape, [3, 3, 1, 1], stride, dilation, 'same');
77 expect(convInfo.batchSize).toEqual(1);
78 expect(convInfo.outHeight).toEqual(3);
79 expect(convInfo.outWidth).toEqual(3);
80 expect(convInfo.outChannels).toEqual(1);
81 expect(convInfo.effectiveFilterWidth).toEqual(3);
82 expect(convInfo.effectiveFilterHeight).toEqual(3);
83 expect(convInfo.padInfo.left).toBe(1);
84 expect(convInfo.padInfo.right).toBe(1);
85 expect(convInfo.padInfo.top).toBe(1);
86 expect(convInfo.padInfo.bottom).toBe(1);
87 });
88 it('2x2 conv over 3x3 array with valid pad with stride 2', () => {
89 const inShape = [1, 3, 3, 1];
90 const stride = 2;
91 const dilation = 1;
92 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilation, 'valid');
93 expect(convInfo.batchSize).toEqual(1);
94 expect(convInfo.outHeight).toEqual(1);
95 expect(convInfo.outWidth).toEqual(1);
96 expect(convInfo.outChannels).toEqual(1);
97 expect(convInfo.effectiveFilterWidth).toEqual(2);
98 expect(convInfo.effectiveFilterHeight).toEqual(2);
99 });
100 it('2x1 conv over 3x3 array with valid pad with stride 1', () => {
101 const inShape = [1, 3, 3, 1];
102 const stride = 1;
103 const dilation = 1;
104 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 1, 1, 1], stride, dilation, 'valid');
105 expect(convInfo.batchSize).toEqual(1);
106 expect(convInfo.outHeight).toEqual(2);
107 expect(convInfo.outWidth).toEqual(3);
108 expect(convInfo.outChannels).toEqual(1);
109 expect(convInfo.effectiveFilterWidth).toEqual(1);
110 expect(convInfo.effectiveFilterHeight).toEqual(2);
111 });
112 it('2x1 conv over 3x3 array with valid pad with strides h=2, w=1', () => {
113 const inShape = [1, 3, 3, 1];
114 const strides = [2, 1];
115 const dilation = 1;
116 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 1, 1, 1], strides, dilation, 'valid');
117 expect(convInfo.batchSize).toEqual(1);
118 expect(convInfo.outHeight).toEqual(1);
119 expect(convInfo.outWidth).toEqual(3);
120 expect(convInfo.outChannels).toEqual(1);
121 expect(convInfo.effectiveFilterWidth).toEqual(1);
122 expect(convInfo.effectiveFilterHeight).toEqual(2);
123 });
124 it('1x2 conv over 3x3 array with valid pad with stride 1', () => {
125 const inShape = [1, 3, 3, 1];
126 const stride = 1;
127 const dilation = 1;
128 const convInfo = conv_util.computeConv2DInfo(inShape, [1, 2, 1, 1], stride, dilation, 'valid');
129 expect(convInfo.batchSize).toEqual(1);
130 expect(convInfo.outHeight).toEqual(3);
131 expect(convInfo.outWidth).toEqual(2);
132 expect(convInfo.outChannels).toEqual(1);
133 expect(convInfo.effectiveFilterWidth).toEqual(2);
134 expect(convInfo.effectiveFilterHeight).toEqual(1);
135 });
136 it('1x2 conv over 3x3 array with valid pad with stride 1, batch=5', () => {
137 const inShape = [5, 3, 3, 1];
138 const stride = 1;
139 const dilation = 1;
140 const convInfo = conv_util.computeConv2DInfo(inShape, [1, 2, 1, 1], stride, dilation, 'valid');
141 expect(convInfo.batchSize).toEqual(5);
142 expect(convInfo.outHeight).toEqual(3);
143 expect(convInfo.outWidth).toEqual(2);
144 expect(convInfo.outChannels).toEqual(1);
145 expect(convInfo.effectiveFilterWidth).toEqual(2);
146 expect(convInfo.effectiveFilterHeight).toEqual(1);
147 });
148 it('2x2 conv over 3x3 array with same pad with dilations 2', () => {
149 const inShape = [1, 3, 3, 1];
150 const stride = 1;
151 const dilations = 2;
152 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilations, 'same');
153 expect(convInfo.batchSize).toEqual(1);
154 expect(convInfo.outHeight).toEqual(3);
155 expect(convInfo.outWidth).toEqual(3);
156 expect(convInfo.outChannels).toEqual(1);
157 // pad evenly on all sides
158 expect(convInfo.padInfo.left).toBe(1);
159 expect(convInfo.padInfo.right).toBe(1);
160 expect(convInfo.padInfo.top).toBe(1);
161 expect(convInfo.padInfo.bottom).toBe(1);
162 expect(convInfo.effectiveFilterWidth).toEqual(3);
163 expect(convInfo.effectiveFilterHeight).toEqual(3);
164 });
165 it('2x1 conv over 3x3 array with same pad with dilations 2', () => {
166 const inShape = [1, 3, 3, 1];
167 const stride = 1;
168 const dilations = 2;
169 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 1, 1, 1], stride, dilations, 'same');
170 expect(convInfo.batchSize).toEqual(1);
171 expect(convInfo.outHeight).toEqual(3);
172 expect(convInfo.outWidth).toEqual(3);
173 expect(convInfo.outChannels).toEqual(1);
174 // pad top and bottom
175 expect(convInfo.padInfo.left).toBe(0);
176 expect(convInfo.padInfo.right).toBe(0);
177 expect(convInfo.padInfo.top).toBe(1);
178 expect(convInfo.padInfo.bottom).toBe(1);
179 expect(convInfo.effectiveFilterWidth).toEqual(1);
180 expect(convInfo.effectiveFilterHeight).toEqual(3);
181 });
182 it('3x4 conv over 8x8 array with same pad with dilations h=4 w=3', () => {
183 const inShape = [1, 8, 8, 1];
184 const stride = 1;
185 const dilations = [4, 3];
186 const convInfo = conv_util.computeConv2DInfo(inShape, [3, 4, 1, 1], stride, dilations, 'same');
187 expect(convInfo.batchSize).toEqual(1);
188 expect(convInfo.outHeight).toEqual(8);
189 expect(convInfo.outWidth).toEqual(8);
190 expect(convInfo.outChannels).toEqual(1);
191 expect(convInfo.effectiveFilterWidth).toEqual(10);
192 expect(convInfo.effectiveFilterHeight).toEqual(9);
193 expect(convInfo.padInfo.left).toBe(4);
194 expect(convInfo.padInfo.right).toBe(5);
195 expect(convInfo.padInfo.top).toBe(4);
196 expect(convInfo.padInfo.bottom).toBe(4);
197 });
198 it('2x1 conv over 3x3 array with valid pad with dilations 2', () => {
199 const inShape = [1, 3, 3, 1];
200 const stride = 1;
201 const dilations = 2;
202 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 1, 1, 1], stride, dilations, 'valid');
203 expect(convInfo.batchSize).toEqual(1);
204 expect(convInfo.outHeight).toEqual(1);
205 expect(convInfo.outWidth).toEqual(3);
206 expect(convInfo.outChannels).toEqual(1);
207 expect(convInfo.effectiveFilterWidth).toEqual(1);
208 expect(convInfo.effectiveFilterHeight).toEqual(3);
209 });
210 it('2x2 conv over 3x3 array with valid pad with dilations 2', () => {
211 const inShape = [1, 3, 3, 1];
212 const stride = 1;
213 const dilations = 2;
214 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilations, 'valid');
215 expect(convInfo.batchSize).toEqual(1);
216 expect(convInfo.outHeight).toEqual(1);
217 expect(convInfo.outWidth).toEqual(1);
218 expect(convInfo.outChannels).toEqual(1);
219 expect(convInfo.effectiveFilterWidth).toEqual(3);
220 expect(convInfo.effectiveFilterHeight).toEqual(3);
221 });
222 it('2x2 conv over 4x4 array with valid pad with dilations 2', () => {
223 const inShape = [1, 4, 4, 1];
224 const stride = 1;
225 const dilations = 2;
226 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, 1, 1], stride, dilations, 'valid');
227 expect(convInfo.batchSize).toEqual(1);
228 expect(convInfo.outHeight).toEqual(2);
229 expect(convInfo.outWidth).toEqual(2);
230 expect(convInfo.outChannels).toEqual(1);
231 expect(convInfo.effectiveFilterWidth).toEqual(3);
232 expect(convInfo.effectiveFilterHeight).toEqual(3);
233 });
234});
235describe('conv_util computeConv3DInfo', () => {
236 it('1x1x1 conv over 1x1x1 array with same pad', () => {
237 const inShape = [1, 1, 1, 1, 1];
238 const stride = 1;
239 const dilation = 1;
240 const convInfo = conv_util.computeConv3DInfo(inShape, [1, 1, 1, 1, 1], stride, dilation, 'same');
241 expect(convInfo.batchSize).toEqual(1);
242 expect(convInfo.outDepth).toEqual(1);
243 expect(convInfo.outHeight).toEqual(1);
244 expect(convInfo.outWidth).toEqual(1);
245 expect(convInfo.outChannels).toEqual(1);
246 });
247 it('2x2x2 conv over 3x3x3 array with same pad', () => {
248 const inShape = [1, 3, 3, 3, 1];
249 const stride = 1;
250 const dilation = 1;
251 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilation, 'same');
252 expect(convInfo.batchSize).toEqual(1);
253 expect(convInfo.outDepth).toEqual(3);
254 expect(convInfo.outHeight).toEqual(3);
255 expect(convInfo.outWidth).toEqual(3);
256 expect(convInfo.outChannels).toEqual(1);
257 // Should produce non-even padding with extra pixel at the back/right/bottom
258 expect(convInfo.padInfo.front).toBe(0);
259 expect(convInfo.padInfo.back).toBe(1);
260 expect(convInfo.padInfo.left).toBe(0);
261 expect(convInfo.padInfo.right).toBe(1);
262 expect(convInfo.padInfo.top).toBe(0);
263 expect(convInfo.padInfo.bottom).toBe(1);
264 });
265 it('2x2x2 conv over 3x3x3 array with same pad', () => {
266 const inShape = [1, 3, 3, 3, 1];
267 const stride = 1;
268 const dilation = 1;
269 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilation, 'same');
270 expect(convInfo.batchSize).toEqual(1);
271 expect(convInfo.outDepth).toEqual(3);
272 expect(convInfo.outHeight).toEqual(3);
273 expect(convInfo.outWidth).toEqual(3);
274 expect(convInfo.outChannels).toEqual(1);
275 });
276 it('2x2x2 conv over 3x3x3 array with valid pad', () => {
277 const inShape = [1, 3, 3, 3, 1];
278 const stride = 1;
279 const dilation = 1;
280 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilation, 'valid');
281 expect(convInfo.batchSize).toEqual(1);
282 expect(convInfo.outDepth).toEqual(2);
283 expect(convInfo.outHeight).toEqual(2);
284 expect(convInfo.outWidth).toEqual(2);
285 expect(convInfo.outChannels).toEqual(1);
286 });
287 it('3x3x3 conv over 5x5x5 array with same pad with stride 2', () => {
288 const inShape = [1, 5, 5, 5, 1];
289 const stride = 2;
290 const dilation = 1;
291 const convInfo = conv_util.computeConv3DInfo(inShape, [3, 3, 3, 1, 1], stride, dilation, 'same');
292 expect(convInfo.batchSize).toEqual(1);
293 expect(convInfo.outDepth).toEqual(3);
294 expect(convInfo.outHeight).toEqual(3);
295 expect(convInfo.outWidth).toEqual(3);
296 expect(convInfo.outChannels).toEqual(1);
297 expect(convInfo.padInfo.front).toBe(1);
298 expect(convInfo.padInfo.back).toBe(1);
299 expect(convInfo.padInfo.left).toBe(1);
300 expect(convInfo.padInfo.right).toBe(1);
301 expect(convInfo.padInfo.top).toBe(1);
302 expect(convInfo.padInfo.bottom).toBe(1);
303 });
304 it('2x2x2 conv over 3x3x3 array with valid pad with stride 2', () => {
305 const inShape = [1, 3, 3, 3, 1];
306 const stride = 2;
307 const dilation = 1;
308 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilation, 'valid');
309 expect(convInfo.batchSize).toEqual(1);
310 expect(convInfo.outDepth).toEqual(1);
311 expect(convInfo.outHeight).toEqual(1);
312 expect(convInfo.outWidth).toEqual(1);
313 expect(convInfo.outChannels).toEqual(1);
314 });
315 it('2x1x1 conv over 3x3x3 array with valid pad with stride 1', () => {
316 const inShape = [1, 3, 3, 3, 1];
317 const stride = 1;
318 const dilation = 1;
319 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 1, 1, 1, 1], stride, dilation, 'valid');
320 expect(convInfo.batchSize).toEqual(1);
321 expect(convInfo.outDepth).toEqual(2);
322 expect(convInfo.outHeight).toEqual(3);
323 expect(convInfo.outWidth).toEqual(3);
324 expect(convInfo.outChannels).toEqual(1);
325 });
326 it('2x1x1 conv over 3x3x3 array with valid pad with strides d=2, h=1, w=1', () => {
327 const inShape = [1, 3, 3, 3, 1];
328 const strides = [2, 1, 1];
329 const dilation = 1;
330 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 1, 1, 1, 1], strides, dilation, 'valid');
331 expect(convInfo.batchSize).toEqual(1);
332 expect(convInfo.outDepth).toEqual(1);
333 expect(convInfo.outHeight).toEqual(3);
334 expect(convInfo.outWidth).toEqual(3);
335 expect(convInfo.outChannels).toEqual(1);
336 });
337 it('1x2x2 conv over 3x3x3 array with valid pad with stride 1', () => {
338 const inShape = [1, 3, 3, 3, 1];
339 const stride = 1;
340 const dilation = 1;
341 const convInfo = conv_util.computeConv3DInfo(inShape, [1, 2, 2, 1, 1], stride, dilation, 'valid');
342 expect(convInfo.batchSize).toEqual(1);
343 expect(convInfo.outDepth).toEqual(3);
344 expect(convInfo.outHeight).toEqual(2);
345 expect(convInfo.outWidth).toEqual(2);
346 expect(convInfo.outChannels).toEqual(1);
347 });
348 it('1x2x2 conv over 3x3x3 array with valid pad with stride 1, batch=5', () => {
349 const inShape = [5, 3, 3, 3, 1];
350 const stride = 1;
351 const dilation = 1;
352 const convInfo = conv_util.computeConv3DInfo(inShape, [1, 2, 2, 1, 1], stride, dilation, 'valid');
353 expect(convInfo.batchSize).toEqual(5);
354 expect(convInfo.outDepth).toEqual(3);
355 expect(convInfo.outHeight).toEqual(2);
356 expect(convInfo.outWidth).toEqual(2);
357 expect(convInfo.outChannels).toEqual(1);
358 });
359 it('2x2x2 conv over 3x3x3 array with same pad with dilations 2', () => {
360 const inShape = [1, 3, 3, 3, 1];
361 const stride = 1;
362 const dilations = 2;
363 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilations, 'same');
364 expect(convInfo.batchSize).toEqual(1);
365 expect(convInfo.outDepth).toEqual(3);
366 expect(convInfo.outHeight).toEqual(3);
367 expect(convInfo.outWidth).toEqual(3);
368 expect(convInfo.outChannels).toEqual(1);
369 // pad evenly on all sides
370 expect(convInfo.padInfo.front).toBe(1);
371 expect(convInfo.padInfo.back).toBe(1);
372 expect(convInfo.padInfo.left).toBe(1);
373 expect(convInfo.padInfo.right).toBe(1);
374 expect(convInfo.padInfo.top).toBe(1);
375 expect(convInfo.padInfo.bottom).toBe(1);
376 });
377 it('2x1x1 conv over 3x3x3 array with same pad with dilations 2', () => {
378 const inShape = [1, 3, 3, 3, 1];
379 const stride = 1;
380 const dilations = 2;
381 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 1, 1, 1, 1], stride, dilations, 'same');
382 expect(convInfo.batchSize).toEqual(1);
383 expect(convInfo.outDepth).toEqual(3);
384 expect(convInfo.outHeight).toEqual(3);
385 expect(convInfo.outWidth).toEqual(3);
386 expect(convInfo.outChannels).toEqual(1);
387 // pad top and bottom
388 expect(convInfo.padInfo.front).toBe(1);
389 expect(convInfo.padInfo.back).toBe(1);
390 expect(convInfo.padInfo.left).toBe(0);
391 expect(convInfo.padInfo.right).toBe(0);
392 expect(convInfo.padInfo.top).toBe(0);
393 expect(convInfo.padInfo.bottom).toBe(0);
394 });
395 it('3x4x4 conv over 8x8 array with same pad with dilations d=4 h=3 w=3', () => {
396 const inShape = [1, 8, 8, 8, 1];
397 const stride = 1;
398 const dilations = [4, 3, 3];
399 const convInfo = conv_util.computeConv3DInfo(inShape, [3, 4, 4, 1, 1], stride, dilations, 'same');
400 expect(convInfo.batchSize).toEqual(1);
401 expect(convInfo.outDepth).toEqual(8);
402 expect(convInfo.outHeight).toEqual(8);
403 expect(convInfo.outWidth).toEqual(8);
404 expect(convInfo.outChannels).toEqual(1);
405 expect(convInfo.padInfo.front).toBe(4);
406 expect(convInfo.padInfo.back).toBe(4);
407 expect(convInfo.padInfo.left).toBe(4);
408 expect(convInfo.padInfo.right).toBe(5);
409 expect(convInfo.padInfo.top).toBe(4);
410 expect(convInfo.padInfo.bottom).toBe(5);
411 });
412 it('2x1x1 conv over 3x3x3 array with valid pad with dilations 2', () => {
413 const inShape = [1, 3, 3, 3, 1];
414 const stride = 1;
415 const dilations = 2;
416 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 1, 1, 1, 1], stride, dilations, 'valid');
417 expect(convInfo.batchSize).toEqual(1);
418 expect(convInfo.outDepth).toEqual(1);
419 expect(convInfo.outHeight).toEqual(3);
420 expect(convInfo.outWidth).toEqual(3);
421 expect(convInfo.outChannels).toEqual(1);
422 });
423 it('2x2x2 conv over 3x3x3 array with valid pad with dilations 2', () => {
424 const inShape = [1, 3, 3, 3, 1];
425 const stride = 1;
426 const dilations = 2;
427 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilations, 'valid');
428 expect(convInfo.batchSize).toEqual(1);
429 expect(convInfo.outDepth).toEqual(1);
430 expect(convInfo.outHeight).toEqual(1);
431 expect(convInfo.outWidth).toEqual(1);
432 expect(convInfo.outChannels).toEqual(1);
433 });
434 it('2x2x2 conv over 4x4x4 array with valid pad with dilations 2', () => {
435 const inShape = [1, 4, 4, 4, 1];
436 const stride = 1;
437 const dilations = 2;
438 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, 1, 1], stride, dilations, 'valid');
439 expect(convInfo.batchSize).toEqual(1);
440 expect(convInfo.outDepth).toEqual(2);
441 expect(convInfo.outHeight).toEqual(2);
442 expect(convInfo.outWidth).toEqual(2);
443 expect(convInfo.outChannels).toEqual(1);
444 });
445});
446describe('conv_util computeConv2DInfo with depthwise=true', () => {
447 it('1x1 filter over 1x1 array with same pad', () => {
448 const inChannels = 1;
449 const inShape = [1, 1, 1, inChannels];
450 const fSize = 1;
451 const chMul = 1;
452 const stride = 1;
453 const dilation = 1;
454 const pad = 'same';
455 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null, true);
456 expect(convInfo.batchSize).toEqual(1);
457 expect(convInfo.outHeight).toEqual(1);
458 expect(convInfo.outWidth).toEqual(1);
459 expect(convInfo.outChannels).toEqual(1);
460 expect(convInfo.effectiveFilterWidth).toEqual(1);
461 expect(convInfo.effectiveFilterHeight).toEqual(1);
462 });
463 it('2x2 filter over 3x3 array with same pad, chMul=3, depth=2', () => {
464 const inChannels = 2;
465 const batchSize = 1;
466 const inSize = 3;
467 const inShape = [batchSize, inSize, inSize, inChannels];
468 const fSize = 2;
469 const chMul = 3;
470 const stride = 1;
471 const dilation = 1;
472 const pad = 'same';
473 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null, true);
474 expect(convInfo.batchSize).toEqual(1);
475 expect(convInfo.outHeight).toEqual(3);
476 expect(convInfo.outWidth).toEqual(3);
477 expect(convInfo.outChannels).toEqual(6);
478 expect(convInfo.effectiveFilterWidth).toEqual(2);
479 expect(convInfo.effectiveFilterHeight).toEqual(2);
480 });
481 it('2x2 filter over 3x3 array with valid pad, chMul=3, depth=2', () => {
482 const inChannels = 2;
483 const batchSize = 1;
484 const inSize = 3;
485 const inShape = [batchSize, inSize, inSize, inChannels];
486 const fSize = 2;
487 const chMul = 3;
488 const stride = 1;
489 const dilation = 1;
490 const pad = 'valid';
491 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null, true);
492 expect(convInfo.batchSize).toEqual(1);
493 expect(convInfo.outHeight).toEqual(2);
494 expect(convInfo.outWidth).toEqual(2);
495 expect(convInfo.outChannels).toEqual(6);
496 expect(convInfo.effectiveFilterWidth).toEqual(2);
497 expect(convInfo.effectiveFilterHeight).toEqual(2);
498 });
499});
500describe('conv_util computeConv3DInfo with depthwise=true', () => {
501 it('1x1x1 filter over 1x1x1 array with same pad', () => {
502 const inChannels = 1;
503 const inShape = [1, 1, 1, 1, inChannels];
504 const fSize = 1;
505 const chMul = 1;
506 const stride = 1;
507 const dilation = 1;
508 const pad = 'same';
509 const convInfo = conv_util.computeConv3DInfo(inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation, pad, true);
510 expect(convInfo.batchSize).toEqual(1);
511 expect(convInfo.outDepth).toEqual(1);
512 expect(convInfo.outHeight).toEqual(1);
513 expect(convInfo.outWidth).toEqual(1);
514 expect(convInfo.outChannels).toEqual(1);
515 });
516 it('2x2x2 filter over 3x3x3 array with same pad, chMul=3, depth=2', () => {
517 const inChannels = 2;
518 const batchSize = 1;
519 const inSize = 3;
520 const inShape = [batchSize, inSize, inSize, inSize, inChannels];
521 const fSize = 2;
522 const chMul = 3;
523 const stride = 1;
524 const dilation = 1;
525 const pad = 'same';
526 const convInfo = conv_util.computeConv3DInfo(inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation, pad, true);
527 expect(convInfo.batchSize).toEqual(1);
528 expect(convInfo.outDepth).toEqual(3);
529 expect(convInfo.outHeight).toEqual(3);
530 expect(convInfo.outWidth).toEqual(3);
531 expect(convInfo.outChannels).toEqual(6);
532 });
533 it('2x2x2 filter over 3x3x3 array with valid pad, chMul=3, depth=2', () => {
534 const inChannels = 2;
535 const batchSize = 1;
536 const inSize = 3;
537 const inShape = [batchSize, inSize, inSize, inSize, inChannels];
538 const fSize = 2;
539 const chMul = 3;
540 const stride = 1;
541 const dilation = 1;
542 const pad = 'valid';
543 const convInfo = conv_util.computeConv3DInfo(inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation, pad, true);
544 expect(convInfo.batchSize).toEqual(1);
545 expect(convInfo.outDepth).toEqual(2);
546 expect(convInfo.outHeight).toEqual(2);
547 expect(convInfo.outWidth).toEqual(2);
548 expect(convInfo.outChannels).toEqual(6);
549 });
550});
551describe('conv_util computeConv2DInfo channelsFirst', () => {
552 it('2x2 conv over 3x3 array with same pad', () => {
553 const inDepth = 2;
554 const outDepth = 4;
555 const inShape = [1, inDepth, 3, 3];
556 const stride = 1;
557 const dilation = 1;
558 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, inDepth, outDepth], stride, dilation, 'same', null, false, 'channelsFirst');
559 expect(convInfo.batchSize).toEqual(1);
560 expect(convInfo.outHeight).toEqual(3);
561 expect(convInfo.outWidth).toEqual(3);
562 expect(convInfo.outChannels).toEqual(4);
563 expect(convInfo.outShape).toEqual([1, 4, 3, 3]);
564 expect(convInfo.effectiveFilterWidth).toEqual(2);
565 expect(convInfo.effectiveFilterHeight).toEqual(2);
566 // Should produce non-even padding with extra pixel at the right/bottom.
567 expect(convInfo.padInfo.left).toBe(0);
568 expect(convInfo.padInfo.right).toBe(1);
569 expect(convInfo.padInfo.top).toBe(0);
570 expect(convInfo.padInfo.bottom).toBe(1);
571 });
572 it('2x2 conv over 3x3 array with valid pad', () => {
573 const inDepth = 6;
574 const outDepth = 16;
575 const inShape = [1, inDepth, 3, 3];
576 const stride = 1;
577 const dilation = 1;
578 const convInfo = conv_util.computeConv2DInfo(inShape, [2, 2, inDepth, outDepth], stride, dilation, 'valid', null, false, 'channelsFirst');
579 expect(convInfo.batchSize).toEqual(1);
580 expect(convInfo.outHeight).toEqual(2);
581 expect(convInfo.outWidth).toEqual(2);
582 expect(convInfo.outChannels).toEqual(16);
583 expect(convInfo.outShape).toEqual([1, 16, 2, 2]);
584 expect(convInfo.effectiveFilterWidth).toEqual(2);
585 expect(convInfo.effectiveFilterHeight).toEqual(2);
586 // Should produce no padding.
587 expect(convInfo.padInfo.left).toBe(0);
588 expect(convInfo.padInfo.right).toBe(0);
589 expect(convInfo.padInfo.top).toBe(0);
590 expect(convInfo.padInfo.bottom).toBe(0);
591 });
592});
593describe('conv_util computeConv3DInfo channelsFirst', () => {
594 it('2x2x2 conv over 3x3x3 array with same pad', () => {
595 const inDepth = 2;
596 const outDepth = 4;
597 const inShape = [1, inDepth, 3, 3, 3];
598 const stride = 1;
599 const dilation = 1;
600 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, inDepth, outDepth], stride, dilation, 'same', false, 'channelsFirst');
601 expect(convInfo.batchSize).toEqual(1);
602 expect(convInfo.outDepth).toEqual(3);
603 expect(convInfo.outHeight).toEqual(3);
604 expect(convInfo.outWidth).toEqual(3);
605 expect(convInfo.outChannels).toEqual(4);
606 expect(convInfo.outShape).toEqual([1, 4, 3, 3, 3]);
607 // Should produce non-even padding with extra pixel at the back/right/bottom
608 expect(convInfo.padInfo.front).toBe(0);
609 expect(convInfo.padInfo.back).toBe(1);
610 expect(convInfo.padInfo.left).toBe(0);
611 expect(convInfo.padInfo.right).toBe(1);
612 expect(convInfo.padInfo.top).toBe(0);
613 expect(convInfo.padInfo.bottom).toBe(1);
614 });
615 it('2x2x2 conv over 3x3x3 array with valid pad', () => {
616 const inDepth = 6;
617 const outDepth = 16;
618 const inShape = [1, inDepth, 3, 3, 3];
619 const stride = 1;
620 const dilation = 1;
621 const convInfo = conv_util.computeConv3DInfo(inShape, [2, 2, 2, inDepth, outDepth], stride, dilation, 'valid', false, 'channelsFirst');
622 expect(convInfo.batchSize).toEqual(1);
623 expect(convInfo.outDepth).toEqual(2);
624 expect(convInfo.outHeight).toEqual(2);
625 expect(convInfo.outWidth).toEqual(2);
626 expect(convInfo.outChannels).toEqual(16);
627 expect(convInfo.outShape).toEqual([1, 16, 2, 2, 2]);
628 // Should produce no padding.
629 expect(convInfo.padInfo.front).toBe(0);
630 expect(convInfo.padInfo.back).toBe(0);
631 expect(convInfo.padInfo.left).toBe(0);
632 expect(convInfo.padInfo.right).toBe(0);
633 expect(convInfo.padInfo.top).toBe(0);
634 expect(convInfo.padInfo.bottom).toBe(0);
635 });
636});
637describe('conv_util computeConv2DInfo roundingMode', () => {
638 const inChannels = 6;
639 const batchSize = 1;
640 const inSize = 5;
641 const inShape = [batchSize, inSize, inSize, inChannels];
642 const fSize = 2;
643 const chMul = 12;
644 const stride = 2;
645 const dilation = 1;
646 const pad = 1;
647 it('Default truncate the output dimension of Conv Layer', () => {
648 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad);
649 expect(convInfo.outShape).toEqual([batchSize, 3, 3, chMul]);
650 });
651 it('Floor the output dimension of Conv Layer', () => {
652 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, 'floor');
653 expect(convInfo.outShape).toEqual([batchSize, 3, 3, chMul]);
654 });
655 it('Round the output dimension of Conv Layer', () => {
656 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, 'round');
657 expect(convInfo.outShape).toEqual([batchSize, 4, 4, chMul]);
658 });
659 it('Ceil the output dimension of Conv Layer', () => {
660 const convInfo = conv_util.computeConv2DInfo(inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, 'ceil');
661 expect(convInfo.outShape).toEqual([batchSize, 4, 4, chMul]);
662 });
663});
664describe('conv_util computePoolInfo roundingMode', () => {
665 const inChannels = 6;
666 const batchSize = 1;
667 const inSize = 5;
668 const inShape = [batchSize, inSize, inSize, inChannels];
669 const fSize = 2;
670 const stride = 2;
671 const dilation = 1;
672 const pad = 1;
673 it('Default truncate the output dimension of Pool Layer', () => {
674 const poolInfo = conv_util.computePool2DInfo(inShape, [fSize, fSize], stride, pad, dilation, 'floor');
675 expect(poolInfo.outShape).toEqual([batchSize, 3, 3, inChannels]);
676 });
677 it('Floor the output dimension of Pool Layer', () => {
678 const poolInfo = conv_util.computePool2DInfo(inShape, [fSize, fSize], stride, pad, dilation, 'floor');
679 expect(poolInfo.outShape).toEqual([batchSize, 3, 3, inChannels]);
680 });
681 it('Round the output dimension of Pool Layer', () => {
682 const poolInfo = conv_util.computePool2DInfo(inShape, [fSize, fSize], stride, pad, dilation, 'round');
683 expect(poolInfo.outShape).toEqual([batchSize, 4, 4, inChannels]);
684 });
685 it('Ceil the output dimension of Pool Layer', () => {
686 const poolInfo = conv_util.computePool2DInfo(inShape, [fSize, fSize], stride, pad, dilation, 'ceil');
687 expect(poolInfo.outShape).toEqual([batchSize, 4, 4, inChannels]);
688 });
689});
690describe('conv_util computePool3dInfo', () => {
691 it('1x1x1 pool over 1x1x1 array with valid pad', () => {
692 const inShape = [1, 1, 1, 1, 1];
693 const filterSize = 1;
694 const stride = 1;
695 const dilation = 1;
696 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'valid');
697 expect(convInfo.batchSize).toEqual(1);
698 expect(convInfo.outDepth).toEqual(1);
699 expect(convInfo.outHeight).toEqual(1);
700 expect(convInfo.outWidth).toEqual(1);
701 expect(convInfo.outChannels).toEqual(1);
702 expect(convInfo.effectiveFilterDepth).toEqual(1);
703 expect(convInfo.effectiveFilterWidth).toEqual(1);
704 expect(convInfo.effectiveFilterHeight).toEqual(1);
705 });
706 it('1x1x1 pool over 3x3x3 array with valid pad', () => {
707 const inShape = [1, 3, 3, 3, 1];
708 const filterSize = 1;
709 const stride = 1;
710 const dilation = 1;
711 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'valid');
712 expect(convInfo.batchSize).toEqual(1);
713 expect(convInfo.outDepth).toEqual(3);
714 expect(convInfo.outHeight).toEqual(3);
715 expect(convInfo.outWidth).toEqual(3);
716 expect(convInfo.outChannels).toEqual(1);
717 expect(convInfo.effectiveFilterDepth).toEqual(1);
718 expect(convInfo.effectiveFilterWidth).toEqual(1);
719 expect(convInfo.effectiveFilterHeight).toEqual(1);
720 });
721 it('2x2x2 pool over 3x3x3 array with same pad', () => {
722 const inShape = [1, 3, 3, 3, 1];
723 const filterSize = 2;
724 const stride = 1;
725 const dilation = 1;
726 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'same');
727 expect(convInfo.batchSize).toEqual(1);
728 expect(convInfo.outDepth).toEqual(3);
729 expect(convInfo.outHeight).toEqual(3);
730 expect(convInfo.outWidth).toEqual(3);
731 expect(convInfo.outChannels).toEqual(1);
732 expect(convInfo.effectiveFilterDepth).toEqual(2);
733 expect(convInfo.effectiveFilterWidth).toEqual(2);
734 expect(convInfo.effectiveFilterHeight).toEqual(2);
735 expect(convInfo.padInfo.top).toEqual(0);
736 expect(convInfo.padInfo.bottom).toEqual(1);
737 expect(convInfo.padInfo.left).toEqual(0);
738 expect(convInfo.padInfo.right).toEqual(1);
739 expect(convInfo.padInfo.front).toEqual(0);
740 expect(convInfo.padInfo.back).toEqual(1);
741 expect(convInfo.padInfo.type).toEqual('SAME');
742 });
743 it('2x2x2 pool over 3x3x3 array with valid pad', () => {
744 const inShape = [1, 3, 3, 3, 1];
745 const filterSize = 2;
746 const stride = 1;
747 const dilation = 1;
748 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'valid');
749 expect(convInfo.batchSize).toEqual(1);
750 expect(convInfo.outDepth).toEqual(2);
751 expect(convInfo.outHeight).toEqual(2);
752 expect(convInfo.outWidth).toEqual(2);
753 expect(convInfo.outChannels).toEqual(1);
754 expect(convInfo.effectiveFilterDepth).toEqual(2);
755 expect(convInfo.effectiveFilterWidth).toEqual(2);
756 expect(convInfo.effectiveFilterHeight).toEqual(2);
757 });
758 it('2x2x2 pool over 4x4x4 array with valid pad, stride 2', () => {
759 const inShape = [1, 4, 4, 4, 1];
760 const filterSize = 2;
761 const stride = 2;
762 const dilation = 1;
763 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'valid');
764 expect(convInfo.batchSize).toEqual(1);
765 expect(convInfo.outDepth).toEqual(2);
766 expect(convInfo.outHeight).toEqual(2);
767 expect(convInfo.outWidth).toEqual(2);
768 expect(convInfo.outChannels).toEqual(1);
769 expect(convInfo.effectiveFilterDepth).toEqual(2);
770 expect(convInfo.effectiveFilterWidth).toEqual(2);
771 expect(convInfo.effectiveFilterHeight).toEqual(2);
772 });
773 it('2x2x2 pool over 3x3x3 array with valid pad, dilation 2', () => {
774 const inShape = [1, 3, 3, 3, 1];
775 const filterSize = 2;
776 const stride = 1;
777 const dilation = 2;
778 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 'valid');
779 expect(convInfo.batchSize).toEqual(1);
780 expect(convInfo.outDepth).toEqual(1);
781 expect(convInfo.outHeight).toEqual(1);
782 expect(convInfo.outWidth).toEqual(1);
783 expect(convInfo.outChannels).toEqual(1);
784 expect(convInfo.effectiveFilterDepth).toEqual(3);
785 expect(convInfo.effectiveFilterWidth).toEqual(3);
786 expect(convInfo.effectiveFilterHeight).toEqual(3);
787 });
788 it('2x2x2 pool over 3x3x3 array with pad 1, roundingMode floor', () => {
789 const inShape = [1, 3, 3, 3, 1];
790 const filterSize = 2;
791 const stride = 1;
792 const dilation = 1;
793 const convInfo = conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 1, 'floor');
794 expect(convInfo.batchSize).toEqual(1);
795 expect(convInfo.outDepth).toEqual(4);
796 expect(convInfo.outHeight).toEqual(4);
797 expect(convInfo.outWidth).toEqual(4);
798 expect(convInfo.outChannels).toEqual(1);
799 expect(convInfo.effectiveFilterDepth).toEqual(2);
800 expect(convInfo.effectiveFilterWidth).toEqual(2);
801 expect(convInfo.effectiveFilterHeight).toEqual(2);
802 expect(convInfo.padInfo.top).toEqual(1);
803 expect(convInfo.padInfo.bottom).toEqual(1);
804 expect(convInfo.padInfo.left).toEqual(1);
805 expect(convInfo.padInfo.right).toEqual(1);
806 expect(convInfo.padInfo.front).toEqual(1);
807 expect(convInfo.padInfo.back).toEqual(1);
808 expect(convInfo.padInfo.type).toEqual('NUMBER');
809 });
810 it('throws unknown dataFormat', () => {
811 const inShape = [1, 3, 3, 3, 1];
812 const filterSize = 2;
813 const stride = 1;
814 const dilation = 1;
815 const fakeDataFormat = 'fakeFormat';
816 expect(() => conv_util.computePool3DInfo(inShape, filterSize, stride, dilation, 1, 'floor', fakeDataFormat))
817 .toThrowError();
818 });
819});
820describe('conv_util convertConv2DDataFormat', () => {
821 it('convert NHWC to channelsLast', () => {
822 const dataFormat = 'NHWC';
823 const $dataFormat = conv_util.convertConv2DDataFormat(dataFormat);
824 expect($dataFormat).toEqual('channelsLast');
825 });
826 it('convert NCHW to channelsFirst', () => {
827 const dataFormat = 'NCHW';
828 const $dataFormat = conv_util.convertConv2DDataFormat(dataFormat);
829 expect($dataFormat).toEqual('channelsFirst');
830 });
831 it('throws unknown dataFormat', () => {
832 const dataFormat = 'FakeFormat';
833 expect(() => conv_util.convertConv2DDataFormat(dataFormat))
834 .toThrowError();
835 });
836});
837//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"conv_util_test.js","sourceRoot":"","sources":["../../../../../../tfjs-core/src/ops/conv_util_test.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2017 Google LLC. All Rights Reserved.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n * =============================================================================\n */\n\nimport * as conv_util from './conv_util';\n\ndescribe('conv_util computeConv2DInfo', () => {\n  it('1x1 conv over 1x1 array with same pad', () => {\n    const inShape: [number, number, number, number] = [1, 1, 1, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [1, 1, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('2x2 conv over 3x3 array with same pad', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n    // Should produce non-even padding with extra pixel at the right/bottom.\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2 conv over 3x3 array with same pad', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x2 conv over 3x3 array with valid pad', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('3x3 conv over 5x5 array with same pad with stride 2', () => {\n    const inShape: [number, number, number, number] = [1, 5, 5, 1];\n    const stride = 2;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [3, 3, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(3);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n\n    expect(convInfo.padInfo.left).toBe(1);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(1);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2 conv over 3x3 array with valid pad with stride 2', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 2;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x1 conv over 3x3 array with valid pad with stride 1', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 1, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x1 conv over 3x3 array with valid pad with strides h=2, w=1', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const strides: [number, number] = [2, 1];\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 1, 1, 1], strides, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('1x2 conv over 3x3 array with valid pad with stride 1', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [1, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('1x2 conv over 3x3 array with valid pad with stride 1, batch=5', () => {\n    const inShape: [number, number, number, number] = [5, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [1, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(5);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('2x2 conv over 3x3 array with same pad with dilations 2', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilations, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    // pad evenly on all sides\n    expect(convInfo.padInfo.left).toBe(1);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(1);\n    expect(convInfo.padInfo.bottom).toBe(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(3);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n\n  it('2x1 conv over 3x3 array with same pad with dilations 2', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 1, 1, 1], stride, dilations, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    // pad top and bottom\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(0);\n    expect(convInfo.padInfo.top).toBe(1);\n    expect(convInfo.padInfo.bottom).toBe(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n\n  it('3x4 conv over 8x8 array with same pad with dilations h=4 w=3', () => {\n    const inShape: [number, number, number, number] = [1, 8, 8, 1];\n    const stride = 1;\n    const dilations: [number, number] = [4, 3];\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [3, 4, 1, 1], stride, dilations, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(8);\n    expect(convInfo.outWidth).toEqual(8);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(10);\n    expect(convInfo.effectiveFilterHeight).toEqual(9);\n\n    expect(convInfo.padInfo.left).toBe(4);\n    expect(convInfo.padInfo.right).toBe(5);\n    expect(convInfo.padInfo.top).toBe(4);\n    expect(convInfo.padInfo.bottom).toBe(4);\n  });\n\n  it('2x1 conv over 3x3 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 1, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n\n  it('2x2 conv over 3x3 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number] = [1, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(3);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n\n  it('2x2 conv over 4x4 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number] = [1, 4, 4, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(3);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n});\n\ndescribe('conv_util computeConv3DInfo', () => {\n  it('1x1x1 conv over 1x1x1 array with same pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 1, 1, 1, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [1, 1, 1, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with same pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    // Should produce non-even padding with extra pixel at the back/right/bottom\n    expect(convInfo.padInfo.front).toBe(0);\n    expect(convInfo.padInfo.back).toBe(1);\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with same pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with valid pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('3x3x3 conv over 5x5x5 array with same pad with stride 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 5, 5, 5, 1];\n    const stride = 2;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [3, 3, 3, 1, 1], stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n\n    expect(convInfo.padInfo.front).toBe(1);\n    expect(convInfo.padInfo.back).toBe(1);\n    expect(convInfo.padInfo.left).toBe(1);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(1);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with valid pad with stride 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 2;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x1x1 conv over 3x3x3 array with valid pad with stride 1', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 1, 1, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x1x1 conv over 3x3x3 array with valid pad with strides d=2, h=1, w=1',\n     () => {\n       const inShape: [number, number, number, number, number] =\n           [1, 3, 3, 3, 1];\n       const strides: [number, number, number] = [2, 1, 1];\n       const dilation = 1;\n       const convInfo = conv_util.computeConv3DInfo(\n           inShape, [2, 1, 1, 1, 1], strides, dilation, 'valid');\n       expect(convInfo.batchSize).toEqual(1);\n       expect(convInfo.outDepth).toEqual(1);\n       expect(convInfo.outHeight).toEqual(3);\n       expect(convInfo.outWidth).toEqual(3);\n       expect(convInfo.outChannels).toEqual(1);\n     });\n\n  it('1x2x2 conv over 3x3x3 array with valid pad with stride 1', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [1, 2, 2, 1, 1], stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('1x2x2 conv over 3x3x3 array with valid pad with stride 1, batch=5',\n     () => {\n       const inShape: [number, number, number, number, number] =\n           [5, 3, 3, 3, 1];\n       const stride = 1;\n       const dilation = 1;\n       const convInfo = conv_util.computeConv3DInfo(\n           inShape, [1, 2, 2, 1, 1], stride, dilation, 'valid');\n       expect(convInfo.batchSize).toEqual(5);\n       expect(convInfo.outDepth).toEqual(3);\n       expect(convInfo.outHeight).toEqual(2);\n       expect(convInfo.outWidth).toEqual(2);\n       expect(convInfo.outChannels).toEqual(1);\n     });\n\n  it('2x2x2 conv over 3x3x3 array with same pad with dilations 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilations, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    // pad evenly on all sides\n    expect(convInfo.padInfo.front).toBe(1);\n    expect(convInfo.padInfo.back).toBe(1);\n    expect(convInfo.padInfo.left).toBe(1);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(1);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x1x1 conv over 3x3x3 array with same pad with dilations 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 1, 1, 1, 1], stride, dilations, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    // pad top and bottom\n    expect(convInfo.padInfo.front).toBe(1);\n    expect(convInfo.padInfo.back).toBe(1);\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(0);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(0);\n  });\n\n  it('3x4x4 conv over 8x8 array with same pad with dilations d=4 h=3 w=3',\n     () => {\n       const inShape: [number, number, number, number, number] =\n           [1, 8, 8, 8, 1];\n       const stride = 1;\n       const dilations: [number, number, number] = [4, 3, 3];\n       const convInfo = conv_util.computeConv3DInfo(\n           inShape, [3, 4, 4, 1, 1], stride, dilations, 'same');\n       expect(convInfo.batchSize).toEqual(1);\n       expect(convInfo.outDepth).toEqual(8);\n       expect(convInfo.outHeight).toEqual(8);\n       expect(convInfo.outWidth).toEqual(8);\n       expect(convInfo.outChannels).toEqual(1);\n\n       expect(convInfo.padInfo.front).toBe(4);\n       expect(convInfo.padInfo.back).toBe(4);\n       expect(convInfo.padInfo.left).toBe(4);\n       expect(convInfo.padInfo.right).toBe(5);\n       expect(convInfo.padInfo.top).toBe(4);\n       expect(convInfo.padInfo.bottom).toBe(5);\n     });\n\n  it('2x1x1 conv over 3x3x3 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 1, 1, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x2x2 conv over 4x4x4 array with valid pad with dilations 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 4, 4, 4, 1];\n    const stride = 1;\n    const dilations = 2;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, 1, 1], stride, dilations, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n});\n\ndescribe('conv_util computeConv2DInfo with depthwise=true', () => {\n  it('1x1 filter over 1x1 array with same pad', () => {\n    const inChannels = 1;\n    const inShape: [number, number, number, number] = [1, 1, 1, inChannels];\n    const fSize = 1;\n    const chMul = 1;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'same';\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null,\n        true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('2x2 filter over 3x3 array with same pad, chMul=3, depth=2', () => {\n    const inChannels = 2;\n    const batchSize = 1;\n    const inSize = 3;\n    const inShape: [number, number, number, number] =\n        [batchSize, inSize, inSize, inChannels];\n    const fSize = 2;\n    const chMul = 3;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'same';\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null,\n        true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(6);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x2 filter over 3x3 array with valid pad, chMul=3, depth=2', () => {\n    const inChannels = 2;\n    const batchSize = 1;\n    const inSize = 3;\n    const inShape: [number, number, number, number] =\n        [batchSize, inSize, inSize, inChannels];\n    const fSize = 2;\n    const chMul = 3;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'valid';\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad, null,\n        true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(6);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n});\n\ndescribe('conv_util computeConv3DInfo with depthwise=true', () => {\n  it('1x1x1 filter over 1x1x1 array with same pad', () => {\n    const inChannels = 1;\n    const inShape: [number, number, number, number, number] =\n        [1, 1, 1, 1, inChannels];\n    const fSize = 1;\n    const chMul = 1;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'same';\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation,\n        pad, true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n  });\n\n  it('2x2x2 filter over 3x3x3 array with same pad, chMul=3, depth=2', () => {\n    const inChannels = 2;\n    const batchSize = 1;\n    const inSize = 3;\n    const inShape: [number, number, number, number, number] =\n        [batchSize, inSize, inSize, inSize, inChannels];\n    const fSize = 2;\n    const chMul = 3;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'same';\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation,\n        pad, true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(6);\n  });\n\n  it('2x2x2 filter over 3x3x3 array with valid pad, chMul=3, depth=2', () => {\n    const inChannels = 2;\n    const batchSize = 1;\n    const inSize = 3;\n    const inShape: [number, number, number, number, number] =\n        [batchSize, inSize, inSize, inSize, inChannels];\n    const fSize = 2;\n    const chMul = 3;\n    const stride = 1;\n    const dilation = 1;\n    const pad = 'valid';\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [fSize, fSize, fSize, inChannels, chMul], stride, dilation,\n        pad, true);\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(6);\n  });\n});\n\ndescribe('conv_util computeConv2DInfo channelsFirst', () => {\n  it('2x2 conv over 3x3 array with same pad', () => {\n    const inDepth = 2;\n    const outDepth = 4;\n    const inShape: [number, number, number, number] = [1, inDepth, 3, 3];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, inDepth, outDepth], stride, dilation, 'same', null,\n        false, 'channelsFirst');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(4);\n    expect(convInfo.outShape).toEqual([1, 4, 3, 3]);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n    // Should produce non-even padding with extra pixel at the right/bottom.\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2 conv over 3x3 array with valid pad', () => {\n    const inDepth = 6;\n    const outDepth = 16;\n    const inShape: [number, number, number, number] = [1, inDepth, 3, 3];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [2, 2, inDepth, outDepth], stride, dilation, 'valid', null,\n        false, 'channelsFirst');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(16);\n    expect(convInfo.outShape).toEqual([1, 16, 2, 2]);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n    // Should produce no padding.\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(0);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(0);\n  });\n});\n\ndescribe('conv_util computeConv3DInfo channelsFirst', () => {\n  it('2x2x2 conv over 3x3x3 array with same pad', () => {\n    const inDepth = 2;\n    const outDepth = 4;\n    const inShape: [number, number, number, number, number] =\n        [1, inDepth, 3, 3, 3];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, inDepth, outDepth], stride, dilation, 'same', false,\n        'channelsFirst');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(4);\n    expect(convInfo.outShape).toEqual([1, 4, 3, 3, 3]);\n    // Should produce non-even padding with extra pixel at the back/right/bottom\n    expect(convInfo.padInfo.front).toBe(0);\n    expect(convInfo.padInfo.back).toBe(1);\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(1);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(1);\n  });\n\n  it('2x2x2 conv over 3x3x3 array with valid pad', () => {\n    const inDepth = 6;\n    const outDepth = 16;\n    const inShape: [number, number, number, number, number] =\n        [1, inDepth, 3, 3, 3];\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computeConv3DInfo(\n        inShape, [2, 2, 2, inDepth, outDepth], stride, dilation, 'valid', false,\n        'channelsFirst');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(16);\n    expect(convInfo.outShape).toEqual([1, 16, 2, 2, 2]);\n    // Should produce no padding.\n    expect(convInfo.padInfo.front).toBe(0);\n    expect(convInfo.padInfo.back).toBe(0);\n    expect(convInfo.padInfo.left).toBe(0);\n    expect(convInfo.padInfo.right).toBe(0);\n    expect(convInfo.padInfo.top).toBe(0);\n    expect(convInfo.padInfo.bottom).toBe(0);\n  });\n});\n\ndescribe('conv_util computeConv2DInfo roundingMode', () => {\n  const inChannels = 6;\n  const batchSize = 1;\n  const inSize = 5;\n  const inShape: [number, number, number, number] =\n      [batchSize, inSize, inSize, inChannels];\n  const fSize = 2;\n  const chMul = 12;\n  const stride = 2;\n  const dilation = 1;\n  const pad = 1;\n\n  it('Default truncate the output dimension of Conv Layer', () => {\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad);\n\n    expect(convInfo.outShape).toEqual([batchSize, 3, 3, chMul]);\n  });\n\n  it('Floor the output dimension of Conv Layer', () => {\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad,\n        'floor');\n\n    expect(convInfo.outShape).toEqual([batchSize, 3, 3, chMul]);\n  });\n\n  it('Round the output dimension of Conv Layer', () => {\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad,\n        'round');\n\n    expect(convInfo.outShape).toEqual([batchSize, 4, 4, chMul]);\n  });\n\n  it('Ceil the output dimension of Conv Layer', () => {\n    const convInfo = conv_util.computeConv2DInfo(\n        inShape, [fSize, fSize, inChannels, chMul], stride, dilation, pad,\n        'ceil');\n\n    expect(convInfo.outShape).toEqual([batchSize, 4, 4, chMul]);\n  });\n});\n\ndescribe('conv_util computePoolInfo roundingMode', () => {\n  const inChannels = 6;\n  const batchSize = 1;\n  const inSize = 5;\n  const inShape: [number, number, number, number] =\n      [batchSize, inSize, inSize, inChannels];\n  const fSize = 2;\n  const stride = 2;\n  const dilation = 1;\n  const pad = 1;\n\n  it('Default truncate the output dimension of Pool Layer', () => {\n    const poolInfo = conv_util.computePool2DInfo(\n        inShape, [fSize, fSize], stride, pad, dilation, 'floor');\n\n    expect(poolInfo.outShape).toEqual([batchSize, 3, 3, inChannels]);\n  });\n\n  it('Floor the output dimension of Pool Layer', () => {\n    const poolInfo = conv_util.computePool2DInfo(\n        inShape, [fSize, fSize], stride, pad, dilation, 'floor');\n\n    expect(poolInfo.outShape).toEqual([batchSize, 3, 3, inChannels]);\n  });\n\n  it('Round the output dimension of Pool Layer', () => {\n    const poolInfo = conv_util.computePool2DInfo(\n        inShape, [fSize, fSize], stride, pad, dilation, 'round');\n\n    expect(poolInfo.outShape).toEqual([batchSize, 4, 4, inChannels]);\n  });\n\n  it('Ceil the output dimension of Pool Layer', () => {\n    const poolInfo = conv_util.computePool2DInfo(\n        inShape, [fSize, fSize], stride, pad, dilation, 'ceil');\n\n    expect(poolInfo.outShape).toEqual([batchSize, 4, 4, inChannels]);\n  });\n});\n\ndescribe('conv_util computePool3dInfo', () => {\n  it('1x1x1 pool over 1x1x1 array with valid pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 1, 1, 1, 1];\n    const filterSize = 1;\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('1x1x1 pool over 3x3x3 array with valid pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 1;\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(1);\n    expect(convInfo.effectiveFilterWidth).toEqual(1);\n    expect(convInfo.effectiveFilterHeight).toEqual(1);\n  });\n\n  it('2x2x2 pool over 3x3x3 array with same pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 2;\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'same');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(3);\n    expect(convInfo.outHeight).toEqual(3);\n    expect(convInfo.outWidth).toEqual(3);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(2);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n    expect(convInfo.padInfo.top).toEqual(0);\n    expect(convInfo.padInfo.bottom).toEqual(1);\n    expect(convInfo.padInfo.left).toEqual(0);\n    expect(convInfo.padInfo.right).toEqual(1);\n    expect(convInfo.padInfo.front).toEqual(0);\n    expect(convInfo.padInfo.back).toEqual(1);\n    expect(convInfo.padInfo.type).toEqual('SAME');\n  });\n\n  it('2x2x2 pool over 3x3x3 array with valid pad', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 2;\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(2);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x2x2 pool over 4x4x4 array with valid pad, stride 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 4, 4, 4, 1];\n    const filterSize = 2;\n    const stride = 2;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(2);\n    expect(convInfo.outHeight).toEqual(2);\n    expect(convInfo.outWidth).toEqual(2);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(2);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n  });\n\n  it('2x2x2 pool over 3x3x3 array with valid pad, dilation 2', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 2;\n    const stride = 1;\n    const dilation = 2;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 'valid');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(1);\n    expect(convInfo.outHeight).toEqual(1);\n    expect(convInfo.outWidth).toEqual(1);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(3);\n    expect(convInfo.effectiveFilterWidth).toEqual(3);\n    expect(convInfo.effectiveFilterHeight).toEqual(3);\n  });\n\n  it('2x2x2 pool over 3x3x3 array with pad 1, roundingMode floor', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 2;\n    const stride = 1;\n    const dilation = 1;\n    const convInfo = conv_util.computePool3DInfo(\n        inShape, filterSize, stride, dilation, 1, 'floor');\n    expect(convInfo.batchSize).toEqual(1);\n    expect(convInfo.outDepth).toEqual(4);\n    expect(convInfo.outHeight).toEqual(4);\n    expect(convInfo.outWidth).toEqual(4);\n    expect(convInfo.outChannels).toEqual(1);\n    expect(convInfo.effectiveFilterDepth).toEqual(2);\n    expect(convInfo.effectiveFilterWidth).toEqual(2);\n    expect(convInfo.effectiveFilterHeight).toEqual(2);\n    expect(convInfo.padInfo.top).toEqual(1);\n    expect(convInfo.padInfo.bottom).toEqual(1);\n    expect(convInfo.padInfo.left).toEqual(1);\n    expect(convInfo.padInfo.right).toEqual(1);\n    expect(convInfo.padInfo.front).toEqual(1);\n    expect(convInfo.padInfo.back).toEqual(1);\n    expect(convInfo.padInfo.type).toEqual('NUMBER');\n  });\n\n  it('throws unknown dataFormat', () => {\n    const inShape: [number, number, number, number, number] = [1, 3, 3, 3, 1];\n    const filterSize = 2;\n    const stride = 1;\n    const dilation = 1;\n    const fakeDataFormat = 'fakeFormat' as 'NDHWC' | 'NCDHW';\n    expect(\n        () => conv_util.computePool3DInfo(\n            inShape, filterSize, stride, dilation, 1, 'floor', fakeDataFormat))\n        .toThrowError();\n  });\n});\n\ndescribe('conv_util convertConv2DDataFormat', () => {\n  it('convert NHWC to channelsLast', () => {\n    const dataFormat: 'NHWC'|'NCHW' = 'NHWC';\n    const $dataFormat = conv_util.convertConv2DDataFormat(dataFormat);\n    expect($dataFormat).toEqual('channelsLast');\n  });\n\n  it('convert NCHW to channelsFirst', () => {\n    const dataFormat: 'NHWC'|'NCHW' = 'NCHW';\n    const $dataFormat = conv_util.convertConv2DDataFormat(dataFormat);\n    expect($dataFormat).toEqual('channelsFirst');\n  });\n\n  it('throws unknown dataFormat', () => {\n    const dataFormat = 'FakeFormat';\n    expect(\n        () => conv_util.convertConv2DDataFormat(dataFormat as 'NHWC' | 'NCHW'))\n        .toThrowError();\n  });\n});\n"]}
\No newline at end of file