You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
3.1 KiB
163 lines
3.1 KiB
/* Copyright 2016 Alex 'AdUser' Z (ad_user@runbox.com) |
|
* |
|
* This program is free software; you can redistribute it and/or modify |
|
* it under the terms of the GNU General Public License version 2 as |
|
* published by the Free Software Foundation. |
|
*/ |
|
#include "source.h" |
|
|
|
#include <ctype.h> |
|
#include <sys/socket.h> |
|
#include <netinet/in.h> |
|
#include <arpa/inet.h> |
|
|
|
#define HOST_MAX 48 |
|
#define PORT_MAX 6 |
|
|
|
typedef struct f2b_port_t { |
|
struct f2b_port_t *next; |
|
char host[HOST_MAX]; |
|
char port[PORT_MAX]; |
|
int fd; |
|
} f2b_port_t; |
|
|
|
struct _config { |
|
char name[32]; |
|
char error[256]; |
|
void (*errcb)(char *errstr); |
|
f2b_port_t *ports; |
|
f2b_port_t *current; |
|
}; |
|
|
|
static bool |
|
try_parse_listen_opt(f2b_port_t *port, const char *value) { |
|
char buf[256]; |
|
char *p; |
|
|
|
strlcpy(buf, value, sizeof(buf)); |
|
if (*buf == '[') { |
|
/* IPv6, expected: [XXXX::XXXX:XXXX]:YYYY */ |
|
if ((p = strstr(buf, "]:")) == NULL) { |
|
*p = '\0', p += 2; |
|
strlcpy(port->port, p, sizeof(port->port)); |
|
p = buf + 1; |
|
strlcpy(port->host, p, sizeof(port->host)); |
|
return true; |
|
} |
|
return false; /* can't find port */ |
|
} |
|
if ((p = strchr(buf, ':')) != NULL) { |
|
/* IPv4, expected: XX.XX.XX.XX:YYYY */ |
|
*p = '\0', p += 1; |
|
strlcpy(port->port, p, sizeof(port->port)); |
|
p = buf; |
|
strlcpy(port->host, p, sizeof(port->host)); |
|
return true; |
|
} |
|
if (isdigit(*buf) && strlen(buf) <= 5) { |
|
/* IPv4, expected: YYYY */ |
|
strlcpy(port->host, "0.0.0.0", sizeof(port->host)); |
|
strlcpy(port->port, buf, sizeof(port->port)); |
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
static void |
|
errcb_stub(char *str) { |
|
assert(str != NULL); |
|
(void)(str); |
|
} |
|
|
|
cfg_t * |
|
create(const char *init) { |
|
cfg_t *cfg = NULL; |
|
assert(init != NULL); |
|
if ((cfg = calloc(1, sizeof(cfg_t))) == NULL) |
|
return NULL; |
|
strlcpy(cfg->name, init, sizeof(cfg->name)); |
|
cfg->errcb = &errcb_stub; |
|
return cfg; |
|
} |
|
|
|
bool |
|
config(cfg_t *cfg, const char *key, const char *value) { |
|
assert(cfg != NULL); |
|
assert(key != NULL); |
|
assert(value != NULL); |
|
|
|
if (strcmp(key, "listen") == 0) { |
|
f2b_port_t *port = NULL; |
|
if ((port = calloc(1, sizeof(f2b_port_t))) == NULL) { |
|
strlcpy(cfg->error, "out of memory", sizeof(cfg->error)); |
|
return false; |
|
} |
|
if (try_parse_listen_opt(port, value) == false) { |
|
snprintf(cfg->error, sizeof(cfg->error), "can't parse: %s", value); |
|
free(port); |
|
return false; |
|
} |
|
port->next = cfg->ports; |
|
cfg->ports = port; |
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
bool |
|
ready(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
if (cfg->ports) |
|
return true; |
|
return false; |
|
} |
|
|
|
char * |
|
error(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
return cfg->error; |
|
} |
|
|
|
void |
|
errcb(cfg_t *cfg, void (*cb)(char *errstr)) { |
|
assert(cfg != NULL); |
|
assert(cb != NULL); |
|
|
|
cfg->errcb = cb; |
|
} |
|
|
|
bool |
|
start(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
/* TODO */ |
|
return true; |
|
} |
|
|
|
bool |
|
stop(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
/* TODO */ |
|
return true; |
|
} |
|
|
|
bool |
|
next(cfg_t *cfg, char *buf, size_t bufsize, bool reset) { |
|
assert(cfg != NULL); |
|
assert(buf != NULL); |
|
assert(bufsize > 0); |
|
|
|
/* TODO */ |
|
return false; |
|
} |
|
|
|
void |
|
destroy(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
free(cfg); |
|
}
|
|
|