"Daniel Platz" <mail.to.daniel.pl...@googlemail.com> wrote in message news:63ac1a01-8491-4885-bae7-cb884abb5...@34g2000yqp.googlegroups.com...
Hello,

I would like to pass a two dimensional array to C function in a dll. I
use ctypes to call the function.

I compile the dll with visual studio 2008 express and my C source code
looks like this.

#include <stdio.h>
#ifdef __cplusplus
extern "C" {  // only need to export C interface if
             // used by C++ source code
using namespace std;
#endif


__declspec(dllexport) int print(double** ptr, int ny, int nx)
{
int i, j;
for(i=0; i<ny; i++)
{
for(j=0; j<nx; j++)
{
printf("%.3f \t", *(*(ptr+i)+j));
}
printf("\n");
}
return 0;
}

#ifdef __cplusplus
}
#endif

As you can see the function expects a doube** variable. I tried to
call this function from Python with this code.

import ctypes as ct

matrix = ct.cdll.LoadLibrary('matrix.dll')
getattr(matrix, 'print_matrix')

ptr_type = ct.c_double * 5 * 4
ptr = ptr_type()
matrix.print_matrix(ptr, ct.c_int(4), ct.c_int(5))

I expected to see a matrix showing only zeros since I can address the
single entries in Python by ptr[i][j]. Instead the interpreter returns
with the error message

WindowsError: exception: access violation reading 0x000000
WARNING: Failure executing file: <ctypes_matrix.py>

Furthermore, I am wondering if there is a fast way to use a numpy 2D
array instead or alternatively to cast the ctypes array into a numpy
array.

Has someone an idea to help me?

A two-dimentional array is not equivalent to a double**, so depending on what your requirements are there are two solutions.

1. To work with your original matrix.c code above, some Python code that works is:

-------------------------------------
import ctypes as ct

# I like C types in caps
DOUBLE = ct.c_double
PDOUBLE = ct.POINTER(DOUBLE)
PPDOUBLE = ct.POINTER(PDOUBLE)
INT = ct.c_int

matrix = ct.cdll.LoadLibrary('matrix.dll')
print_matrix = getattr(matrix,'print')

# An array of doubles can be passed to a function that takes double*.
DBL5ARR = DOUBLE * 5
# An array of double* can be passed to your function as double**.
PDBL4ARR = PDOUBLE * 4

# Declare double* array.
ptr = PDBL4ARR()
for i in range(4):
   # fill out each pointer with an array of doubles.
   ptr[i] = DBL5ARR()
   for j in range(5):
       ptr[i][j] = i + j  # just to initialize the actual doubles.

print_matrix(ptr,4,5)
-----------------------------------------------


2. In C, multidimensional arrays are really just single dimentional arrays where the compiler does the math for you, so if you declare your matrix print function like this:

----- matrix2.c ---------------------------
#include <stdio.h>
#ifdef __cplusplus
extern "C" {  // only need to export C interface if
             // used by C++ source code
#endif

__declspec(dllexport) int print(double* ptr, int ny, int nx)
{
   int i, j;
   for(i=0; i<ny; i++)
   {
       for(j=0; j<nx; j++)
       {
           printf("%.3f \t", ptr[i*nx + j]);
       }
       printf("\n");
   }
   return 0;
}

#ifdef __cplusplus
}
#endif
-------------------------------------------------

Then the Python code simplifies a bit to:

------------------------------------------
import ctypes as ct

DOUBLE = ct.c_double
PDOUBLE = ct.POINTER(DOUBLE)
INT = ct.c_int

matrix = ct.cdll.LoadLibrary('matrix2.dll')
print_matrix = getattr(matrix,'print')

ptr = (DOUBLE*5*4)()
for i in range(4):
   for j in range(5):
       ptr[i][j] = i + j

print_matrix(ptr,4,5)
----------------------------------------------

OUTPUT in both cases:
0.000   1.000   2.000   3.000   4.000
1.000   2.000   3.000   4.000   5.000
2.000   3.000   4.000   5.000   6.000
3.000   4.000   5.000   6.000   7.000

HTH,
Mark


--
http://mail.python.org/mailman/listinfo/python-list

Reply via email to