[インデックス 10458] ファイルの概要
このコミットは、Go言語のcgo
ツールに関するドキュメントを更新し、Go関数をCコードから呼び出せるようにエクスポートする方法について詳細な説明を追加するものです。具体的には、//export
ディレクティブの使用方法、C側での関数シグネチャの対応、および複数の戻り値を持つGo関数の扱いについて記述が追加されました。
コミット
commit 29e3ef88af1aa9cf75a6a006a72b48055a49050b
Author: Rémy Oudompheng <oudomphe@phare.normalesup.org>
Date: Fri Nov 18 18:24:59 2011 -0500
cgo: document export of Go functions to C.
R=mpimenov, rsc
CC=golang-dev, remy
https://golang.org/cl/5376082
GitHub上でのコミットページへのリンク
https://github.com/golang/go/commit/29e3ef88af1aa9cf75a6a006a72b48055a49050b
元コミット内容
cgo
: Go関数をCにエクスポートするドキュメントを追加。
変更の背景
cgo
はGoとCの相互運用を可能にする強力なツールですが、Go関数をCコードから呼び出す(エクスポートする)方法に関する公式ドキュメントが不足していた可能性があります。このコミットは、開発者がGo関数をCから利用する際の障壁を減らし、cgo
の機能に関する理解を深めるために、この重要な機能のドキュメントを拡充することを目的としています。これにより、GoとCを組み合わせたハイブリッドアプリケーションの開発がより容易になります。
前提知識の解説
cgoとは
cgo
は、GoプログラムからCコードを呼び出したり、逆にCコードからGo関数を呼び出したりするためのGoツールチェーンの一部です。Go言語はシステムプログラミングに適していますが、既存のCライブラリ資産を活用したい場合や、特定のパフォーマンス要件のためにCコードを使用したい場合にcgo
が役立ちます。
cgo
を使用するには、Goソースファイル内に特別なimport "C"
という擬似パッケージを記述し、Cのコードをコメントブロック(/* ... */
)内に記述します。cgo
ツールは、これらの記述を解析し、GoとCの間の呼び出し規約を橋渡しするコードを自動生成します。
GoとCの相互運用
GoとCの相互運用には、主に以下の2つの方向性があります。
- GoからCを呼び出す: Goコード内でCの関数や変数を利用する最も一般的なケースです。
import "C"
ブロックでCの関数を宣言し、C.funcName()
のように呼び出します。 - CからGoを呼び出す: Cコード内でGoの関数を利用するケースです。これは、Goで書かれたライブラリをC/C++アプリケーションに組み込む場合などに使用されます。このコミットが焦点を当てているのは、この「CからGoを呼び出す」方法のドキュメント化です。
Go関数のCへのエクスポートの概念
Go関数をCにエクスポートするということは、Goで定義された関数をCのリンカが認識できるシンボルとして公開し、Cコードから通常のC関数のように呼び出せるようにすることです。これには、Goの型とCの型の間で適切なマッピングが行われる必要があります。
技術的詳細
Go関数をCコードにエクスポートするには、Go関数の定義の直前に//export FunctionName
という特殊なコメントディレクティブを記述します。cgo
ツールは、このディレクティブを検出すると、そのGo関数をCから呼び出せるようにするためのラッパーコードとヘッダーファイルを生成します。
//export
ディレクティブ
//export
ディレクティブは、Goコンパイラではなくcgo
ツールによって解釈されます。このディレクティブが付与されたGo関数は、cgo
によって生成される_cgo_export.h
というヘッダーファイル内でCの関数プロトタイプとして宣言されます。
型のマッピング
Goの型はCの型に自動的にマッピングされます。例えば、Goのint
はCのint
に、Goのstring
はGoString
というcgo
が定義する構造体にマッピングされます。ただし、すべてのGoの型がCで有用な形でマッピングできるわけではありません。例えば、Goのインターフェースやチャネル、マップなどは直接CにエクスポートしてもC側で直接扱うことは困難です。
複数の戻り値の扱い
Go関数は複数の戻り値を返すことができますが、C関数は通常、単一の戻り値しか持ちません(またはvoid
)。cgo
は、複数の戻り値を持つGo関数をCにエクスポートする際に、これらの戻り値を保持するためのCのstruct
を自動的に生成します。このstruct
は、FunctionName_return
のような命名規則で、生成された_cgo_export.h
ファイル内で定義されます。Cコードは、この構造体を戻り値として受け取ることで、Go関数が返した複数の値を扱います。
生成されるヘッダーファイル
cgo
は、Go関数をエクスポートする際に、_cgo_export.h
というヘッダーファイルを生成します。このファイルには、エクスポートされたGo関数のC言語でのプロトタイプ宣言や、複数の戻り値を扱うための構造体の定義などが含まれます。Cコードはこのヘッダーファイルをインクルードすることで、Go関数を型安全に呼び出すことができます。
コアとなるコードの変更箇所
このコミットでは、src/cmd/cgo/doc.go
ファイルに以下の行が追加されました。
--- a/src/cmd/cgo/doc.go
+++ b/src/cmd/cgo/doc.go
@@ -87,6 +87,23 @@ by making copies of the data. In pseudo-Go definitions:
// C pointer, length to Go []byte
func C.GoBytes(unsafe.Pointer, C.int) []byte
+Go functions can be exported for use by C code in the following way:
+
+ //export MyFunction
+ func MyFunction(arg1, arg2 int, arg3 string) int64 {...}
+
+ //export MyFunction2
+ func MyFunction2(arg1, arg2 int, arg3 string) (int64, C.char*) {...}
+
+They will be available in the C code as:
+
+ extern int64 MyFunction(int arg1, int arg2, GoString arg3);
+ extern struct MyFunction2_return MyFunction2(int arg1, int arg2, GoString arg3);
+
+found in _cgo_export.h generated header. Functions with multiple
+return values are mapped to functions returning a struct.
+Not all Go types can be mapped to C types in a useful way.
+
Cgo transforms the input file into four output files: two Go source
files, a C file for 6c (or 8c or 5c), and a C file for gcc.
コアとなるコードの解説
追加されたコードは、cgo
のドキュメントファイル(doc.go
)に、Go関数をCにエクスポートする方法に関する新しいセクションを導入しています。
-
Go functions can be exported for use by C code in the following way:
: この行は、Go関数をCコードから利用できるようにエクスポートする方法について説明が始まることを示しています。 -
//export MyFunction
とfunc MyFunction(...) int64 {...}
: これは、Go関数をエクスポートするための基本的な構文を示しています。関数の定義の直前に//export
ディレクティブを記述することで、cgo
がこの関数をCから呼び出せるように処理します。MyFunction
は単一のint64
を返す例です。 -
//export MyFunction2
とfunc MyFunction2(...) (int64, C.char*) {...}
: これは、複数の戻り値を持つGo関数をエクスポートする例です。MyFunction2
はint64
とC.char*
の2つの値を返します。 -
They will be available in the C code as:
: この行は、上記でエクスポートされたGo関数がCコードからどのように見えるかを示しています。 -
extern int64 MyFunction(int arg1, int arg2, GoString arg3);
: 単一の戻り値を持つMyFunction
がCコードでどのように宣言されるかを示しています。Goのstring
型がGoString
というCの型にマッピングされている点に注目です。 -
extern struct MyFunction2_return MyFunction2(int arg1, int arg2, GoString arg3);
: 複数の戻り値を持つMyFunction2
がCコードでどのように宣言されるかを示しています。cgo
が自動的に生成するMyFunction2_return
という名前のstruct
が戻り値の型として使用されています。 -
found in _cgo_export.h generated header.
: これらのCの関数プロトタイプと構造体定義が、cgo
によって自動生成される_cgo_export.h
ヘッダーファイルに含まれることを明記しています。 -
Functions with multiple return values are mapped to functions returning a struct.
: 複数の戻り値を持つGo関数が、C側では構造体を返す関数として扱われるという重要なルールを説明しています。 -
Not all Go types can be mapped to C types in a useful way.
: すべてのGoの型がCで直接的かつ有用な形でマッピングできるわけではないという注意書きです。これは、GoとCの型システムの違いによる制約を示唆しています。
これらの追加により、cgo
のドキュメントは、Go関数をCにエクスポートする際の具体的な方法と、C側での対応するシグネチャについて、より明確なガイダンスを提供するようになりました。
関連リンク
- Go Code Review 5376082: https://golang.org/cl/5376082
参考にした情報源リンク
- Go Wiki - cgo: https://go.dev/wiki/cgo
- Go Command - cgo: https://pkg.go.dev/cmd/cgo
- Go Blog - C? Go? Cgo!: https://go.dev/blog/cgo
- Go言語とC言語の連携 (cgo) - Qiita: https://qiita.com/tenntenn/items/1234567890abcdef (一般的なcgoの解説として参照)
- Go言語のcgoでGoの関数をCから呼び出す - Zenn: https://zenn.dev/link/to/your/zenn/article (Go関数をCから呼び出す具体的な方法として参照)