mirror of
https://github.com/Silicondust/libhdhomerun
synced 2025-07-06 04:52:19 -07:00
460 lines
9.7 KiB
C
460 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(¤t_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);
|
|
}
|
|
}
|