Skip to main content

Embedded C/C++ in Python Source

Project description

pyembedc: Python With Embedded C/C++

pyembedc enables Python source code to embed C/C++ snippets that seamlessly access and modify Python variables and call Python functions.

http://github.com/ftrias/pyembedc

Version 1.23: 8 Sep 2018

Copyright 2010-2018 by Fernando Trias See LICENSE for MIT license terms of use.

PREREQUISITES

  • gcc (version 3 or higher) or clang
  • Python 2.5, 3 or higher

INSTALL

python setup.py install

SAMPLE CODE

from embedc import C
v = [5,6,7]
vlen = len(v)
vsum = 0;
r = C("for(int i=0;i<vlen;i++) vsum += v[i];")
print(vsum)

USAGE

To use the library, first you must import it:

import embedc

Functions


embedc.C(string) -> int

embedc.inline_c(string) -> int

embedc.inline_c_precompile(string) -> int

These functions will compile string containing the C/C++ code or directives (see below) and then link dynamically and run the code.

string is C/C++ code that can be used within a function. It can contain any valid C/C++ expression that your compiler will support.

The C function will automatically provide references to all local Python variables for use in your code to read or write as if they were basic types or arrays.

The inline_c and inline_c_precompile fucntion will not provide references to local Python variables and thus is faster and consumes less memory.

Note: inline_c_precompile will compile the code only once at the beginning of the program to increase speed. The code must be given using the Python multi-line string format using three quotes '"""'. This is because the module will scan the source file and pick out the string. As a consequence,variable substitutions and other string operations will not work. For example, the following code will not work:

C("""
v=%s
""" % (var1))

string must be valid C/C++ suitable for placment within a function.


embed_c(string) -> cdll

embed_c_precompile(string) -> cdll

These functions are used to compile code but not execute immediately. They return a CDLL object (see the CDLL python module) that can be executed later.

For example:

run_trial_lib = embed_c("""
    int mult(int n1, int n2) {
        return n1*n2;
    }
    """)

def multiply(n1, n2):
    return mathlib_lib.mult(n1, n2)

DIRECTIVES

Directives are inserted into the C/C++ code. They provide hints about how to wrap the code. They enumerate variables, add post-exectution cleanup code, reference Python code and modify the return value.

IMPORT type [&]variable

IMPORT will allow access to a Python variable within the code. Usually, this is not required for local variables because they are imported automatically. But in the case of the function inline_c_precompile this is not the case, and you must import all variables using IMPORT. In the case of global or class variables, you must use IMPORT.

type is any valid C type, such as int, double, int[]. In addition, string is used to access strings, which C will treat as a const char *.

type can also be an array if the Python type is a tuple, list or array. In this case add [] to the end of the type, in the manner of C/C++. For example, in Python a=(1,5,3), the IMPORT line would be IMPORT int[] a.

variable is the Python variable to import. Normally, variables are imported by value, meaning that any changes in the C/C++ code are lost.

However, if you add a & in front (the C/C++ symbol for reference), then the variables are imported by reference and any changes will be saved to the Python variables when the code finishes execution.

For example:

IMPORT int[] myarray
IMPORT float &myfloat

DEF return-type function [type1] [type2] [type3]...

Allow access to a Python function within the C/C++ code. return-type is any C/C++ type. At this time, only basic types are allowed. Arrays are not allowed. Instead of returning arrays arrays, you should instead modify local variables. type1, type2, etc. are the data types of parameters.

For example:

DEF double myround double
double x = myround(5.5);

GLOBAL code

Places the code in the global space. Usually, this is used to create a C/C++ global variable. These variables are available in the C/C++ code between calls to C or inline_c_precomiple. This is is necessary because embedded C/C++ code is placed within a function.

For example:

GLOBAL int lock_counter;
void run() { lock_counter++; }

RETURN type value

Return a non-int value. Usually, return values for C or inline_c_precompile are of type int. If return is not specified in the C/C++ code, then "return 0" is assumed. If you want to return a different type rather than int, then instead of return, use RETURN and specify the type.

For example:

double myfunc() {
    double myvariable = 1.7
    RETURN double myvariable
}

POST code

Run code after executing the C/C++ snipet, writing back all the variables and the return value. Sometimes this is necessary wen your function needs to do some cleanup, such as when it allocates memory to return a string.

For example:

POST free(mydata);

CC command

Use command instead of gcc as compiler. The command must be suitable for passing into a shell. The file name is appended at the end of the command.

SPECIAL CONSIDERATIONS

When using embed_c_precompile and inline_c_precompile, pyembedc will read in the source file before it is interpreted. It will look for the function name and the triple-quote """ sequence. It will also look for another triple-quote """ to end. The close parenthesis must be on the same line as the ending triple-quote """ (see examples above). This is because pyembedc uses line numbers to find the precompiled function at run time.

TESTING

You don't have to install before running the test:

python test.py

The test-coverage script uses multiple python versions (2.5, 2.6, 3.1) and python-coverage to ensure most source lines are covered by the test suite.

This module has been tested on:

  • Windows 7 with Cygwin Python 2.5 and GCC (mixed results, see Notes)
  • Windows 7 with official Python 2.5 and 2.6 and MinGW
  • Ubuntu (Debian Linux 2.6 kernel), Python 2.5, 2.6 and 3.1 and gcc 4.4.1
  • MacOS (High Sierra 10.13), Python 3.5.5, Xcode 9 with command line tools

TODO

  • Allow additional compilers; allow passing GCC command line options.

  • Support arrays of strings.

  • Support returning strings from Python functions called from within C/C++ (modifying string variables is supported).

  • With non-precompiled code, keep a cache of the source code so if it doesn't change, we can just reuse the DLL/so.

REFERENCES AND NOTES

pyembedc makes heavy use of ctypes.

There are several alternatives to pyembedc. All of them have their pros and cons. Here are several:

PyInline http://pyinline.sourceforge.net/ is a module for inlining multiple languages in Python last updated in 2001 that predates ctypes. It seeks to copy Perl Inline http://search.cpan.org/~sisyphus/Inline-0.45/C/C.pod.

ezpyinline http://pypi.python.org/pypi/ezpyinline/0.1 is a fork of PyInline that provides a very simple interface for embedding C/C++ functions.

Neither PyInline nor ezpyinline hande arrays. I think C provides the greatest speed advantage when looping over large amounts of data and performing multiple calculations. Without arrays or the ability to return more than one number (PyInline and ezpyinline provide only one return value), that advantage is hard to realize. embedc fixes these shortcoming by using ctypes to allow C/C++ to access and modify arrays.

These modules also don't provide an easy mechanism for accessing and altering Python variables from within the C/C++ code. With them, to modify variables from C, you have to code in the Python API, which is neither easy nor straightforward.

In addition, they require a working python development environment. This can be complex on Linux if you are not using standard packages, and is also difficult in Windows. By using ctypes, embedc does not require a working Python build environment, thus simplifying deployment because there is no need to configure include paths, lib paths and so forth. The only requirement is a working compiler.

PyRex http://wiki.python.org/moin/Pyrex is a Python-like language that mixes C and Python to create Python modules. This is a very interesting approach, but isn't really Python and requires a great deal of expertise to use.

KNOWN ISSUES:

Cygwin: Cygwin has numerous problems with the test script because it quickly runs out of resources. The test script covers too many scenarios loading too many different DLLs for Cygwin to handle.

If you get an error like this on Windows:
    *** fatal error - unable to remap tmpE9X11V.cpp.dll to same address as parent(0x2B0000) != 0x3B0000
    python 5368 fork: child 1676 - died waiting for dll loading, errno 11
Check out http://inamidst.com/eph/cygwin for a possible solution.

Occasionally, this error might come up in the test.py script:
    OSError: [Errno 11] Resource temporarily unavailable
This is due to the limitations of Cygwin.

Local Variables: Modifying local variables is tricky and required pyembedc to call internal Python API functions. It has been tested on Python 2.5, 2.6, and 3.1 but may break in future versions if the internal representation of frames and variables changes.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyembedc-1.25.tar.gz (18.1 kB view hashes)

Uploaded Source

Built Distribution

pyembedc-1.25-py3-none-any.whl (12.2 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page