How can a #defined C value be exposed to Python in a Cython module?

Question:

I’d like to make the integer constants (ACTIVE_TAG, etc) defined here:

//island management, m_activationState1
#define ACTIVE_TAG 1
#define ISLAND_SLEEPING 2
#define WANTS_DEACTIVATION 3
#define DISABLE_DEACTIVATION 4
#define DISABLE_SIMULATION 5

available as normal attributes of a Cython defined module I’m working on, so that Python application code can access them (to pass them in to wrapped APIs which are defined in terms of them).

I’ve looked at defining these with cdef as integers or enums, but neither of these approaches actually binds the value to an an attribute in the Cython module. What other options are there?

Answers:

Here’s one way, which, while seemingly tedious, could feasibly be automated for any given .h file as input:

Step 1. Get all the constants you want into a file, say bulletdefs.h, which has the #defines but with leading underscores, e.g:

#define _ACTIVE_TAG 1
#define _ISLAND_SLEEPING 2
#define _WANTS_DEACTIVATION 3
#define _DISABLE_DEACTIVATION 4
#define _DISABLE_SIMULATION 5

Step 2. Insert a section into the pyx file for your module, say bullet.pyx:

cdef extern from "bulletdefs.h":
    cdef int _ACTIVE_TAG
    cdef int _ISLAND_SLEEPING
    cdef int _WANTS_DEACTIVATION
    cdef int _DISABLE_DEACTIVATION
    cdef int _DISABLE_SIMULATION

ACTIVE_TAG = _ACTIVE_TAG
ISLAND_SLEEPING = _ISLAND_SLEEPING
WANTS_DEACTIVATION = _WANTS_DEACTIVATION
DISABLE_DEACTIVATION = _DISABLE_DEACTIVATION
DISABLE_SIMULATION = _DISABLE_SIMULATION

Then, when you compile your module, you should get the intended effect:

Python 2.7.1+ (r271:86832, Apr 11 2011, 18:05:24) 
[GCC 4.5.2] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import bullet
>>> bullet.ACTIVE_TAG
1
>>> bullet.DISABLE_SIMULATION
5
>>>
Answered By: Vinay Sajip

It worked in my case. Maybe can be helpful for someone too:

In my case I needed to export a #define from a linux kernel library. And it worked for my:

# your_pxd_file.pxd
cdef extern from "sched.h": #here are lots of `#define`'s clauses. something like the link: https://github.com/spotify/linux/blob/master/include/linux/sched.h
    cdef enum:
        CLONE_NEWNS

In your .pyx file:

from your_compiled_cython_package cimport CLONE_NEWNS

print(CLONE_NEWNS)

I hope this can be helpful for someone as it was for me =)

Answered By: lucAlucard

Example header:

// myheader.hh
#ifndef MYHEADER
#define MYHEADER

#define NONE 0

#endif

An answer from my experience for my similar case, without const keyword, a variable is created and it’s not constant as of #define. I found some assignment to NONE at the end of the generated C++ file by Cython and it’s not allowed to assign to macro.

cdef extern from "myheader.hh":
    int NONE

This is a real constant:

cdef extern from "myheader.hh":
    const int NONE

Bug in Cython without const keyword: https://github.com/cython/cython/issues/2621

Answered By: Dee
Categories: questions Tags: ,
Answers are sorted by their score. The answer accepted by the question owner as the best is marked with
at the top-right corner.