-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathGenerateESN.m
182 lines (170 loc) · 7.94 KB
/
GenerateESN.m
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
function [esn] = GenerateESN(s,nInputUnits,nInternalUnits,nOutputUnits,varargin)
% Creates an ESN set up for use in multiple-channel output association tasks.
% The number of input, internal, and output
% units have to be set. Any other option is set using the format
% 'name_of_options1',value1,'name_of_option2',value2, etc.
%
%%%%% input arguments:
% nInputUnits: the dimension of the input
% nInternalUnits: size of the Esn
% nOutputUnits: the dimension of the output
%
%%%%% optional arguments:
% 'inputScaling': a nInputUnits x 1 vector
%
% 'inputShift': a nInputUnits x 1 vector.
%
% 'teacherScaling': a nOutputUnits x 1 vector
%
% 'teacherShift': a nOutputUnits x 1 vector.
%
% 'noiseLevel': a small number containing the amount of uniform noise to be
% added when computing the internal states
%
% 'learningMode': a string ('offline_singleTimeSeries', 'offline_multipleTimeSeries' or 'online')
% 1. Case 'offline_singleTimeSeries': trainInput and trainOutput each represent a
% single time series in an array of size sequenceLength x sequenceDimension
% 2. Case 'offline_multipleTimeSeries': trainInput and trainOutput each represent a
% collection of K time series, given in cell arrays of size K x 1, where each cell is an
% array of size individualSequenceLength x sequenceDimension
% 3. Case 'online': trainInput and trainOutput are a single time
% series, output weights are adapted online
%
% 'reservoirActivationFunction': a string ("tanh", "identity", "sigmoid01") ,
%
% 'outputActivationFunction': a string("tanh", "identity", "sigmoid01") ,
%
% 'inverseOutputActivationFunction': the inverse to
% outputActivationFunction, one of 'atanh', 'identity', 'sigmoid01_inv'.
% When choosing the activation function, make sure the inverse
% activation function is corectly set.
%
% 'methodWeightCompute': a string ('pseudoinverse', 'wiener_hopf'). It
% specifies which method to use to compute the output weights given the
% state collection matrix and the teacher
%
% 'spectralRadius': a positive number less than 1.
%
% 'feedbackScaling': a nOutputUnits x 1 vector, indicating the scaling
% factor to be applied on the output before it is fed back into the network
%
% 'type': a string ('plain_esn', 'leaky_esn' or 'twi_esn')
% 'trained': a flag indicating whether the network has been trained already
% 'timeConstants': option used in networks with type == "leaky_esn", "leaky1_esn" and "twi_esn".
% Is given as column vector of size esn.nInternalUnitsm, where each entry
% signifies a time constant for a reservoir neuron.
% 'leakage': option used in networks with type == "leaky_esn" or "twi_esn"
% 'RLS_lambda': option used in online training(learningMode == "online")
% 'RLS_delta': option used in online training(learningMode == "online")
% for more information on the Echo State network approach take a look at
% the following tutorial :
% http://www.faculty.iu-bremen.de/hjaeger/pubs/ESNTutorialRev.pdf
%% set the number of units
rng(s);
esn.nInternalUnits = nInternalUnits;
esn.nInputUnits = nInputUnits;
esn.nOutputUnits = nOutputUnits;
nTotalUnits = nInternalUnits + nInputUnits + nOutputUnits;
esn.nTotalUnits = nTotalUnits;
connectivity = min([10/nInternalUnits 1]);
esn.internalWeights_UnitSR = GenerateInternalWeights(nInternalUnits,connectivity);
%% Input/Output Weight
% Input weight matrix has weight vectors per input unit in colums
upLimit = 2;
esn.inputWeights = 2.0*upLimit*rand(nInternalUnits, nInputUnits)-upLimit;
% Output weight matrix has weights for output units in rows
% includes weights for input-to-output connections
% esn.outputWeights = zeros(nOutputUnits,nInternalUnits);
% esn.outputWeights = zeros(nOutputUnits,2*(nInternalUnits));
% esn.outputWeights = zeros(nOutputUnits,2*(nInternalUnits+nInputUnits));
esn.outputWeights = zeros(nOutputUnits,(nInternalUnits+nInputUnits));
% esn.outputWeights = 2.0*upLimit*rand(nOutputUnits, nInternalUnits+nInputUnits)-upLimit;
% Output feedback weight matrix has weights in columns
% esn.feedbackWeights = ones(nInternalUnits, nOutputUnits);
upLimit = 0.5;
esn.feedbackWeights = 2.0*upLimit*rand(nInternalUnits, nOutputUnits)-upLimit;
%% Init default parameters
if nInputUnits > 0
esn.inputScaling = ones(nInputUnits, 1);
esn.inputShift = zeros(nInputUnits, 1);
else
esn.inputScaling = [];
esn.inputShift = [];
end
if nOutputUnits > 0
esn.teacherScaling = ones(nOutputUnits, 1);
esn.teacherShift = zeros(nOutputUnits, 1);
else
esn.teacherScaling = [];
esn.teacherShift = [];
end
esn.noiseLevel = 0.0 ;
esn.reservoirActivationFunction = 'tanh';
esn.outputActivationFunction = 'identity'; % options: identity or tanh or sigmoid01
esn.methodWeightCompute = 'pseudoinverse' ; % options: pseudoinverse and wiener_hopf
esn.inverseOutputActivationFunction = 'identity' ; % options:identity or atanh or sigmoid01_inv
esn.spectralRadius = 0.5 ;
esn.feedbackScaling = ones(nOutputUnits, 1);
esn.trained = 0 ;
esn.type = 'leaky_esn';
esn.timeConstants = 0.5*ones(esn.nInternalUnits,1);
esn.leakage = 0.82;
esn.learningMode = 'offline_singleTimeSeries';
esn.RLS_lambda = 1 ;
args = varargin;
nargs= length(args);
for i=1:2:nargs
switch args{i},
case 'inputScaling', esn.inputScaling = args{i+1};
case 'inputShift', esn.inputShift= args{i+1};
case 'teacherScaling', esn.teacherScaling = args{i+1};
case 'teacherShift', esn.teacherShift = args{i+1};
case 'noiseLevel', esn.noiseLevel = args{i+1};
case 'learningMode', esn.learningMode = args{i+1};
case 'reservoirActivationFunction',esn.reservoirActivationFunction=args{i+1};
case 'outputActivationFunction',esn.outputActivationFunction= args{i+1};
case 'inverseOutputActivationFunction', esn.inverseOutputActivationFunction=args{i+1};
case 'methodWeightCompute', esn.methodWeightCompute = args{i+1};
case 'spectralRadius', esn.spectralRadius = args{i+1};
case 'feedbackScaling', esn.feedbackScaling = args{i+1};
case 'type' , esn.type = args{i+1};
case 'timeConstants' , esn.timeConstants = args{i+1};
case 'leakage' , esn.leakage = args{i+1};
case 'RLS_lambda' , esn.RLS_lambda = args{i+1};
case 'RLS_delta' , esn.RLS_delta = args{i+1};
otherwise
error('the option does not exist');
end
end
%% Error checking
% check that inputScaling has correct format
if size(esn.inputScaling,1) ~= esn.nInputUnits
error('the size of the inputScaling does not match the number of input units');
end
if size(esn.inputShift,1) ~= esn.nInputUnits
error('the size of the inputScaling does not match the number of input units');
end
if size(esn.teacherScaling,1) ~= esn.nOutputUnits
error('the size of the teacherScaling does not match the number of output units');
end
if size(esn.teacherShift,1) ~= esn.nOutputUnits
error('the size of the teacherShift does not match the number of output units');
end
if length(esn.timeConstants) ~= esn.nInternalUnits
error('timeConstants must be given as column vector of length esn.nInternalUnits');
end
if ~strcmp(esn.learningMode,'offline_singleTimeSeries') &&...
~strcmp(esn.learningMode,'offline_multipleTimeSeries') && ...
~strcmp(esn.learningMode,'online')
error('learningMode should be either "offline_singleTimeSeries", "offline_multipleTimeSeries" or "online" ') ;
end
if ~((strcmp(esn.outputActivationFunction,'identity') && ...
strcmp(esn.inverseOutputActivationFunction,'identity')) || ...
(strcmp(esn.outputActivationFunction,'tanh') && ...
strcmp(esn.inverseOutputActivationFunction,'atanh')) || ...
(strcmp(esn.outputActivationFunction,'sigmoid01') && ...
strcmp(esn.inverseOutputActivationFunction,'sigmoid01_inv')))
error('outputActivationFunction and inverseOutputActivationFunction do not match');
end
esn.internalWeights = esn.spectralRadius * esn.internalWeights_UnitSR;
end