本業プロジェクトのソフトリリースのために前回の投稿から少し時間が空いてしまいました。
前回PDOを使ってモーターのエンコーダー情報の取得を行いました。今回は差動二輪制御AGVの左右2つのモーターの情報を取得します。CANopenNodeには複数のObjectDirectory情報を扱うための定義CO_MULTIPLE_OD がありますが実装例がありません。
CANopenNodeを使ってCAN上に接続された2つのモータードライバ(BLVD-KRD)からエンコーダ情報を取得するため、下記の2つのことに取組みました。
- SYNCイベントに応じてモーターエンコーダ情報を受信する
前回はモータードライバ側のタイマイベントにより定期的にデータを送信するようにしていました。この設定ではモータードライバからデータ送信タイミングの同期が取れていません。CAN OpenではSYNCイベントという複数のデバイスの同期を取るための機能があります。 - 左右2つのモーターのエンコーダ情報を受信する
SYNCイベント送信に応答してBLVD-KRDから送られるエンコーダ情報を受信します。この時左右のドライバをCAN IDにより識別します。
参考資料
本記事ではオリエンタルモーターのホームページからダウンロードできる下記の資料を参照しています。
資料No | ファイル名 | タイトル | 内容 |
---|---|---|---|
1 | HP-5139J.pdf | BLVシリーズ Rタイプ 設置・接続編 | モーターとドライバ、コントローラの接続方法について |
2 | HP-5141J.pdf | BLVシリーズ Rタイプ 機能編 | |
3 | HP-5143E.pdf | BLV Series R Type Driver CANopen Communication Profile | BLVD-KRDに関するObject Dictionary情報 |
4 | https://github.com/CANopenNode | CANopenNode is free and open source CANopen protocol stack. | Apache-2.0 license |
以下、説明内の引用資料は特に断りがない場合上記資料からの抜粋です。
サンプルコード
作成したコードは下記githubにて公開しています。
SYNCイベントの設定
前回はユーティリティソフトMEXE02を使って、BLVD-KRDが自発的にエンコーダ情報を送信するように設定しました。CAN OpenではSYNCプロデューサーが送信したSYNCオブジェクトに応答する形でPDOオブジェクトを送信することで複数のデバイスの同期を取ることができます。
モータードライバの設定
MEX02を使ってBLVD-KRDの1803hのパラメータをSYNCに同期するように書き換えます。
Sub-index 02h(Transmission type)を0x01(synchronous (cyclic every SYNC))に設定します。この設定により、モータードライバはSYNCオブジェクトを受信する毎にTPDOを送信します。TPDOのマッピング設定は前回と同様です。
CANopenNodeの設定(cocommコマンドによる設定)
SYNCの設定
SYNCオブジェクトのプロデューサー(送信)側は、CANopneNodeです。CANopenNode起動後にcocommコマンドを使って1006h(Communication cycle period)にイベントサイクルを設定した後、1005h(COB-ID SYNC message)をenabledにします。CANopenNodeのnode idは1。
$ ./canopend can0 -i 1 -c "local-/tmp/CO_command_socket"
$ ./cocomm "1 w 0x1006 0 u32 100000"
$ ./cocomm "1 w 0x1005 0 x32 0x40000080"
同期開始
ネットワークステータスをOperationalにします。
$ ./cocomm "0 start"
candumpを確認すると下記のようにモータードライバからエンコーダ情報がバスに流れます。
$ candump -x -ta can0
(1692077030.539135) can0 TX - - 080 [0] <- SYNC
(1692077030.540995) can0 RX - - 38A [6] 60 12 31 FA FF FF -> node id 10からのstatus&encoder情報
(1692077030.540997) can0 RX - - 38B [6] 60 12 7D 1B 02 00 -> node id 11からのstatus&encoder情報
(1692077030.639387) can0 TX - - 080 [0]
(1692077030.641247) can0 RX - - 38A [6] 60 12 30 FA FF FF
(1692077030.641249) can0 RX - - 38B [6] 60 12 7C 1B 02 00
(1692077030.739667) can0 TX - - 080 [0]
(1692077030.741494) can0 RX - - 38A [6] 60 12 2C FA FF FF
(1692077030.741495) can0 RX - - 38B [6] 60 12 7B 1B 02 00
順番があべこべですが、既にバス上にモータードライバを2つ接続しており、それぞれのnode idを10と11に設定します(COB-IDは38Ahと38Bh)。なので、プロデューサー(CANopenNode)からのSYNCに応答して、それぞれのドライバから応答が帰ってきています。
SYNCの送出間隔も設定の100msec周期になっています。
(1692077030.539135) can0 TX - - 080 [0]
(1692077030.639387) can0 TX - - 080 [0]
(1692077030.739667) can0 TX - - 080 [0]
CANopenNodeの設定(プログラムからの設定)
次に、CANopenNodeのAPIを使って同期設定を行います。前回の投稿にて組み込んだサンプルプログラムの初期化関数app_prgramStart()に設定を組み込みます。
CO_ReturnError_t app_programStart(uint16_t *bitRate,
uint8_t *nodeId,
uint32_t *errInfo)
{
・・・
//*************************************
// set sync
//*************************************
ODR_t odRet;
// Set SYNC Period
odRet = OD_set_u32(OD_ENTRY_H1006_communicationCyclePeriod, 0, 100000, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1006_communicationCyclePeriod);
return CO_ERROR_OD_PARAMETERS;
}
// Enable SYNC
odRet = OD_set_u32(OD_ENTRY_H1005_COB_ID_SYNCMessage, 0, 0x40000080, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1005_COB_ID_SYNCMessage);
return CO_ERROR_OD_PARAMETERS;
}
・・・
}
canopendを起動後、ネットワークステータスをOperationalにすると、エンコーダ情報が流れてきます。
左右のモータードライバからエンコーダ情報情報をRPDOで受信する
前回RPDOを使ってモータードライバからエンコーダ情報を受信しました。CANopenNodeを使ってRPDOを受信したとき、呼び出されるコールバック関数ではデータの送信元デバイスのnode idなどデバイスを区別する方法がありません。
CANopenNodeのissuesで問い合わせたところ「それがCANopenの仕様なので、送信元デバイスを区別する方法はない。それぞれ別のオブジェクトディクショナリ(OD)を定義して受信してね」という事らしい。そしてCANopenNodeのサンプルプログラムには、ODが1つのサンプルしかなくQAには「複数のODを扱うのはケースによって色々あるからサンプル提供は難しい」という過去のやり取りが_| ̄|○。というところまでが前回。
CANopenNodeのソースを読んで複数のデバイスから同じオブジェクト情報(今回はエンコーダ情報 6064h: Position actual value)を受信するために下記のことを行いました。
- CANopenEditorを使って2つ目のデバイスのODファイルOD_2nd.cを生成
- CANopenNodeのコンパイルオプションにCO_MULTIPLE_ODのdefineを追加
- canopendの初期化時に、新たに定義したODの初期化処理を追加
- サンプルプログラムの初期化app_programStart()でそれぞれのODの設定を行う。この時コールバックハンドラにデバイスを区別する情報を渡す
今回右のモータドライバのnode idを0xA、左のモータドライバのnode idを0xBとしています。
2つ目のオブジェクトディクショナリを定義
CANopenEditorを起動し、2つ目のODに必要なエントリを追加します。2つ目の方は必要なエントリだけ追加しました。本来は、デバイスを管理するノード(PC側)、左右のドライバを分離して3つのODを定義するのが正しい気がしますが、今回はとりあえず1つめのODファイルに通信パラメータを含めて、2つ目のODファイルにはPDO関連のエントリだけ追加しました。
- 「File → Open」 BLVD-KRD_CANopen_V200.edsを開く
- 「Object Directory」タグから、Communication Specific Parametersのすべてのオブジェクトを選択して右クリックっで削除
- 「Insert Profile → DS301_profile.xpd」を選択しそのまま「Insert」
- 「File → Export CanOpenNode...」を選択してOD.hと名前を付けて保存する(右OD)
- 「Object Directory」タグCommunication Specific Parametersの0x1000~0x1400のオブジェクトを選択して右クリックで削除
- 「File → Export CanOpenNode...」を選択してOD_2nd.hと名前を付けて保存する(左OD)
CANopenNodeのコンパイラオプションの変更
canopendのMakefileを修正し、OD_2nd.cの追加とコンパイルオプションを変更します。
SOURCES = \
$(DRV_SRC)/CO_driver.c \
$(DRV_SRC)/CO_error.c \
$(DRV_SRC)/CO_epoll_interface.c \
$(CANOPEN_SRC)/301/CO_ODinterface.c \
$(CANOPEN_SRC)/301/CO_NMT_Heartbeat.c \
$(CANOPEN_SRC)/301/CO_Emergency.c \
$(CANOPEN_SRC)/301/CO_SDOserver.c \
$(CANOPEN_SRC)/301/CO_SDOclient.c \
$(CANOPEN_SRC)/301/CO_TIME.c \
$(CANOPEN_SRC)/301/CO_SYNC.c \
$(CANOPEN_SRC)/301/CO_PDO.c \
$(CANOPEN_SRC)/301/crc16-ccitt.c \
$(CANOPEN_SRC)/301/CO_fifo.c \
$(CANOPEN_SRC)/303/CO_LEDs.c \
$(CANOPEN_SRC)/304/CO_GFC.c \
$(CANOPEN_SRC)/304/CO_SRDO.c \
$(CANOPEN_SRC)/305/CO_LSSslave.c \
$(CANOPEN_SRC)/305/CO_LSSmaster.c \
$(CANOPEN_SRC)/309/CO_gateway_ascii.c \
$(CANOPEN_SRC)/storage/CO_storage.c \
$(CANOPEN_SRC)/extra/CO_trace.c \
$(CANOPEN_SRC)/CANopen.c \
$(APPL_SRC)/OD.c \
$(APPL_SRC)/OD_2nd.c \ /* 追加 */
$(DRV_SRC)/CO_main_basic.c \
$(DRV_SRC)/CO_application.c
OPT += -DCO_MULTIPLE_OD /* 追加 */
2つ目のオブジェクトディクショナリ初期化処理の追加
canopendのmain()にOD_2ndの初期化処理を追加します。修正はgithubのsample1ブランチを参照してください。修正は次の通り。
- OD_2nd.hのinclude
- global変数CO_2ndの追加とオブジェクト生成
- CO_2ndをdisableにしてからCAN初期化。node idによるパケットのフィルタリング設定等
- CO_2ndのCANOpen初期化。PDOのコールバックハンドラなどの初期化
- イベントループでCO_2ndの処理追加
コールバックハンドラの設定
CO_application.cのapp_programStart()を修正し、PDOのコールバックハンドラを登録します。
/* 6064h エンコーダ情報受信コールバックハンドラ */
ODR_t H6064_write(OD_stream_t *stream, const void *buf,
OD_size_t count, OD_size_t *countWritten)
{
printf("H6064_write");
if (stream == NULL || stream->object == NULL || buf == NULL || countWritten == NULL) {
printf("\n");
return ODR_DEV_INCOMPAT;
}
// stream->objectには、app_programStart()でハンドラに登録した
// オブジェクトが設定されている
uint16_t *can_id = (uint16_t*)stream->object;
int32_t pos = *(int32_t*)buf;
if (can_id && *can_id == 0x038A)
printf(" posR = %d\n", pos);
else if (can_id && *can_id == 0x038B)
printf(" posL = %d\n", pos);
else
printf("bad parameter\n");
return ODR_OK;
}
// 右のハンドラ情報
OD_extension_t H6041_extentionR = {
.object = NULL,
.read = NULL,
.write = H6041_write,
};
// 左のハンドラ情報
OD_extension_t H6041_extentionL = {
.object = NULL,
.read = NULL,
.write = H6041_write,
};
/** アプリケーション初期化関数 **/
CO_ReturnError_t app_programStart(uint16_t *bitRate,
uint8_t *nodeId,
uint32_t *errInfo)
{
CO_ReturnError_t err = CO_ERROR_NO;
//*************************************
// set sync
//*************************************
ODR_t odRet;
// Set SYNC Period 100000μsec
odRet = OD_set_u32(OD_ENTRY_H1006_communicationCyclePeriod, 0, 100000, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1006_communicationCyclePeriod);
return CO_ERROR_OD_PARAMETERS;
}
// Enable SYNC
odRet = OD_set_u32(OD_ENTRY_H1005_COB_ID_SYNCMessage, 0, 0x40000080, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1005_COB_ID_SYNCMessage);
return CO_ERROR_OD_PARAMETERS;
}
//*************************************
// Set RPDO for Right Motor
// 初期化にて動的にパラメータを設定していますが、CANopenEditorで予め設定すること
// 生成されたod.cを修正することもOKです。
//*************************************
odRet = OD_set_u32(OD_ENTRY_H1400_RPDOCommunicationParameter, 1, 0x0000038A, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1400_RPDOCommunicationParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u8(OD_ENTRY_H1600_RPDOMappingParameter, 0, 2, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1600_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u32(OD_ENTRY_H1600_RPDOMappingParameter, 1, 0x60410010, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1600_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u32(OD_ENTRY_H1600_RPDOMappingParameter, 2, 0x60640020, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_ENTRY_H1600_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
/* コールバックハンドラの登録 */
/* 右のハンドラに渡すオブジェクトを生成してnode id(0x038A)を設定する */
uint16_t *objR = (uint16_t*)malloc(sizeof(uint16_t));
*objR = 0x038A;
H6064_extentionR.object = objR;
OD_extension_init(OD_ENTRY_H6064_positionActualValue, &H6064_extentionR);
//*************************************
// Set RPDO for Left Motor
// 初期化にて動的にパラメータを設定していますが、CANopenEditorで予め設定すること
// 生成されたod_2nd.cを修正することもOKです。
//*************************************
odRet = OD_set_u32(OD_2nd_ENTRY_H1401_RPDOCommunicationParameter, 1, 0x0000038B, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_2nd_ENTRY_H1401_RPDOCommunicationParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u8(OD_2nd_ENTRY_H1601_RPDOMappingParameter, 0, 2, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_2nd_ENTRY_H1601_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u32(OD_2nd_ENTRY_H1601_RPDOMappingParameter, 1, 0x60410010, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_2nd_ENTRY_H1601_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
odRet = OD_set_u32(OD_2nd_ENTRY_H1601_RPDOMappingParameter, 2, 0x60640020, true);
if (odRet != ODR_OK) {
if (errInfo != NULL) *errInfo = OD_getIndex(OD_2nd_ENTRY_H1601_RPDOMappingParameter);
return CO_ERROR_OD_PARAMETERS;
}
/* コールバックハンドラの登録 */
/* 左のハンドラに渡すオブジェクトを生成してnode id(0x038B)を設定する */
uint16_t *objL = (uint16_t*)malloc(sizeof(uint16_t));
*objL = 0x038B;
H6064_extentionL.object = objL;
OD_extension_init(OD_2nd_ENTRY_H6064_positionActualValue, &H6064_extentionL);
return CO_ERROR_NO;
}
canopendプログラムを起動し、cocommを使ってネットワークステータスをOperationalにすると次のようにコンソールにログが出力され、エンコーダ情報が左右区別されて取得できていることが確認できます。
$ ./canopend can0 -i 1 -c "local-/tmp/CO_command_socket" | grep H6064
H6064_write posL = 449249
H6064_write posR = 874
H6064_write posL = 449252
H6064_write posR = 874
H6064_write posL = 449251
H6064_write posR = 876
このときのcandumpは下記の通り。
$ candump -x -ta can0
(1692233198.895197) can0 TX - - 080 [0]
(1692233198.897146) can0 RX - - 38B [6] 60 12 E1 DA 06 00
(1692233198.897148) can0 RX - - 38A [6] 60 12 6A 03 00 00
(1692233198.995500) can0 TX - - 080 [0]
(1692233198.997397) can0 RX - - 38B [6] 60 12 E4 DA 06 00
(1692233198.997400) can0 RX - - 38A [6] 60 12 6A 03 00 00
(1692233199.095734) can0 TX - - 080 [0]
(1692233199.097644) can0 RX - - 38B [6] 60 12 E3 DA 06 00
(1692233199.097646) can0 RX - - 38A [6] 60 12 6C 03 00 00
次回予定
車輪の回転情報からAGVのオドメトリを生成するために必要な左右のモーターの角度情報を取得することができるようになりました。
次回はPDOを使ってモーターを駆動します。