Commit 13430d4e authored by David Gibson's avatar David Gibson

aga,agar: New shortcut1 sample graph and testcases based on it

For all the existing test graphs, the shortest path by cost is also the
shortest path by number of edges.  This patch adds a new test graph where
that is not the case, in order to test that the Dijkstra's algorithm
implementation correctly handles that case.
Signed-off-by: default avatarDavid Gibson <david@gibson.dropbear.id.au>
parent a2eaae42
...@@ -61,8 +61,9 @@ int main(void) ...@@ -61,8 +61,9 @@ int main(void)
struct grid_graph gg1, gg2; struct grid_graph gg1, gg2;
struct error_graph eg; struct error_graph eg;
struct traversal1_graph t1g; struct traversal1_graph t1g;
struct shortcut1_graph s1g;
plan_tests(2 + 7 + 35 + 30 + 30 + 42 + 9 + 30); plan_tests(2 + 7 + 35 + 30 + 30 + 42 + 9 + 30 + 9);
trivial_graph_init(&tg); trivial_graph_init(&tg);
test_adjacency("trivial", &tg.sg, trivial_adjacency); test_adjacency("trivial", &tg.sg, trivial_adjacency);
...@@ -91,5 +92,8 @@ int main(void) ...@@ -91,5 +92,8 @@ int main(void)
traversal1_graph_init(&t1g); traversal1_graph_init(&t1g);
test_adjacency("traversal1 graph", &t1g.sg, traversal1_adjacency); test_adjacency("traversal1 graph", &t1g.sg, traversal1_adjacency);
shortcut1_graph_init(&s1g);
test_adjacency("shortcut1 graph", &s1g.sg, shortcut1_adjacency);
return exit_status(); return exit_status();
} }
...@@ -214,12 +214,32 @@ static void test_traversal1(void) ...@@ -214,12 +214,32 @@ static void test_traversal1(void)
aga_finish(&t1g.sg.g); aga_finish(&t1g.sg.g);
} }
static void test_shortcut1(void)
{
struct shortcut1_graph s1g;
aga_icost_t cost;
struct aga_node *node;
shortcut1_graph_init(&s1g);
ok1(aga_dijkstra_start(&s1g.sg.g, &s1g.sg.nodes[1]) == 0);
ok1(aga_dijkstra_path(&s1g.sg.g, &s1g.sg.nodes[3],
&cost, &node, NULL));
ok1(cost == 2);
ok1(node == &s1g.sg.nodes[2]);
ok1(aga_dijkstra_path(&s1g.sg.g, &s1g.sg.nodes[2],
&cost, &node, NULL));
ok1(cost == 1);
ok1(node == &s1g.sg.nodes[1]);
aga_finish(&s1g.sg.g);
}
int main(void) int main(void)
{ {
plan_tests(7 + 20 plan_tests(7 + 20
+ FULL_LEN * (1 + FULL_LEN*4) + FULL_LEN * (1 + FULL_LEN*4)
+ CHAIN_LEN * (1 + CHAIN_LEN*2) + CHAIN_LEN * (1 + CHAIN_LEN*2)
+ 12 + 32); + 12 + 32 + 7);
test_trivial(); test_trivial();
test_parallel(); test_parallel();
...@@ -227,6 +247,7 @@ int main(void) ...@@ -227,6 +247,7 @@ int main(void)
test_chain(); test_chain();
test_error(); test_error();
test_traversal1(); test_traversal1();
test_shortcut1();
return exit_status(); return exit_status();
} }
#include "config.h"
#include <assert.h>
#include <ccan/container_of/container_of.h>
#include <ccan/ptrint/ptrint.h>
#include <ccan/aga/aga.h>
#include "simple-graph.h"
static ptrint_t *shortcut1_first_edge(const struct aga_graph *g,
const struct aga_node *n)
{
struct shortcut1_graph *s1g = container_of(g, struct shortcut1_graph,
sg.g);
int ni = n - s1g->sg.nodes;
switch (ni) {
case 1:
case 2:
return int2ptr(1);
case 3:
return NULL;
default:
assert(0);
}
}
static ptrint_t *shortcut1_next_edge(const struct aga_graph *g,
const struct aga_node *n,
ptrint_t *e)
{
struct shortcut1_graph *s1g = container_of(g, struct shortcut1_graph,
sg.g);
int ni = n - s1g->sg.nodes;
int index = ptr2int(e);
switch (ni) {
case 1:
if (index == 1)
return int2ptr(2);
assert(index == 2);
return NULL;
case 2:
assert(index == 1);
return NULL;
default:
assert(0);
}
}
static int shortcut1_edge_info(const struct aga_graph *g,
const struct aga_node *n,
ptrint_t *e, struct aga_edge_info *ei)
{
struct shortcut1_graph *s1g = container_of(g, struct shortcut1_graph,
sg.g);
int ni = n - s1g->sg.nodes;
int index = ptr2int(e);
switch (ni) {
case 1:
if (index == 1) {
ei->to = &s1g->sg.nodes[3];
ei->icost = 3;
} else {
assert(index == 2);
ei->to = &s1g->sg.nodes[2];
}
break;
case 2:
assert(index == 1);
ei->to = &s1g->sg.nodes[3];
break;
default:
assert(0);
}
return 0;
}
void shortcut1_graph_init(struct shortcut1_graph *s1g)
{
simple_graph_init(&s1g->sg, shortcut1_first_edge,
shortcut1_next_edge,
shortcut1_edge_info);
}
...@@ -215,4 +215,24 @@ static const struct adjacency_list traversal1_adjacency[] = { ...@@ -215,4 +215,24 @@ static const struct adjacency_list traversal1_adjacency[] = {
{}, {},
}; };
/* Shortcut-1 graph
*
* A ---- (3) -----> C
* \ /
* (1)-> B --(1)
*
* This provides an example of a graph where the lowest cost path from
* (A) to (C) is not the path with the smallest number od edges.
*/
struct shortcut1_graph {
struct simple_graph sg;
};
void shortcut1_graph_init(struct shortcut1_graph *s1g);
static const struct adjacency_list shortcut1_adjacency[] = {
{1, {3, 2}},
{2, {3}},
{3, {}},
{},
};
#endif /* _TEST_GRAPHS_H */ #endif /* _TEST_GRAPHS_H */
...@@ -55,8 +55,9 @@ int main(void) ...@@ -55,8 +55,9 @@ int main(void)
struct chain_graphr cgr; struct chain_graphr cgr;
struct grid_graphr ggr1, ggr2; struct grid_graphr ggr1, ggr2;
struct error_graphr egr; struct error_graphr egr;
struct shortcut1_graphr s1gr;
plan_tests(1 + 5 + 30 + 22 + 21 + 33 + 6); plan_tests(1 + 5 + 30 + 22 + 21 + 33 + 6 + 6);
trivial_graphr_init(&tgr); trivial_graphr_init(&tgr);
test_adjacency("trivial", &tgr.gr, trivial_adjacencyr); test_adjacency("trivial", &tgr.gr, trivial_adjacencyr);
...@@ -82,5 +83,8 @@ int main(void) ...@@ -82,5 +83,8 @@ int main(void)
error_graphr_init(&egr); error_graphr_init(&egr);
test_adjacency("error graph", &egr.gr, error_adjacencyr); test_adjacency("error graph", &egr.gr, error_adjacencyr);
shortcut1_graphr_init(&s1gr);
test_adjacency("shortcut1 graph", &s1gr.gr, shortcut1_adjacencyr);
return exit_status(); return exit_status();
} }
...@@ -219,12 +219,31 @@ static void test_traversal1(void) ...@@ -219,12 +219,31 @@ static void test_traversal1(void)
tal_free(sr); tal_free(sr);
} }
static void test_shortcut1(void)
{
struct shortcut1_graphr s1gr;
struct agar_state *sr;
aga_icost_t cost;
const void *node;
shortcut1_graphr_init(&s1gr);
ok1(sr = agar_dijkstra_new(NULL, &s1gr.gr, int2ptr(1)));
ok1(agar_dijkstra_path(sr, int2ptr(3), &cost, &node, NULL));
ok1(cost == 2);
ok1(node == int2ptr(2));
ok1(agar_dijkstra_path(sr, int2ptr(2), &cost, &node, NULL));
ok1(cost == 1);
ok1(node == int2ptr(1));
tal_free(sr);
}
int main(void) int main(void)
{ {
plan_tests(6 + 23 plan_tests(6 + 23
+ FULL_LEN * (FULL_LEN*4 - 1) + FULL_LEN * (FULL_LEN*4 - 1)
+ CHAIN_LEN * (1 + CHAIN_LEN*2) + CHAIN_LEN * (1 + CHAIN_LEN*2)
+ 12 + 32); + 12 + 32 + 7);
test_trivial(); test_trivial();
test_parallel(); test_parallel();
...@@ -232,6 +251,7 @@ int main(void) ...@@ -232,6 +251,7 @@ int main(void)
test_chain(); test_chain();
test_error(); test_error();
test_traversal1(); test_traversal1();
test_shortcut1();
return exit_status(); return exit_status();
} }
#include "config.h"
#include <assert.h>
#include <ccan/container_of/container_of.h>
#include <ccan/ptrint/ptrint.h>
#include <ccan/agar/agar.h>
#include "simple-graphr.h"
static const void *shortcut1_first_edge_r(const struct agar_graph *gr,
const void *nr)
{
int ni = ptr2int(nr);
switch (ni) {
case 1:
case 2:
return int2ptr(1);
case 3:
return NULL;
default:
assert(0);
}
}
static const void *shortcut1_next_edge_r(const struct agar_graph *gr,
const void *nr, const void *e)
{
int ni = ptr2int(nr);
int index = ptr2int(e);
switch (ni) {
case 1:
if (index == 1)
return int2ptr(2);
assert(index == 2);
return NULL;
case 2:
assert(index == 1);
return NULL;
default:
assert(0);
}
}
static int shortcut1_edge_info_r(const struct agar_graph *gr,
const void *nr, const void *e,
struct agar_edge_info *eir)
{
int ni = ptr2int(nr);
int index = ptr2int(e);
switch (ni) {
case 1:
if (index == 1) {
eir->to = int2ptr(3);
eir->icost = 3;
} else {
assert(index == 2);
eir->to = int2ptr(2);
}
break;
case 2:
assert(index == 1);
eir->to = int2ptr(3);
break;
default:
assert(0);
}
return 0;
}
void shortcut1_graphr_init(struct shortcut1_graphr *s1gr)
{
agar_init_graph(&s1gr->gr, shortcut1_first_edge_r,
shortcut1_next_edge_r,
shortcut1_edge_info_r);
}
...@@ -199,4 +199,24 @@ static const struct adjacency_listr traversal1_adjacency[] = { ...@@ -199,4 +199,24 @@ static const struct adjacency_listr traversal1_adjacency[] = {
{}, {},
}; };
/* Shortcut-1 graph
*
* A ---- (3) -----> C
* \ /
* (1)-> B --(1)
*
* This provides an example of a graph where the lowest cost path from
* (A) to (C) is not the path with the smallest number od edges.
*/
struct shortcut1_graphr {
struct agar_graph gr;
};
void shortcut1_graphr_init(struct shortcut1_graphr *s1gr);
static const struct adjacency_listr shortcut1_adjacencyr[] = {
{1, {3, 2}},
{2, {3}},
{3, {}},
{},
};
#endif /* _SIMPLE_GRAPHR_H */ #endif /* _SIMPLE_GRAPHR_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment