File: LuaFunction.cpp

package info (click to toggle)
freespace2 24.2.0%2Brepack-1
  • links: PTS, VCS
  • area: non-free
  • in suites: forky, sid
  • size: 43,716 kB
  • sloc: cpp: 595,001; ansic: 21,741; python: 1,174; sh: 457; makefile: 248; xml: 181
file content (253 lines) | stat: -rw-r--r-- 6,136 bytes parent folder | download | duplicates (2)
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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
#include "LuaFunction.h"
#include "LuaException.h"

#include "LuaHeaders.h"

namespace luacpp {
namespace {

int stdFunctionDestructor(lua_State* L)
{
	// The userdata is the function object so we call the destructor here to free up the internal resources
	const auto luaPtr = lua_touserdata(L, 1);
	const auto funcPtr = static_cast<LuaFunctionObject*>(luaPtr);

	funcPtr->~LuaFunctionObject();

	return 0;
}

int stdFunctionCaller(lua_State* L)
{
	const auto numArgs = lua_gettop(L);

	// Push the actual function object on the stack
	lua_pushvalue(L, lua_upvalueindex(1));

	const auto funcObj = static_cast<LuaFunctionObject*>(lua_touserdata(L, -1));
	lua_pop(L, 1); // We have the pointer so we can remove this from the stack again

	LuaValueList params;
	params.reserve(numArgs);

	for (int i = 1; i <= numArgs; ++i) {
		LuaValue val;
		val.setReference(UniqueLuaReference::create(L, i));
		params.push_back(val);
	}

	const auto ret = (*funcObj)(L, params);

	// Prepare to return values. Clear the stack
	lua_settop(L, 0);
	for (const auto& retVal : ret)
	{
		retVal.pushValue(L);
	}

	// Done!
	return static_cast<int>(ret.size());
}

}

LuaFunction LuaFunction::createFromCFunction(lua_State* L, lua_CFunction function, const LuaValueList& upvalues)
{
	LuaFunction func;

	for (auto& val : upvalues) {
		val.pushValue(L);
	}

	lua_pushcclosure(L, function, (int)upvalues.size());

	func.setReference(UniqueLuaReference::create(L));

	lua_pop(L, 1);

	return func;
}
LuaFunction LuaFunction::createFromStdFunction(lua_State* L, LuaFunctionObject function) {
	auto userdataMetatable = LuaTable::create(L);
	userdataMetatable.addValue("__gc", stdFunctionDestructor);

	// Create storage for the function object in Lua and initialize our data there via placement new
	auto functionStorage = lua_newuserdata(L, sizeof(function));
	new (functionStorage) LuaFunctionObject(std::move(function));

	userdataMetatable.pushValue(L);
	lua_setmetatable(L, -2);

	LuaValue funcValue;
	funcValue.setReference(UniqueLuaReference::create(L));
	lua_pop(L, 1); // Value is referenced. We can remove it from the stack now

	return LuaFunction::createFromCFunction(L, stdFunctionCaller, { funcValue });
}

LuaFunction LuaFunction::createFromCode(lua_State* L, std::string const& code, std::string const& name) {
	int load_err = luaL_loadbuffer(L, code.c_str(), code.length(), name.c_str());

	if (!load_err) {
		LuaFunction func;

		func.setReference(UniqueLuaReference::create(L));

		lua_pop(L, 1);

		return func;
	} else {
		// Get the error message
		size_t len;
		const char* err = lua_tolstring(L, -1, &len);

		lua_pop(L, 1);

		throw LuaException(std::string(err, len));
	}
}

LuaFunction::LuaFunction() : LuaValue(), _errorFunction(nullptr) {
}

LuaFunction::LuaFunction(const LuaFunction&) = default;
LuaFunction& LuaFunction::operator=(const LuaFunction&) = default;

LuaFunction::LuaFunction(LuaFunction&&) noexcept = default;
LuaFunction& LuaFunction::operator=(LuaFunction&&) noexcept = default;

LuaFunction::~LuaFunction() = default;

bool LuaFunction::setEnvironment(const LuaTable& table) {
	if (!table.getReference()->isValid()) {
		throw LuaException("Table reference is not valid!");
	}

	this->pushValue(_luaState);
	table.pushValue(_luaState);

	bool ret = lua_setfenv(_luaState, -2) != 0;

	// Pop the function again
	lua_pop(_luaState, 1);

	return ret;
}

LuaValueList LuaFunction::operator()(lua_State* L, const LuaValueList& args) const {
	return this->call(L, args);
}

void LuaFunction::setReference(const LuaReference& ref) {
	if (ref == nullptr)
	{
		// If not a valid reference then let the base class handle everything
		LuaValue::setReference(ref);
		return;
	}

	lua_State* L = ref->getState();

	ref->pushValue(L);

	if (lua_type(L, -1) != LUA_TFUNCTION) {
		lua_pop(L, 1);
		throw LuaException("Reference does not refer to a function!");
	} else {
		lua_pop(L, 1);
		LuaValue::setReference(ref);
	}
}

LuaValueList LuaFunction::call(lua_State* L, const LuaValueList& args) const {
	int err_idx = 0;
	int stackTop;

	if (_errorFunction) {
		// push the error function
		_errorFunction->pushValue(L);
		err_idx = lua_gettop(L);
		stackTop = err_idx;
	} else {
		stackTop = lua_gettop(L);
	}

	if(!lua_checkstack(L, (int)args.size() + 1))
		throw LuaException("Lua Stack Overflow!");

	// Push the function onto the stack
	this->pushValue(L, true);

	// Push the arguments onto the stack
	for (const auto& arg : args) {
		arg.pushValue(L, true);
	}

	// actually call the function now!
	int err = lua_pcall(L, (int) args.size(), LUA_MULTRET, err_idx);

	if (!err) {
		int numReturn = lua_gettop(L) - stackTop;
		LuaValueList values;
		values.reserve(numReturn);

		LuaValue val;
		for (int i = 0; i < numReturn; ++i) {
			if (convert::popValue(L, val)) {
				// Add values at the begin as the last return value is on top
				// of the stack.
				values.insert(values.begin(), val);
			}
		}

		if (err_idx != 0) {
			// Remove the error function
			lua_pop(L, 1);
		}

		return values;
	} else {
		// Make sure that there is exactly one parameter left on the stack
		// If the error function didn't return anything then this will push nil
		// If it pushed more than one value then this will discard all of them except the last one
		lua_settop(L, stackTop + 1);

		std::string err_msg;
		if (!lua_isstring(L, -1)) {
			err_msg = "Invalid lua value on stack!";
			lua_pop(L, 1); // Remove the value on the stack
		} else {
			if (!convert::popValue(L, err_msg)) {
				err_msg = "Failed to get error message from Lua stack!";
			}
		}

		if (err_idx != 0) {
			// Pop the error function
			lua_pop(L, 1);
		}

		// Throw exception with generated message
		throw LuaException(err_msg);
	}
}

bool ::luacpp::convert::popValue(lua_State* luaState, LuaFunction& target, int stackposition, bool remove) {
	if (!internal::isValidIndex(luaState, stackposition)) {
		return false;
	}

	if (!lua_isfunction(luaState, stackposition)) {
		return false;
	} else {
		target.setReference(UniqueLuaReference::create(luaState, stackposition));

		if (remove) {
			lua_remove(luaState, stackposition);
		}

		return true;
	}
}

}