libhdhomerun/hdhomerun_debug.c

461 lines
9.7 KiB
C

/*
* hdhomerun_debug.c
*
* Copyright © 2007-2016 Silicondust USA Inc. <www.silicondust.com>.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/*
* The debug logging includes optional support for connecting to the
* Silicondust support server. This option should not be used without
* being explicitly enabled by the user. Debug information should be
* limited to information useful to diagnosing a problem.
* - Silicondust.
*/
#include "hdhomerun.h"
#if !defined(HDHOMERUN_DEBUG_HOST)
#define HDHOMERUN_DEBUG_HOST "debug.silicondust.com"
#endif
#if !defined(HDHOMERUN_DEBUG_PORT)
#define HDHOMERUN_DEBUG_PORT 8002
#endif
#define HDHOMERUN_DEBUG_CONNECT_RETRY_TIME 30000
#define HDHOMERUN_DEBUG_CONNECT_TIMEOUT 10000
#define HDHOMERUN_DEBUG_SEND_TIMEOUT 10000
struct hdhomerun_debug_message_t
{
struct hdhomerun_debug_message_t *next;
char buffer[2048];
};
struct hdhomerun_debug_t
{
thread_task_t thread;
volatile bool enabled;
volatile bool terminate;
char *prefix;
thread_mutex_t print_lock;
thread_mutex_t queue_lock;
thread_mutex_t send_lock;
thread_cond_t queue_cond;
struct hdhomerun_debug_message_t *queue_head;
struct hdhomerun_debug_message_t *queue_tail;
uint32_t queue_depth;
uint64_t connect_delay;
char *file_name;
FILE *file_fp;
struct hdhomerun_sock_t *sock;
};
static void hdhomerun_debug_thread_execute(void *arg);
struct hdhomerun_debug_t *hdhomerun_debug_create(void)
{
struct hdhomerun_debug_t *dbg = (struct hdhomerun_debug_t *)calloc(1, sizeof(struct hdhomerun_debug_t));
if (!dbg) {
return NULL;
}
thread_mutex_init(&dbg->print_lock);
thread_mutex_init(&dbg->queue_lock);
thread_mutex_init(&dbg->send_lock);
thread_cond_init(&dbg->queue_cond);
if (!thread_task_create(&dbg->thread, &hdhomerun_debug_thread_execute, dbg)) {
free(dbg);
return NULL;
}
return dbg;
}
void hdhomerun_debug_destroy(struct hdhomerun_debug_t *dbg)
{
if (!dbg) {
return;
}
dbg->terminate = true;
thread_cond_signal(&dbg->queue_cond);
thread_task_join(dbg->thread);
if (dbg->prefix) {
free(dbg->prefix);
}
if (dbg->file_name) {
free(dbg->file_name);
}
if (dbg->file_fp) {
fclose(dbg->file_fp);
}
if (dbg->sock) {
hdhomerun_sock_destroy(dbg->sock);
}
thread_cond_dispose(&dbg->queue_cond);
thread_mutex_dispose(&dbg->print_lock);
thread_mutex_dispose(&dbg->queue_lock);
thread_mutex_dispose(&dbg->send_lock);
free(dbg);
}
/* Send lock held by caller */
static void hdhomerun_debug_close_internal(struct hdhomerun_debug_t *dbg)
{
if (dbg->file_fp) {
fclose(dbg->file_fp);
dbg->file_fp = NULL;
}
if (dbg->sock) {
hdhomerun_sock_destroy(dbg->sock);
dbg->sock = NULL;
}
}
void hdhomerun_debug_close(struct hdhomerun_debug_t *dbg, uint64_t timeout)
{
if (!dbg) {
return;
}
if (timeout > 0) {
hdhomerun_debug_flush(dbg, timeout);
}
thread_mutex_lock(&dbg->send_lock);
hdhomerun_debug_close_internal(dbg);
dbg->connect_delay = 0;
thread_mutex_unlock(&dbg->send_lock);
}
void hdhomerun_debug_set_filename(struct hdhomerun_debug_t *dbg, const char *filename)
{
if (!dbg) {
return;
}
thread_mutex_lock(&dbg->send_lock);
if (!filename && !dbg->file_name) {
thread_mutex_unlock(&dbg->send_lock);
return;
}
if (filename && dbg->file_name) {
if (strcmp(filename, dbg->file_name) == 0) {
thread_mutex_unlock(&dbg->send_lock);
return;
}
}
hdhomerun_debug_close_internal(dbg);
dbg->connect_delay = 0;
if (dbg->file_name) {
free(dbg->file_name);
dbg->file_name = NULL;
}
if (filename) {
dbg->file_name = strdup(filename);
}
thread_mutex_unlock(&dbg->send_lock);
}
void hdhomerun_debug_set_prefix(struct hdhomerun_debug_t *dbg, const char *prefix)
{
if (!dbg) {
return;
}
thread_mutex_lock(&dbg->print_lock);
if (dbg->prefix) {
free(dbg->prefix);
dbg->prefix = NULL;
}
if (prefix) {
dbg->prefix = strdup(prefix);
}
thread_mutex_unlock(&dbg->print_lock);
}
void hdhomerun_debug_enable(struct hdhomerun_debug_t *dbg)
{
if (!dbg) {
return;
}
if (dbg->enabled) {
return;
}
dbg->enabled = true;
thread_cond_signal(&dbg->queue_cond);
}
void hdhomerun_debug_disable(struct hdhomerun_debug_t *dbg)
{
if (!dbg) {
return;
}
dbg->enabled = false;
}
bool hdhomerun_debug_enabled(struct hdhomerun_debug_t *dbg)
{
if (!dbg) {
return false;
}
return dbg->enabled;
}
void hdhomerun_debug_flush(struct hdhomerun_debug_t *dbg, uint64_t timeout)
{
if (!dbg) {
return;
}
timeout = getcurrenttime() + timeout;
while (getcurrenttime() < timeout) {
thread_mutex_lock(&dbg->queue_lock);
struct hdhomerun_debug_message_t *message = dbg->queue_head;
thread_mutex_unlock(&dbg->queue_lock);
if (!message) {
return;
}
msleep_approx(16);
}
}
void hdhomerun_debug_printf(struct hdhomerun_debug_t *dbg, const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
hdhomerun_debug_vprintf(dbg, fmt, args);
va_end(args);
}
void hdhomerun_debug_vprintf(struct hdhomerun_debug_t *dbg, const char *fmt, va_list args)
{
if (!dbg) {
return;
}
struct hdhomerun_debug_message_t *message = (struct hdhomerun_debug_message_t *)malloc(sizeof(struct hdhomerun_debug_message_t));
if (!message) {
return;
}
message->next = NULL;
char *ptr = message->buffer;
char *end = message->buffer + sizeof(message->buffer) - 2;
*end = 0;
/*
* Timestamp.
*/
time_t current_time = time(NULL);
ptr += strftime(ptr, end - ptr, "%Y%m%d-%H:%M:%S ", localtime(&current_time));
if (ptr > end) {
ptr = end;
}
/*
* Debug prefix.
*/
thread_mutex_lock(&dbg->print_lock);
if (dbg->prefix) {
hdhomerun_sprintf(ptr, end, "%s ", dbg->prefix);
ptr = strchr(ptr, 0);
}
thread_mutex_unlock(&dbg->print_lock);
/*
* Message text.
*/
hdhomerun_vsprintf(ptr, end, fmt, args);
ptr = strchr(ptr, 0);
/*
* Force newline.
*/
if (ptr[-1] != '\n') {
hdhomerun_sprintf(ptr, end, "\n");
}
/*
* Enqueue.
*/
thread_mutex_lock(&dbg->queue_lock);
if (dbg->queue_tail) {
dbg->queue_tail->next = message;
} else {
dbg->queue_head = message;
}
dbg->queue_tail = message;
dbg->queue_depth++;
bool signal_thread = dbg->enabled || (dbg->queue_depth > 1024 + 100);
thread_mutex_unlock(&dbg->queue_lock);
if (signal_thread) {
thread_cond_signal(&dbg->queue_cond);
}
}
/* Send lock held by caller */
static bool hdhomerun_debug_output_message_file(struct hdhomerun_debug_t *dbg, struct hdhomerun_debug_message_t *message)
{
if (!dbg->file_fp) {
uint64_t current_time = getcurrenttime();
if (current_time < dbg->connect_delay) {
return false;
}
dbg->connect_delay = current_time + 30*1000;
dbg->file_fp = fopen(dbg->file_name, "a");
if (!dbg->file_fp) {
return false;
}
}
fprintf(dbg->file_fp, "%s", message->buffer);
fflush(dbg->file_fp);
return true;
}
/* Send lock held by caller */
static bool hdhomerun_debug_output_message_sock(struct hdhomerun_debug_t *dbg, struct hdhomerun_debug_message_t *message)
{
if (!dbg->sock) {
uint64_t current_time = getcurrenttime();
if (current_time < dbg->connect_delay) {
return false;
}
dbg->connect_delay = current_time + HDHOMERUN_DEBUG_CONNECT_RETRY_TIME;
dbg->sock = hdhomerun_sock_create_tcp();
if (!dbg->sock) {
return false;
}
uint32_t remote_addr = hdhomerun_sock_getaddrinfo_addr(dbg->sock, HDHOMERUN_DEBUG_HOST);
if (remote_addr == 0) {
hdhomerun_debug_close_internal(dbg);
return false;
}
if (!hdhomerun_sock_connect(dbg->sock, remote_addr, HDHOMERUN_DEBUG_PORT, HDHOMERUN_DEBUG_CONNECT_TIMEOUT)) {
hdhomerun_debug_close_internal(dbg);
return false;
}
}
size_t length = strlen(message->buffer);
if (!hdhomerun_sock_send(dbg->sock, message->buffer, length, HDHOMERUN_DEBUG_SEND_TIMEOUT)) {
hdhomerun_debug_close_internal(dbg);
return false;
}
return true;
}
static bool hdhomerun_debug_output_message(struct hdhomerun_debug_t *dbg, struct hdhomerun_debug_message_t *message)
{
thread_mutex_lock(&dbg->send_lock);
bool ret;
if (dbg->file_name) {
ret = hdhomerun_debug_output_message_file(dbg, message);
} else {
ret = hdhomerun_debug_output_message_sock(dbg, message);
}
thread_mutex_unlock(&dbg->send_lock);
return ret;
}
static void hdhomerun_debug_pop_and_free_message(struct hdhomerun_debug_t *dbg)
{
thread_mutex_lock(&dbg->queue_lock);
struct hdhomerun_debug_message_t *message = dbg->queue_head;
dbg->queue_head = message->next;
if (!dbg->queue_head) {
dbg->queue_tail = NULL;
}
dbg->queue_depth--;
thread_mutex_unlock(&dbg->queue_lock);
free(message);
}
static void hdhomerun_debug_thread_execute(void *arg)
{
struct hdhomerun_debug_t *dbg = (struct hdhomerun_debug_t *)arg;
while (!dbg->terminate) {
thread_mutex_lock(&dbg->queue_lock);
struct hdhomerun_debug_message_t *message = dbg->queue_head;
uint32_t queue_depth = dbg->queue_depth;
thread_mutex_unlock(&dbg->queue_lock);
if (!message) {
thread_cond_wait(&dbg->queue_cond);
continue;
}
if (queue_depth > 1024) {
hdhomerun_debug_pop_and_free_message(dbg);
continue;
}
if (!dbg->enabled) {
thread_cond_wait(&dbg->queue_cond);
continue;
}
if (!hdhomerun_debug_output_message(dbg, message)) {
msleep_approx(1000);
continue;
}
hdhomerun_debug_pop_and_free_message(dbg);
}
}