Yoba Perl
perl.cpp
1 #include "yobaperl/perl.hpp"
2 #include "yobaperl/perl_stack.hpp"
3 #include "yobaperl/callbacks.hpp"
4 
5 namespace yoba {
6 
7 
8 
9 int Perl::_last_id = -1;
10 std::vector<Perl *> Perl::_perls;
11 
12 
13 
14 Perl::Perl(bool do_init)
15  : _callbacks(*this)
16 {
17 #ifndef YOBAPERL_MULTIPLICITY
18  if(_last_id != -1)
19  throw PerlException(0, "Only one Perl instance allowed without -DYOBAPERL_MULTIPLICITY");
20 #endif
21 
22  _id = ++_last_id;
23  _perls.push_back(this);
24 
25  if(do_init)
26  init();
27 }
28 
29 Perl::Perl(Perl && to_move)
30  : _interpreter(to_move._interpreter),
31  _id(to_move._id),
32  _is_exceptions_enabled(to_move.isExceptionsEnabled()),
33  _is_warnings_enabled(to_move.isWarningsEnabled()),
34  _callbacks(std::move(to_move._callbacks))
35 {
36  to_move._interpreter = nullptr; // Skip destuctor
37 }
38 
40 {
41  // Skip destruction if object moved
42  if(_interpreter)
43  {
44  perl_destruct(_interpreter);
45  perl_free(_interpreter);
46  }
47 }
48 
49 
50 
52 {
53  return Scalar(*this, _createSV(), /* ref++ */false);
54 }
55 
56 Scalar Perl::newNamedScalar(const std::string & name)
57 {
58  return Scalar(*this, _getOrCreateNamedSV(name), /* ref++ */true);
59 }
60 
62 {
63  return Array(*this, _createAV(), /* ref++ */false);
64 }
65 
66 Array Perl::newArray(IV begin, IV end)
67 {
68  YOBAPERL_ASSERT(end > begin);
69 
70  Array result = newArray();
71  result.reserve(end - begin);
72  for(IV i = begin; i <= end; i++)
73  result.push(newScalar<IV>(i));
74 
75  return result;
76 }
77 
78 Array Perl::newNamedArray(const std::string & name)
79 {
80  return Array(*this, _getOrCreateNamedAV(name), /* ref++ */true);
81 }
82 
84 {
85  return Hash(*this, _createHV(), /* ref++ */false);
86 }
87 
88 Hash Perl::newNamedHash(const std::string & name)
89 {
90  return Hash(*this, _getOrCreateNamedHV(name), /* ref++ */true);
91 }
92 
93 Scalar Perl::newObject(const std::string & class_name)
94 {
95  return callMethod<Scalar>(class_name + "::new", newScalar(class_name));
96 }
97 
98 Scalar Perl::newObject(const std::string & class_name, const Scalar & param)
99 {
100  return callMethod<Scalar>(class_name + "::new", newScalar(class_name), param);
101 }
102 
103 Scalar Perl::newObject(const std::string & class_name, const Array & params)
104 {
105  return callMethod<Scalar>(class_name + "::new", newScalar(class_name), params);
106 }
107 
108 
109 
110 Scalar Perl::getScalar(const std::string & name)
111 {
112  return Scalar(*this, _getNamedSV(name), /* ref++ */true);
113 }
114 
115 Array Perl::getArray(const std::string & name)
116 {
117  return Array(*this, _getNamedAV(name), /* ref++ */true);
118 }
119 
120 Hash Perl::getHash(const std::string & name)
121 {
122  return Hash(*this, _getNamedHV(name), /* ref++ */true);
123 }
124 
125 Code Perl::getSubroutine(const std::string & name)
126 {
127  return Code(*this, _getNamedCV(name), /* ref++ */true);
128 }
129 
130 Code Perl::getMethod(const std::string & name)
131 {
132  return Code(*this, MUTABLE_CV(newScalar(name).detachSV()), true);
133 }
134 
135 ScalarPtr Perl::getScalarPtr(const std::string & name)
136 {
137  return ScalarPtr(new Scalar(*this, _getNamedSV(name), /* ref++ */true));
138 }
139 
140 ArrayPtr Perl::getArrayPtr(const std::string & name)
141 {
142  return ArrayPtr(new Array(*this, _getNamedAV(name), /* ref++ */true));
143 }
144 
145 HashPtr Perl::getHashPtr(const std::string & name)
146 {
147  return HashPtr(new Hash(*this, _getNamedHV(name), /* ref++ */true));
148 }
149 
150 CodePtr Perl::getSubroutinePtr(const std::string & name)
151 {
152  return CodePtr(new Code(*this, _getNamedCV(name), /* ref++ */true));
153 }
154 
155 
156 
158 {
159  static const char * argv[] = { "", "-f", "-e", "0" };
160  static const int argc = 4;
161 
162  // First object
163  if(getId() == 0)
164  {
165  PERL_SYS_INIT((int *)&argc, (char ***)&argv);
166  std::atexit(Perl_sys_term);
167  }
168 
169  _interpreter = perl_alloc();
170  PERL_SET_CONTEXT(_interpreter);
171 
172  perl_construct(_interpreter);
173 
174  PL_exit_flags |= PERL_EXIT_DESTRUCT_END;
175  PL_perl_destruct_level = 2;
176 
177  if(0 != perl_parse(_interpreter, xs_init, argc, (char **)argv, (char **)NULL))
178  throw PerlException(getId(), "Init error");
179 
180 #ifdef YOBAPERL_MULTIPLICITY
181  Hash(*this, PL_modglobal, true).insert("YobaPerl::id", newScalar(getId()));
182 #endif
183 
184  return *this;
185 }
186 
187 Perl & Perl::lib(const std::string & path)
188 {
189  getArray("INC").unshift(newScalar(path));
190  return *this;
191 }
192 
193 Perl & Perl::use(const std::string & name)
194 {
195 
196  // load_module(0, newSVpv(name.c_str(),name.length()), NULL, NULL);
197  eval("use " + name);
198 
199  if(hasError())
200  {
201  if(isExceptionsEnabled())
202  throw PerlException(getId(), getError());
203 
204  if(isWarningsEnabled())
205  warn("%s", getError().c_str());
206  }
207 
208  return *this;
209 }
210 
211 //TODO Untested
212 Perl & Perl::no(const std::string & name)
213 {
214  load_module(PERL_LOADMOD_DENY, newSVpv(name.c_str(),name.length()), NULL, NULL);
215  return *this;
216 }
217 
218 Perl & Perl::require(const std::string & name)
219 {
220 // require_pv(name.c_str());
221  eval(std::string("require ") + name);
222 
223  if(hasError())
224  {
225  if(isExceptionsEnabled())
226  throw PerlException(getId(), getError());
227 
228  if(isWarningsEnabled())
229  warn("%s", getError().c_str());
230  }
231 
232  return *this;
233 }
234 
236 {
237  _is_exceptions_enabled = state;
238  return *this;
239 }
240 
242 {
243  _is_warnings_enabled = state;
244  return *this;
245 }
246 
247 bool Perl::hasError() const
248 {
249  return SvTRUE_NN(ERRSV);
250 }
251 
252 std::string Perl::getError() const
253 {
254  return SvPVx_nolen_const(ERRSV);
255 }
256 
257 
258 
259 Perl & Perl::registerVoidToVoid(const std::string & name, priv::Callbacks::VoidToVoidCB function)
260 {
261  _callbacks.setVoidToVoid(name, function);
262  return *this;
263 }
264 
265 Perl & Perl::registerVoidToScalar(const std::string & name, priv::Callbacks::VoidToScalarCB function)
266 {
267  _callbacks.setVoidToScalar(name, function);
268  return *this;
269 }
270 
271 Perl & Perl::registerScalarToVoid(const std::string & name, priv::Callbacks::ScalarToVoidCB function)
272 {
273  _callbacks.setScalarToVoid(name, function);
274  return *this;
275 }
276 
277 Perl & Perl::registerScalarToScalar(const std::string & name, priv::Callbacks::ScalarToScalarCB function)
278 {
279  _callbacks.setScalarToScalar(name, function);
280  return *this;
281 }
282 
283 CV * Perl::registerStatic(const std::string & sub_name, void(*function)(PerlInterpreter *, CV *))
284 {
285  CV * sub = Perl_newXS_len_flags(getInterpreter(), sub_name.c_str(), sub_name.length(), function,
286  /*filename*/NULL, /*proto*/NULL, /*svp*/NULL, /*flags*/0);
287  YOBAPERL_ASSERT(sub);
288  return sub;
289 }
290 
291 priv::Callbacks & Perl::getCallbacks()
292 {
293  return _callbacks;
294 }
295 
296 
297 
298 int Perl::getId() const
299 {
300  return _id;
301 }
302 
303 PerlInterpreter * Perl::getInterpreter() const
304 {
305  return _interpreter;
306 }
307 
308 SV * Perl::getNamedSV(const std::string & name) const
309 {
310  return _getNamedSV(name);
311 }
312 
313 AV * Perl::getNamedAV(const std::string & name) const
314 {
315  return _getNamedAV(name);
316 }
317 
318 HV * Perl::getNamedHV(const std::string & name) const
319 {
320  return _getNamedHV(name);
321 }
322 
323 CV * Perl::getNamedCV(const std::string & name) const
324 {
325  return _getNamedCV(name);
326 }
327 
328 bool Perl::isExceptionsEnabled() const
329 {
330  return _is_exceptions_enabled;
331 }
332 
333 bool Perl::isWarningsEnabled() const
334 {
335  return _is_warnings_enabled;
336 }
337 
338 std::string Perl::deparse(const Code & code)
339 {
340  // Static block
341  if(!_code_deparser)
342  {
343  require("B::Deparse");
344  _code_deparser.reset(new Scalar(newObject("B::Deparse", newArray({ "-p", "-sC" }))));
345 // _code_deparser.reset(new Scalar(eval<Scalar>("B::Deparse->new('-p', '-sC')")));
346  }
347 
348  return callMethod<Scalar>("coderef2text", *_code_deparser, code.makeRef()).toString();
349 }
350 
352 {
353  PERL_SET_CONTEXT(_interpreter);
354  return *this;
355 }
356 
358 {
359  return *_perls.at(id);
360 }
361 
362 Perl & Perl::getInstance(PerlInterpreter * _interpreter)
363 {
364 #ifdef YOBAPERL_MULTIPLICITY
365  static const std::string name = "YobaPerl::id";
366 
367  SV ** elements = hv_fetch(PL_modglobal, name.c_str(), name.length(), NULL);
368  YOBAPERL_ASSERT(elements);
369 
370  SV * element = *elements;
371  YOBAPERL_ASSERT(element);
372 
373  return getInstanceById(SvIV(element));
374 #else
375  return getInstanceById(0);
376 #endif
377 }
378 
379 
380 
381 VariablePtr Perl::operator[] (std::string query)
382 {
383  if(query.size() < 2)
384  throw PerlException(getId(), "[]");
385 
386  char prefix = query[0];
387  query.erase(0, 1);
388 
389  switch(prefix)
390  {
391  case '$': return getScalarPtr(query);
392  case '@': return getArrayPtr(query);
393  case '%': return getHashPtr(query);
394  case '&': return getSubroutinePtr(query);
395 
396  default: throw PerlException(getId(), "[]");
397  }
398 }
399 
400 
401 
402 SV * Perl::_createSV(STRLEN size)
403 {
404  SV * result = newSV(size);
405  YOBAPERL_ASSERT(result);
406  return result;
407 }
408 
409 AV * Perl::_createAV(SSize_t size, SV ** elements)
410 {
411  AV * result = nullptr;
412  if(size)
413  result = av_make(size, elements);
414  else
415  result = newAV();
416 
417  YOBAPERL_ASSERT(result);
418  return result;
419 }
420 
421 HV * Perl::_createHV()
422 {
423  HV * result = newHV();
424  YOBAPERL_ASSERT(result);
425  return result;
426 }
427 
428 SV * Perl::_getOrCreateNamedSV(const std::string & name) const
429 {
430  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), GV_ADD, SVt_PV);
431  YOBAPERL_ASSERT(gv);
432  return GvSV(gv);
433 }
434 
435 AV * Perl::_getOrCreateNamedAV(const std::string & name) const
436 {
437  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), GV_ADD, SVt_PVAV);
438  YOBAPERL_ASSERT(gv);
439  return GvAV(gv);
440 }
441 
442 HV * Perl::_getOrCreateNamedHV(const std::string & name) const
443 {
444  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), GV_ADD, SVt_PVHV);
445  YOBAPERL_ASSERT(gv);
446  return GvHV(gv);
447 }
448 
449 SV * Perl::_getNamedSV(const std::string & name) const
450 {
451  if(isExceptionsEnabled() || isWarningsEnabled())
452  {
453  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), /*flags*/0, SVt_PV);
454  if(gv == NULL)
455  {
456  std::string error = "Undefined symbol: $" + name;
457 
458  if(isExceptionsEnabled())
459  throw PerlException(getId(), error);
460 
461  if(isWarningsEnabled())
462  warn("%s", error.c_str());
463  }
464  }
465 
466  return _getOrCreateNamedSV(name);
467 }
468 
469 AV * Perl::_getNamedAV(const std::string & name) const
470 {
471  if(isExceptionsEnabled() || isWarningsEnabled())
472  {
473  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), /*flags*/0, SVt_PVAV);
474 
475  if(gv == NULL)
476  {
477  std::string error = "Undefined symbol: @" + name;
478 
479  if(isExceptionsEnabled())
480  throw PerlException(getId(), error);
481 
482  if(isWarningsEnabled())
483  warn("%s", error.c_str());
484  }
485 
486  return GvAV(gv);
487  }
488 
489  return _getOrCreateNamedAV(name);
490 }
491 
492 HV * Perl::_getNamedHV(const std::string & name) const
493 {
494  if(isExceptionsEnabled() || isWarningsEnabled())
495  {
496  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), /*flags*/0, SVt_PVHV);
497 
498  if(gv == NULL)
499  {
500  std::string error = "Undefined symbol: %" + name;
501 
502  if(isExceptionsEnabled())
503  throw PerlException(getId(), error);
504 
505  if(isWarningsEnabled())
506  warn("%s", error.c_str());
507  }
508 
509  return GvHV(gv);
510  }
511 
512  return _getOrCreateNamedHV(name);
513 }
514 
515 CV * Perl::_getNamedCV(const std::string & name) const
516 {
517  GV * gv = gv_fetchpvn_flags(name.c_str(), name.length(), /*flags*/0, SVt_PVCV);
518 
519  if(gv == NULL)
520  {
521  //TODO Complete
522  std::string error = "Undefined symbol: &" + name;
523 
524 // if(isExceptionsEnabled())
525  throw PerlException(getId(), error);
526 
527 // if(isWarningsEnabled())
528 // warn("%s", error.c_str());
529  }
530 
531  return GvCV(gv);
532 }
533 
534 
535 
536 } // namespace yoba
Array reference
Definition: array.hpp:26
bool hasError() const
Check Perl error.
Definition: perl.cpp:247
CodePtr getSubroutinePtr(const std::string &name)
Place subroutine reference in heap.
Definition: perl.cpp:150
std::string deparse(const Code &code)
Convert subroutine to string using B::Deparse.
Definition: perl.cpp:338
AV * getNamedAV(const std::string &name) const
Get raw array by name.
Definition: perl.cpp:313
Hash getHash(const std::string &name)
Get hash by name.
Definition: perl.cpp:120
Perl & init()
Initialize the Perl interpreter.
Definition: perl.cpp:157
Array getArray(const std::string &name)
Get array by name.
Definition: perl.cpp:115
Array & reserve(SSize_t size)
Reserve space.
Definition: array.cpp:213
ArrayPtr getArrayPtr(const std::string &name)
Place array reference in heap.
Definition: perl.cpp:140
~Perl()
Destructor.
Definition: perl.cpp:39
Array & push(const Scalar &scalar)
Add element to end.
Definition: array.cpp:75
Definition: array.cpp:5
HashPtr getHashPtr(const std::string &name)
Place hash reference in heap.
Definition: perl.cpp:145
STL namespace.
Perl & setExceptionsEnabled(bool state)
Enable/disable all exceptions. Default: enabled.
Definition: perl.cpp:235
Perl & no(const std::string &name)
Unload a module.
Definition: perl.cpp:212
HV * getNamedHV(const std::string &name) const
Get raw hash by name.
Definition: perl.cpp:318
static Perl & getInstanceById(int perl_id)
Get Perl instance by object id.
Definition: perl.cpp:357
Array newNamedArray(const std::string &name)
Create empty array as @name.
Definition: perl.cpp:78
Scalar getScalar(const std::string &name)
Get scalar by name.
Definition: perl.cpp:110
Perl & setWarningsEnabled(bool state)
Enable/disable all warnings. Default: disabled.
Definition: perl.cpp:241
Hash reference
Definition: hash.hpp:22
Code getSubroutine(const std::string &name)
Get subroutine by name.
Definition: perl.cpp:125
Perl & use(const std::string &name)
Import a module.
Definition: perl.cpp:193
Hash & insert(const std::string &key, const Scalar &value)
Add element.
Definition: hash.cpp:57
SV * getNamedSV(const std::string &name) const
Get raw scalar by name.
Definition: perl.cpp:308
VariablePtr operator[](std::string query)
Universal operator.
Definition: perl.cpp:381
ReturnT eval(const std::string &code)
Evaluate string in generic context.
Array newArray()
Create anonymous empty array.
Definition: perl.cpp:61
Scalar newNamedScalar(const std::string &name)
Create empty scalar as $name.
Definition: perl.cpp:56
PerlInterpreter * getInterpreter() const
Raw interpreter.
Definition: perl.cpp:303
Scalar newScalar()
Create anonymous empty scalar.
Definition: perl.cpp:51
Main class.
Definition: perl.hpp:32
Perl & require(const std::string &name)
Load external file.
Definition: perl.cpp:218
Code getMethod(const std::string &name)
Get class method by name.
Definition: perl.cpp:130
Hash newNamedHash(const std::string &name)
Create empty hash as %name.
Definition: perl.cpp:88
Scalar newObject(const std::string &class_name)
Create new object.
Definition: perl.cpp:93
static Perl & getInstance(PerlInterpreter *_interpreter)
Get Perl instance by associated interpreter.
Definition: perl.cpp:362
int getId() const
Perl object id.
Definition: perl.cpp:298
Perl exception.
Subroutine reference
Definition: code.hpp:15
Perl(bool do_init=true)
Default constructor.
Definition: perl.cpp:14
Array & unshift(const Scalar &scalar)
Add element to begin.
Definition: array.cpp:105
Perl & lib(const std::string &path)
Add folder to module search path.
Definition: perl.cpp:187
CV * getNamedCV(const std::string &name) const
Get raw subroutine by name.
Definition: perl.cpp:323
Perl & setContext()
Set interpreter context.
Definition: perl.cpp:351
Scalar reference
Definition: scalar.hpp:24
ScalarPtr getScalarPtr(const std::string &name)
Place scalar reference in heap.
Definition: perl.cpp:135
Hash newHash()
Create anonymous empty hash.
Definition: perl.cpp:83
Scalar makeRef() const
Take reference.
Definition: code.cpp:20