aboutsummaryrefslogtreecommitdiff
path: root/src/script/common/c_internal.h
blob: d2131d1ad3ee4d9b9ae005fc4eefe1d5bd5bff94 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
/*
Minetest
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>

This program 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 program 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 program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/

/******************************************************************************/
/******************************************************************************/
/* WARNING!!!! do NOT add this header in any include file or any code file    */
/*             not being a modapi file!!!!!!!!                                */
/******************************************************************************/
/******************************************************************************/

#pragma once

extern "C" {
#include <lua.h>
#include <lauxlib.h>
}

#include "common/c_types.h"


/*
	Define our custom indices into the Lua registry table.

	Lua 5.2 and above define the LUA_RIDX_LAST macro. Only numbers above that
	may be used for custom indices, anything else is reserved.

	Lua 5.1 / LuaJIT do not use any numeric indices (only string indices),
	so we can use numeric indices freely.
*/
#ifdef LUA_RIDX_LAST
#define CUSTOM_RIDX_BASE ((LUA_RIDX_LAST)+1)
#else
#define CUSTOM_RIDX_BASE 1
#endif

#define CUSTOM_RIDX_SCRIPTAPI           (CUSTOM_RIDX_BASE)
#define CUSTOM_RIDX_GLOBALS_BACKUP      (CUSTOM_RIDX_BASE + 1)
#define CUSTOM_RIDX_CURRENT_MOD_NAME    (CUSTOM_RIDX_BASE + 2)
#define CUSTOM_RIDX_BACKTRACE           (CUSTOM_RIDX_BASE + 3)

// Pushes the error handler onto the stack and returns its index
#define PUSH_ERROR_HANDLER(L) \
	(lua_rawgeti((L), LUA_REGISTRYINDEX, CUSTOM_RIDX_BACKTRACE), lua_gettop((L)))

#define PCALL_RESL(L, RES) {                            \
	int result_ = (RES);                                \
	if (result_ != 0) {                                 \
		script_error((L), result_, NULL, __FUNCTION__); \
	}                                                   \
}

#define script_run_callbacks(L, nargs, mode) \
	script_run_callbacks_f((L), (nargs), (mode), __FUNCTION__)

// What script_run_callbacks does with the return values of callbacks.
// Regardless of the mode, if only one callback is defined,
// its return value is the total return value.
// Modes only affect the case where 0 or >= 2 callbacks are defined.
enum RunCallbacksMode
{
	// Returns the return value of the first callback
	// Returns nil if list of callbacks is empty
	RUN_CALLBACKS_MODE_FIRST,
	// Returns the return value of the last callback
	// Returns nil if list of callbacks is empty
	RUN_CALLBACKS_MODE_LAST,
	// If any callback returns a false value, the first such is returned
	// Otherwise, the first callback's return value (trueish) is returned
	// Returns true if list of callbacks is empty
	RUN_CALLBACKS_MODE_AND,
	// Like above, but stops calling callbacks (short circuit)
	// after seeing the first false value
	RUN_CALLBACKS_MODE_AND_SC,
	// If any callback returns a true value, the first such is returned
	// Otherwise, the first callback's return value (falseish) is returned
	// Returns false if list of callbacks is empty
	RUN_CALLBACKS_MODE_OR,
	// Like above, but stops calling callbacks (short circuit)
	// after seeing the first true value
	RUN_CALLBACKS_MODE_OR_SC,
	// Note: "a true value" and "a false value" refer to values that
	// are converted by readParam<bool> to true or false, respectively.
};

std::string script_get_backtrace(lua_State *L);
int script_exception_wrapper(lua_State *L, lua_CFunction f);
void script_error(lua_State *L, int pcall_result, const char *mod, const char *fxn);
void script_run_callbacks_f(lua_State *L, int nargs,
	RunCallbacksMode mode, const char *fxn);
void log_deprecated(lua_State *L, const std::string &message);
} void log_add_output_maxlev(ILogOutput *out, enum LogMessageLevel lev) { for(int i=0; i<=lev; i++) log_outputs[i].push_back(out); } void log_add_output_all_levs(ILogOutput *out) { for(int i=0; i<LMT_NUM_VALUES; i++) log_outputs[i].push_back(out); } void log_remove_output(ILogOutput *out) { for(int i=0; i<LMT_NUM_VALUES; i++){ std::vector<ILogOutput*>::iterator it = std::find(log_outputs[i].begin(), log_outputs[i].end(), out); if(it != log_outputs[i].end()) log_outputs[i].erase(it); } } void log_set_lev_silence(enum LogMessageLevel lev, bool silence) { MutexAutoLock lock(log_thread_name_mutex); for (std::vector<ILogOutput *>::iterator it = log_outputs[lev].begin(); it != log_outputs[lev].end(); ++it) { ILogOutput *out = *it; out->silence = silence; } } void log_register_thread(const std::string &name) { threadid_t id = get_current_thread_id(); MutexAutoLock lock(log_thread_name_mutex); log_thread_names[id] = name; } void log_deregister_thread() { threadid_t id = get_current_thread_id(); MutexAutoLock lock(log_thread_name_mutex); log_thread_names.erase(id); } static std::string get_lev_string(enum LogMessageLevel lev) { switch(lev){ case LMT_ERROR: return "ERROR"; case LMT_ACTION: return "ACTION"; case LMT_INFO: return "INFO"; case LMT_VERBOSE: return "VERBOSE"; case LMT_NUM_VALUES: break; } return "(unknown level)"; } void log_printline(enum LogMessageLevel lev, const std::string &text) { MutexAutoLock lock(log_thread_name_mutex); std::string threadname = "(unknown thread)"; std::map<threadid_t, std::string>::const_iterator i; i = log_thread_names.find(get_current_thread_id()); if(i != log_thread_names.end()) threadname = i->second; std::string levelname = get_lev_string(lev); std::ostringstream os(std::ios_base::binary); os << getTimestamp() << ": " << levelname << "["<<threadname<<"]: " << text; for(std::vector<ILogOutput*>::iterator i = log_outputs[lev].begin(); i != log_outputs[lev].end(); ++i) { ILogOutput *out = *i; if (out->silence) continue; out->printLog(os.str()); out->printLog(os.str(), lev); out->printLog(lev, text); } } class Logbuf : public std::streambuf { public: Logbuf(enum LogMessageLevel lev): m_lev(lev) { } ~Logbuf() { } int overflow(int c) { bufchar(c); return c; } std::streamsize xsputn(const char *s, std::streamsize n) { for(int i=0; i<n; i++) bufchar(s[i]); return n; } void printbuf() { log_printline(m_lev, m_buf); #ifdef __ANDROID__ __android_log_print(android_log_level_mapping[m_lev], PROJECT_NAME, "%s", m_buf.c_str()); #endif } void bufchar(char c) { if(c == '\n' || c == '\r'){ if(m_buf != "") printbuf(); m_buf = ""; return; } m_buf += c; } private: enum LogMessageLevel m_lev; std::string m_buf; }; Logbuf errorbuf(LMT_ERROR); Logbuf actionbuf(LMT_ACTION); Logbuf infobuf(LMT_INFO); Logbuf verbosebuf(LMT_VERBOSE); std::ostream errorstream(&errorbuf); std::ostream actionstream(&actionbuf); std::ostream infostream(&infobuf); std::ostream verbosestream(&verbosebuf); bool log_trace_level_enabled = false;