***** Wesnoth 1.12.6 で シナリオ開始時の文字列と シナリオ中の会話の文字列を取りだす方法調査 第1回目報告文書 ***
2016/5/30
// この文書のライセンスはWesnothやWesnoth本家配布のゲームWesnothのソースや文書と同じライセンスとします
// ライセンスが複数ある場合より自由度の高いものに従うものとします
// この文書による被害には一切責任を持ちません、この情報をもとにやるなら自己責任でお願いします。
// それほど危険な情報じゃないけど自己防衛の為に書いとます
**** 調べた環境
Wesnoth のバージョンは 1.12.6
Debian Linux Sid amd64 環境で以下の調査と検証を行ないました
(この環境でのDebianでもwesnothパッケージのバージョンは 1:1.12.6-1 )
**** 使ったツール
GNU Global(ソースを調べるのに)
GNU GDB (デバッガ)これが主役
Nemiver (これもデバッガ 最初これ使ってたけど途中からGDBにした)
*** 調査の目的
シナリオ開始時の英文と、シナリオプレイ中の台詞をテキストデータとして取りだす
それをChronium
*** 調査の経緯
これは省略
ソースを真面目に読まずに適当にやったためまわりみちしました
なので省略
*** 上手くいった調査方法
まず wesnothのデバッグ情報を含んだパッケージ wesnoth-1.12-dbg を 以下のコマンドでインストール
sudo aptitude install wesnoth-1.12-dbg
ソースも必要みたいだったので
/build/wesnoth-1.12-65cmTO
を作成して(このディレクトリはデバッガがソースないって文句いってきたディレクトリ)
そこで
apt-get source wesnoth
としてソースをダウンロード
*** 最初はGNU Golobalでソース調べてたので
cd wesnoth-1.12-1.12.6/src/
htags -gansx
としてソースを調べやすくするHTMLファイルも生成した
*** 上手くいった調査方法
まずWesnothを立ちあげてから
調べたいメッセージ表示中に
gdb等のデバッガでattachして
btをかけたりソースを調べて
それらしい文字列が含まれてそうな変数やクラスを
表示して見つけて
ブレイクポイントとして適切そうなものを調べた
*** wesnoth を起動して gdbを起動した後に入れたコマンド
実際はもっと色々試行錯誤したけど
それ書いても面白くもなんともないので
最終上手くいったものは
shell ps x | grep wesnoth
でattach すべき 実行してるwesnoth の pid を調べます
attach 調べたpid番号(22909みたいな数字)
その後
b storyscreen::part_ui::part_ui
b gui2::show_wml_message
でブレイクポイントを指定しておきます
あとはそれぞれのブレイクポイントで停止してから
command 1
p p->text_
c
end
を入力して
c
そして2番目のブレイクポイントで止まったら
command 2
p message
c
end
を入力して
c
これで後は画面がいっぱいになったときに
エンターを押すだけでgdbのターミナルにwesnothのシナリオ開始時のテキスト文と
シナリオ中の会話文がテキストとして表示されます
*** 今後
入れるタイミングとかあるし
gdbを起動した時にwesnothが起動してないなら
起動してpid調べるとかして
調べたpidでattachするとか面倒くさい作業が必要だったので
これを自動化するには
元のソースを改造して出力するようにするか
あるいは gdbの操作自体を expect のような自動化ツールを使ってやれば可能となりました
パッチあてるにしても必要な箇所が上で調べれたので
やるかやらんかわかりません
文字列の取りだしには成功したっぽいので
それが出来たらその文字列を自動翻訳にかけるなり
マウスでかざしたら楽に辞書ひける環境で使えるようにするなり
好きにできるので
また時間があったらやるかもしれません
ここまでで結構楽しかったし
当初の目的は達成されたので比較的満足してしまってる今日この頃
*** パッチを作てくれる人向けの場所説明
gdb で info b
として表示すると以下になってます
Num Type Disp Enb Address What
1 breakpoint keep y 0x0000000000dd0e90 in storyscreen::part_ui::part_ui(storyscreen::part&, display&, gui::button&, gui::button&, gui::button&) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/storyscreen/render.cpp:80
2 breakpoint keep y 0x0000000000c22220 in gui2::show_wml_message(bool, CVideo&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*, unsigned int, std::vector<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, int*) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/gui/dialogs/wml_message.cpp:175
なので
storyscreen::part_ui::part_ui(storyscreen::part&, display&, gui::button&, gui::button&, gui::button&) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/storyscreen/render.cpp:80
と
gui2::show_wml_message(bool, CVideo&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*, unsigned int, std::vector<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, int*) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/gui/dialogs/wml_message.cpp:175
でなんとかすれば文字列をコピペする形で外に取りだすことは簡単だと思います
パッケージを改造するとバージョン上げるごとに
パッチをあてるのが面倒なので
私はもうちょっとgdbと他のツールでなんとかする方向で頑張るかもしれません
***その他の改造について
上の方法で調べれば、勝利条件とかポップアップダイアログの
職種や人種のマニュアルの文字も取りだして自動飜訳するために
ソースの何処をいじれば良いのか調べれると思います
今回GDBをメインで調べてみた感じだと
チートコード的なことをGDBで出来そう
ソースを調べるにはGNU GLOBAL使って
ヒットポイントが減って自分側なら
自動で増やすとか非確定簡単にできそう
ソース公開されてる分
クローズドソースのものよりずっとやりやすいし
誰かやってくれる人がいることを期待しつつ
この調査報告書を終ります
しばらくして誰もやってくれなくて
私がやる気になったら私がやるかも
得にGDBを利用した方はそれほど手間かけずに出来そうだし
つまり他の誰かがやってくれることを
非常に期待してます
buynnnmmm1
***** Wesnoth 1.12.6 で シナリオ開始時の文字列と シナリオ中の会話の文字列を取りだす方法調査 第1回目報告文書 ***

2016/5/30


// この文書のライセンスはWesnothやWesnoth本家配布のゲームWesnothのソースや文書と同じライセンスとします
// ライセンスが複数ある場合より自由度の高いものに従うものとします
// この文書による被害には一切責任を持ちません、この情報をもとにやるなら自己責任でお願いします。
// それほど危険な情報じゃないけど自己防衛の為に書いとます


**** 調べた環境
Wesnoth のバージョンは 1.12.6 
Debian Linux Sid amd64 環境で以下の調査と検証を行ないました
(この環境でのDebianでもwesnothパッケージのバージョンは 1:1.12.6-1 )


**** 使ったツール
GNU Global(ソースを調べるのに)
GNU GDB (デバッガ)これが主役
Nemiver (これもデバッガ 最初これ使ってたけど途中からGDBにした)


*** 調査の目的
シナリオ開始時の英文と、シナリオプレイ中の台詞をテキストデータとして取りだす
それをChronium


*** 調査の経緯
これは省略
ソースを真面目に読まずに適当にやったためまわりみちしました
なので省略


*** 上手くいった調査方法
まず wesnothのデバッグ情報を含んだパッケージ wesnoth-1.12-dbg を 以下のコマンドでインストール

sudo aptitude install wesnoth-1.12-dbg

ソースも必要みたいだったので

/build/wesnoth-1.12-65cmTO

を作成して(このディレクトリはデバッガがソースないって文句いってきたディレクトリ)

そこで

apt-get source wesnoth

としてソースをダウンロード


*** 最初はGNU Golobalでソース調べてたので

cd wesnoth-1.12-1.12.6/src/
htags -gansx

としてソースを調べやすくするHTMLファイルも生成した


*** 上手くいった調査方法

まずWesnothを立ちあげてから
調べたいメッセージ表示中に
gdb等のデバッガでattachして
btをかけたりソースを調べて
それらしい文字列が含まれてそうな変数やクラスを
表示して見つけて
ブレイクポイントとして適切そうなものを調べた


*** wesnoth を起動して gdbを起動した後に入れたコマンド
実際はもっと色々試行錯誤したけど
それ書いても面白くもなんともないので
最終上手くいったものは

shell ps x | grep wesnoth

でattach すべき 実行してるwesnoth の pid を調べます

attach 調べたpid番号(22909みたいな数字)

その後

b storyscreen::part_ui::part_ui
b gui2::show_wml_message

でブレイクポイントを指定しておきます

あとはそれぞれのブレイクポイントで停止してから

command 1
p p->text_
c
end

を入力して

c

そして2番目のブレイクポイントで止まったら

command 2
p message
c
end

を入力して

c

これで後は画面がいっぱいになったときに
エンターを押すだけでgdbのターミナルにwesnothのシナリオ開始時のテキスト文と
シナリオ中の会話文がテキストとして表示されます



*** 今後
入れるタイミングとかあるし
gdbを起動した時にwesnothが起動してないなら
起動してpid調べるとかして
調べたpidでattachするとか面倒くさい作業が必要だったので

これを自動化するには
元のソースを改造して出力するようにするか
あるいは gdbの操作自体を expect のような自動化ツールを使ってやれば可能となりました

パッチあてるにしても必要な箇所が上で調べれたので
やるかやらんかわかりません

文字列の取りだしには成功したっぽいので
それが出来たらその文字列を自動翻訳にかけるなり
マウスでかざしたら楽に辞書ひける環境で使えるようにするなり
好きにできるので
また時間があったらやるかもしれません

ここまでで結構楽しかったし
当初の目的は達成されたので比較的満足してしまってる今日この頃


*** パッチを作てくれる人向けの場所説明

gdb で info b
として表示すると以下になってます

Num     Type           Disp Enb Address            What
1       breakpoint     keep y   0x0000000000dd0e90 in storyscreen::part_ui::part_ui(storyscreen::part&, display&, gui::button&, gui::button&, gui::button&) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/storyscreen/render.cpp:80
2       breakpoint     keep y   0x0000000000c22220 in gui2::show_wml_message(bool, CVideo&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*, unsigned int, std::vector<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, int*) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/gui/dialogs/wml_message.cpp:175


なので

storyscreen::part_ui::part_ui(storyscreen::part&, display&, gui::button&, gui::button&, gui::button&) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/storyscreen/render.cpp:80


と

gui2::show_wml_message(bool, CVideo&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >*, unsigned int, std::vector<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, int*) at /build/wesnoth-1.12-65cmTO/wesnoth-1.12-1.12.6/src/gui/dialogs/wml_message.cpp:175

でなんとかすれば文字列をコピペする形で外に取りだすことは簡単だと思います

パッケージを改造するとバージョン上げるごとに
パッチをあてるのが面倒なので
私はもうちょっとgdbと他のツールでなんとかする方向で頑張るかもしれません


***その他の改造について
上の方法で調べれば、勝利条件とかポップアップダイアログの
職種や人種のマニュアルの文字も取りだして自動飜訳するために
ソースの何処をいじれば良いのか調べれると思います

今回GDBをメインで調べてみた感じだと
チートコード的なことをGDBで出来そう
ソースを調べるにはGNU GLOBAL使って
ヒットポイントが減って自分側なら
自動で増やすとか非確定簡単にできそう
ソース公開されてる分
クローズドソースのものよりずっとやりやすいし

誰かやってくれる人がいることを期待しつつ
この調査報告書を終ります

しばらくして誰もやってくれなくて
私がやる気になったら私がやるかも
得にGDBを利用した方はそれほど手間かけずに出来そうだし

つまり他の誰かがやってくれることを
非常に期待してます


buynnnmmm1