ホスト PC から Movado を介してモータを動かすには,PC のシリアル通信ポートから Movado へ指令パケットを送信(また必要に応じて Movado から返信パケットを受信)することが必要です.それを実現するには,たとえば以下にあげる2つの方法が考えられます.
ここでは,これら2つの方法について解説します.エンジニア(手に機械油がついている方など)には前者を,アーティスト(手に絵の具がついている方など)には後者をおすすめします.
まず,シリアル通信ポート(/dev/ttyUSB0 や COM3 など)の制御やデータ送受信を行なうための基本インタフェースとして,以下の 5 つの関数を準備します.
void com_init (char *dev); void com_quit (); void com_send (unsigned char *data, int n); void com_send_byte (unsigned char data); int com_rec_byte (unsigned char *data);
com_init は,パス名 dev によって参照されるシリアル通信ポートを開き,Movado の通信プロトコルに合わせて初期設定を行ないます.com_quit はシリアル通信ポートを閉じます.com_send は n バイトのデータを送信し,com_send_byte は 1 バイトのデータを送信します.com_rec_byte は 1 バイトのデータを受信します.受信に成功した場合は 1 を,失敗(500ms で時間切れ)の場合は 0 を返します.いずれの関数も,致命的なエラーの場合は異常終了します.
これら基本インタフェースの実装方法は,Linux,Mac,Windows など,使用するオペレーティングシステムによって異なります.以下に,Linux(Ubuntu 8.10, 9.04)および Mac OS X(Leopard)の場合,Windows XP(Win32 API)の場合に分けて,実装例を紹介します.
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/uio.h>
#include <unistd.h>
#include <termios.h>
#include <stdio.h>
static int ComFD = -1; // file descriptor
void com_error (char *func, char *mess)
{
fprintf(stderr, "*** com_%s: %s\n", func, mess);
fprintf(stderr, "Press ENTER to exit: "); fflush(stderr);
getchar();
exit(1);
}
void com_init (char *dev)
{
static struct termios termios_data;
int ret;
// open serial port
ComFD = open(dev, O_RDWR | O_NOCTTY | O_NONBLOCK);
if (ComFD < 0)
com_error("init", "can't open the port");
// set attribute (settings and baud)
ret = tcgetattr(ComFD, &termios_data);
termios_data.c_iflag = 0;
termios_data.c_oflag = 0;
termios_data.c_lflag = 0;
termios_data.c_cflag = CS8 | CREAD | CLOCAL;
termios_data.c_cc[VMIN] = 0; // "read" byte by byte
termios_data.c_cc[VTIME] = 5; // timeout 500ms
ret = cfsetspeed(&termios_data, B38400);
if (ret < 0)
com_error("init", "cannot set speed");
ret = tcsetattr(ComFD, TCSANOW, &termios_data);
if (ret < 0)
com_error("init", "cannot set attribute");
// turn off "NONBOLCK"
ret = fcntl(ComFD, F_SETFL, 0);
if (ret < 0)
com_error("init", "can't fcntl (to unset NONBLOCK)");
// flush input/output
ret = tcflush(ComFD, TCIOFLUSH);
if (ret < 0)
com_error("init", "cannot flush");
}
void com_quit ()
{
int ret;
// close the port
ret = close(ComFD);
if (ret < 0)
com_error("quit", "cannot close the port");
}
void com_send (unsigned char *data, int n)
{
int ret, ret_drain;
// write data to the port
ret = write(ComFD, data, n);
if (ret < 0)
com_error("send", "cannot write data to the port");
// drain out the write-buffer
ret_drain = tcdrain(ComFD);
if (ret_drain < 0)
com_error("send", "cannot drain the data");
// incomplete?
if (ret < n) {
// still some data to send
com_send(data + ret, n - ret);
}
}
void com_send_byte (unsigned char data)
{
com_send(&data, 1);
}
int com_rec_byte (unsigned char *data)
{
int ret;
// read data from the port
ret = read(ComFD, data, 1);
if (ret < 0)
com_error("rec_byte", "cannot read byte from the port");
// return 1 on success
// return 0 on timeout
return ret;
}
#include <windows.h>
#include <stdio.h>
HANDLE ComFH; // file handle
void com_error (char *func, char *mess)
{
fprintf(stderr, "*** com_%s: %s\n", func, mess);
fprintf(stderr, "Press ENTER to exit: "); fflush(stderr);
getchar();
exit(1);
}
void com_init (char *dev)
{
DCB dcb; // device settings
COMMTIMEOUTS t_out; // timeout settings
int ret;
// open serial port
ComFH = CreateFile(dev, GENERIC_READ | GENERIC_WRITE, 0, 0,
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
if (ComFH == INVALID_HANDLE_VALUE)
com_error("init", "cannot CreateFile");
// set attribute (baud, etc.)
FillMemory(&dcb, sizeof(dcb), 0);
dcb.DCBlength = sizeof(dcb);
ret = BuildCommDCB("38400,n,8,1", &dcb);
if (ret == 0)
com_error("init", "cannot BuildCommDCB");
dcb.fRtsControl = RTS_CONTROL_DISABLE;
ret = SetCommState(ComFH, &dcb);
if (ret == 0)
com_error("init", "cannot SetCommState");
// set timeout (500ms)
t_out.ReadIntervalTimeout = MAXDWORD;
t_out.ReadTotalTimeoutMultiplier = MAXDWORD;
t_out.ReadTotalTimeoutConstant = 500;
t_out.WriteTotalTimeoutMultiplier = MAXDWORD;
t_out.WriteTotalTimeoutConstant = 500;
ret = SetCommTimeouts(ComFH, &t_out);
if (ret == 0)
com_error("init", "cannot SetCommTimeouts");
}
void com_quit ()
{
int ret;
ret = CloseHandle(ComFH);
if (ret == 0)
com_error("quit", "cannot CloseHandle");
}
void com_send (unsigned char *data, int n)
{
int ret;
DWORD n_done;
ret = WriteFile(ComFH, data, n, &n_done, NULL);
if (ret == 0)
com_error("send", "cannot WriteFile");
if ((int) n_done < n) {
// still some data to send
Sleep(100);
com_send(data + n_done, n - n_done);
}
}
void com_send_byte (unsigned char data)
{
com_send(&data, 1);
}
int com_rec_byte (unsigned char *data)
{
int ret;
DWORD n_done;
// read data from the port
ret = ReadFile(ComFH, data, 1, &n_done, NULL);
if (ret == 0)
com_error("rec_byte", "cannot ReadFile");
// return 1 on success
// return 0 on timeout
return n_done;
}
com_init に与えるデバイス名は,Linux の場合は,たとえば "/dev/ttyUSB0" あるいは "/dev/ttyS0",Mac の場合は,たとえば "/dev/tty.KeySerial1",そして Windows の場合は,たとえば "COM3" のように指定します.Windows では,コントロール パネル > システム > ハードウェア > デバイス マネージャ > ポートを参照して,適切なデバイス名を見つけてください.
Movado への指令(および Movado からの返信)は 6 バイトのパケットの形をとります.その先頭バイトの MSB を 1 とし,後続するバイトの MSB を 0 とすることで,パケットのフレーム同期を保証しています.この部分を実装すると,つぎにあげる seq.c のようになります.
#include "com.c"
void seq_send (unsigned char *packet)
{
// set marker for "beginning-of-packet"
packet[0] |= 0x80;
// send a six-byte packet
com_send(packet, 6);
}
void seq_rec (unsigned char *packet)
{
int i;
// find "beginning-of-packet"
do {
com_rec_byte(&(packet[0]));
} while (packet[0] < 0x80);
// read the packet body
i = 1;
while (i < 6) {
com_rec_byte(&(packet[i]));
if (packet[i] >= 0x80) {
// unexpected restart
packet[0] = packet[i];
i = 0;
}
i++;
}
}
Movado のパケットは,アドレス(addr)・モータ番号(n)・命令(inst)・4 バイト引数(quad)からなります.アドレスは 5 ビット,モータ番号は 2 ビット,命令は 3 ビット,4 バイト引数は 32 ビットの整数データです.これら整数データとの変換部分を実装すると,つぎにあげる pac.c のようになります.なお,パケットの詳細構造については「制御ソフト Movado」のページを参照してください.
#include "seq.c"
void pac_send (int addr, int n, int inst, int quad)
{
unsigned char data[6];
data[0] = 0x80 | ((addr & 0x1f) << 2) | (n & 0x03);
data[1] = ((inst & 0x07) << 4)
| ((quad & 0x80000000)? 0x08: 0)
| ((quad & 0x00800000)? 0x04: 0)
| ((quad & 0x00008000)? 0x02: 0)
| ((quad & 0x00000080)? 0x01: 0);
data[2] = (quad >> 24) & 0x7f;
data[3] = (quad >> 16) & 0x7f;
data[4] = (quad >> 8) & 0x7f;
data[5] = quad & 0x7f;
seq_send(data);
}
void pac_rec (int *addr, int *n, int *inst, int *quad)
{
unsigned char data[6];
seq_rec(data);
*addr = (data[0] & 0x7c) >> 2;
*n = (data[0] & 0x03);
*inst = (data[1] >> 4);
*quad = ((data[2] | ((data[1] & 0x08)? 0x80: 0)) << 24)
| ((data[3] | ((data[1] & 0x04)? 0x80: 0)) << 16)
| ((data[4] | ((data[1] & 0x02)? 0x80: 0)) << 8)
| ((data[5] | ((data[1] & 0x01)? 0x80: 0)));
}
void pac_do (int addr, int n, int inst, int *quad)
{
// send command
pac_send(addr, n, inst, *quad);
// receive response (if needed)
if (inst >= 5 || ((inst == 4) && ((*quad) & 0x80000000)))
pac_rec(&addr, &n, &inst, quad);
}
pac_send によって送信した指令パケットのうち,inst が 5(getacc)・6(getvel)・7(getpos)のいずれかであるか,あるいは inst が 4(sub)かつ quad の MSB が 1 のもの(get_upper, get_lower, get_word など)については,Movado から返信パケットがホスト PC に返されますので,pac_rec によって受信する必要があります.それ以外の場合は,返信パケットが返されません.pac_do は,返信パケットの有無を考慮したパケット処理を実装したものです.
上述の com.c, seq.c, pac.c(とくに関数 com_init, com_quit, pac_do)をとおして,ホスト PC から Movado を自由に操作することができます.そこで,これら3つのファイルを連結し,Linux・Mac・Windows で共通して使えるようにしたプログラム c-movado.c を用意しました.その利用方法は,下にあげたテストプログラム c-movado-test.c を参考にしてください.
- c-movado.c : C-Movado 基本インタフェース(Linux・Mac・Windows 共通)
- c-movado-test.c : テストプログラム(Linux・Mac・Windows 共通)
#include "c-movado.c"
#if defined(linux)
// Linux
#define DEF_DEV "/dev/ttyUSB0" // example
//
#elif defined(__APPLE__) && defined(__MACH__)
// Mac OS X
#define DEF_DEV "/dev/cu.I-O DATA USB-RSAQ5" // example
//
#elif defined(_WIN32)
// Windows
#define DEF_DEV "COM3" // example
#define sleep(s) Sleep((s)*1000)
//
#endif
int main (int argc, char **argv)
{
int addr = 1;
int n = 0;
int quad;
// specify your serial port
com_init(DEV_DEV);
// move 0x01:0 to 64.0 rad
quad = 0x00400000;
pac_do(addr, n, 0, &quad);
sleep(2);
// getpos 0x01:0
pac_do(addr, n, 7, &quad);
printf("getpos: %08x\n", quad);
// move 0x01:0 to 0.0 rad
quad = 0x00000000;
pac_do(addr, n, 0, &quad);
sleep(2);
// getpos 0x01:0
pac_do(addr, n, 7, &quad);
printf("getpos: %08x\n", quad);
// close serial port
com_quit();
return 0;
}
以下は,Linux および Mac でのコンパイル・実行例です.quad は 32 ビット固定小数点形式(整数部 16 ビット・小数部 16 ビット)で表現されています.この例では,addr=1, n=0 のモータについて,位置 64.0 [rad] への回転を指令し,2 秒後に現在位置を問い合わせたところ,63.948 [rad] の返答がありました.つづいて位置 0.0 [rad] への回転を指令し,2 秒後に現在位置を問い合わせたところ,0.037 [rad] の返答がありました.
unix> cc c-movado-test.c -o c-movado-test unix> c-movado-test getpos: 003ff2e2 getpos: 0000096c unix>
Windows 上では,Microsoft Visual C++ 2008 Express Edition 上で Win32 コンソールアプリーケションとしてビルドし,同様の動作を確認しました.この開発環境は,Microsoft から無償で提供されています.
たとえば move 指令は,目標位置(単位は rad)を 32 ビット固定小数点データ(整数部 16 ビット・小数部 16 ビット)で与えます.しかしホスト PC 内部では,ふつう浮動小数点形式(たとえば double)で位置・速度などのデータを扱うことが多いので,このままでは不便です.そこで,double 形式で位置・速度などを指定できる機能を考えます.(ここでは単位を rad としています.)
#include "c-movado.c"
void movado_move (int addr, int n, double pos)
{
int quad;
quad = pos * 65536;
pac_do(addr, n, 0, &quad);
}
また,Movado からの返信についても,たとえば getpos 指令によって得られる現在位置は 32 ビット固定小数点形式(整数部 16 ビット・小数部 16 ビット)によって表現されています.これについても,つぎのような関数を用意して,ホスト PC で処理しやすい double 形式で取得できると便利です.
#include "c-movado.c"
double movado_getpos (int addr, int n)
{
int quad;
double pos;
pac_do(addr, n, 7, &quad);
pos = quad / 65536.0;
return pos;
}
このようなラッパーをかけることで,基本インタフェース c-movado.c をより使いやすいインタフェースにしたものが,つぎにあげる c-movado-link.c です.インタフェース関数の一覧もあげておきます.これが CareBots プロジェクトにおける公式の C-Movado インタフェースです.
- c-movado-link.c : C-Movado インタフェース(Linux・Mac・Windows 共通)
- c-movado-link-test.c : テストプログラム(Linux・Mac・Windows 共通)
void movado_move (int addr, int n, double pos); void movado_movac (int addr, int n, double pos, double acc); void movado_movel (int addr, int n, double pos, double vel); void movado_setacc (int addr, int n, double acc); void movado_setvel (int addr, int n, double vel); void movado_setpos (int addr, int n, double pos); double movado_getacc (int addr, int n); double movado_getvel (int addr, int n); double movado_getpos (int addr, int n); int movado_sub (int addr, int n, int jnst, int index, int word);
Max はデータフロー型のプログラミングによるマルチメディア制作環境です.処理単位となる「オブジェクト」を相互にリンクすることで,全体的なデータ処理の流れを「パッチャー」とよばれる図式として視覚的・対話的にプログラミングしていきます.Max は Cycling'74 から Max/MSP/Jitter というパッケージとして販売されています.現在のところ Mac OS X と Windows XP/Vista に対応しています.また,Max の基本機能の多くは,同じ開発者によるオープンソースのソフトウェア Pd (Pure Data) でも利用することができます.Pd は Mac・Windows だけでなく Linux にも対応しています.
max-movado は,Movado への指令パケットの送信(および Movado からの返信パケットの受信)を行なうための Max プログラムです.つぎにあげる Max パッチャー max-movado.maxpat をダウンロードし,Max から参照パスの通ったフォルダに格納することで,max-movado を利用することができます.
Max 上で新しいパッチャーを作成し,そこに新規オブジェクトをつくり,その名前を max-movado としてください.これで max-movado オブジェクトが生成されます.参照パスが正しく設定されていれば,max-movado オブジェクトの上側にはインレット(入力端子),下側にはアウトレット(出力端子)が見えるはずです.
上の例では max-movado に引数 a を与えてありますが,これは max-movado が使用するシリアルポートの識別子です.まず引数なしで max-movado を生成すると,Max Window にシリアルポートのデバイス名と識別子がリストされます.このリストから使用するシリアルポートを見つけ,その識別子を max-movado の引数として加えてください.
この max-movado オブジェクトに,指令パケットに相当するメッセージを送り込むことで,Max から Movado を制御することができます.たとえば,アドレス(addr)が 1,モータ番号(n)が 0 のモータを,位置 100.0 [rad] まで台形速度制御によって回転させることを考えます.まず,(1 0 move 100.) というメッセージを用意し,そのアウトレットから max-movado のインレットへのリンクを張ります.パッチャーの編集モードを終了し,このメッセージをクリックすると,その内容が max-movado のインレットに送り込まれます.その結果,目的とするモータは位置 100.0 [rad] まで回転します.
同じように,いくつかのメッセージを用意し,それを max-movado へ送信することで,モータをさまざまな位置まで回転させることができます.つぎの例では,モータを位置 100.0 [rad],0.0 [rad],−100.0 [rad] に動かします.
max-movado のアウトレットからは,Movado からの返信パケットがメッセージとして出力されます.メッセージが出力されるのは,get 系の指令と一部の副指令の場合で,その他の場合は何も出力されません.これを確かめるために,新しく空のメッセージを生成し,その右インレット(代入用)に max-movado のアウトレットからのリンクを接続してみます.
move 指令によってモータが動きますが,max-movado からは何も出力されません.一方,getpos 指令を送信すると,max-movado から現在位置を格納した返信メッセージが出力されます.上の例では,(1 0 move 100.) の後で (1 0 getpos) を max-movado に送り込んだところ,(1 0 getpos 100.003555) という返信メッセージが得られています.
このように max-movado とメッセージをやりとりすることで,Max から Movado を自由に操ることができます.とにかく max-movado の使い方を身体で覚えたい方は,つぎにあげる max-movado-help.maxpat を参照してください.
max-movado のメッセージについて詳しく知りたい方は,まず下表にあげるメッセージ構造(シンタックス)のリストをご覧ください.それぞれの意味(セマンティクス)については「制御ソフト Movado」を参照してください.
- (addr n move pos)
- (addr n movac acc pos)
- (addr n movel vel pos)
- (addr n setacc acc)
- (addr n setvel vel)
- (addr n setpos pos)
- (addr n sub jnst index word)
- (addr n getacc)
- (addr n getvel)
- (addr n getpos)
さらに max-movado の内部処理を概観したい方は,下に max-movado パッチャーを図示しましたので,参考にしてください.左上端にインレット,左下端にアウトレット,赤色の部分が Movado に送信される 6 バイトのパケット,青色の部分が Movado から受信された 6 バイトのパケットです.黄色の serial というオブジェクトはシリアル通信ポートに相当するオブジェクト(Max 標準)です.
pd-movado は,Movado への指令パケットの送信(および Movado からの返信パケットの受信)を行なうための Pd プログラムで,機能拡張版の Pd-extended での利用を前提としています.つぎにあげる Pd パッチ pd-movado.pd をダウンロードし,Pd から参照パスの通ったフォルダに格納することで,pd-movado を利用することができます.
- ダウンロード: pd-movado.pd(Pd-Movado インタフェース)
pd-movado の使い方(入出力メッセージの形式と意味)は max-movado と全く同じです.前出の「Max によるプログラミング」を参照してください.とにかく pd-movado の使い方を身体で覚えたい方は,つぎにあげる pd-movado-help.pd を参照してください.(英字でなく数字でシリアル通信ポートを指定することに注意してください.)
- ダウンロード: pd-movado-help.pd(Pd-Movado 使用例)
pd-movado の内部処理を概観したい方は,下に pd-movado パッチャーを図示しましたので,参考にしてください.左上端にインレット,左下端にアウトレット,中央付近の comport というオブジェクトがシリアル通信ポートに相当するオブジェクト(Pd-extended 標準)です.