tmuxに独自機能を追加する
December 06, 2011 at 01:05 AM | categories: advent calendar, tmux |これはターミナルマルチプレクサ Advent Calendar 2011の6日目7日目です。(netmark.jpさん,matsuuさん、すいません、出遅れました。)
5日目はyoshikawさんの開発版GNU Screenを使ってみようでした。
6日目はnetmark.jpさんのGNU screenをもうちょっとだけ便利に使おう!でした。
この記事ではtmuxのいじり方を簡単に紹介してみます。
はじめに
tmuxは比較的若いプロジェクトなこともあってか、かゆいところに手が届かないシチュエーションがたまにあります。 個人的に不満だったのはバッファの挙動です。
tmuxでバッファに文字列をコピーした場合、list-buffersに新しいものほど前方になるように格納されていきます。 choose-bufferコマンドを使えばある程度簡単に過去のものもさかのぼれますがしばらくすると埋もれてしまいます。 tmuxのバッファにはホスト名とかよくつかうコマンドとかをいれておいて番号を指定して最速でとりだしたいのに、 場所が固定されていないといちいち探さなくてはならず不便に感じていました。 なんとかできないかと思ってソースをいじってみたら、意外と簡単だったので簡単な解説をまじえつつ手順を紹介してみます。
目次
準備
ソースをとってくる
本家サイトからソースをおとします。とりあえず現在の最新版(1.5)でいいと思います。
依存パッケージを入れる
tmatsuuさんの記事にもちょっとありましたが、 tmuxはlibeventとnlcursesに依存しているのでビルドするにはライブラリのヘッダも必要です。 ubuntuだと以下パッケージを入れとけばたぶん大丈夫です。
sudo apt-get install libevent-dev libncurses5-dev
ビルド
./configure && make
ここまでできたら準備OKです。
ソース構成
結構たくさんファイルがありますが、ファイル名を見れば大体雰囲気がつかめると思います。
何か機能を追加したいだけなら、いじる必要のあるファイルはごく一部です。
- tmux.h
-
- ほとんどのコマンドは全部ここに宣言がまとめられてるのでほぼ必ずいじります。
- options-table.c
- オプション追加する場合はここに追加します。値の肩などを定義するテーブルが宣言されているので、中身を見れば大体わかると思います。
いじる必要はないですがarray.hはいろんなところでつかわれてるので目を通しとくとよいと思います。 Cでリストが使いたい場合は他のとこでも流用できそうな便利なヘッダです。
あとは必要に応じて関係しそうなところをみていけばよいです。
いじってみる
バッファまわりを改造したいので、paste.cというファイルが主な対象になります。
細かい解説は省略しますが、paste_stackという構造体がグローバルで共有しているリストの本体で、array.hで宣言されているマクロで値がとりだせるようになっています。 本来の動作であるpaste_addを参考にしつつpaste_add_tailなんて関数を追加します。 後に追加した文字列ほど後ろにいってほしいですが、直前のものだけは先頭にあったほうが便利なのでちょっと変則的な感じになってますが、やっていることはシンプルです。
diff -u tmux-1.5/paste.c tmux-1.5-patched/paste.c --- tmux-1.5/paste.c 2011-07-09 18:42:38.000000000 +0900 +++ tmux-1.5-patched/paste.c 2011-12-06 23:36:31.921805251 +0900 @@ -119,6 +119,38 @@ pb->size = size; } +/* + * Add an item onto the tail of the stack, freeing the bottom if at limit. Note + * that the caller is responsible for allocating data. + */ +void +paste_add_tail(struct paste_stack *ps, char *data, size_t size, u_int limit) +{ + struct paste_buffer *pb; + + if (size == 0) + return; + + while (ARRAY_LENGTH(ps) >= limit) { + pb = ARRAY_LAST(ps); + xfree(pb->data); + xfree(pb); + ARRAY_TRUNC(ps, 1); + } + + pb = xmalloc(sizeof *pb); + + if (ARRAY_LENGTH(ps) > 0) { + ARRAY_ADD(ps, ARRAY_FIRST(ps)); + ARRAY_SET(ps, 0, pb); + } else { + ARRAY_ADD(ps, pb); + } + ARRAY_INSERT(ps, 0, pb); + + pb->data = data; + pb->size = size; +}
ここで追加した関数はtmux.hで宣言しておきます
diff -u tmux-1.5/tmux.h tmux-1.5-patched/tmux.h --- tmux-1.5/tmux.h 2011-07-09 18:42:38.000000000 +0900 +++ tmux-1.5-patched/tmux.h 2011-12-06 23:36:31.925805201 +0900 @@ -1487,6 +1487,7 @@ int paste_free_top(struct paste_stack *); int paste_free_index(struct paste_stack *, u_int); void paste_add(struct paste_stack *, char *, size_t, u_int); +void paste_add_tail(struct paste_stack *, char *, size_t, u_int); int paste_replace(struct paste_stack *, u_int, char *, size_t); char *paste_print(struct paste_buffer *, size_t);
また、今回の機能は設定ファイルでon/offを切り替えられるようにしておきたいので、options-table.cも編集します。
diff -u tmux-1.5/options-table.c tmux-1.5-patched/options-table.c --- tmux-1.5/options-table.c 2011-07-09 18:42:38.000000000 +0900 +++ tmux-1.5-patched/options-table.c 2011-12-06 23:36:31.921805251 +0900 @@ -57,6 +57,11 @@ .default_num = 20 }, + { .name = "reverse-buffer", + .type = OPTIONS_TABLE_FLAG, + .default_num = 0 + }, + { .name = "escape-time", .type = OPTIONS_TABLE_NUMBER, .minimum = 0,
実際にバッファの制御関数を呼び出しているのはwindow-copy.cなので、さっき追加したオプションの値をみて切り替えられるようにします。
diff -u tmux-1.5/window-copy.c tmux-1.5-patched/window-copy.c --- tmux-1.5/window-copy.c 2011-07-09 18:42:38.000000000 +0900 +++ tmux-1.5-patched/window-copy.c 2011-12-06 23:36:31.921805251 +0900 @@ -1348,7 +1348,11 @@ /* Add the buffer to the stack. */ limit = options_get_number(&global_options, "buffer-limit"); - paste_add(&global_buffers, buf, off, limit); + if (options_get_number(&global_options, "reverse-buffer")) { + paste_add(&global_buffers, buf, off, limit); + } else { + paste_add(&global_buffers, buf, off, limit); + } }
あとは設定ファイルに以下を追加してmakeしたtmuxを起動してみます。
set -g reverse-buffer on
バッファにコピーした文字列がlist-buffersの後ろに追加されていけば成功です。 簡単ですよね?
まとめ
以上、tmuxにreverse-bufferというオプションを追加して、バッファの挙動を変更できるようにしてみました。
端末制御のプログラムは普段あまり触れる機会がないし複雑そうなイメージがあって敬遠しがちですが、ちょっとした機能を追加するだけだったら意外と簡単だったりするのでどんどんいじってみるといいと思います。
blog comments powered by Disqus
About Me
mojavy |
Recent posts
95/5 Mbps とは
(August 30, 2015 at 04:22 PM)組み込み用プログラミング言語のパフォーマンス比較
(April 21, 2015 at 01:10 AM)最近読んだ本
(April 05, 2015 at 01:23 PM)Phabricatorを使ったワークフローについて
(March 02, 2015 at 08:55 PM)dnsimpleでダイナミックDNSをつかう
(December 23, 2014 at 08:02 PM)www2014のアドテク関連のResearch Trackメモ
(October 06, 2014 at 09:05 PM)flappymacs がMELPAに登録されました
(July 16, 2014 at 01:07 AM)EmacsでFlappy Birdっぽいもの書きました
(July 10, 2014 at 08:01 PM)
Recent Popular posts
Popular posts
Categories
- C (rss) (3)
- R (rss) (1)
- adtech (rss) (1)
- advent calendar (rss) (2)
- algorithms (rss) (2)
- android (rss) (2)
- aws (rss) (1)
- blog (rss) (2)
- blogofile (rss) (3)
- books (rss) (1)
- c++ (rss) (1)
- chef (rss) (4)
- common lisp (rss) (10)
- debian (rss) (2)
- dns (rss) (1)
- elasticsearch (rss) (1)
- elf (rss) (1)
- elisp (rss) (1)
- emacs (rss) (5)
- english (rss) (1)
- game (rss) (2)
- gearman (rss) (1)
- git (rss) (1)
- github (rss) (1)
- gitlab (rss) (1)
- golang (rss) (2)
- history (rss) (1)
- impress.js (rss) (1)
- internet (rss) (1)
- ios (rss) (3)
- jekyll (rss) (1)
- jenkins (rss) (1)
- linux (rss) (4)
- lisp (rss) (2)
- ltsv (rss) (1)
- lua (rss) (1)
- mac (rss) (3)
- mach-o (rss) (1)
- memo (rss) (2)
- mustache (rss) (1)
- note (rss) (1)
- objective-c (rss) (4)
- os (rss) (1)
- osx (rss) (2)
- others (rss) (1)
- paco (rss) (1)
- pdf (rss) (1)
- php (rss) (2)
- postfix (rss) (1)
- programming (rss) (12)
- project management (rss) (1)
- python (rss) (5)
- quicklinks (rss) (6)
- raspberry pi (rss) (2)
- redmine (rss) (1)
- reveal.js (rss) (1)
- ruby (rss) (10)
- sbcl (rss) (2)
- security (rss) (1)
- shell (rss) (2)
- smtp (rss) (1)
- solr (rss) (1)
- statistics (rss) (2)
- tips (rss) (10)
- tmux (rss) (3)
- toml (rss) (1)
- tools (rss) (1)
- twitter (rss) (1)
- ubuntu (rss) (1)
- unix (rss) (5)
- v8 (rss) (1)
- web (rss) (7)
- xcode (rss) (1)
- zeromq (rss) (2)
Archives
- August 2015 (1)
- April 2015 (2)
- March 2015 (1)
- December 2014 (1)
- October 2014 (1)
- July 2014 (3)
- March 2014 (6)
- February 2014 (4)
- November 2013 (3)
- October 2013 (4)
- September 2013 (2)
- July 2013 (2)
- June 2013 (2)
- May 2013 (1)
- April 2013 (6)
- March 2013 (3)
- February 2013 (8)
- January 2013 (5)
- December 2012 (1)
- November 2012 (6)
- October 2012 (7)
- August 2012 (1)
- July 2012 (9)
- June 2012 (1)
- April 2012 (1)
- December 2011 (2)
- November 2011 (2)