16 for (
int ij = 0; ij < n_terms; ij++) {
17 const double dx = pos[2 * terms[ij].
i] - pos[2 * terms[ij].
j];
18 const double dy = pos[2 * terms[ij].i + 1] - pos[2 * terms[ij].j + 1];
19 const double r = hypot(
dx,
dy) - terms[ij].d;
20 stress += terms[ij].w * (r * r);
28 for (
int i = n_terms - 1; i >= 1; i--) {
39 size_t n_nodes = 0, n_edges = 0;
41 assert(
ND_id(np) == n_nodes);
56 assert(n_edges <= INT_MAX);
59 n_nodes = 0, n_edges = 0;
61 assert(n_edges <= INT_MAX);
62 graph->sources[n_nodes] = n_edges;
71 graph->targets[n_edges] = (size_t)
ND_id(target);
73 assert(
graph->weights[n_edges] > 0);
78 assert(n_nodes ==
graph->n);
79 assert(n_edges <= INT_MAX);
81 graph->sources[n_nodes] = n_edges;
90 for (
size_t i = 0; i <
graph->n; i++) {
92 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
93 size_t j =
graph->targets[x];
99 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
100 size_t j =
graph->targets[x];
103 for (
size_t y =
graph->sources[j]; y <
graph->sources[j + 1]; y++) {
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]; y++) {
116 size_t k =
graph->targets[y];
120 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
121 size_t j =
graph->targets[x];
144 agwarningf(
"circuit model not yet supported in Gmode=sgd, reverting to "
145 "shortpath model\n");
149 agwarningf(
"mds model not yet supported in Gmode=sgd, reverting to "
150 "shortpath model\n");
156 fprintf(stderr,
"calculating shortest paths and setting up stress terms:");
160 int n_fixed = 0, n_terms = 0;
161 for (
int i = 0; i < n; i++) {
164 n_terms += n - n_fixed;
171 for (
int i = 0; i < n; i++) {
176 assert(
offset == n_terms);
183 float w_min = terms[0].
w, w_max = terms[0].
w;
184 for (
int ij = 1; ij < n_terms; ij++) {
185 w_min = fminf(w_min, terms[ij].w);
186 w_max = fmaxf(w_max, terms[ij].w);
192 const double eta_max = 1.0 / w_min;
193 const double eta_min =
Epsilon / w_max;
194 const double lambda = log(eta_max / eta_min) / (
MaxIter - 1);
199 double *
const pos =
gv_calloc(2 * n,
sizeof(
double));
200 bool *unfixed =
gv_calloc(n,
sizeof(
bool));
201 for (
int i = 0; i < n; i++) {
210 fprintf(stderr,
"solving model:");
215 for (
int t = 0; t <
MaxIter; t++) {
217 const double eta = eta_max * exp(-lambda * t);
218 for (
int ij = 0; ij < n_terms; ij++) {
220 const double mu = fmin(eta * terms[ij].w, 1);
222 const double dx = pos[2 * terms[ij].
i] - pos[2 * terms[ij].
j];
223 const double dy = pos[2 * terms[ij].i + 1] - pos[2 * terms[ij].j + 1];
224 const double mag = hypot(
dx,
dy);
226 const double r = (
mu * (mag - terms[ij].d)) / (2 * mag);
227 const double r_x = r *
dx;
228 const double r_y = r *
dy;
230 if (unfixed[terms[ij].i]) {
231 pos[2 * terms[ij].i] -= r_x;
232 pos[2 * terms[ij].i + 1] -= r_y;
234 if (unfixed[terms[ij].j]) {
235 pos[2 * terms[ij].j] += r_x;
236 pos[2 * terms[ij].j + 1] += r_y;
244 fprintf(stderr,
"\nfinished in %.2f sec\n",
elapsed_sec());
249 for (
int 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 double calculate_stress(double *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)