1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
|
/*
* $Id: network_cable.c,v 1.2 2009-06-15 08:41:40 vrsieh Exp $
*
* Copyright (C) 2009 FAUmachine Team <info@faumachine.org>.
* This program is free software. You can redistribute it and/or modify it
* under the terms of the GNU General Public License, either version 2 of
* the License, or (at your option) any later version. See COPYING.
*/
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "random100.h"
#include "sig_integer.h"
#include "network_cable.h"
#define COMP_(x) network_cable_ ## x
struct cpssp {
unsigned int packet_loss;
struct sig_eth *port_end0;
struct sig_eth *port_end1;
};
static void
COMP_(packet_loss_set)(void *_cpssp, int val)
{
struct cpssp *cpssp = (struct cpssp *) _cpssp;
cpssp->packet_loss = val;
}
static void
COMP_(connect)(void *_cpssp, const char *port, void *_sig)
{
static const struct sig_integer_funcs packet_loss_funcs = {
.set = COMP_(packet_loss_set),
};
struct cpssp *cpssp = (struct cpssp *) _cpssp;
struct sig_integer *sig = (struct sig_integer *) _sig;
if (strcmp(port, "packet_loss") == 0) {
sig_integer_connect_in(sig, cpssp, &packet_loss_funcs);
} else {
assert(0); /* FIXME */
}
}
static void
COMP_(disconnect)(void *_cpssp, const char *port)
{
// struct cpssp *cpssp = (struct cpssp *) _cpssp;
/* FIXME */
if (strcmp(port, "packet_loss") == 0) {
// sig_integer_disconnect_in(sig, cpssp);
} else {
assert(0); /* FIXME */
}
}
static void
COMP_(recv0)(void *_cpssp, const void *buf, unsigned int buflen)
{
struct cpssp *cpssp = (struct cpssp *) _cpssp;
if (random100() < cpssp->packet_loss) {
/* Packet lost due to fault injection. */
return;
}
sig_eth_send(cpssp->port_end1, cpssp, buf, buflen);
}
static void
COMP_(recv1)(void *_cpssp, const void *buf, unsigned int buflen)
{
struct cpssp *cpssp = (struct cpssp *) _cpssp;
if (random100() < cpssp->packet_loss) {
/* Packet lost due to fault injection. */
return;
}
sig_eth_send(cpssp->port_end0, cpssp, buf, buflen);
}
void *
COMP_(create)(
const char *name,
struct sig_manage *manage,
struct sig_eth *port_end0,
struct sig_eth *port_end1
)
{
static const struct sig_manage_funcs manage_funcs = {
.connect = COMP_(connect),
.disconnect = COMP_(disconnect),
};
static const struct sig_eth_funcs end0_funcs = {
.recv = COMP_(recv0),
};
static const struct sig_eth_funcs end1_funcs = {
.recv = COMP_(recv1),
};
struct cpssp *cpssp;
cpssp = malloc(sizeof(*cpssp));
assert(cpssp);
cpssp->packet_loss = 0;
/* Call */
sig_manage_connect(manage, cpssp, &manage_funcs);
cpssp->port_end0 = port_end0;
sig_eth_connect(port_end0, cpssp, &end0_funcs);
cpssp->port_end1 = port_end1;
sig_eth_connect(port_end1, cpssp, &end1_funcs);
/* Out */
/* In */
return cpssp;
}
void
COMP_(destroy)(void *_cpssp)
{
struct cpssp *cpssp = _cpssp;
/* FIXME */
free(cpssp);
}
|