16 for (ij=0; ij<n_terms; ij++) {
17 float dx = pos[2*terms[ij].
i] - pos[2*terms[ij].
j];
18 float dy = pos[2*terms[ij].i+1] - pos[2*terms[ij].j+1];
19 float r = hypotf(
dx,
dy) - terms[ij].d;
20 stress += terms[ij].w * (r * r);
27 for (i=n_terms-1; i>=1; i--) {
40 size_t n_nodes = 0, n_edges = 0;
42 assert(
ND_id(np) == n_nodes);
57 assert(n_edges <= INT_MAX);
60 n_nodes = 0, n_edges = 0;
62 assert(n_edges <= INT_MAX);
63 graph->sources[n_nodes] = n_edges;
70 graph->targets[n_edges] = (size_t)
ND_id(target);
72 assert(
graph->weights[n_edges] > 0);
77 assert(n_nodes ==
graph->n);
78 assert(n_edges <= INT_MAX);
80 graph->sources[n_nodes] = n_edges;
89 for (
size_t i = 0; i <
graph->n; i++) {
91 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
92 size_t j =
graph->targets[x];
98 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
99 size_t j =
graph->targets[x];
102 for (
size_t y =
graph->sources[j]; y <
graph->sources[j + 1];
104 size_t k =
graph->targets[y];
114 assert(
graph->weights[x] > 0);
115 for (
size_t y =
graph->sources[j]; y <
graph->sources[j + 1];
117 size_t k =
graph->targets[y];
121 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
122 size_t j =
graph->targets[x];
147 agwarningf(
"circuit model not yet supported in Gmode=sgd, reverting to shortpath model\n");
151 agwarningf(
"mds model not yet supported in Gmode=sgd, reverting to shortpath model\n");
157 fprintf(stderr,
"calculating shortest paths and setting up stress terms:");
161 int i, n_fixed = 0, n_terms = 0;
162 for (i=0; i<n; i++) {
165 n_terms += n-n_fixed;
172 for (i=0; i<n; i++) {
177 assert(
offset == n_terms);
184 float w_min = terms[0].
w, w_max = terms[0].
w;
186 for (ij=1; ij<n_terms; ij++) {
187 w_min = fminf(w_min, terms[ij].w);
188 w_max = fmaxf(w_max, terms[ij].w);
192 float eta_max = 1 / w_min;
193 float eta_min =
Epsilon / w_max;
194 float lambda = log(eta_max/eta_min) / (
MaxIter-1);
199 float *pos =
gv_calloc(2 * n,
sizeof(
float));
200 bool *unfixed =
gv_calloc(n,
sizeof(
bool));
201 for (i=0; i<n; i++) {
210 fprintf(stderr,
"solving model:");
218 float eta = eta_max * exp(-lambda * t);
219 for (ij=0; ij<n_terms; ij++) {
221 float mu = eta * terms[ij].
w;
225 float dx = pos[2*terms[ij].
i] - pos[2*terms[ij].
j];
226 float dy = pos[2*terms[ij].i+1] - pos[2*terms[ij].j+1];
227 float mag = hypotf(
dx,
dy);
229 float r = (
mu * (mag-terms[ij].d)) / (2*mag);
233 if (unfixed[terms[ij].i]) {
234 pos[2*terms[ij].i] -= r_x;
235 pos[2*terms[ij].i+1] -= r_y;
237 if (unfixed[terms[ij].j]) {
238 pos[2*terms[ij].j] += r_x;
239 pos[2*terms[ij].j+1] += r_y;
247 fprintf(stderr,
"\nfinished in %.2f sec\n",
elapsed_sec());
252 for (i=0; i<n; i++) {
Memory allocation wrappers that exit on failure.
static void * gv_calloc(size_t nmemb, size_t size)
static void * gv_alloc(size_t size)
API for compacted arrays of booleans.
static bitarray_t bitarray_new(size_t size_bits)
create an array of the given element length
static bool bitarray_get(bitarray_t self, size_t index)
get the value of the given element
static void bitarray_set(bitarray_t *self, size_t index, bool value)
set or clear the value of the given element
static void bitarray_reset(bitarray_t *self)
free underlying resources and leave a bit array empty
int agnnodes(Agraph_t *g)
Agedge_t * agnxtedge(Agraph_t *g, Agedge_t *e, Agnode_t *n)
Agedge_t * agfstedge(Agraph_t *g, Agnode_t *n)
void agwarningf(const char *fmt,...)
#define GD_neato_nlist(g)
Agnode_t * agnxtnode(Agraph_t *g, Agnode_t *n)
Agnode_t * agfstnode(Agraph_t *g)
Agraph_t * graph(char *name)
swig_ptr_object_handlers offset
int dijkstra_sgd(graph_sgd *graph, int source, term_sgd *terms)
NEATOPROCS_API void initial_positions(graph_t *, int)
unsigned long rk_interval(unsigned long max, rk_state *state)
void rk_seed(unsigned long seed, rk_state *state)
static void free_adjacency(graph_sgd *graph)
static void fisheryates_shuffle(term_sgd *terms, int n_terms, rk_state *rstate)
static float calculate_stress(float *pos, term_sgd *terms, int n_terms)
void sgd(graph_t *G, int model)
static graph_sgd * extract_adjacency(graph_t *G, int model)
static bool intersect(Ppoint_t a, Ppoint_t b, Ppoint_t c, Ppoint_t d)