359 lines
11 KiB
OpenEdge ABL
359 lines
11 KiB
OpenEdge ABL
// supports Java and Python
|
|
|
|
|
|
/* example.i */
|
|
#if defined(SWIGPYTHON)
|
|
%module(directors = "1", package="yap4py") yap
|
|
#else
|
|
%module(directors = "1") yap
|
|
#endif
|
|
|
|
// Language independent exception handler
|
|
%include exception.i
|
|
%include stdint.i
|
|
%include std_string.i
|
|
%include std_vector.i
|
|
|
|
#if __ANDROID__
|
|
%include "arrays_java.i";
|
|
// %inline %{
|
|
static YAPTerm Temp[1];
|
|
// %}
|
|
// %typemap(javapackage) std::vector<YAPTerm> "pt.up.yap.YAPTerm"
|
|
//%template(VectorOfTerm) std::vector<YAPTerm>;
|
|
#endif
|
|
|
|
%feature("novaluewrapper") std::vector<Term>;
|
|
|
|
%ignore *::operator[];
|
|
|
|
class YAPAtom;
|
|
class YAPPredicate;
|
|
class YAPEngine;
|
|
|
|
|
|
#if defined(SWIGPYTHON)
|
|
|
|
%pythoncode %{
|
|
YAPError = _yap.YAPError
|
|
%}
|
|
|
|
%typemap(typecheck) Term* {
|
|
$1 = PySequence_Check($input);
|
|
}
|
|
|
|
// Map a Python sequence into any sized C double array
|
|
%typemap(in) Term* {
|
|
int i;
|
|
if (!PySequence_Check($input)) {
|
|
PyErr_SetString(PyExc_TypeError,"Expecting a sequence");
|
|
$1 = nullptr;
|
|
} else {
|
|
int sz = PyObject_Length($input);
|
|
std::vector<Term> v(sz);
|
|
for (i =0; i < sz; i++) {
|
|
PyObject *o = PySequence_GetItem($input,i);
|
|
v[i] = Term(pythonToYAP(o));
|
|
//Py_DECREF(o);
|
|
}
|
|
$1 = &v[0];
|
|
}
|
|
}
|
|
|
|
|
|
//%typemap(check) YAPEngine::fun(Term) { $1 = 1; }
|
|
//%typemap(check) YAPEngine::fun(YAPTerm) { $1 = 0; }
|
|
|
|
|
|
%typemap(in) Int { if (PyLong_Check($input)) {
|
|
$1 = PyLong_AsLong($input);}
|
|
}
|
|
|
|
//%typemap(in) double { if (PyFloat_Check($input)) {
|
|
// $1 = PyFloat_AsDouble($input); } }
|
|
|
|
//%typemap(in) char const * { if (PyUnicode_Check($input)) {
|
|
// $1 = PyUnicode_AsUTF8($input); } }
|
|
|
|
//%typemap(in) YAPTerm { $1 = new YAPTerm(pythonToYAP($input)); PyErr_Clear(); }
|
|
%typemap(in) YAP_Term { $1 = pythonToYAP($input); PyErr_Clear(); }
|
|
%typemap(in) Term { $1 = pythonToYAP($input); PyErr_Clear(); }
|
|
%typemap(in) YAPTerm { YAPTerm(($1 = pythonToYAP($input))); PyErr_Clear(); }
|
|
|
|
%typecheck(2) Int { $1 = PyLong_Check($input); }
|
|
%typecheck(3) double { $1 = PyFloat_Check($input); }
|
|
%typecheck(2) const char * { $1 = PyUnicode_Check($input); }
|
|
|
|
%typecheck(1) Term { $1 = !PyUnicode_Check($input); }
|
|
%typecheck(1) YAP_Term { $1 = PyUnicode_Check($input); }
|
|
|
|
%typecheck(0) YAPTerm { $1 = !PyUnicode_Check($input); }
|
|
|
|
|
|
%typemap(out) YAP_Term { return $result = yap_to_python($1, false, 0, true); }
|
|
|
|
%typemap(out) Term { return $result = yap_to_python($1, false, 0, true); }
|
|
|
|
%typemap(out) std::vector<Term> {
|
|
size_t len = $1.size();
|
|
$result = PyList_New(len);
|
|
for (size_t i = 0; i< len; i++) {
|
|
PyObject *o = yap_to_python($1[i],false,0,true);
|
|
PyList_SetItem($result,i,o);
|
|
|
|
}
|
|
return $result; }
|
|
|
|
// Language independent exception handler
|
|
|
|
%exception {
|
|
try {
|
|
$action
|
|
} catch (YAPError &e) {
|
|
YAPPycatch(e);
|
|
SWIG_fail;
|
|
}
|
|
}
|
|
|
|
%typecheck(2) Int { $1 = PyLong_Check($input); }
|
|
%typecheck(3) double { $1 = PyFloat_Check($input); }
|
|
%typecheck(2) const char * { $1 = PyUnicode_Check($input); }
|
|
|
|
%typecheck(1) Term { $1 = !PyUnicode_Check($input); }
|
|
%typecheck(1) YAP_Term { $1 = PyUnicode_Check($input); }
|
|
|
|
%typecheck(0) YAPTerm { $1 = !PyUnicode_Check($input); }
|
|
|
|
#else
|
|
|
|
%typemap(in) arity_t { (jlong)($input); }
|
|
|
|
|
|
|
|
%typemap(in) jlong %{
|
|
$1 = (jlong)$input;
|
|
%}
|
|
|
|
%typemap(out) arity_t { *(jlong *)&$result = $1; }
|
|
|
|
// Language independent exception handler
|
|
// simplified version
|
|
%include <exception.i>
|
|
|
|
%exception {
|
|
try {
|
|
$action
|
|
} catch (const std::out_of_range& e) {
|
|
SWIG_exception(SWIG_IndexError, e.what());
|
|
} catch (const std::exception& e) {
|
|
SWIG_exception(SWIG_RuntimeError, e.what());
|
|
} catch (...) {
|
|
SWIG_exception(SWIG_RuntimeError, "unknown exception");
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
%{
|
|
/* Put header files here or function declarations like below */
|
|
#include "yapi.hh"
|
|
|
|
|
|
extern "C" {
|
|
extern void Yap_PrintException(yap_error_descriptor_t *i);
|
|
|
|
#if THREADS
|
|
#define Yap_regp regcache
|
|
#endif
|
|
|
|
// we cannot consult YapInterface.h, that conflicts with what we
|
|
// declare, though
|
|
// it shouldn't
|
|
}
|
|
|
|
extern void init_sqlite();
|
|
|
|
|
|
#define X_API
|
|
|
|
|
|
#include <cmath>
|
|
#include <gmpxx.h>
|
|
|
|
extern "C"{
|
|
#include "Yap.h"
|
|
|
|
X_API extern Term YAP_MkCharPTerm( char *n);
|
|
|
|
#if defined(SWIGPYTHON)
|
|
|
|
#include <py4yap.h>
|
|
|
|
X_API extern PyObject * pYAPError;
|
|
|
|
extern inline PyObject *AtomToPy(const char *s) {
|
|
if (strcmp(s, "true") == 0)
|
|
return Py_True;
|
|
if (strcmp(s, "false") == 0)
|
|
return Py_False;
|
|
if (strcmp(s, "none") == 0)
|
|
return Py_None;
|
|
if (strcmp(s, "[]") == 0)
|
|
return PyList_New(0);
|
|
else if (strcmp(s, "{}") == 0)
|
|
return PyDict_New();
|
|
/* return __main__,s */
|
|
else if (PyObject_HasAttrString(py_Main, s)) {
|
|
return PyObject_GetAttrString(py_Main, s);
|
|
}
|
|
// no way to translate
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
YAPPycatch(YAPError &e)
|
|
{
|
|
|
|
yap_error_number en = e.getID();
|
|
switch (e.getErrorClass()) {
|
|
case YAPC_NO_ERROR:
|
|
break;
|
|
/// bad domain, "first argument often is the predicate.
|
|
case DOMAIN_ERROR: {
|
|
switch (en) {
|
|
case DOMAIN_ERROR_OUT_OF_RANGE:
|
|
pYAPError = PyExc_GeneratorExit;
|
|
break;
|
|
case DOMAIN_ERROR_NOT_LESS_THAN_ZERO:
|
|
pYAPError = PyExc_IndexError;
|
|
break;
|
|
case DOMAIN_ERROR_CLOSE_OPTION:
|
|
case DOMAIN_ERROR_ENCODING:
|
|
case DOMAIN_ERROR_PROLOG_FLAG:
|
|
case DOMAIN_ERROR_ABSOLUTE_FILE_NAME_OPTION:
|
|
case DOMAIN_ERROR_READ_OPTION:
|
|
case DOMAIN_ERROR_SET_STREAM_OPTION:
|
|
pYAPError = PyExc_KeyError;
|
|
break;
|
|
case DOMAIN_ERROR_FILE_ERRORS:
|
|
case DOMAIN_ERROR_FILE_TYPE:
|
|
case DOMAIN_ERROR_IO_MODE:
|
|
case DOMAIN_ERROR_SOURCE_SINK:
|
|
case DOMAIN_ERROR_STREAM_POSITION:
|
|
pYAPError = PyExc_IOError;
|
|
break;
|
|
default:
|
|
pYAPError = PyExc_ValueError;
|
|
}
|
|
} break;
|
|
/// bad arithmetic
|
|
case EVALUATION_ERROR: {
|
|
switch (en) {
|
|
case EVALUATION_ERROR_FLOAT_OVERFLOW:
|
|
case EVALUATION_ERROR_INT_OVERFLOW:
|
|
pYAPError = PyExc_OverflowError;
|
|
break;
|
|
case EVALUATION_ERROR_FLOAT_UNDERFLOW:
|
|
case EVALUATION_ERROR_UNDERFLOW:
|
|
case EVALUATION_ERROR_ZERO_DIVISOR:
|
|
pYAPError = PyExc_ArithmeticError;
|
|
break;
|
|
default:
|
|
pYAPError = PyExc_RuntimeError;
|
|
}
|
|
} break;
|
|
/// missing object (I/O mostly)
|
|
case EXISTENCE_ERROR:
|
|
pYAPError = PyExc_NotImplementedError;
|
|
break;
|
|
/// should be bound
|
|
case INSTANTIATION_ERROR_CLASS:
|
|
pYAPError = PyExc_RuntimeError;
|
|
break;
|
|
/// bad access, I/O
|
|
case PERMISSION_ERROR: {
|
|
switch (en) {
|
|
case PERMISSION_ERROR_INPUT_BINARY_STREAM:
|
|
case PERMISSION_ERROR_INPUT_PAST_END_OF_STREAM:
|
|
case PERMISSION_ERROR_INPUT_STREAM:
|
|
case PERMISSION_ERROR_INPUT_TEXT_STREAM:
|
|
case PERMISSION_ERROR_OPEN_SOURCE_SINK:
|
|
case PERMISSION_ERROR_OUTPUT_BINARY_STREAM:
|
|
case PERMISSION_ERROR_REPOSITION_STREAM:
|
|
case PERMISSION_ERROR_OUTPUT_STREAM:
|
|
case PERMISSION_ERROR_OUTPUT_TEXT_STREAM:
|
|
pYAPError = PyExc_OverflowError;
|
|
break;
|
|
default:
|
|
pYAPError = PyExc_RuntimeError;
|
|
}
|
|
} break;
|
|
/// something that could not be represented into a type
|
|
case REPRESENTATION_ERROR:
|
|
pYAPError = PyExc_RuntimeError;
|
|
break;
|
|
/// not enough ....
|
|
case RESOURCE_ERROR:
|
|
pYAPError = PyExc_RuntimeError;
|
|
break;
|
|
/// bad text
|
|
case SYNTAX_ERROR_CLASS:
|
|
pYAPError = PyExc_SyntaxError;
|
|
break;
|
|
/// OS or internal
|
|
case SYSTEM_ERROR_CLASS:
|
|
pYAPError = PyExc_RuntimeError;
|
|
break;
|
|
/// bad typing
|
|
case TYPE_ERROR:
|
|
pYAPError = PyExc_TypeError;
|
|
break;
|
|
/// should be unbound
|
|
case UNINSTANTIATION_ERROR_CLASS:
|
|
pYAPError = PyExc_RuntimeError;
|
|
break;
|
|
/// escape hatch
|
|
default:
|
|
break;
|
|
}
|
|
PyErr_SetString(pYAPError, e.text().c_str());
|
|
}
|
|
#endif
|
|
}
|
|
|
|
%}
|
|
|
|
/* turn on director wrapping Callback */
|
|
//%feature("director") YAPCallback;
|
|
|
|
%include "yapa.hh"
|
|
|
|
%include "yapie.hh"
|
|
|
|
%include "yapt.hh"
|
|
|
|
%include "yapdb.hh"
|
|
|
|
%include "yapq.hh"
|
|
|
|
|
|
namespace std {
|
|
|
|
%template(TermVector) vector<Term>;
|
|
%feature("novaluewrapper") vector<Term>;
|
|
|
|
%template(YAPTermVector) vector<YAPTerm>;
|
|
%feature("novaluewrapper") vector<YAPTerm>;
|
|
|
|
|
|
|
|
};
|
|
|
|
%init %{
|
|
PyObject * pYAPError = PyErr_NewException("_yap.YAPError", NULL, NULL);
|
|
Py_INCREF(pYAPError);
|
|
PyModule_AddObject(m, "YAPError", pYAPError);
|
|
|
|
%}
|