layer1/PConv.h (227 lines of code) (raw):
/*
A* -------------------------------------------------------------------
B* This file contains source code for the PyMOL computer program
C* copyright 1998-2000 by Warren Lyford Delano of DeLano Scientific.
D* -------------------------------------------------------------------
E* It is unlawful to modify or remove this copyright notice.
F* -------------------------------------------------------------------
G* Please see the accompanying LICENSE file for further information.
H* -------------------------------------------------------------------
I* Additional authors of this source file include:
-*
-*
-*
Z* -------------------------------------------------------------------
*/
#ifndef _H_PConv
#define _H_PConv
#include"os_python.h"
#include"PyMOLGlobals.h"
#include"Base.h"
#include"OVLexicon.h"
#include <map>
#include <set>
#include <string>
#include <vector>
/* Convenient conversion routines for C<->Python data interchange
Note that all of these routines assume that we have the global
interpreter lock - blocking all other threads.
There are three ways to get it:
- call PBlock() [followe by PUnblock() when done]
- call PBlockAndUnlockAPI - [followed by PLockAPIAndUnblock() when
done]
- or in response to a call to the PM API, you will have the main
Python thread by default. [Note that within an
APIEntry(),APIExit() block the lock is released, so these
functions should be called outside of that block].
*/
// CPythonVal macros
#define CPythonVal PyObject
#define CPythonVal_PyString_Check PyString_Check
#define CPythonVal_PyList_Check PyList_Check
#define CPythonVal_PyList_Size PyList_Size
#define CPythonVal_PyList_GetItem(G, list, i) PyList_GetItem(list, i)
#define CPythonVal_PyDict_GetItemString(G, p, key) PyDict_GetItemString(p, key)
#define CPythonVal_PConvPyIntToInt PConvPyIntToInt
#define CPythonVal_PConvPyIntToInt_From_List(G, list, i, ptr) PConvPyIntToInt(PyList_GetItem(list, i), ptr)
#define CPythonVal_PConvPyFloatToFloat_From_List(G, list, i, ptr) PConvPyFloatToFloat(PyList_GetItem(list, i), ptr)
#define CPythonVal_PConvPyListToIntArrayInPlace(G, obj, ff, ll) PConvPyListToIntArrayInPlace(obj, ff, ll)
#define CPythonVal_PConvPyListToIntArrayInPlace_From_List(G, list, i, ff, ll) PConvPyListToIntArrayInPlace(PyList_GetItem(list, i), ff, ll)
#define CPythonVal_PConvPyListToFloatArrayInPlaceAutoZero_From_List(G, list, i, ...) \
PConvPyListToFloatArrayInPlaceAutoZero(PyList_GetItem(list, i), __VA_ARGS__)
#define CPythonVal_PConvPyListToFloatVLANoneOkay_From_List(G, list, i, f) PConvPyListToFloatVLANoneOkay(PyList_GetItem(list, i), f)
#define CPythonVal_PConvPyListToLabPosVLA(G, obj, vla_ptr) PConvPyListToLabPosVLA(obj, vla_ptr)
#define CPythonVal_PConvPyStrToStr_From_List(G, list, i, ptr, l) PConvPyStrToStr(PyList_GetItem(list, i), ptr, l)
#define CPythonVal_Free(obj)
#define CPythonVal_FreeAll(PYOBJECT)
#define CPythonVal_New(G, PYOBJECT) PYOBJECT
#define CPythonVal_Append_List(LIST, ITEM) PyList_Append(LIST, ITEM)
#define CPythonVal_New_List() PyList_New(0)
#define CPythonVal_New_Tuple(SIZE) PyTuple_New(SIZE)
#define CPythonVal_Tuple_SetItem(TUPLE, ITEM, VAL) PyTuple_SetItem(TUPLE, ITEM, VAL)
#define CPythonVal_New_String(BUF, LEN) PyString_FromStringAndSize(BUF, LEN)
#define CPythonVal_New_Boolean(VAL) (VAL ? PyBool_FromLong(1) : PyBool_FromLong(0))
#define CPythonVal_New_Integer(VAL) PyInt_FromLong(VAL)
#define CPythonVal_New_Float(VAL) PyFloat_FromDouble(VAL)
#define CPythonVal_IsNone(PYOBJECT) (PYOBJECT == Py_None)
/* == error-checking routines: true = success, false = failure. */
/* NOTE: the string routines will write strings up to the specified
* length, PLUS a NULL...so watch out for array overruns */
int PConvAttrToStrMaxLen(PyObject * obj, const char *attr, char *str, ov_size ll);
int PConvPyListToBitmask(PyObject * obj, int *bitmask, ov_size ll);
int PConvPyListToExtent(PyObject * obj, float *mn, float *mx);
int PConvAttrToFloatArrayInPlace(PyObject * obj, const char *attr, float *ff, ov_size ll);
int PConvAttrToIntArrayInPlace(PyObject * obj, const char *attr, int *ff, ov_size ll);
int PConvAttrToPtr(PyObject * obj, const char *name, void **cobj);
int PConvCObjectToPtr(PyObject * obj, void **ptr);
int PConvPyListToStrVLAList(PyObject * obj, char **vla, int *n_str);
int PConvPyListToStringVLA(PyObject * obj, char **vla_ptr);
#define PConvPyListToIntVLA(obj, f) PConvPyListToIntArrayImpl(obj, f, true)
int PConvPyStrToStr(PyObject * obj, char *ptr, int l);
#ifndef _PYMOL_NOPY
int PConvPyStrToStrPtr(PyObject * obj, const char **ptr);
#endif
int PConvPyStrToLexRef(PyObject * obj, OVLexicon * lex, int *lex_ref);
int PConvPyFloatToFloat(PyObject * obj, float *ptr);
int PConvPyIntToChar(PyObject * obj, char *ptr);
int PConvPyIntToInt(PyObject * obj, int *ptr);
int PConvPyBoolToInt(PyObject * obj, int *ptr);
int PConvPyListToLabPosVLA(PyObject * obj, LabPosType ** vla_ptr);
/* Jenarix conventions -- returns before args */
ov_status PConvPyTupleToIntVLA(int **result, PyObject * tuple);
ov_status PConvPyTupleToFloatVLA(float **result, PyObject * tuple);
/* === end === */
/* categories below... */
PyObject *PConvFloatVLAToPyList(const float *vla);
PyObject *PConvFloatVLAToPyTuple(float *vla);
PyObject *PConvIntVLAToPyList(const int *vla);
PyObject *PConvIntVLAToPyTuple(int *vla);
PyObject *PConvIntArrayToPyList(const int *f, int l, bool dump_binary=false);
PyObject *PConvSIntArrayToPyList(const short int *f, int l);
PyObject *PConvSCharArrayToPyList(const signed char *f, int l);
PyObject *PConvLabPosVLAToPyList(const LabPosType * vla, int l);
void PConvFloat3ToPyObjAttr(PyObject * obj, const char *attr, const float *v);
void PConvFloatToPyObjAttr(PyObject * obj, const char *attr, float f);
void PConvIntToPyObjAttr(PyObject * obj, const char *attr, int i);
void PConvInt2ToPyObjAttr(PyObject * obj, const char *attr, const int *v);
void PConvStringToPyObjAttr(PyObject * obj, const char *attr, const char *f);
int PConvPyObjectToFloat(PyObject * object, float *value);
int PConvPyObjectToInt(PyObject * object, int *value);
int PConvPyObjectToChar(PyObject * object, char *value);
/* NOTE: the string routines will write strings up to the specified
* length, PLUS a NULL...so watch out for array overruns */
int PConvPyObjectToStrMaxLen(PyObject * object, char *value, int ln);
int PConvPyObjectToStrMaxClean(PyObject * object, char *value, int ln);
PyObject *PConvStringListToPyList(int l, const char * const *str);
PyObject *PConvStringVLAToPyList(const char *str);
void PConv44PyListTo44f(PyObject * src, float *dest); /* note loss of precision */
#define PConvPyListToFloatVLA(obj, f) PConvPyListToFloatArrayImpl(obj, f, true)
int PConvPyListToFloatVLANoneOkay(PyObject * obj, float **f);
int PConvPyList3ToFloatVLA(PyObject * obj, float **f);
#define PConvPyListToFloatArray(obj, f) PConvPyListToFloatArrayImpl(obj, f, false)
int PConvPyListToFloatArrayImpl(PyObject * obj, float **f, bool as_vla);
int PConvPyListToDoubleArray(PyObject * obj, double **f);
int PConvPyListToFloatArrayInPlace(PyObject * obj, float *ff, ov_size ll);
int PConvPyListOrTupleToFloatArrayInPlace(PyObject * obj, float *ff, ov_size ll);
int PConvPyListToFloatArrayInPlaceAutoZero(PyObject * obj, float *ii, ov_size ll);
int PConvPyListToDoubleArrayInPlace(PyObject * obj, double *ff, ov_size ll);
PyObject *PConvFloatArrayToPyList(const float *f, int l, bool dump_binary=false);
PyObject *PConvFloatArrayToPyListNullOkay(const float *f, int l);
PyObject *PConvDoubleArrayToPyList(const double *f, int l);
#define PConvPyListToIntArray(obj, f) PConvPyListToIntArrayImpl(obj, f, false)
int PConvPyListToIntArrayImpl(PyObject * obj, int **f, bool as_vla);
int PConvPyListToIntArrayInPlace(PyObject * obj, int *ff, ov_size ll);
int PConvPyListToIntArrayInPlaceAutoZero(PyObject * obj, int *ii, ov_size ll);
int PConvPyListToSIntArrayInPlaceAutoZero(PyObject * obj, short int *ii, ov_size ll);
int PConvPyListToSCharArrayInPlaceAutoZero(PyObject * obj, signed char *ii, ov_size ll);
PyObject *PConv3DIntArrayTo3DPyList(int ***array, int *dim);
PyObject *PConvPickleLoads(PyObject * str);
PyObject *PConvPickleDumps(PyObject * obj);
PyObject *PConvAutoNone(PyObject * result); /* automatically own Py_None */
PyObject *PConvIntToPyDictItem(PyObject * dict, const char *key, int i);
/* ============================================================ */
/*
* PConvToPyObject: Convert any standart type (primitives
* and c++ std library) to a python object.
*
* Return value: New reference.
*/
inline PyObject * PConvToPyObject(PyObject * v) {
return v;
}
inline PyObject * PConvToPyObject(int v) {
return PyInt_FromLong(v);
}
inline PyObject * PConvToPyObject(float v) {
return PyFloat_FromDouble(v);
}
inline PyObject * PConvToPyObject(double v) {
return PyFloat_FromDouble(v);
}
inline PyObject * PConvToPyObject(const std::string &v) {
return PyString_FromString(v.c_str());
}
inline PyObject * PConvToPyObject(const char * v) {
#ifndef _PYMOL_NOPY
if (!v) {
Py_RETURN_NONE;
}
#endif
return PyString_FromString(v);
}
inline PyObject * PConvToPyObject(const float * v, int n) {
return PConvFloatArrayToPyList((float*)v, n);
}
template <class T>
PyObject * PConvToPyObject(const std::vector<T> &v) {
int n = v.size();
PyObject * o = PyList_New(n);
for (int i = 0; i < n; ++i) {
PyList_SetItem(o, i, PConvToPyObject(v[i]));
}
return o;
}
/*
* Convert a set to a Python list
*/
template <class T>
PyObject * PConvToPyObject(const std::set<T> &v) {
size_t i = 0, n = v.size();
PyObject * o = PyList_New(n);
for (auto it = v.begin(); it != v.end(); ++it) {
PyList_SET_ITEM(o, i++, PConvToPyObject(*it));
}
return o;
}
/*
* Convert a map to a flat Python list
*
* {k1: v1, k2: v2, ...} -> [k1, v1, k2, v2, ...]
*/
template <class K, class V>
PyObject * PConvToPyObject(const std::map<K, V> &v) {
size_t i = 0, n = v.size();
PyObject * o = PyList_New(n * 2);
for (auto it = v.begin(); it != v.end(); ++it) {
PyList_SET_ITEM(o, i++, PConvToPyObject(it->first));
PyList_SET_ITEM(o, i++, PConvToPyObject(it->second));
}
return o;
}
/* ============================================================ */
/*
* PConvFromPyObject: Templated conversion of a python object to a
* standart type (primitives and c++ std library).
*/
inline bool PConvFromPyObject(PyMOLGlobals *, PyObject * obj, int &out) {
out = PyInt_AsLong(obj);
return true;
}
inline bool PConvFromPyObject(PyMOLGlobals *, PyObject * obj, float &out) {
out = PyFloat_AsDouble(obj);
return true;
}
inline bool PConvFromPyObject(PyMOLGlobals *, PyObject * obj, double &out) {
out = PyFloat_AsDouble(obj);
return true;
}
inline bool PConvFromPyObject(PyMOLGlobals *, PyObject * obj, std::string &out) {
out = PyString_AsSomeString(obj);
return true;
}
inline bool PConvFromPyObject(PyMOLGlobals *, PyObject * obj, float * out) {
return PConvPyListToFloatArrayInPlace(obj, out, 0);
}
template <class T>
bool PConvFromPyObject(PyMOLGlobals * G, PyObject * obj, std::vector<T> &out) {
if (!PyList_Check(obj))
return false;
int n = PyList_Size(obj);
out.clear();
out.reserve(n);
for (int i = 0; i < n; ++i) {
PyObject *item = PyList_GET_ITEM(obj, i);
T t;
if (!PConvFromPyObject(G, item, t))
return false;
out.push_back(t);
}
return true;
}
/*
* Convert a Python list to a set
*/
template <class T>
bool PConvFromPyObject(PyMOLGlobals * G, PyObject * obj, std::set<T> &out) {
if (!PyList_Check(obj))
return false;
int n = PyList_Size(obj);
out.clear();
for (int i = 0; i < n; ++i) {
PyObject *item = PyList_GET_ITEM(obj, i);
T t;
if (!PConvFromPyObject(G, item, t))
return false;
out.insert(t);
}
return true;
}
/*
* Convert a flat Python list to a map, even indices are keys and odd
* indices are values.
*
* [a, b, c, d, ...] -> {a: b, c: d, ...}
*/
template <class K, class V>
bool PConvFromPyObject(PyMOLGlobals * G, PyObject * obj, std::map<K, V> &out) {
if (!PyList_Check(obj))
return false;
int n = PyList_Size(obj);
out.clear();
for (int i = 0; i < n - 1;) {
PyObject *key = PyList_GET_ITEM(obj, i++);
PyObject *value = PyList_GET_ITEM(obj, i++);
K k;
if (!PConvFromPyObject(G, key, k))
return false;
if (!PConvFromPyObject(G, value, out[k]))
return false;
}
return true;
}
/* ============================================================ */
#endif