| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | #ifdef HAVE_CONFIG_H |
| | #include <config.h> |
| | #endif |
| |
|
| | #include <stdio.h> |
| | #include <stdlib.h> |
| | #include <stdarg.h> |
| | #include <string.h> |
| | #include <errno.h> |
| |
|
| | #include <pocketsphinx.h> |
| |
|
| | #include "util/filename.h" |
| | #include "util/ckd_alloc.h" |
| |
|
| | static FILE* logfp = NULL; |
| | static int logfp_disabled = FALSE; |
| |
|
| | #if defined(__ANDROID__) |
| | #include <android/log.h> |
| | static void |
| | err_logcat_cb(void* user_data, err_lvl_t level, const char *fmt, ...); |
| | #elif defined(_WIN32_WCE) |
| | #include <windows.h> |
| | #define vsnprintf _vsnprintf |
| | static void |
| | err_wince_cb(void* user_data, err_lvl_t level, const char *fmt, ...); |
| | #endif |
| |
|
| | #if defined(__ANDROID__) |
| | static err_cb_f err_cb = err_logcat_cb; |
| | #elif defined(_WIN32_WCE) |
| | static err_cb_f err_cb = err_wince_cb; |
| | #else |
| | static err_cb_f err_cb = err_logfp_cb; |
| | #endif |
| | static void* err_user_data; |
| | static err_lvl_t min_loglevel = ERR_WARN; |
| | static const char *err_level[ERR_MAX] = |
| | { |
| | "DEBUG", "INFO", "WARN", "ERROR", "FATAL" |
| | }; |
| |
|
| | int |
| | err_set_loglevel(err_lvl_t lvl) |
| | { |
| | int rv = min_loglevel; |
| | min_loglevel = lvl; |
| | return rv; |
| | } |
| |
|
| | const char * |
| | err_set_loglevel_str(char const *lvl) |
| | { |
| | const char *rv = err_level[min_loglevel]; |
| | int i; |
| |
|
| | if (lvl == NULL) |
| | return NULL; |
| | if (!strncmp(lvl, "ERR_", 4)) |
| | lvl += 4; |
| | for (i = 0; i < ERR_MAX; ++i) { |
| | if (!strcmp(lvl, err_level[i])) { |
| | min_loglevel = i; |
| | return rv; |
| | } |
| | } |
| | return NULL; |
| | } |
| |
|
| | void |
| | err_msg(err_lvl_t lvl, const char *path, long ln, const char *fmt, ...) |
| | { |
| |
|
| | char msg[1024]; |
| | va_list ap; |
| |
|
| | if (!err_cb) |
| | return; |
| | if (lvl < min_loglevel) |
| | return; |
| |
|
| | va_start(ap, fmt); |
| | vsnprintf(msg, sizeof(msg), fmt, ap); |
| | va_end(ap); |
| |
|
| | if (path) { |
| | const char *fname = path2basename(path); |
| | if (lvl == ERR_INFO) |
| | err_cb(err_user_data, lvl, "%s: %s(%ld): %s", err_level[lvl], fname, ln, msg); |
| | else |
| | err_cb(err_user_data, lvl, "%s: \"%s\", line %ld: %s", err_level[lvl], fname, ln, msg); |
| | } else { |
| | err_cb(err_user_data, lvl, "%s", msg); |
| | } |
| | } |
| |
|
| | #ifdef _WIN32_WCE |
| | void |
| | err_msg_system(err_lvl_t lvl, const char *path, long ln, const char *fmt, ...) |
| | { |
| | va_list ap; |
| | LPVOID error_wstring; |
| | DWORD error; |
| | char msg[1024]; |
| | char error_string[1024]; |
| |
|
| | if (!err_cb) |
| | return; |
| | if (lvl < min_loglevel) |
| | return; |
| |
|
| | error = GetLastError(); |
| | FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | |
| | FORMAT_MESSAGE_FROM_SYSTEM | |
| | FORMAT_MESSAGE_IGNORE_INSERTS, |
| | NULL, |
| | error, |
| | 0, |
| | (LPTSTR) &error_wstring, |
| | 0, |
| | NULL); |
| | wcstombs(error_string, error_wstring, 1023); |
| | LocalFree(error_wstring); |
| |
|
| | va_start(ap, fmt); |
| | vsnprintf(msg, sizeof(msg), fmt, ap); |
| | va_end(ap); |
| |
|
| | if (path) { |
| | const char *fname = path2basename(path); |
| | if (lvl == ERR_INFO) |
| | err_cb(err_user_data, lvl, "%s: %s(%ld): %s: %s\n", err_prefix[lvl], fname, ln, msg, error_string); |
| | else |
| | err_cb(err_user_data, lvl, "%s: \"%s\", line %ld: %s: %s\n", err_prefix[lvl], fname, ln, msg, error_string); |
| | } else { |
| | err_cb(err_user_data, lvl, "%s: %s\n", msg, error_string); |
| | } |
| | } |
| | #else |
| | void |
| | err_msg_system(err_lvl_t lvl, const char *path, long ln, const char *fmt, ...) |
| | { |
| | int local_errno = errno; |
| | |
| | char msg[1024]; |
| | va_list ap; |
| |
|
| | if (!err_cb) |
| | return; |
| | if (lvl < min_loglevel) |
| | return; |
| |
|
| | va_start(ap, fmt); |
| | vsnprintf(msg, sizeof(msg), fmt, ap); |
| | va_end(ap); |
| |
|
| | if (path) { |
| | const char *fname = path2basename(path); |
| | if (lvl == ERR_INFO) |
| | err_cb(err_user_data, lvl, "%s: %s(%ld): %s: %s\n", err_level[lvl], fname, ln, msg, strerror(local_errno)); |
| | else |
| | err_cb(err_user_data, lvl, "%s: \"%s\", line %ld: %s: %s\n", err_level[lvl], fname, ln, msg, strerror(local_errno)); |
| | } else { |
| | err_cb(err_user_data, lvl, "%s: %s\n", msg, strerror(local_errno)); |
| | } |
| | } |
| | #endif |
| |
|
| | #if defined(__ANDROID__) |
| | static void |
| | err_logcat_cb(void *user_data, err_lvl_t lvl, const char *fmt, ...) |
| | { |
| | static const int android_level[ERR_MAX] = {ANDROID_LOG_DEBUG, ANDROID_LOG_INFO, |
| | ANDROID_LOG_INFO, ANDROID_LOG_WARN, ANDROID_LOG_ERROR, ANDROID_LOG_ERROR}; |
| |
|
| | va_list ap; |
| | va_start(ap, fmt); |
| | __android_log_vprint(android_level[lvl], "cmusphinx", fmt, ap); |
| | va_end(ap); |
| | } |
| | #elif defined(_WIN32_WCE) |
| | static void |
| | err_wince_cb(void *user_data, err_lvl_t lvl, const char *fmt, ...) |
| | { |
| | char msg[1024]; |
| | WCHAR *wmsg; |
| | size_t size; |
| | va_list ap; |
| |
|
| | va_start(ap, fmt); |
| | _vsnprintf(msg, sizeof(msg), fmt, ap); |
| | va_end(ap); |
| |
|
| | size = mbstowcs(NULL, msg, 0) + 1; |
| | wmsg = ckd_calloc(size, sizeof(*wmsg)); |
| | mbstowcs(wmsg, msg, size); |
| |
|
| | OutputDebugStringW(wmsg); |
| | ckd_free(wmsg); |
| | } |
| | #else |
| | void |
| | err_logfp_cb(void *user_data, err_lvl_t lvl, const char *fmt, ...) |
| | { |
| | va_list ap; |
| | FILE *fp = err_get_logfp(); |
| |
|
| | (void)user_data; |
| | (void)lvl; |
| | |
| | if (!fp) |
| | return; |
| | |
| | va_start(ap, fmt); |
| | vfprintf(fp, fmt, ap); |
| | va_end(ap); |
| | fflush(fp); |
| | } |
| | #endif |
| |
|
| | int |
| | err_set_logfile(const char *path) |
| | { |
| | FILE *newfp; |
| |
|
| | if ((newfp = fopen(path, "a")) == NULL) |
| | return -1; |
| | err_set_logfp(newfp); |
| | return 0; |
| | } |
| |
|
| | void |
| | err_set_logfp(FILE *stream) |
| | { |
| | if (logfp != NULL && logfp != stdout && logfp != stderr) |
| | fclose(logfp); |
| | if (stream == NULL) { |
| | logfp_disabled = TRUE; |
| | logfp = NULL; |
| | return; |
| | } |
| | logfp_disabled = FALSE; |
| | logfp = stream; |
| | return; |
| | } |
| |
|
| | FILE * |
| | err_get_logfp(void) |
| | { |
| | if (logfp_disabled) |
| | return NULL; |
| | if (logfp == NULL) |
| | return stderr; |
| |
|
| | return logfp; |
| | } |
| |
|
| | void |
| | err_set_callback(err_cb_f cb, void* user_data) |
| | { |
| | err_cb = cb; |
| | err_user_data= user_data; |
| | } |
| |
|