forked from vdumoulin/conv_arithmetic
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconv_arithmetic.tex
1170 lines (1012 loc) · 54.7 KB
/
conv_arithmetic.tex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
\documentclass[notitlepage]{report}
\usepackage{amsmath,amsfonts,amsthm,amssymb}
\usepackage{authblk}
\usepackage[T1]{fontenc}
\usepackage{graphicx}
\usepackage[utf8]{inputenc}
\usepackage[framemethod=tikz]{mdframed}
\usepackage{natbib}
\usepackage{subcaption}
\usepackage{tikz}
\usepackage{xcolor}
\usepackage{epigraph}
\usepackage{float}
\usepackage{hyperref}
\definecolor{blue}{RGB}{38,139,210}
\definecolor{cyan}{RGB}{42,161,152}
\definecolor{violet}{RGB}{108,113,196}
\definecolor{red}{RGB}{220,50,47}
\definecolor{base01}{RGB}{88,110,117}
\definecolor{base02}{RGB}{7,54,66}
\definecolor{base03}{RGB}{0,43,54}
\usetikzlibrary{calc,shapes,positioning}
\newcommand{\todo}[1]{\textcolor{red}{TODO: #1}}
\newtheorem{relationship}{Relationship}
\providecommand*{\relationshipautorefname}{Relationship}
\surroundwithmdframed[
topline=false,
bottomline=false,
middlelinewidth=0.5pt,
linecolor=base01,
roundcorner=5pt,
innertopmargin=0pt,
leftmargin=15pt,
rightmargin=15pt,
nobreak=true,
]{relationship}
\setcounter{MaxMatrixCols}{16}
\let\originalepigraph\epigraph
\renewcommand\epigraph[2]{\originalepigraph{\textit{#1}}{\textsc{#2}}}
% Use arabic numbers for thanks
\makeatletter
\let\@fnsymbol\@arabic
\makeatother
\title{A guide to convolution arithmetic for deep learning}
\author[$\bigstar$]{Vincent Dumoulin\thanks{[email protected]}}
\author[$\bigstar\dagger$]{Francesco Visin\thanks{[email protected]}}
\affil[$\bigstar$]{MILA, Universit\'{e} de Montr\'{e}al}
\affil[$\dagger$]{AIRLab, Politecnico di Milano}
\date{\today}
\begin{document}
\maketitle
\thispagestyle{empty}
\clearpage
\setlength{\epigraphwidth}{0.4\textwidth}
\epigraph{All models are wrong, but some are useful.}{George E. P. Box}
\clearpage
\renewcommand{\abstractname}{Acknowledgements}
\begin{abstract}
The authors of this guide would like to thank David Warde-Farley, Guillaume
Alain and Caglar Gulcehre for their valuable feedback.
Special thanks to Ethan Schoonover, creator of the Solarized color
scheme,\footnote{\url{http://ethanschoonover.com/solarized}} whose colors
were used for the figures.
\end{abstract}
\renewcommand{\abstractname}{Feedback}
\begin{abstract}
Your feedback is welcomed! We did our best to be as precise, informative and
up to the point as possible, but should there be anything you feel might be
an error or could be rephrased to be more precise or comprehensible, please
don't refrain from contacting us. Likewise, drop us a line if you think
there is something that might fit this technical report and you would like
us to discuss -- we will make our best effort to update this document.
\end{abstract}
\renewcommand{\abstractname}{Source code and animations}
\begin{abstract}
The code used to generate this guide along with its figures is available on
GitHub.\footnote{\url{https://github.com/vdumoulin/conv_arithmetic}} There
the reader can also find an animated version of the figures.
\end{abstract}
\tableofcontents
\chapter{Introduction}
Deep convolutional neural networks (CNNs) have been at the heart of spectacular
advances in deep learning. Although CNNs have been used as early as the nineties
to solve character recognition tasks \citep{le1997reading}, their current
widespread application is due to much more recent work, when a deep CNN was used
to beat state-of-the-art in the ImageNet image classification challenge
\citep{krizhevsky2012imagenet}.
Convolutional neural networks therefore constitute a very useful tool for
machine learning practitioners. However, learning to use CNNs for the first time
is generally an intimidating experience. A convolutional layer's output shape is
affected by the shape of its input as well as the choice of kernel shape, zero
padding and strides, and the relationship between these properties is not
trivial to infer. This contrasts with fully-connected layers, whose output size
is independent of the input size. Additionally, CNNs also usually feature a {\em
pooling\/} stage, adding yet another level of complexity with respect to
fully-connected networks. Finally, so-called transposed convolutional layers
(also known as fractionally strided convolutional layers) have been employed in
more and more work as of late \citep{zeiler2011adaptive,zeiler2014visualizing,
long2015fully,radford2015unsupervised,visin15,im2016generating}, and their
relationship with convolutional layers has been explained with various degrees
of clarity.
This guide's objective is twofold:
\begin{enumerate}
\item Explain the relationship between convolutional layers and transposed
convolutional layers.
\item Provide an intuitive understanding of the relationship between input
shape, kernel shape, zero padding, strides and output shape in
convolutional, pooling and transposed convolutional layers.
\end{enumerate}
In order to remain broadly applicable, the results shown in this guide are
independent of implementation details and apply to all commonly used machine
learning frameworks, such as Theano
\citep{bergstra2010theano,bastien2012theano}, Torch \citep{collobert2011torch7},
Tensorflow \citep{abaditensorflow} and Caffe \citep{jia2014caffe}.
This chapter briefly reviews the main building blocks of CNNs, namely discrete
convolutions and pooling. For an in-depth treatment of the subject, see Chapter
9 of the Deep Learning textbook \citep{Goodfellow-et-al-2016-Book}.
\section{Discrete convolutions}
The bread and butter of neural networks is \emph{affine transformations}: a
vector is received as input and is multiplied with a matrix to produce an
output (to which a bias vector is usually added before passing the result
through a nonlinearity). This is applicable to any type of input, be it an
image, a sound clip or an unordered collection of features: whatever their
dimensionality, their representation can always be flattened into a vector
before the transformation.
Images, sound clips and many other similar kinds of data have an intrinsic
structure. More formally, they share these important properties:
\begin{itemize}
\item They are stored as multi-dimensional arrays.
\item They feature one or more axes for which ordering matters (e.g., width
and height axes for an image, time axis for a sound clip).
\item One axis, called the channel axis, is used to access different views
of the data (e.g., the red, green and blue channels of a color image, or
the left and right channels of a stereo audio track).
\end{itemize}
These properties are not exploited when an affine transformation is applied; in
fact, all the axes are treated in the same way and the topological information
is not taken into account. Still, taking advantage of the implicit structure of
the data may prove very handy in solving some tasks, like computer vision and
speech recognition, and in these cases it would be best to preserve it. This is
where discrete convolutions come into play.
A discrete convolution is a linear transformation that preserves this notion of
ordering. It is sparse (only a few input units contribute to a given output
unit) and reuses parameters (the same weights are applied to multiple locations
in the input).
\autoref{fig:numerical_no_padding_no_strides} provides an example of a discrete
convolution. The light blue grid is called the {\em input feature map}. To keep
the drawing simple, a single input feature map is represented, but it is not
uncommon to have multiple feature maps stacked one onto another.\footnote{%
An example of this is what was referred to earlier as {\em channels\/} for
images and sound clips.}
A {\em kernel\/} (shaded area) of value
\begin{figure}[H]
\centering
\begin{tikzpicture}[scale=.4,every node/.style={minimum size=1cm}, on grid]
\draw[fill=base02,opacity=0.4] (0,0) rectangle (3,3);
\draw[draw=base03,thick] (0,0) grid (3,3);
\node (00) at (0.5,2.5) {\tiny 0};
\node (01) at (1.5,2.5) {\tiny 1};
\node (02) at (2.5,2.5) {\tiny 2};
\node (10) at (0.5,1.5) {\tiny 2};
\node (11) at (1.5,1.5) {\tiny 2};
\node (12) at (2.5,1.5) {\tiny 0};
\node (20) at (0.5,0.5) {\tiny 0};
\node (21) at (1.5,0.5) {\tiny 1};
\node (22) at (2.5,0.5) {\tiny 2};
\end{tikzpicture}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_00.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_01.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_02.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_03.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_04.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_05.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_06.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_07.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_no_padding_no_strides_08.pdf}
\caption{\label{fig:numerical_no_padding_no_strides} Computing the output
values of a discrete convolution.}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_00.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_01.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_02.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_03.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_04.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_05.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_06.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_07.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_padding_strides_08.pdf}
\caption{\label{fig:numerical_padding_strides} Computing the output values
of a discrete convolution for $N = 2$, $i_1 = i_2 = 5$, $k_1 = k_2 = 3$,
$s_1 = s_2 = 2$, and $p_1 = p_2 = 1$.}
\end{figure}
\noindent slides across the input feature map. At each location, the product
between each element of the kernel and the input element it overlaps is computed
and the results are summed up to obtain the output in the current location. The
procedure can be repeated using different kernels to form as many output feature
maps as desired (\autoref{fig:full_picture}). The final outputs of this procedure
are called {\em output feature maps}.\footnote{%
While there is a distinction between convolution and cross-correlation from
a signal processing perspective, the two become interchangeable when the
kernel is learned. For the sake of simplicity and to stay consistent with
most of the machine learning literature, the term {\em convolution\/}
will be used in this guide.}
If there are multiple input feature maps, the kernel will have to be
3-dimensional -- or, equivalently each one of the feature maps will be
convolved with a distinct kernel -- and the resulting feature maps will
be summed up elementwise to produce the output feature map.
The convolution depicted in \autoref{fig:numerical_no_padding_no_strides} is an
instance of a 2-D convolution, but it can be generalized to N-D convolutions.
For instance, in a 3-D convolution, the kernel would be a {\em cuboid\/} and
would slide across the height, width and depth of the input feature map.
The collection of kernels defining a discrete convolution has a shape
corresponding to some permutation of $(n, m, k_1, \ldots, k_N)$, where
\begin{equation*}
\begin{split}
n &\equiv \text{number of output feature maps},\\
m &\equiv \text{number of input feature maps},\\
k_j &\equiv \text{kernel size along axis $j$}.
\end{split}
\end{equation*}
The following properties affect the output size $o_j$ of a convolutional layer
along axis $j$:
\begin{itemize}
\item $i_j$: input size along axis $j$,
\item $k_j$: kernel size along axis $j$,
\item $s_j$: stride (distance between two consecutive positions of the
kernel) along axis $j$,
\item $p_j$: zero padding (number of zeros concatenated at the beginning and
at the end of an axis) along axis $j$.
\end{itemize}
\noindent For instance, \autoref{fig:numerical_padding_strides} shows a $3
\times 3$ kernel applied to a $5 \times 5$ input padded with a $1 \times 1$
border of zeros using $2 \times 2$ strides.
Note that strides constitute a form of \emph{subsampling}. As an alternative to
being interpreted as a measure of how much the kernel is translated, strides
can also be viewed as how much of the output is retained. For instance, moving
the kernel by hops of two is equivalent to moving the kernel by hops of one but
retaining only odd output elements (\autoref{fig:strides_subsampling}).
\begin{figure}[p]
\centering
\begin{tikzpicture}[scale=.35,every node/.style={minimum size=1cm}, on grid]
\begin{scope}[xshift=0cm,yshift=0cm]
\begin{scope}[xshift=0cm,yshift=0cm]
\draw[draw=base03,fill=violet,thick]
(0,0) grid (5,5) rectangle (0,0);
\end{scope}
\begin{scope}[xshift=0.5cm,yshift=0.5cm]
\draw[draw=base03,fill=blue,thick]
(0,0) grid (5,5) rectangle (0,0);
\end{scope}
\end{scope}
\foreach \x in {-10,1,11} {%
\begin{scope}[xshift=\x cm,yshift=10cm]
\begin{scope}[xshift=0cm,yshift=0cm]
\draw[draw=base03,fill=violet,thick]
(0,0) grid (3,3) rectangle (0,0);
\end{scope}
\begin{scope}[xshift=0.5cm,yshift=0.5cm]
\draw[draw=base03,fill=blue,thick]
(0,0) grid (3,3) rectangle (0,0);
\end{scope}
\end{scope}
\begin{scope}[xshift=\x cm,yshift=20cm]\begin{scope}[xshift=0.5cm]
\draw[draw=base03,fill=cyan,thick]
(0,0) grid (3,3) rectangle (0,0);
\end{scope}\end{scope}
}
\begin{scope}[xshift=1cm,yshift=30cm]
\foreach \s in {0.0,0.5,1.0} {%
\begin{scope}[xshift=\s cm,yshift=\s cm]
\draw[draw=base03,fill=cyan,thick]
(0,0) grid (3,3) rectangle (0,0);
\end{scope}
}
\end{scope}
\draw[->, thick] (-0.5,2.5) to (-8.5,9.5);
\draw[->, thick] (3,6) to (3,9.5);
\draw[->, thick] (6,3.5) to (12.5,9.5);
\draw[thick] (-8,14.5) to (-8,16);
\draw[->, thick] (-8,18) to (-8,19.5);
\node[thick] (p1) at (-8,17) {$+$};
\draw[thick] (3,14.5) to (3,16);
\draw[->, thick] (3,18) to (3,19.5);
\node[thick] (p2) at (3,17) {$+$};
\draw[thick] (13,14.5) to (13,16);
\draw[->, thick] (13,18) to (13,19.5);
\node[thick] (p3) at (13,17) {$+$};
\draw[->, thick] (-8,23.5) to (2,29.5);
\draw[->, thick] (3,23.5) to (2.5,29.5);
\draw[->, thick] (13,23.5) to (3,29.5);
\end{tikzpicture}
\caption{\label{fig:full_picture} A convolution mapping from two input
feature maps to three output feature maps using a $3 \times 2 \times 3
\times 3$ collection of kernels $\mathbf{w}$. In the left pathway, input
feature map 1 is convolved with kernel $\mathbf{w}_{1,1}$ and input
feature map 2 is convolved with kernel $\mathbf{w}_{1,2}$, and the
results are summed together elementwise to form the first output feature
map. The same is repeated for the middle and right pathways to form the
second and third feature maps, and all three output feature maps are
grouped together to form the output.}
\end{figure}
\begin{figure}[p]
\centering
\begin{tikzpicture}[scale=.35,every node/.style={minimum size=1cm}, on grid]
\begin{scope}[xshift=0,yshift=0cm]
\begin{scope}[xshift=0cm,yshift=0cm]
\draw[draw=base03,fill=blue,thick] (0,0) grid (5,5) rectangle (0,0);
\draw[fill=base02, opacity=0.4] (0,2) rectangle (3,5);
\end{scope}
\begin{scope}[xshift=7cm,yshift=1.5cm]
\draw[draw=base03,fill=cyan,thick] (0,0) grid (2,2) rectangle (0,0);
\end{scope}
\end{scope}
\draw[draw=base03, ->, thick] (2.6,3.5) to (4.5,3.5);
\draw[draw=base03, ->, thick] (1.5,2.4) to (1.5,0.5);
\draw[draw=base03, ->, thick] (5.25, 2.5) to (6.75, 2.5);
\begin{scope}[xshift=12cm,yshift=0cm]
\begin{scope}[xshift=0cm,yshift=0cm]
\draw[draw=base03,fill=blue,thick] (0,0) grid (5,5) rectangle (0,0);
\draw[fill=base02, opacity=0.4] (0,2) rectangle (3,5);
\end{scope}
\begin{scope}[xshift=7cm,yshift=1cm]
\draw[draw=base03,fill=cyan,thick] (0,0) grid (3,3) rectangle (0,0);
\draw[draw=base03] (1,0) -- (2,1) -- (2,0) -- (1,1);
\draw[draw=base03] (0,1) -- (1,2) -- (1,1) -- (0,2);
\draw[draw=base03] (1,1) -- (2,2) -- (2,1) -- (1,2);
\draw[draw=base03] (2,1) -- (3,2) -- (3,1) -- (2,2);
\draw[draw=base03] (1,2) -- (2,3) -- (2,2) -- (1,3);
\end{scope}
\begin{scope}[xshift=12cm,yshift=1.5cm]
\draw[draw=base03,fill=cyan,thick] (0,0) grid (2,2) rectangle (0,0);
\end{scope}
\end{scope}
\draw[draw=base03, ->, thick] (14.6,3.5) to (15.5,3.5);
\draw[draw=base03, ->, thick] (15.6,3.5) to (16.5,3.5);
\draw[draw=base03, ->, thick] (13.5,2.4) to (13.5,1.5);
\draw[draw=base03, ->, thick] (13.5,1.4) to (13.5,0.5);
\draw[draw=base03, ->, thick] (17.25, 2.5) to (18.75, 2.5);
\draw[draw=base03, ->, thick] (22.25, 2.5) to (23.75, 2.5);
\end{tikzpicture}
\caption{\label{fig:strides_subsampling} An alternative way of viewing
strides. Instead of translating the $3 \times 3$ kernel by increments of
$s = 2$ (left), the kernel is translated by increments of $1$ and only
one in $s = 2$ output elements is retained (right).}
\end{figure}
\section{Pooling}
In addition to discrete convolutions themselves, {\em pooling\/} operations
make up another important building block in CNNs. Pooling operations reduce
the size of feature maps by using some function to summarize subregions, such
as taking the average or the maximum value.
Pooling works by sliding a window across the input and feeding the content of
the window to a {\em pooling function}. In some sense, pooling works very much
like a discrete convolution, but replaces the linear combination described by
the kernel with some other function. \autoref{fig:numerical_average_pooling}
provides an example for average pooling, and \autoref{fig:numerical_max_pooling}
does the same for max pooling.
The following properties affect the output size $o_j$ of a pooling layer
along axis $j$:
\begin{itemize}
\item $i_j$: input size along axis $j$,
\item $k_j$: pooling window size along axis $j$,
\item $s_j$: stride (distance between two consecutive positions of the
pooling window) along axis $j$.
\end{itemize}
\begin{figure}[p]
\centering
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_00.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_01.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_02.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_03.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_04.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_05.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_06.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_07.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_average_pooling_08.pdf}
\caption{\label{fig:numerical_average_pooling} Computing the output values
of a $3 \times 3$ average pooling operation on a $5 \times 5$ input
using $1 \times 1$ strides.}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_00.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_01.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_02.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_03.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_04.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_05.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_06.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_07.pdf}
\includegraphics[width=0.32\textwidth]{pdf/numerical_max_pooling_08.pdf}
\caption{\label{fig:numerical_max_pooling} Computing the output values of a
$3 \times 3$ max pooling operation on a $5 \times 5$ input using $1
\times 1$ strides.}
\end{figure}
\chapter{Convolution arithmetic}
The analysis of the relationship between convolutional layer properties is eased
by the fact that they don't interact across axes, i.e., the choice of kernel
size, stride and zero padding along axis $j$ only affects the output size of
axis $j$. Because of that, this chapter will focus on the following simplified
setting:
\begin{itemize}
\item 2-D discrete convolutions ($N = 2$),
\item square inputs ($i_1 = i_2 = i$),
\item square kernel size ($k_1 = k_2 = k$),
\item same strides along both axes ($s_1 = s_2 = s$),
\item same zero padding along both axes ($p_1 = p_2 = p$).
\end{itemize}
This facilitates the analysis and the visualization, but keep in mind that the
results outlined here also generalize to the N-D and non-square cases.
\section{No zero padding, unit strides}
The simplest case to analyze is when the kernel just slides across every
position of the input (i.e., $s = 1$ and $p = 0$).
\autoref{fig:no_padding_no_strides} provides an example for $i = 4$ and $k =
3$.
One way of defining the output size in this case is by the number of possible
placements of the kernel on the input. Let's consider the width axis: the kernel
starts on the leftmost part of the input feature map and slides by steps of one
until it touches the right side of the input. The size of the output will be
equal to the number of steps made, plus one, accounting for the initial position
of the kernel (\autoref{fig:no_padding_no_strides_explained}). The same logic
applies for the height axis.
More formally, the following relationship can be inferred:
\begin{relationship}\label{rel:no_padding_no_strides}
For any $i$ and $k$, and for $s = 1$ and $p = 0$,
\begin{equation*}
o = (i - k) + 1.
\end{equation*}
\end{relationship}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_03.pdf}
\caption{\label{fig:no_padding_no_strides} (No padding, unit strides)
Convolving a $3 \times 3$ kernel over a $4 \times 4$ input using unit
strides (i.e., $i = 4$, $k = 3$, $s = 1$ and $p = 0$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_03.pdf}
\caption{\label{fig:arbitrary_padding_no_strides} (Arbitrary padding, unit
strides) Convolving a $4 \times 4$ kernel over a $5 \times 5$ input
padded with a $2 \times 2$ border of zeros using unit strides (i.e.,
$i = 5$, $k = 4$, $s = 1$ and $p = 2$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_03.pdf}
\caption{\label{fig:same_padding_no_strides} (Half padding, unit strides)
Convolving a $3 \times 3$ kernel over a $5 \times 5$ input using half
padding and unit strides (i.e., $i = 5$, $k = 3$, $s = 1$ and $p = 1$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/full_padding_no_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/full_padding_no_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/full_padding_no_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/full_padding_no_strides_03.pdf}
\caption{\label{fig:full_padding_no_strides} (Full padding, unit strides)
Convolving a $3 \times 3$ kernel over a $5 \times 5$ input using full
padding and unit strides (i.e., $i = 5$, $k = 3$, $s = 1$ and $p = 2$).}
\end{figure}
\section{Zero padding, unit strides}
To factor in zero padding (i.e., only restricting to $s = 1$), let's consider
its effect on the effective input size: padding with $p$ zeros changes the
effective input size from $i$ to $i + 2p$. In the general case,
\autoref{rel:no_padding_no_strides} can then be used to infer the following
relationship:
\begin{relationship}\label{rel:arbitrary_padding_no_strides}
For any $i$, $k$ and $p$, and for $s = 1$,
\begin{equation*}
o = (i - k) + 2p + 1.
\end{equation*}
\end{relationship}
\noindent \autoref{fig:arbitrary_padding_no_strides} provides an example for $i
= 5$, $k = 4$ and $p = 2$.
In practice, two specific instances of zero padding are used quite extensively
because of their respective properties. Let's discuss them in more detail.
\subsection{Half (same) padding}
Having the output size be the same as the input size (i.e., $o = i$) can be a
desirable property:
\begin{relationship}\label{rel:same_padding_no_strides}
For any $i$ and for $k$ odd ($k = 2n + 1, \quad n \in \mathbb{N}$), $s = 1$ and
$p = \lfloor k / 2 \rfloor = n$,
\begin{equation*}
\begin{split}
o &= i + 2 \lfloor k / 2 \rfloor - (k - 1) \\
&= i + 2n - 2n \\
&= i.
\end{split}
\end{equation*}
\end{relationship}
\noindent This is sometimes referred to as {\em half\/} (or {\em same\/})
padding. \autoref{fig:same_padding_no_strides} provides an example for
$i = 5$, $k = 3$ and (therefore) $p = 1$.
\subsection{Full padding}
While convolving a kernel generally {\em decreases\/} the output size with
respect to the input size, sometimes the opposite is required. This can be
achieved with proper zero padding:
\begin{relationship}\label{rel:full_padding_no_strides}
For any $i$ and $k$, and for $p = k - 1$ and $s = 1$,
\begin{equation*}
\begin{split}
o &= i + 2(k - 1) - (k - 1) \\
&= i + (k - 1).
\end{split}
\end{equation*}
\end{relationship}
\noindent This is sometimes referred to as {\em full\/} padding, because in this
setting every possible partial or complete superimposition of the kernel on the
input feature map is taken into account. \autoref{fig:full_padding_no_strides}
provides an example for $i = 5$, $k = 3$ and (therefore) $p = 2$.
\section{No zero padding, non-unit strides}
All relationships derived so far only apply for unit-strided convolutions.
Incorporating non unitary strides requires another inference leap. To
facilitate the analysis, let's momentarily ignore zero padding (i.e., $s > 1$
and $p = 0$). \autoref{fig:no_padding_strides} provides an example for $i =
5$, $k = 3$ and $s = 2$.
Once again, the output size can be defined in terms of the number of possible
placements of the kernel on the input. Let's consider the width axis: the
kernel starts as usual on the leftmost part of the input, but this time it
slides by steps of size $s$ until it touches the right side of the input. The
size of the output is again equal to the number of steps made, plus one,
accounting for the initial position of the kernel
(\autoref{fig:no_padding_strides_explained}). The same logic applies for the
height axis.
From this, the following relationship can be inferred:
\begin{relationship}\label{rel:no_padding_strides}
For any $i$, $k$ and $s$, and for $p = 0$,
\begin{equation*}
o = \left\lfloor \frac{i - k}{s} \right\rfloor + 1.
\end{equation*}
\end{relationship}
\noindent The floor function accounts for the fact that sometimes the last
possible step does {\em not\/} coincide with the kernel reaching the end of the
input, i.e., some input units are left out (see
\autoref{fig:padding_strides_odd} for an example of such a case).
\section{Zero padding, non-unit strides}
The most general case (convolving over a zero padded input using non-unit
strides) can be derived by applying \autoref{rel:no_padding_strides} on an
effective input of size $i + 2p$, in analogy to what was done for
\autoref{rel:arbitrary_padding_no_strides}:
\begin{relationship}\label{rel:padding_strides}
For any $i$, $k$, $p$ and $s$,
\begin{equation*}
o = \left\lfloor \frac{i + 2p - k}{s} \right\rfloor + 1.
\end{equation*}
\end{relationship}
\noindent As before, the floor function means that in some cases a convolution
will produce the same output size for multiple input sizes. More specifically,
if $i + 2p - k$ is a multiple of $s$, then any input size $j = i + a, \quad a
\in \{0,\ldots,s - 1\}$ will produce the same output size. Note that this
ambiguity applies only for $s > 1$.
\autoref{fig:padding_strides} shows an example with $i = 5$, $k = 3$, $s = 2$
and $p = 1$, while \autoref{fig:padding_strides_odd} provides an example for
$i = 6$, $k = 3$, $s = 2$ and $p = 1$. Interestingly, despite having different
input sizes these convolutions share the same output size. While this doesn't
affect the analysis for {\em convolutions}, this will complicate the analysis
in the case of {\em transposed convolutions}.
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/no_padding_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_strides_03.pdf}
\caption{\label{fig:no_padding_strides} (No zero padding, arbitrary
strides) Convolving a $3 \times 3$ kernel over a $5 \times 5$ input
using $2 \times 2$ strides (i.e., $i = 5$, $k = 3$, $s = 2$ and
$p = 0$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_03.pdf}
\caption{\label{fig:padding_strides} (Arbitrary padding and strides)
Convolving a $3 \times 3$ kernel over a $5 \times 5$ input padded with
a $1 \times 1$ border of zeros using $2 \times 2$ strides (i.e.,
$i = 5$, $k = 3$, $s = 2$ and $p = 1$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_odd_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_odd_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_odd_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/padding_strides_odd_03.pdf}
\caption{\label{fig:padding_strides_odd} (Arbitrary padding and strides)
Convolving a $3 \times 3$ kernel over a $6 \times 6$ input padded with
a $1 \times 1$ border of zeros using $2 \times 2$ strides (i.e.,
$i = 6$, $k = 3$, $s = 2$ and $p = 1$). In this case, the bottom row
and right column of the zero padded input are not covered by the
kernel.}
\end{figure}
\begin{figure}[p]
\centering
\begin{subfigure}[t]{0.48\textwidth}
\centering
\begin{tikzpicture}[scale=.35,every node/.style={minimum size=1cm},
on grid]
\draw[fill=blue] (0,0) rectangle (5,5);
\draw[draw=base03, thick] (0,0) grid (5,5);
\draw[fill=base02, opacity=0.4] (0,2) rectangle (3,5);
\draw[step=10mm, base03, thick] (0,2) grid (3,5);
\draw[draw=base03, ->, thick] (2.6,3.5) to (3.5,3.5);
\draw[draw=base03, ->, thick] (3.6,3.5) to (4.5,3.5);
\draw[draw=base03, ->, thick] (1.5,2.4) to (1.5,1.5);
\draw[draw=base03, ->, thick] (1.5,1.4) to (1.5,0.5);
\end{tikzpicture}
\caption{\label{fig:no_padding_no_strides_explained} The kernel has to
slide two steps to the right to touch the right side of the input
(and equivalently downwards). Adding one to account for the
initial kernel position, the output size is $3 \times 3$.}
\end{subfigure}
~
\begin{subfigure}[t]{0.48\textwidth}
\centering
\begin{tikzpicture}[scale=.35,every node/.style={minimum size=1cm},
on grid]
\draw[fill=blue] (0,0) rectangle (5,5);
\draw[draw=base03, thick] (0,0) grid (5,5);
\draw[fill=base02, opacity=0.4] (0,2) rectangle (3,5);
\draw[step=10mm, base03, thick] (0,2) grid (3,5);
\draw[draw=base03, ->, thick] (2.5,3.5) to (4.5,3.5);
\draw[draw=base03, ->, thick] (1.5,2.5) to (1.5,0.5);
\end{tikzpicture}
\caption{\label{fig:no_padding_strides_explained} The kernel has to
slide one step of size two to the right to touch the right side of
the input (and equivalently downwards). Adding one to account for
the initial kernel position, the output size is $2 \times 2$.}
\end{subfigure}
\caption{Counting kernel positions.}
\end{figure}
\chapter{Pooling arithmetic}
In a neural network, pooling layers provide invariance to small translations of
the input. The most common kind of pooling is \emph{max pooling}, which
consists in splitting the input in (usually non-overlapping) patches and
outputting the maximum value of each patch. Other kinds of pooling exist, e.g.,
mean or average pooling, which all share the same idea of aggregating the input
locally by applying a non-linearity to the content of some patches \citep{%
boureau-cvpr-10,boureau-icml-10,boureau-iccv-11,ICML2011Saxe_551}.
Some readers may have noticed that the treatment of convolution arithmetic only
relies on the assumption that some function is repeatedly applied onto subsets
of the input. This means that the relationships derived in the previous chapter
can be reused in the case of pooling arithmetic. Since pooling does not involve
zero padding, the relationship describing the general case is as follows:
\begin{relationship}\label{rel:pooling}
For any $i$, $k$ and $s$,
\begin{equation*}
o = \left\lfloor \frac{i - k}{s} \right\rfloor + 1.
\end{equation*}
\end{relationship}
\noindent This relationship holds for any type of pooling.
\chapter{Transposed convolution arithmetic}
The need for transposed convolutions generally arises from the desire to use a
transformation going in the opposite direction of a normal convolution, i.e.,
from something that has the shape of the output of some convolution to
something that has the shape of its input while maintaining a connectivity
pattern that is compatible with said convolution. For instance, one might use
such a transformation as the decoding layer of a convolutional autoencoder or to
project feature maps to a higher-dimensional space.
Once again, the convolutional case is considerably more complex than the
fully-connected case, which only requires to use a weight matrix whose shape
has been transposed. However, since every convolution boils down to an
efficient implementation of a matrix operation, the insights gained from the
fully-connected case are useful in solving the convolutional case.
Like for convolution arithmetic, the dissertation about transposed convolution
arithmetic is simplified by the fact that transposed convolution properties
don't interact across axes.
The chapter will focus on the following setting:
\begin{itemize}
\item 2-D transposed convolutions ($N = 2$),
\item square inputs ($i_1 = i_2 = i$),
\item square kernel size ($k_1 = k_2 = k$),
\item same strides along both axes ($s_1 = s_2 = s$),
\item same zero padding along both axes ($p_1 = p_2 = p$).
\end{itemize}
\noindent Once again, the results outlined generalize to the N-D and non-square
cases.
\section{Convolution as a matrix operation}
Take for example the convolution represented in
\autoref{fig:no_padding_no_strides}. If the input and output were to be unrolled
into vectors from left to right, top to bottom, the convolution could be
represented as a sparse matrix $\mathbf{C}$ where the non-zero elements are the
elements $w_{i,j}$ of the kernel (with $i$ and $j$ being the row and column of
the kernel respectively):
\begin{equation*}
\resizebox{.98\hsize}{!}{$
\begin{pmatrix}
w_{0,0} & w_{0,1} & w_{0,2} & 0 & w_{1,0} & w_{1,1} & w_{1,2} & 0 &
w_{2,0} & w_{2,1} & w_{2,2} & 0 & 0 & 0 & 0 & 0 \\
0 & w_{0,0} & w_{0,1} & w_{0,2} & 0 & w_{1,0} & w_{1,1} & w_{1,2} &
0 & w_{2,0} & w_{2,1} & w_{2,2} & 0 & 0 & 0 & 0 \\
0 & 0 & 0 & 0 & w_{0,0} & w_{0,1} & w_{0,2} & 0 &
w_{1,0} & w_{1,1} & w_{1,2} & 0 & w_{2,0} & w_{2,1} & w_{2,2} & 0 \\
0 & 0 & 0 & 0 & 0 & w_{0,0} & w_{0,1} & w_{0,2} &
0 & w_{1,0} & w_{1,1} & w_{1,2} & 0 & w_{2,0} & w_{2,1} & w_{2,2} \\
\end{pmatrix}$}
\end{equation*}
This linear operation takes the input matrix flattened as a 16-dimensional
vector and produces a 4-dimensional vector that is later reshaped as the $2
\times 2$ output matrix.
Using this representation, the backward pass is easily obtained by transposing
$\mathbf{C}$; in other words, the error is backpropagated by multiplying the
loss with $\mathbf{C}^T$. This operation takes a 4-dimensional vector as input
and produces a 16-dimensional vector as output, and its connectivity pattern is
compatible with $\mathbf{C}$ by construction.
Notably, the kernel $\mathbf{w}$ defines both the matrices $\mathbf{C}$ and
$\mathbf{C}^T$ used for the forward and backward passes.
\section{Transposed convolution}
Let's now consider what would be required to go the other way around, i.e., map
from a 4-dimensional space to a 16-dimensional space, while keeping the
connectivity pattern of the convolution depicted in
\autoref{fig:no_padding_no_strides}. This operation is known as a {\em
transposed convolution}.
Transposed convolutions -- also called {\em fractionally strided convolutions\/}
-- work by swapping the forward and backward passes of a convolution. One way to
put it is to note that the kernel defines a convolution, but whether it's a
direct convolution or a transposed convolution is determined by how the forward
and backward passes are computed.
For instance, although the kernel $\mathbf{w}$ defines a convolution whose
forward and backward passes are computed by multiplying with $\mathbf{C}$ and
$\mathbf{C}^T$ respectively, it {\em also\/} defines a transposed convolution
whose forward and backward passes are computed by multiplying with
$\mathbf{C}^T$ and $(\mathbf{C}^T)^T = \mathbf{C}$ respectively.\footnote{The
transposed convolution operation can be thought of as the gradient of {\em
some\/} convolution with respect to its input, which is usually how
transposed convolutions are implemented in practice.}
Finally note that it is always possible to emulate a transposed convolution with
a direct convolution. The disadvantage is that it usually involves adding many
columns and rows of zeros to the input, resulting in a much less efficient
implementation.
Building on what has been introduced so far, this chapter will proceed somewhat
backwards with respect to the convolution arithmetic chapter, deriving the
properties of each transposed convolution by referring to the direct
convolution with which it shares the kernel, and defining the equivalent direct
convolution.
\section{No zero padding, unit strides, transposed}
The simplest way to think about a transposed convolution is by computing the
output shape of the direct convolution for a given input shape first, and then
inverting the input and output shapes for the transposed convolution.
Let's consider the convolution of a $3 \times 3$ kernel on a $4 \times 4$
input with unitary stride and no padding (i.e., $i = 4$, $k = 3$, $s = 1$ and
$p = 0$). As depicted in \autoref{fig:no_padding_no_strides}, this produces a
$2 \times 2$ output. The transpose of this convolution will then have an output
of shape $4 \times 4$ when applied on a $2 \times 2$ input.
Another way to obtain the result of a transposed convolution is to apply an
equivalent -- but much less efficient -- direct convolution. The example
described so far could be tackled by convolving a $3 \times 3$ kernel over a
$2 \times 2$ input padded with a $2 \times 2$ border of zeros using unit
strides (i.e., $i' = 2$, $k' = k$, $s' = 1$ and $p' = 2$), as shown in
\autoref{fig:no_padding_no_strides_transposed}. Notably, the kernel's and
stride's sizes remain the same, but the input of the transposed convolution is
now zero padded.\footnote{Note that although
equivalent to applying the transposed matrix, this visualization adds a lot
of zero multiplications in the form of zero padding. This is done here for
illustration purposes, but it is inefficient, and software implementations
will normally not perform the useless zero multiplications.}
One way to understand the logic behind zero padding is to consider the
connectivity pattern of the transposed convolution and use it to guide the
design of the equivalent convolution. For example, the top left pixel of the
input of the direct convolution only contribute to the top left pixel of the
output, the top right pixel is only connected to the top right output pixel,
and so on.
To maintain the same connectivity pattern in the equivalent convolution it is
necessary to zero pad the input in such a way that the first (top-left)
application of the kernel only touches the top-left pixel, i.e., the padding
has to be equal to the size of the kernel minus one.
Proceeding in the same fashion it is possible to determine similar observations
for the other elements of the image, giving rise to the following relationship:
\begin{relationship}\label{rel:no_padding_no_strides_transposed}
A convolution described by $s = 1$, $p = 0$ and $k$ has an associated
transposed convolution described by $k' = k$, $s' = s$ and $p' = k - 1$ and its
output size is
\begin{equation*}
o' = i' + (k - 1).
\end{equation*}
\end{relationship}
Interestingly, this corresponds to a fully padded convolution with unit
strides.
\section{Zero padding, unit strides, transposed}
Knowing that the transpose of a non-padded convolution is equivalent to
convolving a zero padded input, it would be reasonable to suppose that the
transpose of a zero padded convolution is equivalent to convolving an input
padded with {\em less\/} zeros.
It is indeed the case, as shown in
\autoref{fig:arbitrary_padding_no_strides_transposed} for $i = 5$, $k = 4$ and
$p = 2$.
Formally, the following relationship applies for zero padded convolutions:
\begin{relationship}\label{rel:arbitrary_padding_no_strides_transposed}
A convolution described by $s = 1$, $k$ and $p$ has an
associated transposed convolution described by $k' = k$, $s' = s$ and $p' = k -
p - 1$ and its output size is
\begin{equation*}
o' = i' + (k - 1) - 2p.
\end{equation*}
\end{relationship}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_transposed_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_transposed_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_transposed_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/no_padding_no_strides_transposed_03.pdf}
\caption{\label{fig:no_padding_no_strides_transposed} The transpose of
convolving a $3 \times 3$ kernel over a $4 \times 4$ input using unit
strides (i.e., $i = 4$, $k = 3$, $s = 1$ and $p = 0$). It is equivalent
to convolving a $3 \times 3$ kernel over a $2 \times 2$ input padded
with a $2 \times 2$ border of zeros using unit strides (i.e., $i' = 2$,
$k' = k$, $s' = 1$ and $p' = 2$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_transposed_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_transposed_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_transposed_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/arbitrary_padding_no_strides_transposed_03.pdf}
\caption{\label{fig:arbitrary_padding_no_strides_transposed} The transpose
of convolving a $4 \times 4$ kernel over a $5 \times 5$ input padded
with a $2 \times 2$ border of zeros using unit strides (i.e., $i = 5$,
$k = 4$, $s = 1$ and $p = 2$). It is equivalent to convolving a $4
\times 4$ kernel over a $6 \times 6$ input padded with a $1 \times 1$
border of zeros using unit strides (i.e., $i' = 6$, $k' = k$, $s' = 1$
and $p' = 1$).}
\end{figure}
\begin{figure}[p]
\centering
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_transposed_00.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_transposed_01.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_transposed_02.pdf}
\includegraphics[width=0.24\textwidth]{pdf/same_padding_no_strides_transposed_03.pdf}
\caption{\label{fig:same_padding_no_strides_transposed} The transpose of
convolving a $3 \times 3$ kernel over a $5 \times 5$ input using half
padding and unit strides (i.e., $i = 5$, $k = 3$, $s = 1$ and $p = 1$).
It is equivalent to convolving a $3 \times 3$ kernel over a $5 \times 5$
input using half padding and unit strides (i.e., $i' = 5$, $k' = k$, $s'
= 1$ and $p' = 1$).}
\end{figure}
\subsection{Half (same) padding, transposed}
By applying the same inductive reasoning as before, it is reasonable to expect
that the equivalent convolution of the transpose of a half padded convolution
is itself a half padded convolution, given that the output size of a half
padded convolution is the same as its input size. Thus the following relation
applies:
\begin{relationship}\label{rel:half_padding_no_strides_transposed}
A convolution described by $k = 2n + 1, \quad n \in \mathbb{N}$, $s = 1$ and $p
= \lfloor k / 2 \rfloor = n$ has an associated transposed convolution described
by $k' = k$, $s' = s$ and $p' = p$ and its output size is
\begin{equation*}
\begin{split}
o' &= i' + (k - 1) - 2p \\
&= i' + 2n - 2n \\
&= i'.