forked from google/svcca
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdft_ccas.py
167 lines (137 loc) · 6.61 KB
/
dft_ccas.py
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
# Copyright 2016 Google Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
Code for use with large conv layers using DFT (discrete fourier transform).
The functions in this script provide a scalable method for computing the cca
similarity between large convolutional layers. The main function, fourier_ccas,
takes in two sets of convolutional activations, of shapes [dataset_size,
height1, width1, num_channels1, [dataset_size, height2, width2, num_channels2]
and computes the cca similarity between them. The results are exact
when the dataset over which correlations are computed is translation invariant.
However, due to the strided nature of convolutional and pooling layers, image
datasets are close to translation invariant, and very good results can still
be achieved without taking correlations over a translation invariant dataset.
See https://arxiv.org/abs/1706.05806 for details.
This function can also be used to compute cca similarity between conv
layers and fully connected layers (or neurons). We may want to compare
similarity between convolutional feature maps at a layer and a particular class.
Again assuming
translation invariance of the original dataset, the fourier_ccas function can
be used for this (reshaping the vector to be (dataset_size, 1, 1, 1)), and will
output the correlation of the vector with the dc component of the DFT.
This can be seen as a lower bound on the correlation of the vector with the
channels.
"""
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
import numpy as np
import pandas as pd
import cca_core as cca_core
def fft_resize(images, resize=False, new_size=None):
"""Function for applying DFT and resizing.
This function takes in an array of images, applies the 2-d fourier transform
and resizes them according to new_size, keeping the frequencies that overlap
between the two sizes.
Args:
images: a numpy array with shape
[batch_size, height, width, num_channels]
resize: boolean, whether or not to resize
new_size: a tuple (size, size), with height and width the same
Returns:
im_fft_downsampled: a numpy array with shape
[batch_size, (new) height, (new) width, num_channels]
"""
assert len(images.shape) == 4, ("expecting images to be"
"[batch_size, height, width, num_channels]")
im_complex = images.astype("complex64")
im_fft = np.fft.fft2(im_complex, axes=(1, 2))
# resizing images
if resize:
# get fourier frequencies to threshold
assert (im_fft.shape[1] == im_fft.shape[2]), ("Need images to have same"
"height and width")
# downsample by threshold
width = im_fft.shape[2]
new_width = new_size[0]
freqs = np.fft.fftfreq(width, d=1.0 / width)
idxs = np.flatnonzero((freqs >= -new_width / 2.0) & (freqs <
new_width / 2.0))
im_fft_downsampled = im_fft[:, :, idxs, :][:, idxs, :, :]
else:
im_fft_downsampled = im_fft
return im_fft_downsampled
def fourier_ccas(conv_acts1, conv_acts2, return_coefs=False,
compute_dirns=False, verbose=False):
"""Computes cca similarity between two conv layers with DFT.
This function takes in two sets of convolutional activations, conv_acts1,
conv_acts2 After resizing the spatial dimensions to be the same, applies fft
and then computes the ccas.
Finally, it applies the inverse fourier transform to get the CCA directions
and neuron coefficients.
Args:
conv_acts1: numpy array with shape
[batch_size, height1, width1, num_channels1]
conv_acts2: numpy array with shape
[batch_size, height2, width2, num_channels2]
compute_dirns: boolean, used to determine whether results also
contain actual cca directions.
Returns:
all_results: a pandas dataframe, with cca results for every spatial
location. Columns are neuron coefficients (combinations
of neurons that correspond to cca directions), the cca
correlation coefficients (how well aligned directions
correlate) x and y idxs (for computing cca directions
on the fly if compute_dirns=False), and summary
statistics. If compute_dirns=True, the cca directions
are also computed.
"""
height1, width1 = conv_acts1.shape[1], conv_acts1.shape[2]
height2, width2 = conv_acts2.shape[1], conv_acts2.shape[2]
if height1 != height2 or width1 != width2:
height = min(height1, height2)
width = min(width1, width2)
new_size = [height, width]
resize = True
else:
height = height1
width = width1
new_size = None
resize = False
# resize and preprocess with fft
fft_acts1 = fft_resize(conv_acts1, resize=resize, new_size=new_size)
fft_acts2 = fft_resize(conv_acts2, resize=resize, new_size=new_size)
# loop over spatial dimensions and get cca coefficients
all_results = pd.DataFrame()
for i in xrange(height):
for j in xrange(width):
results_dict = cca_core.get_cca_similarity(
fft_acts1[:, i, j, :].T, fft_acts2[:, i, j, :].T, compute_dirns,
verbose=verbose)
# apply inverse FFT to get coefficients and directions if specified
if return_coefs:
results_dict["neuron_coeffs1"] = np.fft.ifft2(
results_dict["neuron_coeffs1"])
results_dict["neuron_coeffs2"] = np.fft.ifft2(
results_dict["neuron_coeffs2"])
else:
del results_dict["neuron_coeffs1"]
del results_dict["neuron_coeffs2"]
if compute_dirns:
results_dict["cca_dirns1"] = np.fft.ifft2(results_dict["cca_dirns1"])
results_dict["cca_dirns2"] = np.fft.ifft2(results_dict["cca_dirns2"])
# accumulate results
results_dict["location"] = (i, j)
all_results = all_results.append(results_dict, ignore_index=True)
return all_results