mirror of
https://github.com/irungentoo/toxcore.git
synced 2024-03-22 13:30:51 +08:00
220 lines
5.3 KiB
C
220 lines
5.3 KiB
C
/*
|
|
* util.c -- Utilities.
|
|
*
|
|
* This file is donated to the Tox Project.
|
|
* Copyright 2013 plutooo
|
|
*
|
|
* Copyright (C) 2013 Tox project All Rights Reserved.
|
|
*
|
|
* This file is part of Tox.
|
|
*
|
|
* Tox 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 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Tox is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Tox. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <time.h>
|
|
|
|
/* for CLIENT_ID_SIZE */
|
|
#include "DHT.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
|
/* don't call into system billions of times for no reason */
|
|
static uint64_t unix_time_value;
|
|
static uint64_t unix_base_time_value;
|
|
|
|
void unix_time_update()
|
|
{
|
|
if (unix_base_time_value == 0)
|
|
unix_base_time_value = ((uint64_t)time(NULL) - (current_time_monotonic() / 1000ULL));
|
|
|
|
unix_time_value = (current_time_monotonic() / 1000ULL) + unix_base_time_value;
|
|
}
|
|
|
|
uint64_t unix_time()
|
|
{
|
|
return unix_time_value;
|
|
}
|
|
|
|
int is_timeout(uint64_t timestamp, uint64_t timeout)
|
|
{
|
|
return timestamp + timeout <= unix_time();
|
|
}
|
|
|
|
|
|
/* id functions */
|
|
bool id_equal(uint8_t *dest, uint8_t *src)
|
|
{
|
|
return memcmp(dest, src, CLIENT_ID_SIZE) == 0;
|
|
}
|
|
|
|
uint32_t id_copy(uint8_t *dest, uint8_t *src)
|
|
{
|
|
memcpy(dest, src, CLIENT_ID_SIZE);
|
|
return CLIENT_ID_SIZE;
|
|
}
|
|
|
|
void host_to_net(uint8_t *num, uint16_t numbytes)
|
|
{
|
|
#ifndef WORDS_BIGENDIAN
|
|
uint32_t i;
|
|
uint8_t buff[numbytes];
|
|
|
|
for (i = 0; i < numbytes; ++i) {
|
|
buff[i] = num[numbytes - i - 1];
|
|
}
|
|
|
|
memcpy(num, buff, numbytes);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
/* state load/save */
|
|
int load_state(load_state_callback_func load_state_callback, void *outer,
|
|
uint8_t *data, uint32_t length, uint16_t cookie_inner)
|
|
{
|
|
if (!load_state_callback || !data) {
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "load_state() called with invalid args.\n");
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
|
|
uint16_t type;
|
|
uint32_t length_sub, cookie_type;
|
|
uint32_t size_head = sizeof(uint32_t) * 2;
|
|
|
|
while (length >= size_head) {
|
|
memcpy(&length_sub, data, sizeof(length_sub));
|
|
memcpy(&cookie_type, data + sizeof(length_sub), sizeof(cookie_type));
|
|
data += size_head;
|
|
length -= size_head;
|
|
|
|
if (length < length_sub) {
|
|
/* file truncated */
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "state file too short: %u < %u\n", length, length_sub);
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
if ((cookie_type >> 16) != cookie_inner) {
|
|
/* something is not matching up in a bad way, give up */
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "state file garbeled: %04hx != %04hx\n", (cookie_type >> 16), cookie_inner);
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
type = cookie_type & 0xFFFF;
|
|
|
|
if (-1 == load_state_callback(outer, data, length_sub, type))
|
|
return -1;
|
|
|
|
data += length_sub;
|
|
length -= length_sub;
|
|
}
|
|
|
|
return length == 0 ? 0 : -1;
|
|
};
|
|
|
|
#ifdef LOGGING
|
|
time_t starttime = 0;
|
|
size_t logbufferprelen = 0;
|
|
char *logbufferpredata = NULL;
|
|
char *logbufferprehead = NULL;
|
|
char logbuffer[512];
|
|
static FILE *logfile = NULL;
|
|
void loginit(uint16_t port)
|
|
{
|
|
if (logfile)
|
|
fclose(logfile);
|
|
|
|
if (!starttime) {
|
|
unix_time_update();
|
|
starttime = unix_time();
|
|
}
|
|
|
|
struct tm *tm = localtime(&starttime);
|
|
|
|
/* "%F %T" might not be Windows compatible */
|
|
if (strftime(logbuffer + 32, sizeof(logbuffer) - 32, "%F %T", tm))
|
|
sprintf(logbuffer, "%u-%s.log", ntohs(port), logbuffer + 32);
|
|
else
|
|
sprintf(logbuffer, "%u-%lu.log", ntohs(port), starttime);
|
|
|
|
logfile = fopen(logbuffer, "w");
|
|
|
|
if (logbufferpredata) {
|
|
if (logfile)
|
|
fprintf(logfile, "%s", logbufferpredata);
|
|
|
|
free(logbufferpredata);
|
|
logbufferpredata = NULL;
|
|
}
|
|
|
|
};
|
|
void loglog(char *text)
|
|
{
|
|
if (logfile) {
|
|
fprintf(logfile, "%4u %s", (uint32_t)(unix_time() - starttime), text);
|
|
fflush(logfile);
|
|
|
|
return;
|
|
}
|
|
|
|
/* log messages before file was opened: store */
|
|
|
|
size_t len = strlen(text);
|
|
|
|
if (!starttime) {
|
|
unix_time_update();
|
|
starttime = unix_time();
|
|
|
|
logbufferprelen = 1024 + len - (len % 1024);
|
|
logbufferpredata = malloc(logbufferprelen);
|
|
logbufferprehead = logbufferpredata;
|
|
}
|
|
|
|
/* loginit() called meanwhile? (but failed to open) */
|
|
if (!logbufferpredata)
|
|
return;
|
|
|
|
if (len + (logbufferprehead - logbufferpredata) + 16U < logbufferprelen) {
|
|
size_t logpos = logbufferprehead - logbufferpredata;
|
|
size_t lennew = logbufferprelen * 1.4;
|
|
logbufferpredata = realloc(logbufferpredata, lennew);
|
|
logbufferprehead = logbufferpredata + logpos;
|
|
logbufferprelen = lennew;
|
|
}
|
|
|
|
int written = sprintf(logbufferprehead, "%4u %s", (uint32_t)(unix_time() - starttime), text);
|
|
logbufferprehead += written;
|
|
}
|
|
|
|
void logexit()
|
|
{
|
|
if (logfile) {
|
|
fclose(logfile);
|
|
logfile = NULL;
|
|
}
|
|
};
|
|
#endif
|
|
|