2018-02-17 10:39:23 +01:00
|
|
|
/*
|
2010-08-08 10:06:34 +02:00
|
|
|
+-------------------+
|
|
|
|
| | Enno Rehling <enno@eressea.de>
|
|
|
|
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
| (c) 1998 - 2008 | Katja Zedel <katze@felidae.kn-bremen.de>
|
|
|
|
| | Henning Peters <faroul@beyond.kn-bremen.de>
|
|
|
|
+-------------------+
|
|
|
|
|
|
|
|
This program may not be used, modified or distributed
|
|
|
|
without prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
2017-12-29 06:13:28 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <platform.h>
|
|
|
|
#endif
|
|
|
|
|
2014-08-27 21:09:39 +02:00
|
|
|
#include "helpers.h"
|
|
|
|
#include "vortex.h"
|
2017-02-27 09:48:24 +01:00
|
|
|
#include "alchemy.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
#include <util/base36.h>
|
2017-05-01 13:26:17 +02:00
|
|
|
#include <util/event.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/functions.h>
|
2017-09-21 16:26:53 +02:00
|
|
|
#include <util/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/log.h>
|
2017-12-28 18:55:45 +01:00
|
|
|
#include <util/macros.h>
|
2015-11-27 15:23:48 +01:00
|
|
|
#include <util/parser.h>
|
2014-07-26 22:52:25 +02:00
|
|
|
#include <util/resolve.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-04-30 22:15:18 +02:00
|
|
|
#include <kernel/callbacks.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <kernel/config.h>
|
2017-04-30 18:23:42 +02:00
|
|
|
#include <kernel/callbacks.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <kernel/equipment.h>
|
|
|
|
#include <kernel/faction.h>
|
2012-05-29 21:17:25 +02:00
|
|
|
#include <kernel/spell.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <kernel/race.h>
|
2017-02-26 13:19:47 +01:00
|
|
|
#include <kernel/resources.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <kernel/unit.h>
|
|
|
|
#include <kernel/building.h>
|
|
|
|
#include <kernel/item.h>
|
|
|
|
#include <kernel/region.h>
|
2014-07-26 22:52:25 +02:00
|
|
|
|
|
|
|
#include <storage.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <tolua.h>
|
2012-07-01 19:17:47 +02:00
|
|
|
#include <lua.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <assert.h>
|
2014-03-15 19:29:11 +01:00
|
|
|
#include <string.h>
|
2014-07-26 22:52:25 +02:00
|
|
|
#include <stdlib.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
static int
|
2012-05-30 05:13:54 +02:00
|
|
|
lua_giveitem(unit * s, unit * d, const item_type * itype, int n, struct order *ord)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
|
|
|
char fname[64];
|
2017-12-10 12:23:28 +01:00
|
|
|
int result = -1, len;
|
2015-01-30 20:37:14 +01:00
|
|
|
const char *iname = itype->rtype->_name;
|
|
|
|
|
2017-12-28 18:55:45 +01:00
|
|
|
UNUSED_ARG(ord);
|
2015-01-30 20:37:14 +01:00
|
|
|
assert(s != NULL);
|
2017-12-10 12:23:28 +01:00
|
|
|
len = snprintf(fname, sizeof(fname), "%s_give", iname);
|
|
|
|
if (len > 0 && (size_t)len < sizeof(fname)) {
|
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
tolua_pushusertype(L, s, TOLUA_CAST "unit");
|
|
|
|
tolua_pushusertype(L, d, TOLUA_CAST "unit");
|
|
|
|
tolua_pushstring(L, iname);
|
|
|
|
lua_pushinteger(L, n);
|
|
|
|
|
|
|
|
if (lua_pcall(L, 4, 1, 0) != 0) {
|
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("unit %s calling '%s': %s.\n", unitname(s), fname, error);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = (int)lua_tonumber(L, -1);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-12-10 12:23:28 +01:00
|
|
|
log_error("unit %s trying to call '%s' : not a function.\n", unitname(s), fname);
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-02-26 13:19:47 +01:00
|
|
|
static int limit_resource_lua(const region * r, const resource_type * rtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
char fname[64];
|
2017-12-10 12:23:28 +01:00
|
|
|
int result = -1, len;
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
|
|
|
|
2017-12-10 12:23:28 +01:00
|
|
|
len = snprintf(fname, sizeof(fname), "%s_limit", rtype->_name);
|
|
|
|
if (len > 0 && (size_t)len < sizeof(fname)) {
|
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
tolua_pushusertype(L, (void *)r, TOLUA_CAST "region");
|
2015-01-30 20:37:14 +01:00
|
|
|
|
2017-12-10 12:23:28 +01:00
|
|
|
if (lua_pcall(L, 1, 1, 0) != 0) {
|
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("limit(%s) calling '%s': %s.\n", regionname(r, NULL), fname, error);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = (int)lua_tonumber(L, -1);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-12-10 12:23:28 +01:00
|
|
|
log_error("limit(%s) calling '%s': not a function.\n", regionname(r, NULL), fname);
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-02-26 13:19:47 +01:00
|
|
|
produce_resource_lua(region * r, const resource_type * rtype, int norders)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
|
|
|
char fname[64];
|
2017-12-10 12:23:28 +01:00
|
|
|
int len;
|
|
|
|
|
|
|
|
len = snprintf(fname, sizeof(fname), "%s_produce", rtype->_name);
|
|
|
|
if (len > 0 && (size_t)len < sizeof(fname)) {
|
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
tolua_pushusertype(L, (void *)r, TOLUA_CAST "region");
|
|
|
|
lua_pushinteger(L, norders);
|
|
|
|
|
|
|
|
if (lua_pcall(L, 2, 0, 0) != 0) {
|
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("produce(%s) calling '%s': %s.\n", regionname(r, NULL), fname, error);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("produce(%s) calling '%s': not a function.\n", regionname(r, NULL), fname);
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void push_param(lua_State * L, char c, spllprm * param)
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
if (c == 'u')
|
|
|
|
tolua_pushusertype(L, param->data.u, "unit");
|
|
|
|
else if (c == 'b')
|
|
|
|
tolua_pushusertype(L, param->data.b, "building");
|
|
|
|
else if (c == 's')
|
|
|
|
tolua_pushusertype(L, param->data.sh, "ship");
|
|
|
|
else if (c == 'r')
|
|
|
|
tolua_pushusertype(L, param->data.sh, "region");
|
|
|
|
else if (c == 'c')
|
|
|
|
tolua_pushstring(L, param->data.s);
|
|
|
|
else {
|
|
|
|
log_error("unsupported syntax %c.\n", c);
|
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/** callback to use lua for spell functions */
|
2017-04-30 18:41:30 +02:00
|
|
|
static int lua_callspell(castorder * co, const char *fname)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
|
|
|
unit *caster = co_get_caster(co);
|
|
|
|
region * r = co_get_region(co);
|
|
|
|
int result = -1;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
int nparam = 4;
|
|
|
|
tolua_pushusertype(L, r, TOLUA_CAST "region");
|
|
|
|
tolua_pushusertype(L, caster, TOLUA_CAST "unit");
|
2015-06-08 20:53:40 +02:00
|
|
|
lua_pushinteger(L, co->level);
|
|
|
|
lua_pushnumber(L, co->force);
|
2015-01-30 20:37:14 +01:00
|
|
|
if (co->sp->parameter && co->par->length) {
|
|
|
|
const char *synp = co->sp->parameter;
|
|
|
|
int i = 0;
|
|
|
|
++nparam;
|
|
|
|
lua_newtable(L);
|
|
|
|
while (*synp && i < co->par->length) {
|
|
|
|
spllprm *param = co->par->param[i];
|
|
|
|
char c = *synp;
|
|
|
|
if (c == '+') {
|
|
|
|
push_param(L, *(synp - 1), param);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
push_param(L, c, param);
|
|
|
|
++synp;
|
|
|
|
}
|
|
|
|
lua_rawseti(L, -2, ++i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lua_pcall(L, nparam, 1, 0) != 0) {
|
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("spell(%s) calling '%s': %s.\n", unitname(caster), fname, error);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = (int)lua_tonumber(L, -1);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
int ltype = lua_type(L, -1);
|
|
|
|
log_error("spell(%s) calling '%s': not a function, has type %d.\n", unitname(caster), fname, ltype);
|
|
|
|
lua_pop(L, 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2011-03-07 08:02:35 +01:00
|
|
|
lua_changeresource(unit * u, const struct resource_type *rtype, int delta)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
2017-12-10 12:23:28 +01:00
|
|
|
int len, result = -1;
|
2015-01-30 20:37:14 +01:00
|
|
|
char fname[64];
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-12-10 12:23:28 +01:00
|
|
|
len = snprintf(fname, sizeof(fname), "%s_changeresource", rtype->_name);
|
|
|
|
if (len > 0 && (size_t)len < sizeof(fname)) {
|
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
tolua_pushusertype(L, u, TOLUA_CAST "unit");
|
|
|
|
lua_pushinteger(L, delta);
|
2018-02-17 10:39:23 +01:00
|
|
|
|
2017-12-10 12:23:28 +01:00
|
|
|
if (lua_pcall(L, 2, 1, 0) != 0) {
|
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("change(%s) calling '%s': %s.\n", unitname(u), fname, error);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = (int)lua_tonumber(L, -1);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-12-10 12:23:28 +01:00
|
|
|
log_error("change(%s) calling '%s': not a function.\n", unitname(u), fname);
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/** callback for an item-use function written in lua. */
|
2017-02-27 04:22:28 +01:00
|
|
|
static int
|
2018-02-17 10:39:23 +01:00
|
|
|
lua_use_item(unit *u, const item_type *itype, const char * fname, int amount, struct order *ord)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_State *L = (lua_State *)global.vm_state;
|
2017-12-10 12:23:28 +01:00
|
|
|
int len, result = 0;
|
2015-01-30 20:37:14 +01:00
|
|
|
char fname[64];
|
2018-02-25 20:41:53 +01:00
|
|
|
int (*callout)(unit *, const item_type *, int, struct order *);
|
2017-02-27 09:48:24 +01:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_getglobal(L, fname);
|
|
|
|
if (lua_isfunction(L, -1)) {
|
|
|
|
tolua_pushusertype(L, (void *)u, TOLUA_CAST "unit");
|
2015-06-08 20:53:40 +02:00
|
|
|
lua_pushinteger(L, amount);
|
2015-11-27 15:23:48 +01:00
|
|
|
lua_pushstring(L, getstrtoken());
|
2015-12-30 21:20:59 +01:00
|
|
|
tolua_pushusertype(L, (void *)ord, TOLUA_CAST "order");
|
|
|
|
if (lua_pcall(L, 4, 1, 0) != 0) {
|
2015-01-30 20:37:14 +01:00
|
|
|
const char *error = lua_tostring(L, -1);
|
|
|
|
log_error("use(%s) calling '%s': %s.\n", unitname(u), fname, error);
|
|
|
|
}
|
|
|
|
else {
|
2018-02-17 10:39:23 +01:00
|
|
|
int result = (int)lua_tonumber(L, -1);
|
2015-01-30 20:37:14 +01:00
|
|
|
lua_pop(L, 1);
|
2018-02-17 10:39:23 +01:00
|
|
|
return result;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
}
|
2017-03-04 20:59:43 +01:00
|
|
|
lua_pop(L, 1);
|
2018-02-17 10:39:23 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
use_item_callback(unit *u, const item_type *itype, int amount, struct order *ord)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
char fname[64];
|
|
|
|
|
|
|
|
len = snprintf(fname, sizeof(fname), "use_%s", itype->rtype->_name);
|
|
|
|
if (len > 0 && (size_t)len < sizeof(fname)) {
|
|
|
|
int result;
|
2018-02-25 17:58:45 +01:00
|
|
|
int(*callout)(unit *, const item_type *, int, struct order *);
|
2018-02-17 10:39:23 +01:00
|
|
|
|
|
|
|
/* check if we have a register_item_use function */
|
|
|
|
callout = (int(*)(unit *, const item_type *, int, struct order *))get_function(fname);
|
|
|
|
if (callout) {
|
|
|
|
return callout(u, itype, amount, ord);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if we have a matching lua function */
|
|
|
|
result = lua_use_item(u, itype, fname, amount, ord);
|
|
|
|
if (result != 0) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the item is a potion, try use_potion, the generic function for
|
|
|
|
* potions that add an effect: */
|
|
|
|
if (itype->flags & ITF_POTION) {
|
|
|
|
return use_potion(u, itype, amount, ord);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("no such callout: %s", fname);
|
|
|
|
}
|
|
|
|
log_error("use(%s) calling '%s': not a function.\n", unitname(u), fname);
|
2017-12-10 12:23:28 +01:00
|
|
|
}
|
2018-02-17 10:39:23 +01:00
|
|
|
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-05-01 13:26:17 +02:00
|
|
|
/* compat code for old data files */
|
2017-09-21 16:26:53 +02:00
|
|
|
static int caldera_read(trigger *t, gamedata *data)
|
2017-05-01 13:26:17 +02:00
|
|
|
{
|
|
|
|
UNUSED_ARG(t);
|
2017-09-21 16:26:53 +02:00
|
|
|
READ_INT(data->store, NULL);
|
2017-05-01 13:26:17 +02:00
|
|
|
return AT_READ_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct trigger_type tt_caldera = {
|
|
|
|
"caldera",
|
|
|
|
NULL, NULL, NULL, NULL,
|
|
|
|
caldera_read
|
|
|
|
};
|
|
|
|
|
2017-12-09 21:20:20 +01:00
|
|
|
|
2018-02-09 21:20:43 +01:00
|
|
|
static int building_action_read(variant *var, void *owner, gamedata *data)
|
2017-12-09 21:20:20 +01:00
|
|
|
{
|
|
|
|
struct storage *store = data->store;
|
|
|
|
|
2017-12-28 18:55:45 +01:00
|
|
|
UNUSED_ARG(owner);
|
2018-02-09 21:20:43 +01:00
|
|
|
UNUSED_ARG(var);
|
2017-12-28 18:55:45 +01:00
|
|
|
|
2017-12-09 21:20:20 +01:00
|
|
|
if (data->version < ATTRIBOWNER_VERSION) {
|
|
|
|
READ_INT(data->store, NULL);
|
|
|
|
}
|
|
|
|
READ_TOK(store, NULL, 0);
|
|
|
|
READ_TOK(store, NULL, 0);
|
|
|
|
return AT_READ_DEPR;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void register_tolua_helpers(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-05-01 13:26:17 +02:00
|
|
|
tt_register(&tt_caldera);
|
2014-08-27 21:09:39 +02:00
|
|
|
at_register(&at_direction);
|
2017-12-09 21:20:20 +01:00
|
|
|
at_deprecate("lcbuilding", building_action_read);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-04-30 18:23:42 +02:00
|
|
|
callbacks.cast_spell = lua_callspell;
|
2018-02-17 10:39:23 +01:00
|
|
|
callbacks.use_item = use_item_callback;
|
2017-05-09 08:18:20 +02:00
|
|
|
callbacks.produce_resource = produce_resource_lua;
|
|
|
|
callbacks.limit_resource = limit_resource_lua;
|
2017-05-08 22:44:18 +02:00
|
|
|
|
|
|
|
register_function((pf_generic)lua_changeresource, "lua_changeresource");
|
2017-04-30 22:15:18 +02:00
|
|
|
register_item_give(lua_giveitem, "lua_giveitem");
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|