はじめに
bashには次の2つの理由によって、組み込みコマンド(builtin command)というものが存在します。
- スクリプトの高速化のため。組み込みコマンドであれば通常のコマンドを実行する場合に比べてプロセスの生成コストが削減できる
- bash自身の状態を変更させるため。例えばcdコマンドを/bin/cdとして用意してbashから当該コマンドを実行しても、当該コマンドのpwdが変更されるだけで、bashのそれは変更されない
今回は前者に焦点を合わせて、その効果と、組み込みコマンドの自作方法について述べます。
予備知識: 組込みコマンドによるスクリプト高速化の効果
組込みコマンドそのものの存在と存在意義について既にご存知のかたは、この節を飛ばしてもらって構いません。
例えば皆さんがbashスクリプトからechoコマンドを実行した場合、通常は/bin/echoコマンドではなくbashの組込みコマンドechoを実行します。目的は前述の通り、高速化です。
組込みコマンドによる高速化の効果を体験してみましょう。改行のみを出力する/bin/echoコマンドを一万回実行したときの経過時間を計測した結果を示します。出力そのものは不要なので/dev/nullに捨てています。
$ time (for ((i=0;i<10000;i++)) ; do /bin/echo >/dev/null; done)
real 0m5.760s
user 0m1.744s
sys 0m1.084s
$
所要時間はおよそ5.8秒でした。
同じことをbash組み込みコマンドで、つまり通常みなさんがbashから実行するechoでやってみましょう。
$ time (for ((i=0;i<10000;i++)) ; do echo >/dev/null; done)
real 0m0.086s
user 0m0.072s
sys 0m0.012s
$
こちらの所要時間はおよそ0.086秒と、通常のコマンドを実行する場合に比べて約1.4%の時間で処理を終えられました。bashはこのようにして様々な頻出コマンドを高速化しています。
次のようにすれば組み込みコマンドのリストを得られます。興味のあるかたはご確認ください。
$ enable
enable .
enable :
enable [
...
enable unalias
enable unset
enable wait
$
それぞれ高速化のためのものか、あるいは組込みコマンドでなければ実現不可能な機能なのかを考えてみるのもおもしろいかもしれません。
コマンド作成の実行環境
本記事執筆時点でのdebian/testing(stretch)最新版
必要なパッケージ
bash-builtins
組み込みコマンドの作成
簡単のため、みなさんのスクリプトにおいて、内部で頻繁に呼ばれるhello worldプログラムがボトルネックになっていると仮定して、同コマンドの組み込みコマンドを自作してみましょう。
もとのコマンドのソースは次の通りです。
#include <stdio.h>
int main(void) {
puts("Hello world!");
return 0;
}
実行してみましょう。
$ ./hello
Hello world!
$
これと同じことをする組み込みコマンドのソースは次の通りです。
#include <builtins.h>
#include <shell.h>
#include <stdio.h>
static int myhello(WORD_LIST *list) {
puts("Hello world!");
fflush(stdout);
return EXECUTION_SUCCESS;
}
static char *desc[] = {
"Show a greeting message.",
"",
"It's far faster than launching executable file",
"because it't not necessary to call exec() and fork().",
(char *)NULL
};
struct builtin myhello_struct = {
"myhello", // builtin command name
myhello, // function called when issueing this command
BUILTIN_ENABLED, // initial flag
desc, // long description
"myhello", // short description
0,
};
組込みコマンドの処理に対応する関数myhello(), ドキュメントとなるdesc変数、および、この組み込みコマンドをbashに登録するために必要なmyhello_structを作成する必要があります。それぞれの意味についてはソース内のコメントや後述の参考資料をごらんください。
これをビルドするためのMakefileは次のようになります。
BINARIES := hello myhello
.PHONY: all clean
all: hello myhello
myhello: myhello.c
cc -L $@ -I/usr/include/bash/ -I/usr/include/bash/include -fpic -shared -o myhello.so myhello.c
clean:
rm -rf *.o *.so *~ $(BINARIES)
ビルドしましょう。
$ make
cc hello.c -o hello
cc -L myhello -I/usr/include/bash/ -I/usr/include/bash/include -fpic -shared -o\
myhello.so myhello.c
$ ls
LICENSE Makefile README.md benchmark hello hello.c myhello.c myhello.so
成功です。作成されたmyhello.soという共有ライブラリが組み込みコマンドの実体です。
作成したコマンドの組み込み
コマンドの組み込みは次のようにします。
$ enable -f ./myhello.so myhello
$
これは./myhello.soという共有ライブラリをmyhelloという名前で組み込むという意味です。ファイル名の前の"./"を省略するとコマンドが失敗するので注意してください。
成功したかどうかを確認しましょう。
$ enable | grep myhello
enable myhello
$
myhello組み込みコマンドが正しくbashに認識されています。
ドキュメントも表示できます。
$ help myhello
myhello: myhello
Show a greeting message.
It's far faster than launching executable file
because it't not necessary to call exec() and fork().
$
実行してみましょう。
$ myhello
Hello world!
$
成功です。
効果の確認
それぞれ同じことをする実行ファイル(./hello)とmyhello組み込みコマンドを10000回連続実行した際の所要時間を計測してみましょう。
$ time (for ((i=0;i<10000;i++)) ; do ./hello >/dev/null ; done)
real 0m5.508s
user 0m1.932s
sys 0m0.848s
$ time (for ((i=0;i<10000;i++)) ; do myhello >/dev/null ; done)
real 0m0.087s
user 0m0.076s
sys 0m0.008s
$
組み込みコマンドは実行ファイルの場合に比べて1.5%の所要時間で処理を終えられました。
注意
組み込みコマンドは実行ファイルに比べて高速なのはよいのですが、Cソースを書かなければいけないこと、および、高速作成にビルドを伴うために移植性を保つのが面倒という問題があります。乱用するのは避けて、特定環境でスクリプトそのもののソースを変更せずに特定処理だけを高速化したいというような特殊な場合に使用するのがよいと筆者は考えます。ちょうどrubyやpythonなどのスクリプト言語の一部をCで実装したり、Cプログラムの一部をアセンブリ言語で実装したりするのと似ています。
おわりに
本記事で使用したソースはgithub上にアップロードしていますので、参考にしてください。
参考資料
- /usr/share/doc/bash-builtins以下のファイル
- 入門bash 第3版の付録C