-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathrepeatconn.mod
373 lines (336 loc) · 17 KB
/
repeatconn.mod
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
COMMENT
This repeatconn mechanism written by [email protected] on April 19, 2011.
It decreases the time taken to decide which connections to make between
cell by an order of magnitude, compared to Rob & Viji's code (even the
version that I parallelized). How it works, in context with hoc:
for each cell type (as postsynaptic cell type) - hoc
> for each cell type (as presynaptic cell type) - hoc
> > make a vector of gids of all cells owned by this processor
of the postsynaptic type - hoc
> > send vector, presynaptic cell type axonal distribution,
number of connections desired to NMODL - hoc
> > for each postsynaptic cell of postsynaptic cell type
owned by this processor - NMODL
> > > calculate the distance between every presynaptic cell
of presynaptic cell type and this postsynaptic cell - NMODL
> > > choose the desired number of connections at various distances - NMODL
(as defined by the presynaptic axonal distribution
and total # of desired connections) - NMODL
> > > randomly pick the specific connections, up to the desired
and/or available number for each distance - NMODL
> > > for each desired connection, add the gid
of the presynaptic cell to a vector - NMODL
> > return the filled vector to hoc - NMODL
> > make all the connections listed in the vector - hoc
The way to call this from hoc:
- first, install the vector method with this argument:
install_repeatconn()
- then, create a vector with the parameters for the connections
(parameters described below)
- then, create a vector with the gids of the post synaptic cells
- and finally, create an empty placeholder vector with 3x the elements
as the number of connections desired
- finally, populate the placeholder vector with the connection
information using the following command:
conns2make.repeatconn(params, postcellgids)
params vector has 26 elements:
0 - start gid of the presynaptic cell type
1 - end gid of the presynaptic cell type (assume continuous range)
2 - total number of connections desired from this presynaptic cell type
to this postsynaptic cell type
3 - total number of cells of the postsynaptic type
4 - number of cells of postsynaptic type with gids owned by this host
5 - total distance over which distance distribution applies (um)
6 - resolution of the fit to the distribution, in # steps to take
7 - distribution equation coefficient a (for this presynaptic cell type)
8 - distribution equation coefficient b (for this presynaptic cell type)
9 - distribution equation coefficient c (for this presynaptic cell type)
10 - number of position bins along the X axis for presynaptic cell type
11 - number of position bins along the Y axis for presynaptic cell type
12 - number of position bins along the Z axis for presynaptic cell type
13 - length (um) of bins along the X axis for presynaptic cell type
14 - length (um) of bins along the Y axis for presynaptic cell type
15 - length (um) of bins along the Z axis for presynaptic cell type
16 - height in the Z direction, of the layer in which the
presynaptic cell type exists
17 - number of position bins along the X axis for postsynaptic cell type
18 - number of position bins along the Y axis for postsynaptic cell type
19 - number of position bins along the Z axis for postsynaptic cell type
20 - length (um) of bins along the X axis for presynaptic cell type
21 - length (um) of bins along the Y axis for presynaptic cell type
22 - length (um) of bins along the Z axis for presynaptic cell type
23 - height in the Z direction, of the layer in which the
postsynaptic cell type exists
24 - start gid of the postsynaptic cell type
25 - high index at which to start the random number stream (for each gid)
should be set such that there are no overlaps per each gid, so err
on the side of caution when setting the next high index.
ENDCOMMENT
VERBATIM
# include <stdlib.h>
# include <stdio.h>
# include <math.h>
ENDVERBATIM
NEURON {
SUFFIX nothing
}
VERBATIM
extern double* vector_vec();
extern int vector_capacity();
extern void* vector_arg();
extern double get_x_pos(double gid, double gmin, double BinNumX, double BinNumYZ, double binSizeX);
extern double get_y_pos(double gid, double gmin, double BinNumY, double BinNumZ, double binSizeY);
extern double get_z_pos(double gid, double gmin, double BinNumZ, double binSizeZ, double ZHeight);
//void srand(unsigned seed);
ENDVERBATIM
VERBATIM
static double repeatconn (void* vv) {
int repeatfinal, ny, nz, num_pre, num_post, gmin, gmax, steps, myflaggy, myi, postgmin, stepover, reuse_conns_from_prev_step;
double *x, *y, *z, a, b, c, nconv, ncell, axonal_extent;
reuse_conns_from_prev_step=1;
/* Get hoc vectors into c arrays */
repeatfinal = vector_instance_px(vv, &x); // x is an array corresponding
// to the placeholder vector
// of connections to make
ny = vector_arg_px(1, &y); // y is an array of parameters
nz = vector_arg_px(2, &z); // z is an array of the postsynaptic gids
/* Load the parameters from the param array */
gmin = y[0]; // presynaptic start gid
gmax = y[1]; // presynaptic end gid
num_pre = gmax - gmin + 1; // number of presynaptic cells
nconv = y[2]; // total number of desired connections
ncell = y[3]; // total number of postsynaptic cells
num_post = y[4]; // number of postsynaptic cells owned by this host
axonal_extent = y[5]; // total distance over which distribution fits, the axonal extent of the presynaptic cell type
steps = y[6]; // resolution of the distribution fit (in steps)
a = y[7]; // distribution fit coefficient a, describing the axonal distribution of the presynaptic cell type
b = y[8]; // distribution fit coefficient b, describing the axonal distribution of the presynaptic cell type
c = y[9]; // distribution fit coefficient c, describing the axonal distribution of the presynaptic cell type
postgmin = y[24]; // postsynaptic start gid
stepover = y[26]; // buffer size for number of conns for results vector
myi=2; // myi will give the next index into finalconn
// 0 is reserved for # conns to make
// 1 is reserved for the last high index used by nrnRan4int
/* Get positions of the presynaptic and postsynaptic cells*/
double prepos [num_pre][3];
double postpos [num_post][3];
int cell;
for (cell=0; cell<num_pre; cell++) {
prepos [cell] [0] = get_x_pos(cell+gmin, gmin, y[10], y[11]*y[12], y[13]);
prepos [cell] [1] = get_y_pos(cell+gmin, gmin, y[11], y[12], y[14]);
prepos [cell] [2] = get_z_pos(cell+gmin, gmin, y[12], y[15], y[16]);
}
for (cell=0; cell<num_post; cell++) {
postpos [cell] [0] = get_x_pos(z[cell], postgmin, y[17], y[18]*y[19], y[20]);
postpos [cell] [1] = get_y_pos(z[cell], postgmin, y[18], y[19], y[21]);
postpos [cell] [2] = get_z_pos(z[cell], postgmin, y[19], y[22], y[23]);
}
/* calculate the distribution of desired connections*/
double current_distance [steps], connection_distribution [steps], distribution_denominator, conndist;
int step, feasible_conns_this_step [steps], desired_conns_this_step [steps];
distribution_denominator = 0.0;
int max_fraction_step; max_fraction_step=0;
for (step=0; step<steps; step++) {
current_distance[step] = axonal_extent*1.0*(step+0.5)/(steps); /* current_distance[step] = distance step (in terms of max distance)*/
//current_distance[step] = axonal_extent*1.0*(step+1)/(steps); /* current_distance[step] = distance step (in terms of max distance)*/
//connection_distribution[step] = (1.0/a)*exp(-((current_distance[step]-b)*1.0/c)*((current_distance[step]-b)*1.0/c))*axonal_extent;
connection_distribution[step] = exp(-a*((current_distance[step]-b)*1.0/c)*((current_distance[step]-b)*1.0/c));
if (connection_distribution[step]>connection_distribution[max_fraction_step]) {
max_fraction_step=step;
}
distribution_denominator = distribution_denominator + connection_distribution[step]; // this will be used to normalize the distribution
}
// connection_distribution[step]/distribution_denominator is the fraction of connections to make in this step
if (connection_distribution[max_fraction_step]/distribution_denominator*nconv < 0.5) { //distribution_denominator) nconv=nconn*1.0/ncell
for (step=0; step<steps; step++) {
desired_conns_this_step[step] = round((2.0*connection_distribution[step]/distribution_denominator)*(nconv));// the number of desired
// connections for each
// distance bin step, per cell
/////printf("A. connection_distribution[%d] = %f, desired_conns_this_step[%d] = %d, x=%f out of max distance = %f\n", step, connection_distribution[step], step, desired_conns_this_step[step], current_distance[step], axonal_extent);
//printf("A. connection_distribution[%d]=%f, distribution_denominator=%f, nconv=%f\n", step, connection_distribution[step], distribution_denominator, nconv);
}
} else {
for (step=0; step<steps; step++) {
desired_conns_this_step[step] = round((connection_distribution[step]/distribution_denominator)*(nconv));// the number of desired
// connections for each
// distance bin step, per cell
/////printf("B. connection_distribution[%d] = %f, desired_conns_this_step[%d] = %d, x=%f out of max distance = %f\n", step, connection_distribution[step], step, desired_conns_this_step[step], current_distance[step], axonal_extent);
//printf("B. connection_distribution[%d]=%f, distribution_denominator=%f, nconv=%f\n", step, connection_distribution[step], distribution_denominator, nconv);
}
}
/*for (step=0; step<steps; step++) {
printf("desired_conns_this_step[%d]=%d\n", step, desired_conns_this_step[step]);
}*/
/* for each postsynaptic cell, find the possible connections and
* make the desired number of connections where possible */
int m, n, i, q, goupto, rem, extra, szr, szp [steps];
double distance_between;
u_int32_t idx1, idx2, maxidx1; // high and low index (seeds) for MCell_Ran4
maxidx1 = y[25];
for (n=0; n<num_post; n++) { // for each post cell
int myx = (int)z[n]; // get the gid of the current postsynaptic cell in int form
idx1 = y[25]; // reset the high index for the next postsynaptic
// cell. It should be set to a value that is
// certainly higher than what would have been
// used during previous calls for this low index/gid
// We accomplish that by setting it equal to the
// sum of all previous numconns/ncell (for this
// post cell type, with the previous pre cell types)?
idx2 = myx; // set the low index equal to the gid
/*if (myx==0 || myx==100 || myx==200 || myx==300 || myx==400) {
printf("INDEX: gid=%d, idx1=%d\n", idx2, idx1);
}*/
double sortedpos [num_pre][steps];
for (step=0; step< steps; step++) {
szp [step]=0; // initialize the szp array to 0
// (it holds a number per bin, telling which
// index of the array you are on for that bin)
// when filling the array with available
// connections for each bin
feasible_conns_this_step[step] = desired_conns_this_step[step];
}
double dist;
for(m=0; m<num_pre; m++) { // for each pre cell
// If the presynaptic cell == the postsynaptic cell, skip (no autapses here...)
if ((gmin+m)==myx) {
continue;
}
// calculate the distance between the pre and post cells
//3d distance
//distance_between = sqrt((1.0*prepos[m][0] - postpos[n][0])*(prepos[m][0] - postpos[n][0])+(prepos[m][1] - postpos[n][1])*(prepos[m][1] - postpos[n][1])+(prepos[m][2] - postpos[n][2])*(prepos[m][2] - postpos[n][2]));
//2d distance
distance_between = sqrt((1.0*prepos[m][0] - postpos[n][0])*(prepos[m][0] - postpos[n][0])+(prepos[m][1] - postpos[n][1])*(prepos[m][1] - postpos[n][1]));
for (step=0; step< steps; step++) {
/*if (ncell==2 && num_pre==3) {
printf("distance=%f step=%d stepmax=%f gmin=%d postg=%d\n", distance_between, step, current_distance[step], gmin, postgmin);
}*/
if (distance_between<= current_distance[step]) // if the distance is less than the max distance for that step
{
sortedpos [szp [step]] [step] = m; // add this pre cell to this particular bin's column (the next row, which szp keeps track of)
szp [step]++;
break;
}
}
}
/*if (ncell==2 && num_pre==3) {
for (step=0; step< steps; step++) {
printf("step: %d szp: %d\n", step, szp [step]);
}
}*/
// now, this particular post cell has an array (sortedpos) where each
// column contains a bunch of pre-cell gids whose distances fit within
// that column's "step" or "distance bin"
// There is also a feasible_conns_this_step array that gives the ideal # of connections
// for each step
rem=0;extra=0;
for (step=0; step<steps; step++) { // for each step except the last one
szr = szp [step]; // Get the number of available connections for this step
//if (feasible_conns_this_step[step] + rem> szr) { //If more are wanted than are available,
if (szr < 1) { //Only if there are 0 available in that step, try it in a different step
rem=feasible_conns_this_step[step]+rem-szr;
// check the previous level (closer level) for extras
if (step>0) {
for(i=1; i<=step; i++) {
if (reuse_conns_from_prev_step==0) {
if (szp [step-i] > feasible_conns_this_step[step-i]) {
if (szp [step-i] - feasible_conns_this_step[step-i]>rem) {
extra = rem;
} else {
extra = szp [step-i] - feasible_conns_this_step[step-i];
}
feasible_conns_this_step[step-i] = feasible_conns_this_step[step-i] + extra;
feasible_conns_this_step[step] = feasible_conns_this_step[step] - extra;
rem = rem - extra;
}
} else { // reuse_conns_from_prev_step==1
if (szp [step-i] > 0) {
extra = rem;
feasible_conns_this_step[step-i] = feasible_conns_this_step[step-i] + extra;
feasible_conns_this_step[step] = feasible_conns_this_step[step] - extra;
rem = rem - extra;
}
}
}
}
if (rem>0 && step<steps-1) { // if that still doesn't satisfy all the remainder
for(i=step+1; i<steps; i++) {
if (reuse_conns_from_prev_step==0) {
if (szp [i] > feasible_conns_this_step[i]) {
if (szp [i] - feasible_conns_this_step[i]>rem) {
extra = rem;
} else {
extra = szp [i] - feasible_conns_this_step[i];
}
feasible_conns_this_step[i] = feasible_conns_this_step[i] + extra;
feasible_conns_this_step[step] = feasible_conns_this_step[step] - extra;
rem = rem - extra;
}
} else { // reuse_conns_from_prev_step==1
if (szp [i] > 0) {
extra = rem;
feasible_conns_this_step[i] = feasible_conns_this_step[i] + extra;
feasible_conns_this_step[step] = feasible_conns_this_step[step] - extra;
rem = rem - extra;
}
}
}
}
}
}
/*if (ncell==2 && num_pre==3) {
for (step=0; step< steps; step++) {
printf("step: %d szp: %d feasible_conns_this_step: %d\n", step, szp [step], feasible_conns_this_step[step]);
}
}*/
rem=0;
for (step=0; step<steps; step++) { // for each step
szr = szp [step]; // Get the number of available unique pre-cells for this step
q=0;
if (feasible_conns_this_step[step]>0 && szr>0) { // if this particular step wants any connections
/* Find all the possible connections for each distance level */
/*if (ncell==2 && num_pre==3) {
printf("precells=%d postcells=%d step=%d szr=%d\n", num_pre, ncell, step, szr);
}*/
int r[szr]; // Define an array the length of the number of available unique pre-cells
int rout[feasible_conns_this_step[step]]; // Define an array the length of the number of desired connections
for (i=0; i< szr; i++) {
r[i] = sortedpos [i] [step]; // Fill the array with the available unique pre-cells
}
/* this random routine allows a pre-cell to make multiple connections on the post cell and makes the total number of desired connections*/
u_int32_t randi;
for (i=0; i<feasible_conns_this_step[step]; i++) {
randi = nrnRan4int(&idx1, idx2) % (u_int32_t)szr; // limit to the range of indices in the r array
rout[i] = r[randi];
}
for (q=0; q<feasible_conns_this_step[step]; q++) { // for each one to make, r[q] gives the precell index in the pre_pos array (this program assumes
// the gid range is continuous from gmin to gmax arguments to this mechanism.
// n is the post-cell here.
x [myi] = (rout[q]+gmin)*1.0; // presynaptic gid
x [myi+1*stepover] = (z[n])*1.0; // postsynaptic gid
x [myi+2*stepover] = (step+1)*1.0; // distance step
myi++;
}
}
//if (num_pre>10000) {
// if (z[0]==21504) {
//printf("step=%d, gid=%f, myi=%d\n", step, z[n], myi);
//}
//}
}
if (idx1>maxidx1) { maxidx1=idx1;}
}
x [0] = myi-2; // fill the first element of the array (vector)
// with the total number of connections to make,
// which may be less than the desired number (and
// hence the size of the array)
x [1] = (double)maxidx1;
return repeatfinal;
}
ENDVERBATIM
: This PROCEDURE install_repeatconn() should be called from hoc
: to make the repeatconn procedure available there
PROCEDURE install_repeatconn () {
VERBATIM
install_vector_method("repeatconn", repeatconn);
ENDVERBATIM
}