[インデックス 11508] ファイルの概要
このコミットは、Go言語のリンカである5l
の-v
(verbose)オプションの出力から、不要な情報を削減することを目的としています。具体的には、リンカのベンチマーク時に出力される、重複シンボルに関するメッセージがノイズとなるため、これらをコメントアウトすることで出力を整理しています。
コミット
- コミットハッシュ:
94b796133b3a236530f933362a3feb90af1fcbf2
- 作者: Shenghou Ma minux.ma@gmail.com
- コミット日時: 2012年1月31日 火曜日 10:59:29 -0500
- コミットメッセージ:
5l: make -v option output less nonessential clutter 5l -v is for benchmarking various parts of the loader, but this code in obj.c will clutter the output. I only comment them out, because this is on par with 8l/6l. R=golang-dev CC=golang-dev https://golang.org/cl/5600046
GitHub上でのコミットページへのリンク
https://github.com/golang/go/commit/94b796133b3a236530f933362a3feb90af1fcbf2
元コミット内容
5l: make -v option output less nonessential clutter
5l -v is for benchmarking various parts of the loader, but this code in
obj.c will clutter the output. I only comment them out, because this is
on par with 8l/6l.
R=golang-dev
CC=golang-dev
https://golang.org/cl/5600046
変更の背景
この変更の背景には、Go言語のツールチェーンにおけるリンカのデバッグおよびベンチマークの効率化があります。5l
はGo言語のARMアーキテクチャ向けリンカであり、-v
オプションは通常、詳細なデバッグ情報や処理の進行状況を出力するために使用されます。しかし、リンカの特定の処理(この場合は重複シンボルのスキップ)に関するメッセージが、ベンチマークの際に大量に出力され、本来のベンチマーク結果の分析を妨げる「ノイズ」となっていました。
コミットメッセージにある「5l -v is for benchmarking various parts of the loader
」という記述から、開発者がリンカのパフォーマンス特性を詳細に分析するために-v
オプションを利用していたことが伺えます。この文脈において、不要な出力は分析の妨げとなるため、それらを抑制する必要がありました。
また、「this is on par with 8l/6l
」という記述は、他のアーキテクチャ(8l
はx86-64、6l
はx86-32)向けのリンカでも同様の出力抑制が行われていることを示唆しており、5l
もそれに合わせて一貫性を持たせる意図があったと考えられます。これは、Goツールチェーン全体でのデバッグ出力の標準化の一環とも言えるでしょう。
前提知識の解説
このコミットを理解するためには、以下のGo言語のツールチェーンおよびC言語の基本的な概念に関する知識が必要です。
-
Go言語のリンカ (
5l
,6l
,8l
): Go言語のビルドプロセスにおいて、コンパイルされたオブジェクトファイル(.o
ファイル)を結合し、実行可能なバイナリを生成する役割を担うのがリンカです。Goのリンカは、伝統的なUnix系のリンカ(ld
など)とは異なり、Go言語のランタイムやガベージコレクタ、スケジューラなどの特殊な要件に対応するために独自に実装されています。5l
: ARMアーキテクチャ向けのリンカです。Goのソースコードでは、アーキテクチャごとに異なるリンカが用意されており、5l
はその一つです。6l
: x86-32アーキテクチャ向けのリンカです。8l
: x86-64アーキテクチャ向けのリンカです。 これらのリンカは、それぞれ対応するアーキテクチャのオブジェクトファイルを処理し、最終的な実行ファイルを生成します。
-
debug['v']
: これは、Go言語のツールチェーン内でよく見られるデバッグフラグのパターンです。debug
は通常、コマンドラインオプションや環境変数によって設定されるデバッグレベルを保持するマップまたは配列のような構造体です。debug['v']
は、リンカが詳細な(verbose)出力を生成するかどうかを制御するブール値として機能します。このフラグがtrue
の場合、通常はより多くの情報が標準出力またはエラー出力に表示されます。 -
Bprint(&bso, ...)
:Bprint
は、C言語で書かれたGoツールチェーンのコードベースでよく使用される、バッファリングされた出力関数です。bso
は通常、標準出力(stdout
)または標準エラー出力(stderr
)に関連付けられたバッファリングされたストリームオブジェクトです。この関数は、指定されたフォーマット文字列と引数を使用して、bso
にデータを書き込みます。バッファリングにより、I/O操作のオーバーヘッドを削減し、パフォーマンスを向上させることができます。 -
s->dupok
:s
はシンボル(Sym
構造体)へのポインタであり、dupok
はそのシンボルが重複を許容するかどうかを示すフラグです。リンカは、複数のオブジェクトファイルから同じ名前のシンボルが定義されている場合、通常はエラーを報告します。しかし、一部のシンボル(例えば、弱いシンボルや特定のセクションのシンボル)は重複が許容される場合があります。dupok
フラグは、リンカがそのようなシンボルを検出した際に、エラーとせずにスキップして処理を続行すべきかどうかを判断するために使用されます。 -
goto loop;
: C言語におけるgoto
文は、プログラムの実行フローを特定のラベルにジャンプさせるために使用されます。このコンテキストでは、重複が許容されるシンボルが検出された場合、現在のシンボル処理をスキップして、次のシンボルを処理するためのループの先頭に戻るために使用されています。
技術的詳細
このコミットは、Go言語のリンカ5l
のソースコードsrc/cmd/5l/obj.c
内の特定のデバッグ出力行をコメントアウトすることで、-v
オプション使用時の出力を削減しています。
リンカの処理において、複数のオブジェクトファイルから同じ名前のシンボルが定義されている場合、通常はリンケージエラーとなります。しかし、Go言語のリンカでは、特定のシンボルに対してdupok
(duplicate OK)フラグが設定されている場合、そのシンボルが重複していてもエラーとせず、最初の定義を採用し、後続の重複定義をスキップするロジックが存在します。
変更前のコードでは、このdupok
フラグが設定されたシンボルをスキップする際に、debug['v']
がtrue
(つまり、-v
オプションが有効)であれば、「skipping %s in %s: dupok\n
」(%s
はシンボル名とファイル名)というメッセージをBprint
関数を使って出力していました。このメッセージは、リンカがどのシンボルを重複としてスキップしたかを示すもので、通常のデバッグ時には有用な情報です。
しかし、リンカのパフォーマンスをベンチマークする際には、この「スキップ」メッセージが大量に出力されることがあり、特に大規模なプロジェクトや多数の重複シンボルが存在する場合には、出力が膨大になり、ベンチマーク結果の解析を困難にしていました。コミットメッセージが示唆するように、5l -v
はリンカの各部分のベンチマークを目的としているため、このような「非本質的な雑多な出力(nonessential clutter)」は排除されるべきでした。
この変更は、これらのBprint
呼び出しをC言語のコメントアウト構文//
で無効化することで、-v
オプションが有効であっても、重複シンボルスキップ時のメッセージが出力されないようにしています。これにより、ベンチマーク実行時の出力がクリーンになり、開発者はより本質的なパフォーマンスデータに集中できるようになります。
このアプローチは、8l
や6l
といった他のアーキテクチャ向けリンカでも同様の出力抑制が行われていることと「同等(on par)」であるとされており、Goツールチェーン全体でのデバッグ出力の一貫性を保つ意図も含まれています。
コアとなるコードの変更箇所
変更はsrc/cmd/5l/obj.c
ファイルに対して行われました。
--- a/src/cmd/5l/obj.c
+++ b/src/cmd/5l/obj.c
@@ -558,8 +558,8 @@ loop:
// redefinitions.
s = p->from.sym;
if(s->dupok) {
-\t\t\tif(debug['v'])\n-\t\t\t\tBprint(&bso, \"skipping %s in %s: dupok\\n\", s->name, pn);\n
+//\t\t\tif(debug['v'])\n+//\t\t\t\tBprint(&bso, \"skipping %s in %s: dupok\\n\", s->name, pn);\n
\tgoto loop;\n
}\n
if(s->file == nil)\n
コアとなるコードの解説
変更されたコードブロックは、リンカがシンボルを処理するループ(loop:
ラベル)の中にあります。
s = p->from.sym;
: 現在処理しているプログラム要素p
から、関連するシンボルs
を取得しています。if(s->dupok) { ... }
: この条件文は、取得したシンボルs
がdupok
フラグ(重複を許容するフラグ)を持っているかどうかをチェックしています。- 変更前:
if(debug['v']) Bprint(&bso, "skipping %s in %s: dupok\n", s->name, pn);
debug['v']
がtrue
の場合(つまり、-v
オプションが有効な場合)、Bprint
関数を使って「skipping %s in %s: dupok\n
」というメッセージを出力していました。s->name
はスキップされるシンボルの名前、pn
はそのシンボルが定義されているファイルの名前です。 - 変更後:
元の// if(debug['v']) // Bprint(&bso, "skipping %s in %s: dupok\n", s->name, pn);
if
文とその中のBprint
呼び出しが、C言語の単一行コメント//
によって完全にコメントアウトされています。これにより、debug['v']
がtrue
であっても、このメッセージは一切出力されなくなりました。 goto loop;
:dupok
シンボルが検出された場合、メッセージの出力有無にかかわらず、現在のシンボル処理をスキップして、リンカのメインループの先頭に戻り、次のシンボルの処理に進みます。
この変更は、リンカの機能的な動作には影響を与えず、-v
オプション使用時の出力内容のみを変更するものです。これにより、リンカのベンチマークやデバッグ作業において、よりクリーンで関連性の高い情報のみが得られるようになります。
関連リンク
- Go CL 5600046: https://golang.org/cl/5600046
参考にした情報源リンク
- Go言語のリンカに関する一般的な情報:
- Go linker and object files (Go公式ドキュメントのAssemblerのセクションにリンカの概要が含まれています)
- The Go linker (Russ CoxによるGoリンカに関するブログ記事)
- C言語の
goto
文:- C++ goto Statement (C++の例ですが、C言語の
goto
も同様です)
- C++ goto Statement (C++の例ですが、C言語の
- Go言語のツールチェーンにおけるデバッグフラグの慣習:
- Go言語のソースコードを検索することで、
debug['v']
のようなパターンが他のツールでも使用されていることが確認できます。
- Go言語のソースコードを検索することで、
Bprint
関数に関する情報:- Go言語のツールチェーンのC言語ソースコード内で
Bprint
の定義を検索することで、その機能と使用法を理解できます。 - 例:
src/cmd/internal/obj/util.go
(Goのツールチェーン内のユーティリティ関数)
- Go言語のツールチェーンのC言語ソースコード内で
dupok
シンボルに関する情報:- Go言語のリンカのソースコード(例:
src/cmd/link/internal/ld/sym.go
など)でdupok
フラグの定義と使用箇所を検索することで、その目的と動作を詳細に理解できます。
- Go言語のリンカのソースコード(例: