#include <atf-c.h>
#include <errno.h>
#include <stdio.h>
#include <net/ethernet.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <netinet/ip6.h>
#include "util.h"
#include <netgraph/ng_bridge.h>
static void get_tablesize(char const *source, struct ng_mesg *msg, void *ctx);
struct gettable
{
u_int32_t tok;
int cnt;
};
struct frame4
{
struct ether_header eh;
struct ip ip;
char data[64];
};
struct frame6
{
struct ether_header eh;
struct ip6_hdr ip;
char data[64];
};
static struct frame4 msg4 = {
.ip.ip_v = 4,
.ip.ip_hl = 5,
.ip.ip_ttl = 1,
.ip.ip_p = 254,
.ip.ip_src = {htonl(0x0a00dead)},
.ip.ip_dst = {htonl(0x0a00beef)},
.ip.ip_len = 32,
.eh.ether_type = ETHERTYPE_IP,
.eh.ether_shost = {2, 4, 6},
.eh.ether_dhost = {2, 4, 6},
};
ATF_TC(basic);
ATF_TC_HEAD(basic, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(basic, dummy)
{
ng_counter_t r;
struct gettable rm;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_errors(FAIL);
ng_mkpeer(".", "a", "bridge", "link0");
ng_name("a", "bridge");
ng_connect(".", "b", "bridge:", "link1");
ng_connect(".", "c", "bridge:", "link2");
ng_register_data("a", get_data0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0);
ng_register_data("b", get_data1);
ng_register_data("c", get_data2);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 2;
ng_send_data("b", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 3;
ng_send_data("c", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 1 && r[2] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[5] = 3;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 0 && r[2] == 1);
ng_register_msg(get_tablesize);
rm.tok = ng_send_msg("bridge:", "gettable");
rm.cnt = 0;
ng_handle_events(50, &rm);
ATF_CHECK(rm.cnt == 3);
ng_rmhook(".", "b");
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[5] = 0;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 0 && r[2] == 1);
ng_register_msg(get_tablesize);
rm.tok = ng_send_msg("bridge:", "gettable");
rm.cnt = 0;
ng_handle_events(50, &rm);
ATF_CHECK(rm.cnt == 2);
ng_shutdown("bridge:");
}
ATF_TC(persistence);
ATF_TC_HEAD(persistence, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(persistence, dummy)
{
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_errors(FAIL);
ng_mkpeer(".", "a", "bridge", "link0");
ng_name("a", "bridge");
ng_send_msg("bridge:", "setpersistent");
ng_rmhook(".", "a");
ng_shutdown("bridge:");
}
ATF_TC(loop);
ATF_TC_HEAD(loop, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(loop, dummy)
{
ng_counter_t r;
int i;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge1:");
ng_shutdown("bridge2:");
ng_errors(FAIL);
ng_mkpeer(".", "a", "bridge", "link0");
ng_name("a", "bridge1");
ng_mkpeer(".", "b", "bridge", "link1");
ng_name("b", "bridge2");
ng_register_data("a", get_data0);
ng_register_data("b", get_data1);
ng_connect("bridge1:", "link11", "bridge2:", "link11");
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1);
ng_connect("bridge1:", "link12", "bridge2:", "link12");
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_errors(PASS);
ng_send_data("a", &msg4, sizeof(msg4));
ATF_CHECK_ERRNO(ELOOP, errno != 0);
ng_errors(FAIL);
for (i = 0; i < 10; i++)
if (!ng_handle_event(50, &r))
break;
ATF_CHECK(r[0] == 0 && r[1] == 1);
ng_shutdown("bridge1:");
ng_shutdown("bridge2:");
}
ATF_TC(many_unicasts);
ATF_TC_HEAD(many_unicasts, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(many_unicasts, dummy)
{
ng_counter_t r;
int i;
const int HOOKS = 1000;
struct gettable rm;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_errors(FAIL);
ng_mkpeer(".", "a", "bridge", "link0");
ng_name("a", "bridge");
ng_register_data("a", get_data0);
ng_counter_clear(r);
msg4.eh.ether_shost[3] = 0xff;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0);
msg4.eh.ether_shost[3] = 0;
msg4.eh.ether_dhost[3] = 0xff;
ng_counter_clear(r);
for (i = 1; i <= HOOKS; i++)
{
char hook[20];
snprintf(hook, sizeof(hook), "link%d", i);
ng_connect(".", hook, "bridge:", hook);
ng_register_data(hook, get_data2);
msg4.eh.ether_shost[4] = i >> 8;
msg4.eh.ether_shost[5] = i & 0xff;
ng_errors(PASS);
ng_send_data(hook, &msg4, sizeof(msg4));
ng_errors(FAIL);
if (errno != 0)
break;
ng_handle_events(50, &r);
}
ATF_CHECK(r[0] == HOOKS && r[2] == 0);
ng_register_msg(get_tablesize);
rm.cnt = 0;
ng_errors(PASS);
rm.tok = ng_send_msg("bridge:", "gettable");
ng_errors(FAIL);
if (rm.tok == (u_int32_t)-1)
{
ATF_CHECK_ERRNO(ENOBUFS, 1);
atf_tc_expect_fail("response too large");
}
ng_handle_events(50, &rm);
ATF_CHECK(rm.cnt == HOOKS + 1);
atf_tc_expect_pass();
ng_shutdown("bridge:");
}
ATF_TC(many_broadcasts);
ATF_TC_HEAD(many_broadcasts, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(many_broadcasts, dummy)
{
ng_counter_t r;
int i;
const int HOOKS = 1000;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_errors(FAIL);
ng_mkpeer(".", "a", "bridge", "link0");
ng_name("a", "bridge");
ng_register_data("a", get_data0);
ng_counter_clear(r);
msg4.eh.ether_shost[3] = 0xff;
ng_send_data("a", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0);
msg4.eh.ether_shost[3] = 0;
memset(msg4.eh.ether_dhost, 0xff, sizeof(msg4.eh.ether_dhost));
ng_counter_clear(r);
for (i = 1; i <= HOOKS; i++)
{
char hook[20];
snprintf(hook, sizeof(hook), "link%d", i);
ng_connect(".", hook, "bridge:", hook);
ng_register_data(hook, get_data3);
msg4.eh.ether_shost[4] = i >> 8;
msg4.eh.ether_shost[5] = i & 0xff;
ng_errors(PASS);
ng_send_data(hook, &msg4, sizeof(msg4));
ng_errors(FAIL);
if (errno != 0)
break;
ng_handle_events(50, &r);
}
ATF_CHECK(r[0] > 100 && r[3] > 100);
if (i < HOOKS)
atf_tc_expect_fail("netgraph queue full (%d)", i);
ATF_CHECK(r[0] == HOOKS);
atf_tc_expect_pass();
ng_shutdown("bridge:");
}
ATF_TC(uplink_private);
ATF_TC_HEAD(uplink_private, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(uplink_private, dummy)
{
ng_counter_t r;
struct gettable rm;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_mkpeer(".", "u1", "bridge", "uplink1");
if (errno > 0)
atf_tc_skip("uplinks are not supported.");
ng_errors(FAIL);
ng_name("u1", "bridge");
ng_register_data("u1", get_data1);
ng_connect(".", "u2", "bridge:", "uplink2");
ng_register_data("u2", get_data2);
ng_connect(".", "l0", "bridge:", "link0");
ng_register_data("l0", get_data0);
ng_connect(".", "l3", "bridge:", "link3");
ng_register_data("l3", get_data3);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 0 && r[2] == 1 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
msg4.eh.ether_dhost[5] = 2;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 2;
msg4.eh.ether_dhost[5] = 0;
ng_send_data("u2", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 0 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 3;
msg4.eh.ether_dhost[5] = 0;
ng_send_data("l3", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 0 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[5] = 2;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 0 && r[2] == 1 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[0] = 0xff;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
memset(msg4.eh.ether_dhost, 0xff, sizeof(msg4.eh.ether_dhost));
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_register_msg(get_tablesize);
rm.tok = ng_send_msg("bridge:", "gettable");
rm.cnt = 0;
ng_handle_events(50, &rm);
ATF_CHECK(rm.cnt == 2);
ng_shutdown("bridge:");
}
ATF_TC(uplink_classic);
ATF_TC_HEAD(uplink_classic, conf)
{
atf_tc_set_md_var(conf, "require.user", "root");
}
ATF_TC_BODY(uplink_classic, dummy)
{
ng_counter_t r;
struct gettable rm;
ng_init();
ng_errors(PASS);
ng_shutdown("bridge:");
ng_mkpeer(".", "l0", "bridge", "link0");
if (errno > 0)
atf_tc_skip("uplinks are not supported.");
ng_errors(FAIL);
ng_name("l0", "bridge");
ng_register_data("l0", get_data0);
ng_connect(".", "u1", "bridge:", "uplink1");
ng_register_data("u1", get_data1);
ng_connect(".", "u2", "bridge:", "uplink2");
ng_register_data("u2", get_data2);
ng_connect(".", "l3", "bridge:", "link3");
ng_register_data("l3", get_data3);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
msg4.eh.ether_dhost[5] = 2;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 2;
msg4.eh.ether_dhost[5] = 0;
ng_send_data("u2", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 0 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 3;
msg4.eh.ether_dhost[5] = 0;
ng_send_data("l3", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 0 && r[3] == 0);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[5] = 2;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
msg4.eh.ether_dhost[0] = 0xff;
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 1;
memset(msg4.eh.ether_dhost, 0xff, sizeof(msg4.eh.ether_dhost));
ng_send_data("u1", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 1 && r[1] == 0 && r[2] == 1 && r[3] == 1);
ng_counter_clear(r);
msg4.eh.ether_shost[5] = 0;
ng_send_data("l0", &msg4, sizeof(msg4));
ng_handle_events(50, &r);
ATF_CHECK(r[0] == 0 && r[1] == 1 && r[2] == 1 && r[3] == 1);
ng_register_msg(get_tablesize);
rm.tok = ng_send_msg("bridge:", "gettable");
rm.cnt = 0;
ng_handle_events(50, &rm);
ATF_CHECK(rm.cnt == 2);
ng_shutdown("bridge:");
}
ATF_TP_ADD_TCS(bridge)
{
ATF_TP_ADD_TC(bridge, basic);
ATF_TP_ADD_TC(bridge, loop);
ATF_TP_ADD_TC(bridge, persistence);
ATF_TP_ADD_TC(bridge, many_unicasts);
ATF_TP_ADD_TC(bridge, many_broadcasts);
ATF_TP_ADD_TC(bridge, uplink_private);
ATF_TP_ADD_TC(bridge, uplink_classic);
return atf_no_error();
}
static void
get_tablesize(char const *source, struct ng_mesg *msg, void *ctx)
{
struct gettable *rm = ctx;
struct ng_bridge_host_ary *gt = (void *)msg->data;
fprintf(stderr, "Response from %s to query %d\n", source, msg->header.token);
if (rm->tok == msg->header.token)
rm->cnt = gt->numHosts;
}