19 for (
int ij = 0; ij < n_terms; ij++) {
20 const double dx = pos[2 * terms[ij].
i] - pos[2 * terms[ij].
j];
21 const double dy = pos[2 * terms[ij].i + 1] - pos[2 * terms[ij].j + 1];
22 const double r = hypot(
dx,
dy) - terms[ij].d;
23 stress += terms[ij].w * (r * r);
31 for (
int i = n_terms - 1; i >= 1; i--) {
34 SWAP(&terms[i], &terms[j]);
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;
72 graph->targets[n_edges] = (size_t)
ND_id(target);
74 assert(
graph->weights[n_edges] > 0);
79 assert(n_nodes ==
graph->n);
80 assert(n_edges <= INT_MAX);
82 graph->sources[n_nodes] = n_edges;
91 for (
size_t i = 0; i <
graph->n; i++) {
93 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
94 size_t j =
graph->targets[x];
100 for (
size_t x =
graph->sources[i]; x <
graph->sources[i + 1]; x++) {
101 size_t j =
graph->targets[x];
104 for (
size_t y =
graph->sources[j]; y <
graph->sources[j + 1]; y++) {
105 size_t k =
graph->targets[y];
115 assert(
graph->weights[x] > 0);
116 for (
size_t y =
graph->sources[j]; y <
graph->sources[j + 1]; y++) {
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];
145 agwarningf(
"circuit model not yet supported in Gmode=sgd, reverting to "
146 "shortpath model\n");
150 agwarningf(
"mds model not yet supported in Gmode=sgd, reverting to "
151 "shortpath model\n");
157 fprintf(stderr,
"calculating shortest paths and setting up stress terms:");
161 int n_fixed = 0, n_terms = 0;
162 for (
int i = 0; i < n; i++) {
165 n_terms += n - n_fixed;
172 for (
int i = 0; i < n; i++) {
177 assert(offset == n_terms);
184 float w_min = terms[0].
w, w_max = terms[0].
w;
185 for (
int ij = 1; ij < n_terms; ij++) {
186 w_min = fminf(w_min, terms[ij].w);
187 w_max = fmaxf(w_max, terms[ij].w);
193 const double eta_max = 1.0 / w_min;
194 const double eta_min =
Epsilon / w_max;
195 const double lambda = log(eta_max / eta_min) / (
MaxIter - 1);
200 double *
const pos =
gv_calloc(2 * n,
sizeof(
double));
201 bool *unfixed =
gv_calloc(n,
sizeof(
bool));
202 for (
int i = 0; i < n; i++) {
211 fprintf(stderr,
"solving model:");
216 for (
int t = 0; t <
MaxIter; t++) {
218 const double eta = eta_max * exp(-lambda * t);
219 for (
int ij = 0; ij < n_terms; ij++) {
221 const double mu = fmin(eta * terms[ij].w, 1);
223 const double dx = pos[2 * terms[ij].
i] - pos[2 * terms[ij].
j];
224 const double dy = pos[2 * terms[ij].i + 1] - pos[2 * terms[ij].j + 1];
225 const double mag = hypot(
dx,
dy);
227 const double r = (
mu * (mag - terms[ij].d)) / (2 * mag);
228 const double r_x = r *
dx;
229 const double r_y = r *
dy;
231 if (unfixed[terms[ij].i]) {
232 pos[2 * terms[ij].i] -= r_x;
233 pos[2 * terms[ij].i + 1] -= r_y;
235 if (unfixed[terms[ij].j]) {
236 pos[2 * terms[ij].j] += r_x;
237 pos[2 * terms[ij].j + 1] += r_y;
245 fprintf(stderr,
"\nfinished in %.2f sec\n",
elapsed_sec());
250 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)
Arithmetic helper functions.
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)