eqemu-server/zone/embperl.h
Alex King 331e04fbf8
[Quests] Cleanup Quest Parser Logic (#4025)
* [Quests] Cleanup Quest Parser Logic

# Notes
- Consolidate duplicated code into loops to make it easier to read and easier to add on to.

# Images

* Update quest_parser_collection.cpp

* Push

* Push

* Update embperl.cpp

* Push

* Additional cleanup, use File::Exists utility

* Range based loops

* Update embparser.cpp

* Cleanup

* Update embparser.cpp

---------

Co-authored-by: Akkadius <akkadius1@gmail.com>
2024-02-05 19:53:16 -05:00

153 lines
3.2 KiB
C++

/*
Embperl.h
---------------
eqemu perl wrapper
Eglin
*/
#ifndef EMBPERL_H
#define EMBPERL_H
#ifdef EMBPERL
#include "zone_config.h"
#include <string>
#include <vector>
#include <map>
#include <stdio.h>
#include <string.h>
#include <perlbind/perlbind.h>
namespace perl = perlbind;
#undef Null
#ifdef WIN32
#define snprintf _snprintf
#endif
//perl defines these macros and dosent clean them up, lazy bastards. -- I hate them too!
#ifdef Copy
#undef Copy
#endif
#ifdef list
#undef list
#endif
#ifdef write
#undef write
#endif
#ifdef bool
#undef bool
#endif
#ifdef Zero
#undef Zero
#endif
//These need to be cleaned up on FreeBSD
#ifdef __FreeBSD__
#ifdef do_open
#undef do_open
#endif
#ifdef do_close
#undef do_close
#endif
#endif
//so embedded scripts can use xs extensions (ala 'use socket;')
EXTERN_C void boot_DynaLoader(pTHX_ CV* cv);
EXTERN_C void xs_init(pTHX);
extern const ZoneConfig *Config;
class Embperl
{
private:
//if we fail inside a script evaluation, this will hold the croak msg (not much help if we die during construction, but that's our own fault)
mutable std::string errmsg;
//install a perl func
void init_eval_file(void);
protected:
//the embedded interpreter
PerlInterpreter * my_perl;
void DoInit();
public:
Embperl(void); //This can throw errors! Buyer beware
~Embperl(void);
void Reinit();
//evaluate an expression. throws string errors on fail
int eval(const char* code);
//execute a subroutine. throws lasterr on failure
int dosub(const char* sub_name, const std::vector<std::string>* args = nullptr, int mode = G_SCALAR | G_EVAL);
//put an integer into a perl varable
void seti(const char* variable_name, int val) const
{
SV* t = get_sv(variable_name, true);
sv_setiv(t, val);
}
//put a real into a perl varable
void setd(const char* variable_name, float val) const
{
SV* t = get_sv(variable_name, true);
sv_setnv(t, val);
}
//put a string into a perl varable
void setstr(const char* variable_name, const char* val) const
{
SV* t = get_sv(variable_name, true);
sv_setpv(t, val);
}
// put a pointer into a blessed perl variable
void setptr(const char* variable_name, const char* class_name, void* val) const
{
SV* t = get_sv(variable_name, GV_ADD);
sv_setref_pv(t, class_name, val);
}
// put key-value pairs in hash
void sethash(const char* variable_name, std::map<std::string, std::string>& vals)
{
std::map<std::string, std::string>::iterator it;
// Get hash and clear it.
HV* hv = get_hv(variable_name, TRUE);
hv_clear(hv);
// Iterate through key-value pairs, storing them in hash
for (it = vals.begin(); it != vals.end(); ++it) {
int key_length = static_cast<int>(it->first.length());
SV* val = newSVpv(it->second.c_str(), it->second.length());
// If val was not added to hash, reset reference count
if (!hv_store(hv, it->first.c_str(), key_length, val, 0)) {
val->sv_refcnt = 0;
}
}
}
//loads a file and compiles it into our interpreter (assuming it hasn't already been read in)
//idea borrowed from perlembed
int eval_file(const char* package_name, const char* filename);
//check to see if a sub exists in package
bool SubExists(const char* package, const char* sub);
};
#endif //EMBPERL
#endif //EMBPERL_H