質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

ポインタ

ポインタはアドレスを用いてメモリに格納された値を"参照する"変数です。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

1回答

3698閲覧

[Python/C API] Cで作ったポインタ変数が引数にある関数をPythonで呼び出し使用したい。

toriaezu_NAMA

総合スコア3

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

ポインタ

ポインタはアドレスを用いてメモリに格納された値を"参照する"変数です。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2021/07/27 15:28

やりたいこと・知りたい事

やりたいこと

  • 表題にある通り、Cで作ったポインタ引数のある関数をPythonで使用したい。

そのため、下記のような関数のラッパーファイルを作成するとき、どのように記載したら良いか知りたい。

  • 下記の関数はポインタを使用しない方法でも実装できるが、後学のため、ポインタを関数の引数に置くことを条件としている。

c

1void pointer_func(int n1,int n2,int *sum,int *diff){ 2 *sum = n1 + n2; 3 *diff = (n1 > n2) ? n1 -n2 : n2 - n1; 4} 5 6

作成したラッパーファイル・呼び出し

作成したラッパーファイル

c

1#include "/usr/~~~/python3.8/Python.h" 2 3extern void pointer_func(int n1,int n2,int *sum,int *diff); 4PyObject* f_pointer_func(PyObject* self, PyObject* args){ 5 6 int n1,n2; 7 int* sum; 8 int* diff; 9 10 //変数 int* sum,int* diff に使用できる "" がわからない 11 if (!PyArg_ParseTuple(args, "iiii" ,&n1,&n2,&sum,&diff)){ 12 return NULL; 13 } 14 15 pointer_func(n1,n2,sum,diff); 16 return Py_BuildValue(""); 17} 18 19 20static PyMethodDef methods[] = { 21 {"pointer_func", f_pointer_func, METH_VARARGS, "pointer_ch" }, 22 {NULL} 23}; 24 25static struct PyModuleDef pointer_check = 26{ 27 PyModuleDef_HEAD_INIT, 28 "pointer_check", 29 "", 30 -1, 31 methods 32}; 33 34PyMODINIT_FUNC PyInit_pointer_check(void) 35{ 36 return PyModule_Create(&pointer_check); 37} 38

呼び出し
コンパイルした後、下記のようなファイルを作成し、import と呼び出しをした。

Python

1import pointer_check as ptc 2wa = 0 3sa = 0 4ptc.pointer_func(10,20,wa,sa);

###エラーメッセージ

segmentation fault (core dumped)

調べたこと・試してること

リファレンスの PyArg_ParseTuple() の項目を参照してもポインタ引数を扱うものが見当たらないため質問しました。

別途で
PyCapsule_New がポインタを格納できると記載されていたので、下記の参考資料のURLを見ながら引数単体で使用できないか調べています。

補足情報・参考資料

Python/C API リファレンスマニュアル
PyArg_ParseTuple
https://docs.python.org/ja/3/c-api/arg.html?highlight=pyarg_parsetuple#c.PyArg_ParseTuple
カプセル
https://docs.python.org/ja/3/c-api/capsule.html
PythonのC/C++拡張パターン - ポインタ
https://qiita.com/junkoda/items/2b1eda7569186809ca14

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

まず最初に、Pythonはctypesモジュールを使用しなければ、C言語のポインタを扱えません。

妥当な回答

一番妥当なのは、結果は戻り値として返すやり方です。

c

1PyObject * test01(PyObject *self, PyObject *args) 2{ 3 int n1, n2; 4 int sum, diff; 5 6 if (!PyArg_ParseTuple(args, "ii", &n1, &n2)) { 7 return NULL; 8 } 9 pointer_func(n1, n2, &sum, &diff); 10 return Py_BuildValue("ii", sum, diff); 11}
>>> import hoge >>> hoge.test01(2, 3) (5, 1)

GTK+(C言語のGUIフレームワーク)を例に挙げます。
gtk_tree_selection_get_selectedという関数があります。

どういう関数かの説明は省きますが、第2・3引数は値を受け取るためにポインタになっています。

GTK+をバインディングしたPythonのライブラリはどのような仕様になっているかというと、以下のようになっています。

このように、C言語の関数での第2・3引数で受け取る値は、戻り値として返すようになっています。

ちょっと遊んだ回答

「ctypesモジュールを使わなければ」ポインタは扱えませんが、じゃctypesモジュールを使えばいいんですよ。

c

1/* 2 * import ctypes 3 * addr = ctypes.addressof(obj) 4 * 5 * 相当の処理をする。 6 */ 7int * query_address(PyObject *obj) 8{ 9 static PyObject *ctypes_module = NULL; 10 PyObject *result_obj; 11 unsigned long address; 12 13 if (ctypes_module == NULL) { 14 ctypes_module = PyImport_ImportModule("ctypes"); 15 if (ctypes_module == NULL) { 16 return NULL; 17 } 18 } 19 result_obj = PyObject_CallMethod(ctypes_module, "addressof", "O", obj); 20 if (result_obj == NULL) { 21 return NULL; 22 } 23 address = PyLong_AsUnsignedLong(result_obj); 24 Py_DECREF(result_obj); 25 return (PyErr_Occurred() == NULL? (int *)address: NULL); 26} 27 28PyObject * test02(PyObject *self, PyObject *args) 29{ 30 int n1, n2; 31 PyObject *sum_obj, *diff_obj; 32 int *sum_ptr, *diff_ptr; 33 34 if (!PyArg_ParseTuple(args, "iiOO", &n1, &n2, &sum_obj, &diff_obj)) { 35 return NULL; 36 } 37 sum_ptr = query_address(sum_obj); 38 if (sum_ptr == NULL) { 39 return NULL; 40 } 41 diff_ptr = query_address(diff_obj); 42 if (diff_ptr == NULL) { 43 return NULL; 44 } 45 pointer_func(n1, n2, sum_ptr, diff_ptr); 46 Py_RETURN_NONE; 47} 48
>>> import hoge >>> import ctypes >>> s = ctypes.c_int() >>> d = ctypes.c_int() >>> hoge.test02(2, 3, s, d) >>> print(s.value, d.value) 5 1

ただし、わざわざctypesモジュールを使ってまでやることかは、疑問に思います。
前述の方法のほうがよりスマートでしょう。


PyCapsuleAPIは使ったことがありませんが、ここを読む限り、拡張モジュール同士でのデータのやり取りに使用するためのものであって、目的にあったものではないと思います。

Python はある拡張モジュールの C レベルの情報 (ポインタ) を別のモジュールに渡すための特殊な機構: Capsule (カプセル)を提供しています。 Capsule はポインタ (void *) を記憶する Python のデータ型です。 Capsule は C API を介してのみ生成したりアクセスしたりできますが、他の Python オブジェクトと同じように受け渡しできます。


質問とは関係ありませんが、ちょっと気になった点です。

c

1 return Py_BuildValue("");

おそらく特に戻り値を必要としないためにこのような書き方をしていると思いますが、Pythonではそのような場合にはNoneを返すことを慣例としています。
更に、その慣例を手っ取り早く書くためにPy_RETURN_NONEというマクロが用意されています。

c

1 //return Py_BuildValue(""); 2 Py_RETURN_NONE;

投稿2021/07/28 12:14

編集2021/07/28 12:15
katsuko

総合スコア3538

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

toriaezu_NAMA

2021/07/29 08:04

知りたかった以上の事まで教えて頂き大変勉強になりました。 ありがとうございました。 現状、回答に対して質問内容が見当たりませんので、BAにさせて頂きます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問