2014-06-05 09:34:29 +00:00
|
|
|
/**
|
|
|
|
* Some helper functions
|
|
|
|
*
|
|
|
|
* @author Steffen Vogel <stvogel@eonerc.rwth-aachen.de>
|
|
|
|
* @copyright 2014, Institute for Automation of Complex Power Systems, EONERC
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2014-06-05 09:34:56 +00:00
|
|
|
#include <string.h>
|
2014-06-05 09:34:29 +00:00
|
|
|
#include <time.h>
|
2014-06-05 09:34:56 +00:00
|
|
|
#include <errno.h>
|
2014-06-05 09:35:23 +00:00
|
|
|
#include <unistd.h>
|
2014-06-05 09:34:56 +00:00
|
|
|
#include <netdb.h>
|
2014-06-05 09:35:23 +00:00
|
|
|
|
2014-06-05 09:34:56 +00:00
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
2014-06-05 09:34:29 +00:00
|
|
|
|
|
|
|
#include "config.h"
|
2014-06-05 09:35:23 +00:00
|
|
|
#include "cfg.h"
|
2014-06-05 09:34:29 +00:00
|
|
|
#include "utils.h"
|
|
|
|
|
|
|
|
static const char *log_prefix[] = {
|
|
|
|
"Debug",
|
|
|
|
"Info",
|
|
|
|
"Warning",
|
2014-06-05 09:34:46 +00:00
|
|
|
"Error"
|
2014-06-05 09:34:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void print(enum log_level lvl, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
struct timespec ts;
|
|
|
|
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
|
|
|
clock_gettime(CLOCK_REALTIME, &ts);
|
|
|
|
|
2014-06-05 09:34:48 +00:00
|
|
|
printf("%17.6f [%-7s] ", ts.tv_sec + ts.tv_nsec / 1e9, log_prefix[lvl]);
|
2014-06-05 09:34:29 +00:00
|
|
|
vprintf(fmt, ap);
|
|
|
|
printf("\n");
|
|
|
|
|
|
|
|
va_end(ap);
|
2014-06-05 09:34:56 +00:00
|
|
|
}
|
|
|
|
|
2014-06-05 09:35:33 +00:00
|
|
|
int resolve_addr(const char *addr, struct sockaddr_in *sa, int flags)
|
2014-06-05 09:34:56 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
char *node;
|
|
|
|
char *service;
|
|
|
|
|
|
|
|
/* Split addr into node:service */
|
|
|
|
if (sscanf(addr, "%m[^:]:%ms", &node, &service) != 2)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Check for wildcards */
|
|
|
|
if (!strcmp(node, "*")) {
|
|
|
|
free(node);
|
|
|
|
node = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcmp(service, "*")) {
|
|
|
|
free(service);
|
|
|
|
service = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get IP */
|
|
|
|
struct addrinfo *result;
|
|
|
|
struct addrinfo hint = {
|
|
|
|
.ai_flags = flags,
|
|
|
|
.ai_family = AF_INET,
|
|
|
|
.ai_socktype = SOCK_DGRAM,
|
|
|
|
.ai_protocol = 0
|
|
|
|
};
|
|
|
|
|
2014-06-25 01:53:37 +00:00
|
|
|
ret = getaddrinfo(node, service, &hint, &result);
|
|
|
|
if (ret)
|
2014-06-05 09:34:56 +00:00
|
|
|
error("Failed to lookup address: %s", gai_strerror(ret));
|
|
|
|
|
|
|
|
memcpy(sa, result->ai_addr, result->ai_addrlen);
|
|
|
|
|
|
|
|
free(node);
|
|
|
|
free(service);
|
|
|
|
freeaddrinfo(result);
|
2014-06-05 09:34:29 +00:00
|
|
|
|
2014-06-05 09:34:56 +00:00
|
|
|
return 0;
|
2014-06-05 09:34:29 +00:00
|
|
|
}
|
2014-06-05 09:35:23 +00:00
|
|
|
|
2014-06-05 09:35:38 +00:00
|
|
|
int sockaddr_cmp(struct sockaddr *a, struct sockaddr *b)
|
|
|
|
{
|
|
|
|
if (a->sa_family == b->sa_family) {
|
|
|
|
switch (a->sa_family) {
|
|
|
|
case AF_INET: {
|
|
|
|
struct sockaddr_in *ai = (struct sockaddr_in *) a;
|
|
|
|
struct sockaddr_in *bi = (struct sockaddr_in *) b;
|
|
|
|
|
|
|
|
return memcmp(&ai->sin_addr, &bi->sin_addr, sizeof(struct in_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
case AF_INET6: {
|
|
|
|
struct sockaddr_in6 *ai = (struct sockaddr_in6 *) a;
|
|
|
|
struct sockaddr_in6 *bi = (struct sockaddr_in6 *) b;
|
|
|
|
|
|
|
|
return memcmp(&ai->sin6_addr, &bi->sin6_addr, sizeof(struct in6_addr));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-06-10 19:44:22 +00:00
|
|
|
cpu_set_t to_cpu_set_t(int set)
|
2014-06-05 09:35:34 +00:00
|
|
|
{
|
|
|
|
cpu_set_t cset;
|
|
|
|
|
|
|
|
for (int i = 0; i < sizeof(int) * 8; i++) {
|
|
|
|
if (set & (1L << i))
|
|
|
|
CPU_SET(i, &cset);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cset;
|
|
|
|
}
|