1
0
Fork 0
mirror of https://git.rwth-aachen.de/acs/public/villas/node/ synced 2025-03-16 00:00:02 +01:00
VILLASnode/server/src/node.c

145 lines
2.6 KiB
C
Raw Normal View History

/** Nodes.
*
* @author Steffen Vogel <stvogel@eonerc.rwth-aachen.de>
* @copyright 2014, Institute for Automation of Complex Power Systems, EONERC
*/
#include <string.h>
2014-12-05 12:39:52 +01:00
#include "node.h"
#include "cfg.h"
#include "utils.h"
2014-12-05 12:39:52 +01:00
/* Node types */
#include "socket.h"
#include "gtfpga.h"
#ifdef ENABLE_OPAL_ASYNC
2014-12-05 12:39:52 +01:00
#include "opal.h"
#endif
2014-12-05 12:39:52 +01:00
#define VTABLE(type, name, fnc) { type, name, config_parse_ ## fnc, \
fnc ## _print, \
fnc ## _open, \
fnc ## _close, \
fnc ## _read, \
fnc ## _write }
/** Vtable for virtual node sub types */
static const struct node_vtable vtables[] = {
#ifdef ENABLE_OPAL_ASYNC
VTABLE(OPAL_ASYNC, "opal", opal),
#endif
2015-03-31 13:28:11 +02:00
VTABLE(LOG_FILE), "file", file),
2014-12-05 12:39:52 +01:00
VTABLE(IEEE_802_3, "ieee802.3", socket),
VTABLE(IP, "ip", socket),
VTABLE(UDP, "udp", socket),
VTABLE(TCP, "tcp", socket),
VTABLE(TCPD, "tcpd", socket)
2014-12-05 12:39:52 +01:00
};
/** Linked list of nodes. */
struct list nodes;
2014-12-05 12:39:52 +01:00
struct node * node_lookup_name(const char *str, struct list *nodes)
{
FOREACH(nodes, it) {
if (!strcmp(str, it->node->name))
return it->node;
2014-12-05 12:39:52 +01:00
}
2014-12-05 12:39:52 +01:00
return NULL;
}
2014-12-05 12:39:52 +01:00
struct node_vtable const * node_lookup_vtable(const char *str)
{
2014-12-05 12:39:52 +01:00
for (int i = 0; i < ARRAY_LEN(vtables); i++) {
if (!strcmp(vtables[i].name, str))
return &vtables[i];
}
2014-12-05 12:39:52 +01:00
return NULL;
}
2014-12-05 12:39:52 +01:00
int node_start(struct node *n)
2015-03-21 15:29:00 +01:00
{ INDENT
2015-03-18 16:16:44 +01:00
if (!n->refcnt) {
warn("Node '%s' is unused. Skipping...", n->name);
return -1;
2015-03-18 16:16:44 +01:00
}
2014-12-05 12:39:52 +01:00
char str[256];
node_print(n, str, sizeof(str));
debug(1, "Starting node '%s' of type '%s' (%s)", n->name, n->vt->name, str);
{ INDENT
return n->vt->open(n);
}
2014-12-05 12:39:52 +01:00
}
2014-12-05 12:39:52 +01:00
int node_start_defer(struct node *n)
{
int ret;
2014-12-05 12:39:52 +01:00
if (node_type(n) == TCPD) {
info("Wait for incoming TCP connection from node '%s'...", n->name);
ret = listen(n->socket->sd2, 1);
if (ret < 0)
serror("Failed to listen on socket for node '%s'", n->name);
ret = accept(n->socket->sd2, NULL, NULL);
if (ret < 0)
serror("Failed to accept on socket for node '%s'", n->name);
n->socket->sd = ret;
2014-12-05 12:39:52 +01:00
}
return 0;
}
int node_stop(struct node *n)
2015-03-21 15:29:00 +01:00
{ INDENT
2014-12-05 12:39:52 +01:00
int ret;
info("Stopping node '%s'", n->name);
{ INDENT
ret = n->vt->close(n);
}
return ret;
}
2015-03-21 15:29:00 +01:00
void node_reverse(struct node *n)
{
switch (n->vt->type) {
case IEEE_802_3:
case IP:
case UDP:
case TCP:
SWAP(n->socket->remote, n->socket->local);
break;
default: { }
}
2015-03-21 15:29:00 +01:00
}
struct node * node_create()
{
return alloc(sizeof(struct node));
}
void node_destroy(struct node *n)
{
switch (n->vt->type) {
case IEEE_802_3:
case IP:
case UDP:
case TCP:
free(n->socket->netem);
default: { }
}
free(n->socket);
free(n);
}