2014-11-12 11:43:55 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2014 ccminer team
|
|
|
|
*
|
|
|
|
* Implementation by tpruvot (based on cgminer)
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the Free
|
|
|
|
* Software Foundation; either version 2 of the License, or (at your option)
|
|
|
|
* any later version. See COPYING for more details.
|
|
|
|
*/
|
|
|
|
#define APIVERSION "1.0"
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
# define _WINSOCK_DEPRECATED_NO_WARNINGS
|
|
|
|
# include <winsock2.h>
|
|
|
|
# include <mstcpip.h>
|
|
|
|
#endif
|
|
|
|
|
2014-11-12 11:43:55 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
#include "compat.h"
|
|
|
|
#include "miner.h"
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
#ifndef _MSC_VER
|
2014-11-12 11:43:55 +01:00
|
|
|
# include <errno.h>
|
|
|
|
# include <sys/socket.h>
|
|
|
|
# include <netinet/in.h>
|
|
|
|
# include <arpa/inet.h>
|
|
|
|
# include <netdb.h>
|
|
|
|
# define SOCKETTYPE long
|
|
|
|
# define SOCKETFAIL(a) ((a) < 0)
|
|
|
|
# define INVSOCK -1 /* INVALID_SOCKET */
|
|
|
|
# define INVINETADDR -1 /* INADDR_NONE */
|
|
|
|
# define CLOSESOCKET close
|
|
|
|
# define SOCKETINIT {}
|
|
|
|
# define SOCKERRMSG strerror(errno)
|
|
|
|
#else
|
|
|
|
# define SOCKETTYPE SOCKET
|
|
|
|
# define SOCKETFAIL(a) ((a) == SOCKET_ERROR)
|
|
|
|
# define INVSOCK INVALID_SOCKET
|
|
|
|
# define INVINETADDR INADDR_NONE
|
|
|
|
# define CLOSESOCKET closesocket
|
2014-11-12 12:53:30 +01:00
|
|
|
#define in_addr_t uint32_t
|
2014-11-12 11:43:55 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define GROUP(g) (toupper(g))
|
|
|
|
#define PRIVGROUP GROUP('W')
|
|
|
|
#define NOPRIVGROUP GROUP('R')
|
|
|
|
#define ISPRIVGROUP(g) (GROUP(g) == PRIVGROUP)
|
|
|
|
#define GROUPOFFSET(g) (GROUP(g) - GROUP('A'))
|
|
|
|
#define VALIDGROUP(g) (GROUP(g) >= GROUP('A') && GROUP(g) <= GROUP('Z'))
|
|
|
|
#define COMMANDS(g) (apigroups[GROUPOFFSET(g)].commands)
|
|
|
|
#define DEFINEDGROUP(g) (ISPRIVGROUP(g) || COMMANDS(g) != NULL)
|
|
|
|
struct APIGROUPS {
|
|
|
|
// This becomes a string like: "|cmd1|cmd2|cmd3|" so it's quick to search
|
|
|
|
char *commands;
|
|
|
|
} apigroups['Z' - 'A' + 1]; // only A=0 to Z=25 (R: noprivs, W: allprivs)
|
|
|
|
|
|
|
|
struct IP4ACCESS {
|
|
|
|
in_addr_t ip;
|
|
|
|
in_addr_t mask;
|
|
|
|
char group;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int ips = 1;
|
|
|
|
static struct IP4ACCESS *ipaccess = NULL;
|
|
|
|
|
|
|
|
// Big enough for largest API request
|
|
|
|
// though a PC with 100s of CPUs may exceed the size ...
|
|
|
|
// Current code assumes it can socket send this size also
|
|
|
|
#define MYBUFSIZ 16384
|
|
|
|
|
|
|
|
// Socket is on 127.0.0.1
|
|
|
|
#define QUEUE 10
|
|
|
|
|
|
|
|
#define LOCAL_ADDR_V4 "127.0.0.1"
|
|
|
|
static const char *localaddr = LOCAL_ADDR_V4;
|
|
|
|
static const char *UNAVAILABLE = " - API will not be available";
|
|
|
|
static char *buffer = NULL;
|
2014-11-12 15:26:25 +01:00
|
|
|
static time_t startup = 0;
|
2014-11-12 11:43:55 +01:00
|
|
|
static int bye = 0;
|
|
|
|
|
|
|
|
extern int opt_intensity;
|
|
|
|
extern int opt_n_threads;
|
|
|
|
extern int opt_api_listen;
|
|
|
|
extern uint64_t global_hashrate;
|
|
|
|
extern uint32_t accepted_count;
|
|
|
|
extern uint32_t rejected_count;
|
|
|
|
|
|
|
|
char *opt_api_allow = LOCAL_ADDR_V4;
|
|
|
|
int opt_api_network = 1;
|
|
|
|
#define gpu_threads opt_n_threads
|
|
|
|
|
|
|
|
extern void get_currentalgo(char* buf, int sz);
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
static void gpustatus(int thr_id)
|
|
|
|
{
|
2014-11-12 15:26:25 +01:00
|
|
|
char buf[MYBUFSIZ];
|
2014-11-12 11:43:55 +01:00
|
|
|
float gt;
|
|
|
|
int gf, gp;
|
|
|
|
|
|
|
|
if (thr_id >= 0 && thr_id < gpu_threads) {
|
|
|
|
struct cgpu_info *cgpu = &thr_info[thr_id].gpu;
|
|
|
|
|
|
|
|
#ifdef HAVE_HWMONITORING
|
|
|
|
// todo
|
|
|
|
if (gpu->has_monitoring) {
|
|
|
|
gt = gpu_temp(gpu);
|
|
|
|
gf = gpu_fanspeed(gpu);
|
|
|
|
gp = gpu_fanpercent(gpu);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
2014-11-12 12:53:30 +01:00
|
|
|
{
|
|
|
|
gt = 0.0; gf = gp = 0;
|
|
|
|
}
|
2014-11-12 11:43:55 +01:00
|
|
|
|
|
|
|
// todo: can be 0 if set by algo (auto)
|
|
|
|
if (opt_intensity == 0 && opt_work_size) {
|
|
|
|
int i = 0;
|
|
|
|
uint32_t ws = opt_work_size;
|
|
|
|
while (ws > 1 && i++ < 32)
|
|
|
|
ws = ws >> 1;
|
|
|
|
cgpu->intensity = i;
|
|
|
|
} else {
|
|
|
|
cgpu->intensity = opt_intensity;
|
|
|
|
}
|
|
|
|
|
|
|
|
// todo: per gpu
|
|
|
|
cgpu->accepted = accepted_count;
|
|
|
|
cgpu->rejected = rejected_count;
|
|
|
|
|
|
|
|
cgpu->khashes = stats_get_speed(thr_id) / 1000.0;
|
|
|
|
|
|
|
|
sprintf(buf, "GPU=%d;TEMP=%.1f;FAN=%d;FANP=%d;KHS=%.2f;"
|
2014-11-12 15:26:25 +01:00
|
|
|
"HWF=%d;I=%d|",
|
2014-11-12 11:43:55 +01:00
|
|
|
thr_id, gt, gf, gp, cgpu->khashes,
|
2014-11-12 15:26:25 +01:00
|
|
|
cgpu->hw_errors, cgpu->intensity);
|
2014-11-12 11:43:55 +01:00
|
|
|
|
|
|
|
strcat(buffer, buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
static char *getsummary(char *params)
|
|
|
|
{
|
|
|
|
char algo[64] = "";
|
2014-11-12 15:26:25 +01:00
|
|
|
int uptime = (time(NULL) - startup);
|
|
|
|
double accps = (60.0 * accepted_count) / (uptime ? uptime : 1.0);
|
|
|
|
|
2014-11-12 11:43:55 +01:00
|
|
|
get_currentalgo(algo, sizeof(algo));
|
2014-11-12 15:26:25 +01:00
|
|
|
|
2014-11-12 11:43:55 +01:00
|
|
|
*buffer = '\0';
|
|
|
|
sprintf(buffer, "NAME=%s;VER=%s;API=%s;"
|
2014-11-12 15:26:25 +01:00
|
|
|
"ALGO=%s;KHS=%.2f;ACC=%d;REJ=%d;ACCMN=%.3f;UPTIME=%d|",
|
2014-11-12 11:43:55 +01:00
|
|
|
PACKAGE_NAME, PACKAGE_VERSION, APIVERSION,
|
2014-11-12 15:26:25 +01:00
|
|
|
algo, (double)global_hashrate / 1000.0,
|
|
|
|
accepted_count, rejected_count,
|
|
|
|
accps, uptime);
|
2014-11-12 11:43:55 +01:00
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *getstats(char *params)
|
|
|
|
{
|
|
|
|
*buffer = '\0';
|
|
|
|
for (int i = 0; i < gpu_threads; i++)
|
|
|
|
gpustatus(i);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct CMDS {
|
|
|
|
char *name;
|
|
|
|
char *(*func)(char *);
|
|
|
|
} cmds[] = {
|
|
|
|
{ "summary", getsummary },
|
|
|
|
{ "stats", getstats },
|
|
|
|
};
|
|
|
|
|
|
|
|
#define CMDMAX 2
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
static void send_result(SOCKETTYPE c, char *result)
|
2014-11-12 11:43:55 +01:00
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (result == NULL)
|
|
|
|
result = "";
|
|
|
|
|
|
|
|
// ignore failure - it's closed immediately anyway
|
2014-11-12 12:53:30 +01:00
|
|
|
n = send(c, result, strlen(result) + 1, 0);
|
2014-11-12 11:43:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Interpret [W:]IP[/Prefix][,[R|W:]IP2[/Prefix2][,...]] --api-allow option
|
|
|
|
* special case of 0/0 allows /0 (means all IP addresses)
|
|
|
|
*/
|
|
|
|
#define ALLIP4 "0/0"
|
|
|
|
/*
|
|
|
|
* N.B. IP4 addresses are by Definition 32bit big endian on all platforms
|
|
|
|
*/
|
|
|
|
static void setup_ipaccess()
|
|
|
|
{
|
|
|
|
char *buf, *ptr, *comma, *slash, *dot;
|
|
|
|
int ipcount, mask, octet, i;
|
|
|
|
char group;
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
buf = (char*) calloc(1, strlen(opt_api_allow) + 1);
|
2014-11-12 11:43:55 +01:00
|
|
|
if (unlikely(!buf))
|
|
|
|
proper_exit(1);//, "Failed to malloc ipaccess buf");
|
|
|
|
|
|
|
|
strcpy(buf, opt_api_allow);
|
|
|
|
ipcount = 1;
|
|
|
|
ptr = buf;
|
|
|
|
while (*ptr) if (*(ptr++) == ',')
|
|
|
|
ipcount++;
|
|
|
|
|
|
|
|
// possibly more than needed, but never less
|
2014-11-12 15:26:25 +01:00
|
|
|
ipaccess = (struct IP4ACCESS *) calloc(ipcount, sizeof(struct IP4ACCESS));
|
2014-11-12 11:43:55 +01:00
|
|
|
if (unlikely(!ipaccess))
|
|
|
|
proper_exit(1);//, "Failed to calloc ipaccess");
|
|
|
|
|
|
|
|
ips = 0;
|
|
|
|
ptr = buf;
|
|
|
|
while (ptr && *ptr) {
|
|
|
|
while (*ptr == ' ' || *ptr == '\t')
|
|
|
|
ptr++;
|
|
|
|
|
|
|
|
if (*ptr == ',') {
|
|
|
|
ptr++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
comma = strchr(ptr, ',');
|
|
|
|
if (comma)
|
|
|
|
*(comma++) = '\0';
|
|
|
|
|
|
|
|
group = NOPRIVGROUP;
|
|
|
|
|
|
|
|
if (isalpha(*ptr) && *(ptr+1) == ':') {
|
|
|
|
if (DEFINEDGROUP(*ptr))
|
|
|
|
group = GROUP(*ptr);
|
|
|
|
ptr += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
ipaccess[ips].group = group;
|
|
|
|
|
|
|
|
if (strcmp(ptr, ALLIP4) == 0)
|
|
|
|
ipaccess[ips].ip = ipaccess[ips].mask = 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
slash = strchr(ptr, '/');
|
|
|
|
if (!slash)
|
|
|
|
ipaccess[ips].mask = 0xffffffff;
|
|
|
|
else {
|
|
|
|
*(slash++) = '\0';
|
|
|
|
mask = atoi(slash);
|
|
|
|
if (mask < 1 || mask > 32)
|
|
|
|
goto popipo; // skip invalid/zero
|
|
|
|
|
|
|
|
ipaccess[ips].mask = 0;
|
|
|
|
while (mask-- >= 0) {
|
|
|
|
octet = 1 << (mask % 8);
|
|
|
|
ipaccess[ips].mask |= (octet << (24 - (8 * (mask >> 3))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ipaccess[ips].ip = 0; // missing default to '.0'
|
|
|
|
for (i = 0; ptr && (i < 4); i++) {
|
|
|
|
dot = strchr(ptr, '.');
|
|
|
|
if (dot)
|
|
|
|
*(dot++) = '\0';
|
|
|
|
|
|
|
|
octet = atoi(ptr);
|
|
|
|
if (octet < 0 || octet > 0xff)
|
|
|
|
goto popipo; // skip invalid
|
|
|
|
|
|
|
|
ipaccess[ips].ip |= (octet << (24 - (i * 8)));
|
|
|
|
|
|
|
|
ptr = dot;
|
|
|
|
}
|
|
|
|
|
|
|
|
ipaccess[ips].ip &= ipaccess[ips].mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
ips++;
|
|
|
|
popipo:
|
|
|
|
ptr = comma;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_connect(struct sockaddr_in *cli, char **connectaddr, char *group)
|
|
|
|
{
|
|
|
|
bool addrok = false;
|
|
|
|
|
|
|
|
*connectaddr = inet_ntoa(cli->sin_addr);
|
|
|
|
|
|
|
|
*group = NOPRIVGROUP;
|
|
|
|
if (opt_api_allow) {
|
|
|
|
int client_ip = htonl(cli->sin_addr.s_addr);
|
|
|
|
for (int i = 0; i < ips; i++) {
|
|
|
|
if ((client_ip & ipaccess[i].mask) == ipaccess[i].ip) {
|
|
|
|
addrok = true;
|
|
|
|
*group = ipaccess[i].group;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (opt_api_network)
|
|
|
|
addrok = true;
|
|
|
|
else
|
|
|
|
addrok = (strcmp(*connectaddr, localaddr) == 0);
|
|
|
|
|
|
|
|
return addrok;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void api()
|
|
|
|
{
|
|
|
|
const char *addr = localaddr;
|
|
|
|
short int port = opt_api_listen; // 4068
|
2014-11-12 15:26:25 +01:00
|
|
|
char buf[MYBUFSIZ];
|
2014-11-12 11:43:55 +01:00
|
|
|
int c, n, bound;
|
|
|
|
char *connectaddr;
|
|
|
|
char *binderror;
|
|
|
|
char group;
|
|
|
|
time_t bindstart;
|
|
|
|
struct sockaddr_in serv;
|
|
|
|
struct sockaddr_in cli;
|
|
|
|
socklen_t clisiz;
|
|
|
|
bool addrok = false;
|
|
|
|
long long counter;
|
|
|
|
char *result;
|
|
|
|
char *params;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
SOCKETTYPE *apisock;
|
|
|
|
if (!opt_api_listen && opt_debug) {
|
|
|
|
applog(LOG_DEBUG, "API disabled");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_api_allow) {
|
|
|
|
setup_ipaccess();
|
|
|
|
if (ips == 0) {
|
|
|
|
applog(LOG_WARNING, "API not running (no valid IPs specified)%s", UNAVAILABLE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
apisock = (SOCKETTYPE*) calloc(1, sizeof(*apisock));
|
2014-11-12 11:43:55 +01:00
|
|
|
*apisock = INVSOCK;
|
|
|
|
|
|
|
|
sleep(1);
|
|
|
|
|
|
|
|
*apisock = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (*apisock == INVSOCK) {
|
|
|
|
applog(LOG_ERR, "API initialisation failed (%s)%s", strerror(errno), UNAVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(&serv, 0, sizeof(serv));
|
|
|
|
serv.sin_family = AF_INET;
|
|
|
|
serv.sin_addr.s_addr = inet_addr(localaddr);
|
|
|
|
if (serv.sin_addr.s_addr == (in_addr_t)INVINETADDR) {
|
|
|
|
applog(LOG_ERR, "API initialisation 2 failed (%s)%s", strerror(errno), UNAVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
serv.sin_port = htons(port);
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
// On linux with SO_REUSEADDR, bind will get the port if the previous
|
|
|
|
// socket is closed (even if it is still in TIME_WAIT) but fail if
|
|
|
|
// another program has it open - which is what we want
|
|
|
|
int optval = 1;
|
|
|
|
// If it doesn't work, we don't really care - just show a debug message
|
|
|
|
if (SOCKETFAIL(setsockopt(*apisock, SOL_SOCKET, SO_REUSEADDR, (void *)(&optval), sizeof(optval))))
|
|
|
|
applog(LOG_DEBUG, "API setsockopt SO_REUSEADDR failed (ignored): %s", SOCKERRMSG);
|
|
|
|
#else
|
|
|
|
// On windows a 2nd program can bind to a port>1024 already in use unless
|
|
|
|
// SO_EXCLUSIVEADDRUSE is used - however then the bind to a closed port
|
|
|
|
// in TIME_WAIT will fail until the timeout - so we leave the options alone
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// try for 1 minute ... in case the old one hasn't completely gone yet
|
|
|
|
bound = 0;
|
|
|
|
bindstart = time(NULL);
|
|
|
|
while (bound == 0) {
|
|
|
|
if (bind(*apisock, (struct sockaddr *)(&serv), sizeof(serv)) < 0) {
|
|
|
|
binderror = strerror(errno);
|
|
|
|
if ((time(NULL) - bindstart) > 61)
|
|
|
|
break;
|
|
|
|
else {
|
|
|
|
applog(LOG_ERR, "API bind to port %d failed - trying again in 15sec", port);
|
|
|
|
sleep(15);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
bound = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (bound == 0) {
|
|
|
|
applog(LOG_ERR, "API bind to port %d failed (%s)%s", port, binderror, UNAVAILABLE);
|
|
|
|
free(apisock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SOCKETFAIL(listen(*apisock, QUEUE))) {
|
|
|
|
applog(LOG_ERR, "API initialisation 3 failed (%s)%s", strerror(errno), UNAVAILABLE);
|
|
|
|
CLOSESOCKET(*apisock);
|
|
|
|
free(apisock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-11-12 12:53:30 +01:00
|
|
|
buffer = (char *) calloc(1, MYBUFSIZ + 1);
|
2014-11-12 11:43:55 +01:00
|
|
|
|
|
|
|
counter = 0;
|
|
|
|
while (bye == 0) {
|
|
|
|
counter++;
|
|
|
|
|
|
|
|
clisiz = sizeof(cli);
|
|
|
|
if (SOCKETFAIL(c = accept(*apisock, (struct sockaddr *)(&cli), &clisiz))) {
|
|
|
|
applog(LOG_ERR, "API failed (%s)%s", strerror(errno), UNAVAILABLE);
|
|
|
|
CLOSESOCKET(*apisock);
|
|
|
|
free(apisock);
|
|
|
|
free(buffer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
addrok = check_connect(&cli, &connectaddr, &group);
|
|
|
|
if (opt_protocol)
|
|
|
|
applog(LOG_DEBUG, "API: connection from %s - %s",
|
|
|
|
connectaddr, addrok ? "Accepted" : "Ignored");
|
|
|
|
|
|
|
|
if (addrok) {
|
2014-11-12 15:26:25 +01:00
|
|
|
n = recv(c, &buf[0], MYBUFSIZ - 1, 0);
|
2014-11-12 12:53:30 +01:00
|
|
|
// applog(LOG_DEBUG, "API: recv command: (%d) '%s'", n, buf);
|
|
|
|
if (!SOCKETFAIL(n)) {
|
2014-11-12 11:43:55 +01:00
|
|
|
buf[n] = '\0';
|
|
|
|
params = strchr(buf, '|');
|
|
|
|
if (params != NULL)
|
|
|
|
*(params++) = '\0';
|
|
|
|
|
|
|
|
for (i = 0; i < CMDMAX; i++) {
|
|
|
|
if (strcmp(buf, cmds[i].name) == 0) {
|
|
|
|
result = (cmds[i].func)(params);
|
|
|
|
send_result(c, result);
|
2014-11-12 12:53:30 +01:00
|
|
|
CLOSESOCKET(c);
|
2014-11-12 11:43:55 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CLOSESOCKET(*apisock);
|
|
|
|
free(apisock);
|
|
|
|
free(buffer);
|
|
|
|
}
|
|
|
|
|
2014-11-12 15:26:25 +01:00
|
|
|
/* external access */
|
2014-11-12 11:43:55 +01:00
|
|
|
void *api_thread(void *userdata)
|
|
|
|
{
|
|
|
|
struct thr_info *mythr = (struct thr_info*)userdata;
|
|
|
|
|
2014-11-12 15:26:25 +01:00
|
|
|
startup = time(NULL);
|
2014-11-12 11:43:55 +01:00
|
|
|
api();
|
|
|
|
|
|
|
|
tq_freeze(mythr->q);
|
|
|
|
|
|
|
|
return NULL;
|
2014-11-12 12:53:30 +01:00
|
|
|
}
|