ステップ2: コントローラの作成と導入¶
ステップ1ではコントローラが無かったため、シミュレーションの際にTankモデルの砲身の姿勢を維持することができませんでした。そこでステップ2では姿勢を維持するための最小限のコントローラの作成を通して、コントローラ導入の基本を学ぶことにします。
目次
コントローラの形式について¶
一般的に、コントローラの実装形式は様々なものがあり得ます。特定のロボットシステムやシミュレータが規定する形式もありますし、ROSなどのミドルウェアの規定する形式でコントローラを実装することも一般的となっています。
これに関して、本チュートリアルでは、Choreonoid独自の 「シンプルコントローラ(SimpleController)」の形式でコントローラの実装を行います。シンプルコントローラは、C++言語とChoreonoid内部のデータ構造を用いてコントローラを実装するもので、ROSなどのミドルウェアを使用する場合と比べて覚えることが少なくて済み、コードも比較的シンプルなものになるといった利点があります。
ただしこれはChoreonoidの独自形式であるため、汎用性の面ではROSなどのミドルウェアに劣ります。また、ROSなどのミドルウェアが提供するような通信機能を提供するものでもありません。ChoreonoidはROSと連携する機能も備えていますので、必要に応じてそちらも用いるようにしてください。ROSを利用する方法については ROSとの連携 で解説しています。
コントローラ "TurretController1" の実装¶
シンプルコントローラ形式ではC++のクラスとしてコントローラを実装します。ここでは砲塔(Turret)のピッチ軸を維持するだけの "TurretController1" の実装を行います。まずこのコントローラのソースコードを以下に示します。
1#include <cnoid/SimpleController>
2
3using namespace cnoid;
4
5class TurretController1 : public SimpleController
6{
7 Link* joint;
8 double q_ref;
9 double q_prev;
10 double dt;
11
12public:
13 virtual bool initialize(SimpleControllerIO* io) override
14 {
15 joint = io->body()->link("TURRET_P");
16 joint->setActuationMode(Link::JOINT_TORQUE);
17 io->enableIO(joint);
18 q_ref = q_prev = joint->q();
19
20 dt = io->timeStep();
21
22 return true;
23 }
24
25 virtual bool control() override
26 {
27 // PD gains
28 static const double P = 200.0;
29 static const double D = 50.0;
30
31 double q = joint->q(); // input
32 double dq = (q - q_prev) / dt;
33 double dq_ref = 0.0;
34 joint->u() = P * (q_ref - q) + D * (dq_ref - dq); // output
35 q_prev = q;
36
37 return true;
38 }
39};
40
41CNOID_IMPLEMENT_SIMPLE_CONTROLLER_FACTORY(TurretController1)
以下では、このコントローラをシミュレーションプロジェクトに導入し、シミュレーションを行うまでを解説します。その後、コントローラの実装内容について解説したいと思います。
コントローラのビルド方法¶
C++言語で記述されたコントローラのソースコードをビルド(コンパイル)するにあたって、主に以下の2通りの方法があります。
Choreonoid本体と一緒にビルドを行う
Choreonoid本体とは別にビルドを行う
Choreonoidをソースからビルドしている場合、1の方が手軽ですので、本チュートリアルでは1で解説を進めることにします。
ビルド方法の詳細については コントローラのビルド で解説していますのでそちらを参照してください。実際のコントローラ開発時には、環境や目的に応じて両手法を使い分けることになるかと思います。
いずれの場合も、まずコントローラのソースコードを格納するディレクトリが必要となります。これを 「プロジェクトディレクトリ」 と呼ぶことにします。(プロジェクトディレクトリには本チュートリアルで作成する他のファイルも格納することにします。)
上記の1の方法でビルドを行う場合は、プロジェクトディレクトリをChoreonoid本体のビルドシステムに認識させる必要があります。これについても2通りの方法があります。
Choreonoidのソースディレクトリ内の "ext" ディレクトリに対象のディレクトリを配置する
ChoreonoidのCMakeの設定で "ADDITIONAL_EXT_DIRECTORIES" に対象のディレクトリを指定する
Aを採用する場合は、プロジェクトディレクトリをextディレクトリ以下に作成します。 Bを採用する場合は、プロジェクトディレクトリをどこかに作成した上で、そのディレクトリへのパスを上記の設定項目に指定します。対象ディレクトリが複数ある場合は、セミコロンで区切って入力することができます。
特に理由がなければAの方法を用いるのがよいかと思います。以下ではその方法で進めることにします。
プロジェクトディレクトリの作成¶
ではプロジェクトディレクトリをext以下に作成しましょう。ディレクトリ名は何でもよいですが、ここでは "tank" というディレクトリ名を使うことにします。例えばChoreonoidのソースディレクトリ上で
cd ext
mkdir tank
などとします。
この "tank" ディレクトリの中に、本ステップの冒頭の コントローラ "TurretController1" の実装 で示したソースコードのファイルを作成してください。ファイル名は "TurretController1.cpp" とすることにします。
ここまで進めると、プロジェクトディレクトリとその中のファイルは以下の構成になります。
Choreonoidソースディレクトリ
+ ext
+ tank
- TurretController1.cpp
なお、ステップ1で プロジェクトの保存 を行ったファイルについても、この "tank" ディレクトリにまとめて保存するとよいでしょう。そのように、本チュートリアルのシミュレーションプロジェクトに関わる全てのファイルをまとめておく意図もあるので、このディレクトリを「プロジェクトディレクトリ」と呼んでいます。
注釈
Ubuntuでどのテキストエディタを使ったらよいか分からないという方は、とりあえず標準の "gedit" というテキストエディタを使ってみて下さい。アプリの検索で "gedit"と入力すると出てくる「テキストエディター」というアイコンで起動できます。コマンドラインから "gedit" と入力してもOKです。
注釈
ソースコードのファイルはChoreonoid本体の "sample/tutorial/Tank/" 以下に格納されています。自分で入力するのが面倒な場合は、このファイルを利用してもOKです。本チュートリアルで言及する他のファイルについてもこのディレクトリに格納されていますので、必要に応じてご利用下さい。
CMakeLists.txtの記述¶
次に、プロジェクトディレクトリ "tank" 内に "CMakeLists.txt" というテキストファイルを新たに作成し、ここにコントローラのコンパイルに関する設定を記述します。
といっても今回記述すべき内容は非常にシンプルで、新規作成した "CMakeLists.txt" のファイルに以下の一行を記述すればOKです。
choreonoid_add_simple_controller(TankTutorial_TurretController1 TurretController1.cpp)
ここで用いている "choreonoid_add_simple_controller" という関数は、Choreonoid本体のCMake記述にて予め定義された関数です。この関数に、生成されるコントローラの名前とソースファイルを与えるだけで、コントローラのコンパイルを行うことができます。
今回コントローラ名に "TankTutorial" というプレフィックスをつけています。これは必須というわけではありませんが、今後他のプロジェクトで開発したコントローラと区別しやすくするためにつけています。
ここまで進めると、プロジェクトディレクトリとその中のファイルは以下の構成になります。
1Choreonoidソースディレクトリ
2 + ext
3 + tank
4 - CMakeLists.txt
5 - TurretController1.cpp
コントローラのビルド¶
コントローラのビルドを行いましょう。今回はChoreonoid本体と一緒にビルドする手法を用いているので、再度Choreonoid本体のビルドを行えばOKです。今回CMakeLists.txtが追加されましたので、まずそれを認識させるため、CMakeを再実行しましょう。ステップ1で開いた端末についてはChoreonoid本体のソースディレクトリがカレントディレクトリになっているかと思います。そうでない場合は
cd [Choreonoid本体のソースディレクトリへのパス]
として、Choreonoid本体のソースディレクトリに移動してください。
通常はビルド用の "build" ディレクトリを作成してビルドしているかと思いますので、そちらに移動して、CMakeを再実行します。
cd build
cmake ..
つぎにビルドディレクトリ上で続けて
make
と入力して下さい。(ビルド方法の詳細は ソースコードからのビルドとインストール (Ubuntu Linux編) の Choreonoidのビルド を参照して下さい。)
この際 コントローラのビルド方法 で述べたAかBの条件を満たしていれば、上記のCMakeLists.txtが検出され、その内容も実行されるというわけです。
ビルドに成功すると、ビルドディレクトリの "lib/choreonoid-x.x/simplecontroller" (x.xはChoreonoidのバージョン番号)に、
TankTutorial_TurretController1.so
というファイルが生成されるはずです。これがコントローラ本体のファイルとなります。この拡張子からも分かるように、コントローラの実態は共有ライブラリファイルとなります。コントローラが生成されたディレクトリは今後 「コントローラディレクトリ」 と呼ぶことにします。
コンパイルエラーが出た場合は、エラーメッセージを参考にして、ソースコードやCMakeLists.txtの記述を見直してみて下さい。
注釈
この後 "make install" を実行すると、生成されたコントローラのファイルである "TankTutorial_TurretController1.so" もインストール先にコピーされます。ただし本チュートリアルでは、ステップ1の Choreonoidの起動 で述べたように、"make install" は実行せずに、buildディレクトリ内のファイルを実行する前提で解説していますので、その点にご注意ください。
コントローラの導入¶
ビルドしたコントローラをシミュレーションプロジェクトに導入します。
コントローラアイテムの生成¶
シンプルコントローラは「シンプルコントローラアイテム」によってプロジェクトに導入するようになっていますので、まずは対応するアイテムを生成しましょう。メインメニューの「ファイル」-「新規」から「シンプルコントローラ」を選択して生成します。アイテムの名前はなんでもよいですが、コントローラに合わせて "TurretController" とするとよいでしょう。
生成したアイテムは、下図のように、制御対象のTankアイテムの子アイテムとして配置するようにします。
この配置によって、コントローラの制御対象がTankモデルであることを明示します。これを実現するにあたっては、Tankアイテムを選択状態としてからコントローラアイテムの生成を行ってもよいですし、生成後にこの配置になるようドラッグしてもOKです。
コントローラ本体のセット¶
次に先ほど作成したコントローラの本体をシンプルコントローラアイテムにセットします。
これはシンプルコントローラアイテムの「コントローラモジュール」というプロパティを用いて行います。まず、アイテムツリー上で "TurretController" を選択します。するとこのアイテムのプロパティ一覧がアイテムプロパティビュー上に表示されますので、その中から「コントローラモジュール」というプロパティを探して下さい。そのプロパティの値の部分(デフォルトでは空欄となっている)をダブルクリックすると、モジュールのファイルを入力することができます。
この際、入力用のファイルダイアログを用いて入力するのが手軽です。コントローラモジュールの入力時には下図に示すように値を入力する箇所の右端にアイコンがあります。
このアイコンをクリックすると、ファイル選択のダイアログが表示されます。このダイアログは通常シンプルコントローラ格納用の標準ディレクトリを指しています。そこには先ほど作成した "TankTutorial_TurretController1.so" が格納されているはずですので、これを選択して下さい。
これでコントローラ本体がシンプルコントローラアイテムにセットされました。これでコントローラを機能させることができます。
ここまで設定できたら、またプロジェクトを保存しておきましょう。ファイル名は "step2.cnoid" として、プロジェクトディレクトリに保存しておくとよいかと思います。
シミュレーションの実行¶
以上の設定を行った上でシミュレーションを実行して下さい。すると、ステップ1では重力で下を向いてしまった砲身が、今回は正面を向いたままとなっているはずです。これはコントローラ "TurretController1" によって、姿勢の維持に必要なトルクが砲塔ピッチ軸にかけられているからです。
うまくいかない場合は、メッセージビューも確認してみて下さい。コントローラの設定や稼働に問題があると、シミュレーション開始時にその旨を知らせるメッセージが出力される場合があります。
なお、このコントローラでは砲塔ヨー軸の制御は行っていないため、そちらには力がかかっていません。ステップ1の時と同様に、 インタラクション機能 を用いて砲塔部分をドラッグすると、ヨー軸に関してはフリーで動かせることが分かります。
実装内容の解説¶
今回作成したコントローラ "TurretController1" の実装内容は以下のようになっています。
SimpleControllerクラス¶
まず、シンプルコントローラはChoreonoidで定義されている "SimpleController" クラスを継承したクラスとして実装するようになっています。そこでまず
#include <cnoid/SimpleController>
によって、このクラスが定義されているヘッダをインクルードしています。Choreonoidが提供するヘッダファイルはインクルードディレクトリの "cnoid" サブディレクトリに格納されるようになっており、このように cnoid ディレクトリからのパスとして記述します。拡張子は必要ありません。
また、Choreonoidで定義されているクラスは全て "cnoid" という名前空間に所属しています。ここでは
using namespace cnoid;
によって名前区間を省略できるようにしています。
コントローラのクラス定義は、
class TurretController1 : public SimpleController
{
...
};
によって行われています。SimpleControllerを継承するかたちでTurretController1を定義していることが分かります。
SimpleControllerクラスではいくつかの関数が仮想(バーチャル)関数として定義されており、継承先でそれらの関数をオーバーライドすることでコントローラの処理内容を実装します。通常以下の2つの関数をオーバーライドします。
virtual bool initialize(SimpleControllerIO* io)
virtual bool control()
initialize関数の実装¶
initialize関数はコントローラの初期化を行う関数で、シミュレーション開始の直前に1回だけ実行されます。
この関数に引数として与えられるSimpleControllerIO型は、コントローラの入出力に必要な機能をまとめたクラスとなっています。この詳細は コントローラの実装 の IOオブジェクト をみていただくとして、ここではまず
joint = io->body()->link("TURRET_P");
によって、砲塔ピッチ軸の入出力を行うためのLinkオブジェクトを取得し、joint変数に格納しています。
io->body() によってTankモデル入出力用のBodyオブジェクトを取得し、続けてこのオブジェクトが有するLinkオブジェクトから "TURRET_P" という名前を持つものを取得しています。これは Tankモデルの作成 において記述した 砲塔ピッチ軸部 の関節に対応するものです。
次に
joint->setActuationMode(Link::JOINT_TORQUE);
によって、この関節の 状態変数シンボル を関節トルクに設定しています。これにより、関節トルクを指令値とした制御が可能となります。
また、
io->enableIO(joint);
によって、この関節に対する入出力を有効にしています。この記述は関節のデフォルトの入出力を有効化するものです。今回悪チュエーションモードが関節トルクとなっているので、関節角度を入力し、関節トルクを出力することになります。これによってこの関節に対してPD制御を行うことが可能となります。
関節に対して上記のアクチュエーションモードの設定や入出力の有効化を行わない場合、その関節の制御を行うことはできませんので、ご注意下さい。入出力を設定する関数としては、他に入力飲みを設定する enableInput という関数と、出力のみを設定する enableOutput という関数も利用可能です。
注釈
同様のことを行う関数として、SimpleControllerIOのsetLinkInput、setJointInput、setLinkOutput、setJointOutputといった関数もありますが、これらは古い仕様の関数を互換性のために残しているものですので、今後はenableXXXの関数を使うようにしてください。
他にPD制御に必要な値として、
q_ref = q_prev = joint->q();
によって初期関節角度を取得し、それを変数q_ref、 q_prevに代入しています。q_refは目標関節角で、q_prevは関節角速度計算用の変数です。また、
dt = io->timeStep();
によって変数dtにタイムステップを代入しています。これはシミュレーションの物理計算1回あたりに進める内部の時間を表していて、この時間間隔で次の control 関数が呼ばれることになります。
最後にinitialize関数の戻り値として true を返して、初期化に成功したことをシステムに伝えています。
control関数の実装¶
control関数は実際の制御コードを記述する部分で、シミュレーション中に繰り返し実行されます。
ここでは砲塔ピッチ軸に関するPD制御のコードが書かれているだけです。
static const double P = 200.0;
static const double D = 50.0;
はPゲイン、Dゲインの値で、
double q = joint->q(); // input
によって現在関節角を入力し、
double dq = (q - q_prev) / dt;
によって現在角速度を算出し、
double dq_ref = 0.0;
で目標角速度は0に設定し、
joint->u() = P * (q_ref - q) + D * (dq_ref - dq); // output
によってPD制御で計算したトルク値を関節に出力し、
q_prev = q;
によって次回計算用にq_prevを更新しています。
このように、入出力はLinkオブジェクトの変数を用いて行うことがポイントです。joint->q()、joint->u() はそれぞれ関節角度、関節トルクの変数に対応しています。
最後に、正常終了を表すtrueを戻り値として返しています。これによって制御ループが継続されます。
ファクトリ関数の定義¶
シンプルコントローラのクラスを定義したら、そのオブジェクトを生成する「ファクトリ関数」も所定の形式で定義しておく必要があります。これは、シンプルコントローラアイテムが実行時にコントローラの共有ライブラリを読み込んで、そこからコントローラのオブジェクトを生成するために必要となります。
これはマクロを使って、
CNOID_IMPLEMENT_SIMPLE_CONTROLLER_FACTORY(TurretController1)
と記述することができます。引数としてはこのようにコントローラのクラス名を与えて下さい。