2021-09-23 23:16:29 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 Benjamin Valentin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
|
|
* License v2. See the file LICENSE for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2023-10-07 16:06:57 +02:00
|
|
|
#ifndef CONFIG_USE_NUMERIC_NAMES
|
|
|
|
#define CONFIG_USE_NUMERIC_NAMES 1
|
|
|
|
#endif
|
|
|
|
|
2021-09-23 23:16:29 +02:00
|
|
|
struct node {
|
|
|
|
char name[8];
|
|
|
|
int x;
|
|
|
|
int y;
|
|
|
|
unsigned r;
|
|
|
|
bool rx_from_root;
|
|
|
|
bool tx_to_root;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct world {
|
|
|
|
unsigned w;
|
|
|
|
unsigned h;
|
|
|
|
unsigned num_nodes;
|
|
|
|
struct node *nodes;
|
2023-02-15 16:51:53 +01:00
|
|
|
bool grid;
|
2021-09-23 23:16:29 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned random_range(unsigned lower, unsigned upper)
|
|
|
|
{
|
|
|
|
unsigned range = upper - lower + 1;
|
|
|
|
return lower + rand() % range;
|
|
|
|
}
|
|
|
|
|
2023-02-15 16:51:53 +01:00
|
|
|
static struct node *node_generate(struct node *node, const struct world *w,
|
|
|
|
unsigned range, unsigned idx)
|
2021-09-23 23:16:29 +02:00
|
|
|
{
|
2023-02-15 16:51:53 +01:00
|
|
|
if (w->grid) {
|
|
|
|
float width = w->w;
|
|
|
|
float height = w->h;
|
|
|
|
float num = w->num_nodes;
|
|
|
|
|
|
|
|
/* https://math.stackexchange.com/a/1039514 */
|
|
|
|
float n_x = sqrtf(num * width/height
|
|
|
|
+ powf(width - height, 2)/(4 * height*height))
|
|
|
|
- (width - height)/2;
|
|
|
|
float n_y = num / n_x;
|
|
|
|
|
|
|
|
unsigned step_x = width / n_x;
|
|
|
|
unsigned step_y = height / n_y;
|
|
|
|
|
|
|
|
node->x = (idx * step_x + step_x / 2) % w->w;
|
|
|
|
node->y = ((idx * step_x + step_x / 2) / w->w) * step_y + step_y / 2;
|
|
|
|
} else {
|
|
|
|
node->x = random_range(0, w->w);
|
|
|
|
node->y = random_range(0, w->h);
|
|
|
|
}
|
|
|
|
node->r = range;
|
|
|
|
return node;
|
2021-09-23 23:16:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static double node_distance(const struct node *a, const struct node *b)
|
|
|
|
{
|
|
|
|
return sqrt(pow(a->x - b->x, 2) + pow(a->y - b->y, 2));
|
|
|
|
}
|
|
|
|
|
|
|
|
static double node_distance_weight(const struct node *a, const struct node *b)
|
|
|
|
{
|
2023-10-04 16:30:43 +02:00
|
|
|
double w = 1 - pow(node_distance(a, b), 2) / pow(a->r, 2);
|
2021-09-23 23:16:29 +02:00
|
|
|
|
|
|
|
if (w < 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return w;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void node_name(struct node *n, unsigned idx)
|
|
|
|
{
|
2023-10-07 16:06:57 +02:00
|
|
|
if (CONFIG_USE_NUMERIC_NAMES) {
|
|
|
|
snprintf(n->name, sizeof(n->name), "n%03u", (uint16_t)idx + 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-23 23:16:29 +02:00
|
|
|
char *s = n->name;
|
|
|
|
const char *end = s + sizeof(n->name) - 1;
|
|
|
|
|
|
|
|
do {
|
|
|
|
uint8_t rem = idx % 26;
|
2023-01-25 19:51:52 +01:00
|
|
|
idx /= 26;
|
2021-09-23 23:16:29 +02:00
|
|
|
*s++ = 'A' + rem;
|
|
|
|
} while (idx && s != end);
|
|
|
|
*s = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void world_gen(struct world *w, unsigned num_nodes,
|
|
|
|
unsigned width, unsigned height,
|
|
|
|
unsigned range, unsigned var)
|
|
|
|
{
|
|
|
|
w->w = width;
|
|
|
|
w->h = height;
|
|
|
|
w->num_nodes = num_nodes;
|
|
|
|
w->nodes = calloc(num_nodes, sizeof(*w->nodes));
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < num_nodes; ++i) {
|
2023-02-15 16:51:53 +01:00
|
|
|
node_generate(&w->nodes[i], w, random_range(range - var, range + var), i);
|
2021-09-23 23:16:29 +02:00
|
|
|
node_name(&w->nodes[i], i);
|
|
|
|
}
|
2023-10-07 16:07:43 +02:00
|
|
|
|
|
|
|
if (!w->grid) {
|
|
|
|
/* place first node at origin */
|
|
|
|
w->nodes[0].x = 0;
|
|
|
|
w->nodes[0].y = 0;
|
|
|
|
}
|
2021-09-23 23:16:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned _color(const struct node *n, unsigned base)
|
|
|
|
{
|
|
|
|
if (n->rx_from_root && n->tx_to_root) {
|
|
|
|
return base;
|
|
|
|
} else if (n->rx_from_root) {
|
|
|
|
return base / 2;
|
|
|
|
} else if (n->tx_to_root) {
|
|
|
|
return base / 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _print_nodes(struct node *nodes, unsigned num, unsigned color)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < num; ++i) {
|
|
|
|
printf("# %s\t%d\t%d\t%u\t0x%x\n", nodes[i].name,
|
|
|
|
nodes[i].x, nodes[i].y,
|
|
|
|
nodes[i].r, _color(&nodes[i], color));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* To visualize the network we color the nodes based on whether they have
|
|
|
|
* a (bi-directional) connection to the root node via some other node(s).
|
|
|
|
*/
|
|
|
|
static void _calc_connections(struct node *nodes, unsigned num)
|
|
|
|
{
|
|
|
|
bool changes = true;
|
|
|
|
nodes->rx_from_root = true;
|
|
|
|
nodes->tx_to_root = true;
|
|
|
|
|
|
|
|
/* super basic algorithm - just loop unti there are no more changes
|
|
|
|
* in the node's connection states. */
|
|
|
|
while (changes) {
|
|
|
|
changes = false;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < num; ++i) {
|
|
|
|
const struct node *n = &nodes[i];
|
|
|
|
|
|
|
|
for (unsigned j = 0; j < num; ++j) {
|
|
|
|
struct node *m = &nodes[j];
|
|
|
|
|
|
|
|
/* node is already fully connected */
|
|
|
|
if (m->rx_from_root && m->tx_to_root) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* m can receive from n and n can receive from root */
|
|
|
|
if (node_distance_weight(n, m) > 0) {
|
|
|
|
if (!m->rx_from_root && n->rx_from_root) {
|
|
|
|
m->rx_from_root = true;
|
|
|
|
changes = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* m can send to n and n can send to root */
|
|
|
|
if (node_distance_weight(m, n) > 0) {
|
|
|
|
if (!m->tx_to_root && n->tx_to_root) {
|
|
|
|
m->tx_to_root = true;
|
|
|
|
changes = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-30 21:40:19 +01:00
|
|
|
static void _print_distance(struct node *nodes, unsigned num, bool recursive, bool binary)
|
2021-09-23 23:16:29 +02:00
|
|
|
{
|
|
|
|
struct node *start = nodes;
|
|
|
|
|
2023-10-07 16:08:08 +02:00
|
|
|
if (recursive) {
|
|
|
|
for (unsigned i = 0; i < num; ++i) {
|
|
|
|
printf("%s\n", nodes[i].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-23 23:16:29 +02:00
|
|
|
for (unsigned i = 1; i < num; ++i) {
|
|
|
|
struct node *n = &nodes[i];
|
|
|
|
|
|
|
|
double to_node = node_distance_weight(start, n);
|
|
|
|
double from_node = node_distance_weight(n, start);
|
|
|
|
|
2023-01-30 21:40:19 +01:00
|
|
|
if (binary) {
|
|
|
|
if (to_node >= 0.5) {
|
|
|
|
printf("%s\t%s\n", start->name, n->name);
|
|
|
|
}
|
|
|
|
if (from_node >= 0.5) {
|
|
|
|
printf("%s\t%s\n", n->name, start->name);
|
|
|
|
}
|
|
|
|
} else if (to_node > 0 || from_node > 0) {
|
2021-09-23 23:16:29 +02:00
|
|
|
printf("%s\t%s\t%.2f\t%.2f\n", start->name, n->name, to_node, from_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (recursive) {
|
2023-01-30 21:40:19 +01:00
|
|
|
_print_distance(n, num - i, false, binary);
|
2021-09-23 23:16:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _print_help(const char *name)
|
|
|
|
{
|
|
|
|
puts("Generate a number of nodes that are randomly placed in a rectangular area");
|
|
|
|
printf("usage: %s [-s <seed>]"
|
|
|
|
" [-w <width>]"
|
|
|
|
" [-h <height>]"
|
|
|
|
" [-r <range>]"
|
|
|
|
" [-v <variance of range>]"
|
|
|
|
" [-n <nodes>]"
|
2023-12-17 17:31:21 +01:00
|
|
|
" [-b][-g]"
|
2021-09-23 23:16:29 +02:00
|
|
|
"\n", name);
|
2023-12-17 17:31:21 +01:00
|
|
|
|
|
|
|
puts("\nOptions:");
|
|
|
|
puts("\t-s <seed>\trandom seed used for topology generation");
|
|
|
|
puts("\t-w <width>\twidth of the 2D topology");
|
|
|
|
puts("\t-h <height>\theight of the 2D topology");
|
|
|
|
puts("\t-r <range>\tRadio range of the nodes");
|
|
|
|
puts("\t-v <variance>\tmaximal random variance of radio range");
|
|
|
|
puts("\t-n <nodes>\tnumber of nodes in the topology");
|
|
|
|
puts("\t-b\t\tbinary links: link quality is rounded to 100% or 0%");
|
|
|
|
puts("\t-g\t\tnodes are organized as a grid");
|
2021-09-23 23:16:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
|
|
|
const char *progname = argv[0];
|
|
|
|
|
|
|
|
unsigned width = 100;
|
|
|
|
unsigned height = 100;
|
|
|
|
unsigned seed = time(NULL);
|
|
|
|
unsigned range = 25;
|
|
|
|
unsigned var = 0;
|
|
|
|
unsigned num = 10;
|
2023-01-30 21:40:19 +01:00
|
|
|
bool binary = false;
|
2023-02-15 16:51:53 +01:00
|
|
|
bool grid = false;
|
2021-09-23 23:16:29 +02:00
|
|
|
char c;
|
|
|
|
|
2023-02-15 16:51:53 +01:00
|
|
|
while ((c = getopt(argc, argv, "s:w:h:r:v:n:bg")) != -1) {
|
2021-09-23 23:16:29 +02:00
|
|
|
switch (c) {
|
2023-01-30 21:40:19 +01:00
|
|
|
case 'b':
|
|
|
|
binary = true;
|
|
|
|
break;
|
2023-02-15 16:51:53 +01:00
|
|
|
case 'g':
|
|
|
|
grid = true;
|
|
|
|
break;
|
2021-09-23 23:16:29 +02:00
|
|
|
case 's':
|
|
|
|
seed = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
width = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case 'h':
|
|
|
|
height = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
range = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
var = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
num = atoi(optarg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
_print_help(progname);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
srand(seed);
|
|
|
|
|
2023-02-15 16:51:53 +01:00
|
|
|
struct world w = {
|
|
|
|
.grid = grid,
|
|
|
|
};
|
2021-09-23 23:16:29 +02:00
|
|
|
world_gen(&w, num, width, height, range, var);
|
|
|
|
|
|
|
|
printf("# seed = %u\n", seed);
|
|
|
|
puts("# Connections");
|
2023-01-30 21:40:19 +01:00
|
|
|
_print_distance(w.nodes, w.num_nodes, true, binary);
|
2021-09-23 23:16:29 +02:00
|
|
|
puts("");
|
|
|
|
puts("");
|
|
|
|
puts("# Node\tX\tY\trange\tcolor");
|
|
|
|
_calc_connections(w.nodes, w.num_nodes);
|
|
|
|
_print_nodes(w.nodes, w.num_nodes, rand());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|