version 1.28, 1999/12/20 18:00:55 |
version 1.80, 1999/12/25 04:08:50 |
|
|
\documentclass{jarticle} |
\documentclass{jarticle} |
|
|
\title{\bf Open XM($B%?%$%H%kL$Dj(B)} |
%% $OpenXM: OpenXM/doc/genkou19991125.tex,v 1.79 1999/12/24 21:56:37 tam Exp $ |
\author{ |
|
%Maekawa, Masahide (Oct., 1999 -- : CVS server) \\ |
\usepackage{jssac} |
$BA0@n(B $B$^$5$R$G(B, |
\title{ |
%$B",4A;zD4$YCf!#(B commit $B8+$?$i65$($F$M!#$I$&$;L@F|J9$1$k$1$I!#(B |
1. 意味もない修飾過剰な語句は排除しましょう。\\ |
%Noro, Masayuki (Jan., 1996 -- : OpenXM Protocol, asir99) \\ |
2. せっかく fill しているのをいじらないでくれ。\\ |
$BLnO$(B $B@59T(B, |
3. 田村が遊んでばかりでおればかり仕事をしているのはどう考えても不公平だ。 |
%Ohara, Katsuyoshi (Jan., 1998 -- : ox\_math) \\ |
なんで仕事をしないのか、いい加減仕事をしろ、田村。 |
$B>.86(B $B8yG$(B, \\ |
%↑すみません、家で御飯食べてました。 |
%Okutani, Yukio (Oct., 1999 -- : asir contrib) \\ |
|
$B1|C+(B $B9,IW(B, |
|
%Takayama, Nobuki (Jan., 1996 -- : OpenXM Protocol, kan) \\ |
|
$B9b;3(B $B?.5#(B, |
|
%Tamura, Yasushi (Nov., 1998 -- : OpenMath proxy) \\ |
|
$BEDB<(B $B63;N(B |
|
} |
} |
\date{1999$BG/(B11$B7n(B25$BF|(B} |
|
|
|
%\pagestyle{empty} |
\author{奥 谷 行 央\affil{神戸大学大学院自然科学研究科} |
|
\mail{okutani@math.sci.kobe-u.ac.jp} |
|
\and 小 原 功 任\affil{金沢大学理学部} |
|
\mail{ohara@kappa.s.kanazawa-u.ac.jp} |
|
\and 高 山 信 毅\affil{神戸大学理学部} |
|
\mail{takayama@math.sci.kobe-u.ac.jp} |
|
\and 田 村 恭 士\affil{神戸大学大学院自然科学研究科} |
|
\mail{tamura@math.sci.kobe-u.ac.jp} |
|
\and 野 呂 正 行\affil{富士通研究所} |
|
\mail{noro@para.flab.fujitsu.co.jp} |
|
\and 前 川 将 秀\affil{神戸大学理学部} |
|
\mail{maekawa@math.sci.kobe-u.ac.jp} |
|
} |
|
\art{} |
|
|
\begin{document} |
\begin{document} |
\maketitle |
\maketitle |
|
|
\section{OpenXM $B$N7W;;%b%G%k(B} |
\section{OpenXMとは} |
|
|
OpenXM $B$O?t3X%=%U%H4V$G%a%C%;!<%8$r8r49$9$k$?$a$N5,Ls$G$"$k!#(B |
OpenXM は数学プロセス間でメッセージを交換するための規約である。 |
$B?t3XE*$J%G!<%?$r4^$s$@%a%C%;!<%8$J$I$rMQ$$$F(B |
数学プロセス間でメッセージをやりとりすることにより、 |
$B?t3X%=%U%H4V$G%a%C%;!<%8$r$d$j$H$j$5$;$k$3$H$K$h$j!"(B |
ある数学プロセスから他の数学プロセスを呼び出して計算を行なったり、 |
$B$"$k?t3X%=%U%H$+$iB>$N?t3X%=%U%H$r8F$S=P$7$F7W;;$r9T$J$C$?$j!"(B |
他のマシンで計算を行なわせたりすることが目的である。 |
$BB>$N%^%7%s$G7W;;$r9T$J$o$;$?$j$9$k$3$H$,L\E*$G$"$k!#(B |
なお、 OpenXM とは Open message eXchange protocol for Mathematics の略である。 |
$BH/C<$OLnO$@59T$H9b;3?.5#$K$h$j!"(B asir $B$H(B kan/sm1 $B$r(B |
OpenXM の開発の発端は野呂と高山により、 |
$BAj8_$K8F$S=P$95!G=$r<BAu$7$?$3$H$G$"$k!#(B |
asir と kan/sm1 を相互に呼び出す機能を実装したことである。 |
$B8=:_$NL\I8$O!"%U%j!<$N?t3X%=%U%H$rAj8_$K@\B3$7$F(B |
|
$B9%$-$J8@8l$+$i4JC1$KB>$N?t3X%=%U%H$r;H$($k$h$&$K$9$k$3$H$G$"$k!#(B |
|
$B$J$*!"(B OpenXM $B$H$O(B Open message eXchange protocol for Mathematics $B$N(B |
|
$BN,$G$"$k!#(B |
|
|
|
$BH/C<$H$J$C$?(B asir $B$H(B kan/sm1 $B$G$N<BAu;~$K$O!"(B |
初期の実装では、相手側のローカル言語の文法に従った文字列を送っていた。 |
$B$*8_$$$KAj<jB&$N%3%^%s%IJ8;zNs$rAw$C$F$$$?!#(B |
この方法では相手側のソフトが asir なのか kan/sm1 なのかを判別するなどして、 |
$B$3$NJ}K!$O8=:_$N(B OpenXM $B5,Ls$G$b7A$rJQ$($F2DG=$G$O$"$k$,!"(B |
相手側のローカル言語の文法に合わせた文字列を作成しなければならない。 |
$B;H$$$d$9$$H?LL!"8zN(E*$G$"$k$H$O$$$$Fq$$!#(B |
このローカル言語の文法に従った文字列を送る方法は、 |
$B$5$i$K!"$3$NJ}K!$G$OAj<jB&$N%=%U%H$,(B asir $B$J$N$+(B kan/sm1 $B$J$N$+$r(B |
効率的であるとはいい難いが、使いやすいとも言える。 |
$BH=JL$7$F!"Aj<jB&$K9g$o$;$F%3%^%s%IJ8;zNs$r:n@.$9$kI,MW$,$"$k!#(B |
|
|
|
$B$3$l0J30$NJ}K!$H$7$F!"(B |
現在の OpenXM 規約では共通表現形式によるメッセージを用いている。 |
OpenXM $B5,Ls$G$O6&DLI=8=7A<0$K$h$k%a%C%;!<%8$bMQ0U$7$F$$$k!#(B |
上記の文字列を送る方法の利点を生かすため、 |
OpenXM $B5,LsFH<+$N%G!<%?7A<0$G$"$k(B CMO $B7A<0(B(Common Mathematical Object format) |
OpenXM 規約では共通表現形式の中の文字列として、 |
$B0J30$K$b!"(B MP $B$d(B OpenMath $B$N(B XML, binary $BI=8=7A<0$H$$$C$?B>$N7A<0$r$b(B |
ローカル言語の文法に従った文字列を用いたメッセージの交換も可能となっている。 |
$B07$($k$h$&$K$7$F$"$k!#(B |
|
$B$J$*!"8=:_$N(B OpenXM $B5,Ls$G$O!"(B |
|
$BA0=R$N%3%^%s%IJ8;zNs$b(B CMO $B7A<0$J$I$N2?$i$+$N%G!<%?7A<0$NCf$N(B |
|
$BJ8;zNs$H$7$FI=8=$7$FAw$kI,MW$,$"$k!#(B |
|
|
|
OpenXM $B5,Ls$G$N%a%C%;!<%8$N8r49$O%5!<%P$H%/%i%$%"%s%H$N4V$G9T$J$o$l$k!#(B |
OpenXM 規約では通信の方法に幾らかの自由度があるが、 |
$B%5!<%P$O%9%?%C%/%^%7%s$G$"$k$H2>Dj$5$l$F$*$j!"(B |
現在のところは TCP/IP を用いた通信しか実装されていない。 |
$B%5!<%P$,%/%i%$%"%s%H$+$i<u$1<h$C$?%a%C%;!<%8$O$9$Y$F%9%?%C%/$K@Q$^$l$k!#(B |
そこで、この論文では具体的な実装は TCP/IP を用いていると仮定する。 |
OpenXM $B$N%a%C%;!<%8$NCf$K$O%5!<%P$K9T$J$o$;$?$$F0:n$KBP1~$9$k%G!<%?$,$"$j!"(B |
|
$B$3$N%a%C%;!<%8$r<u$1<h$C$?%5!<%P$O$=$l$KBP1~$9$kF0:n$r(B |
|
$B9T$J$&$3$H$,4|BT$5$l$F$$$k!#(B |
|
$B$?$@$7!"%5!<%P$OL?Na$5$l$J$$8B$j2?$bF0:n$r9T$J$*$&$H$O$7$J$$!#(B |
|
|
|
|
\section{OpenXM のメッセージの構造} |
|
|
\section{OpenXM $B$N%a%C%;!<%8$N9=B$(B} |
通信の方法によってメッセージの構造は変わる。 |
|
前節で仮定したとおり、この論文では TCP/IP の場合についてのみ説明を行なう。 |
|
|
OpenXM $B$N%a%C%;!<%8$OJ#?t$N<oN`$N%G!<%?$rI=8=$9$k$3$H$,2DG=$G$"$k!#(B |
OpenXM 規約で規定されているメッセージはバイトストリームとなっており、 |
$B%a%C%;!<%8$,$I$N$h$&$J%G!<%?$J$N$+$O!"(B |
次のような構造になっている。 |
$B@hF,$K$"$k(B tag $B$r$_$l$PJ,$+$k$h$&$K$J$C$F$$$k!#(B |
|
$B$3$N(B tag $B$r$_$k$3$H$K$h$C$F!"(B |
|
CMO $B7A<0$d!"(B OpenMath $B$N(B XML $BI=8=$J$I$N!"(B |
|
$B%a%C%;!<%8$K4^$^$l$F$$$k%G!<%?$N<oN`$,J,$+$k$h$&$K$J$C$F$$$k!#(B |
|
|
|
$B%a%C%;!<%8$NCf$N<B:]$K%G!<%?$NF~$C$F$$$kItJ,$O(B |
\begin{tabular}{|c|c|} |
tag $B0J9_$K$"$k(B body $B$NItJ,$G$"$k!#(B |
\hline |
body $B$NCf$N%G!<%?$,$I$N$h$&$K3JG<$5$l$F$$$k$+$O(B |
ヘッダ & \hspace{10mm} ボディ \hspace{10mm} \\ |
$B3F%G!<%?7A<0$,$=$l$>$lFHN)$K7h$a$i$l$k$h$&$K$J$C$F$$$k!#(B |
\hline |
$B$b$7!"(B OpenXM $B5,Ls$G%a%C%;!<%8$N$d$j$H$j$r9T$J$$$?$$$,!"(B |
\end{tabular} |
$B$^$@5,Ls$GDj5A$5$l$F$$$J$$%G!<%?7A<0$r;H$$$?$$>l9g$O!"(B |
|
tag $B$r$^$@;H$o$l$F$J$5$=$&$JCM(B |
|
($B%7%9%F%`8GM-$NI=8=$N$?$a$K?d>)$5$l$F$$$kCM$,$"$k(B) |
|
$B$K@_Dj$7!"(B body $B$NItJ,$K%G!<%?$rKd$a9~$a$P$h$$!#(B |
|
$B$J$*!"$9$Y$F$N%a%C%;!<%8$K(B body $B$,I,MW$H$$$&$o$1$G$O$J$/!"(B |
|
body $B$N$J$$%a%C%;!<%8$b(B OpenXM $B5,Ls$K$OB8:_$9$k$3$H$K(B |
|
$BCm0U$7$J$1$l$P$J$i$J$$!#(B |
|
|
|
$B%5!<%P$KBP$9$kF0:n$KBP1~$7$?%G!<%?$O(B SM $B7A<0$H$7$FDj5A$5$l$F$$$k!#(B |
ヘッダの長さは 8 バイトであると定められている。 |
SM $B7A<00J30$N%G!<%?$G$O!"%5!<%P$O<u$1<h$C$?%G!<%?$r%9%?%C%/$K@Q$`(B |
ボディの長さはメッセージごとに異なっているが、 |
$B0J30$NF0:n$r$7$J$$$3$H$K$J$C$F$$$k!#(B |
長さは $0$ でもよい。 |
$B$D$^$j!"(B SM $B7A<0$N%G!<%?$,%G!<%?$r<u$1<h$k0J30$NF0:n$r(B |
|
$B%5!<%P$K9T$J$o$;$kM#0l$N%G!<%?7A<0$G$"$k!#(B |
|
$B$3$N%G!<%?$r<u$1<h$k0J30$NF0:n$NCf$K$O!"(B |
|
$B%G!<%?$K$J$s$i$+$N2C9)$r;\$9F0:n$bF~$C$F$$$k!#(B |
|
$B$3$N%G!<%?$K$J$s$i$+$N2C9)$r;\$9F0:n$NCf$K$O(B |
|
$B?t3XE*$J1i;;$r9T$J$&F0:n$b4^$^$l$F$$$k!#(B |
|
$B0J8e!"%G!<%?$K$J$s$i$+$N2C9)$r;\$9F0:n$N$3$H$r7W;;$H8F$V$3$H$K$9$k!#(B |
|
|
|
|
ヘッダは次の二つの情報を持っている。 |
|
\begin{enumerate} |
|
\item 前半の 4 バイト。メッセージの種類を表わす識別子であり、 |
|
タグと呼ばれる。 |
|
\item 後半の 4 バイト。メッセージにつけられた通し番号である。 |
|
\end{enumerate} |
|
それぞれの 4 バイトは 32 ビット整数とみなされて扱われる。 |
|
この場合に用いられる整数の表現方法については後述するが、 |
|
基本的に表現方法はいくつかの選択肢から選ぶことが可能となっており、 |
|
またその選択は通信路の確立時に一度だけなされることに注意しなければならない。 |
|
現在のOpenXM 規約では、タグ(整数値)として |
|
以下のものが定義されている。 |
|
|
\section{OpenXM $B$N7W;;$N?J9TJ}K!(B} |
\begin{verbatim} |
|
#define OX_COMMAND 513 |
|
#define OX_DATA 514 |
|
#define OX_SYNC_BALL 515 |
|
#define OX_DATA_WITH_LENGTH 521 |
|
#define OX_DATA_OPENMATH_XML 523 |
|
#define OX_DATA_OPENMATH_BINARY 524 |
|
#define OX_DATA_MP 525 |
|
\end{verbatim} |
|
|
$B%/%i%$%"%s%H$,%5!<%P$X$J$s$i$+$N7W;;$r9T$J$o$;$k>l9g!"(B |
ボディの構造はメッセージの種類によって異なる。 |
$B%/%i%$%"%s%H$+$i%5!<%P$X7W;;$5$;$?$$%G!<%?$r%a%C%;!<%8$H$7$FAw$j!"(B |
タグが OX\_COMMAND となっているメッセージはスタックマシンへの命令であり、 |
$B$=$7$F$=$N7k2L$r%5!<%P$+$i%a%C%;!<%8$G<u$1<h$k$3$H$K$h$C$F7W;;$O9T$J$o$l$k!#(B |
それ以外のメッセージは何らかのオブジェクトを表している。 |
$B$?$@$7!"%5!<%P$O7k2L$NAw?.$9$i$bL?Na$5$l$J$1$l$P9T$J$&$3$H$O$J$/!"(B |
この論文では OX\_DATA と OX\_COMMAND で識別される |
$B%/%i%$%"%s%H$O7k2L$r<u$1<h$i$:$K%5!<%P$K<!!9$H(B |
メッセージについてのみ、説明する。 |
$B7W;;$r9T$J$o$;$k$3$H$b2DG=$G$"$k!#(B |
|
|
|
$B%5!<%P$,%/%i%$%"%s%H$+$i<u$1<h$C$?%a%C%;!<%8$O$9$Y$F%9%?%C%/$K@Q$^$l$k!#(B |
既存のメッセージでは対応できない場合は、新しい識別子を定義することで新し |
$B$?$@$7!"$3$N$^$^$G$O<u$1<h$C$?%a%C%;!<%8$K4^$^$l$k%G!<%?$r(B |
い種類のメッセージを作成することができる。この方法は各数学ソフトウェアの |
$B%9%?%C%/$K@Q$_>e$2$F$$$/$@$1$G!"%5!<%P$O7W;;$r9T$J$*$&$H$O$7$J$$!#(B |
固有の表現を含むメッセージを作成したい場合などに有効である。新しい識別子 |
$B<!$$$G%5!<%P$K9T$J$o$;$?$$F0:n$KBP1~$7$?%G!<%?$rAw$k$H!"(B |
の定義方法については、\cite{OpenXM-1999} を参照すること。 |
$B=i$a$F%5!<%P$O7W;;$J$I$N!"$J$s$i$+$NF0:n$r9T$J$&!#(B |
|
$B$3$N$H$-!"I,MW$,$"$l$P%5!<%P$O%9%?%C%/$+$iI,MW$J$@$1%G!<%?$r<h$j=P$9!#(B |
|
$B$3$3$G!"%/%i%$%"%s%H$+$i$NL?Na$K$h$kF0:nCf$K$?$H$(%(%i!<$,H/@8$7$?$H$7$F$b(B |
|
$B%5!<%P$O%(%i!<%*%V%8%'%/%H$r%9%?%C%/$K@Q$`$@$1$G!"(B |
|
$BL@<($5$l$J$$8B$j%(%i!<$rJV$5$J$$$3$H$KCm0U$7$J$1$l$P$J$i$J$$!#(B |
|
|
|
$B7k2L$,@8$8$kF0:n$r%5!<%P$,9T$J$C$?>l9g!"(B |
\section{OpenXM の計算モデル} |
$B%5!<%P$OF0:n$N7k2L$r%9%?%C%/$K@Q$s$G$$$k!#(B |
|
$B%5!<%P$K9T$J$o$;$?F0:n$N7k2L$r%/%i%$%"%s%H$,CN$j$?$$>l9g!"(B |
|
$B%9%?%C%/$+$i%G!<%?$r<h$j=P$7Aw?.$r9T$J$&L?Na$KBP1~$7$?(B SM $B7A<0$N%G!<%?$r(B |
|
$B%5!<%PB&$XAw$l$P$h$$!#(B |
|
|
|
$B%/%i%$%"%s%H$,%5!<%P$X7W;;$r9T$J$o$;!"7k2L$rF@$k$H$$$&<j=g$rDI$C$F$$$/$H!"(B |
OpenXM 規約での計算とはメッセージを交換することである。また、 OpenXM 規 |
$B<!$N$h$&$K$J$k!#(B |
約ではクライアント・サーバモデルを採用しているので、メッセージの交換はサー |
|
バとクライアントの間で行なわれる。クライアントからサーバへメッセージを送 |
|
り、クライアントがサーバからメッセージを受け取ることによって計算の結果が |
|
得られる。このメッセージのやりとりはクライアントの主導で行われる。つまり、 |
|
クライアントは自由にメッセージをサーバに送付してもよいが、サーバからは自 |
|
発的にメッセージが送付されることはない。この原理はサーバはスタックマシン |
|
であることで実現される。スタックマシンの構造については \ref{sec:oxsm} 節 |
|
で述べる。 |
|
|
|
サーバがクライアントから受け取ったオブジェクト(つまり OX\_COMMAND でない |
|
メッセージのボディ)はすべてスタックに積まれる。スタックマシンへの命令 |
|
(OX\_COMMAND で識別されるメッセージのボディ)を受け取ったサーバは命令に対 |
|
応する動作を行なう。このとき、命令によってはスタックからオブジェクトを取 |
|
り出すことがあり、また(各数学システムでの)計算結果をスタックに積むことが |
|
ある。もし、与えられたデータが正しくないなどの理由でエラーが生じた場合に |
|
はサーバはエラーオブジェクトをスタックに積む。計算結果をクライアントが得 |
|
る場合にはスタックマシンの命令 SM\_popCMO または SM\_popString をサーバ |
|
に送らなければならない。これらの命令を受け取ってはじめて、サーバからクラ |
|
イアントへメッセージが送られる。 |
|
|
|
%{\Huge 以下、書き直し} |
|
|
|
まとめると、クライアントがサーバへメッセージを送り、 |
|
計算の結果を得るという手順は以下のようになる。 |
|
|
\begin{enumerate} |
\begin{enumerate} |
\item $B$^$:!"%/%i%$%"%s%H$,%5!<%P$X7W;;$5$;$?$$%G!<%?$rAw$k!#(B |
\item |
$B%5!<%P$OAw$i$l$F$-$?%G!<%?$r%9%?%C%/$K@Q$`!#(B |
まず、クライアントがサーバへオブジェクトを送る。サーバは送られてきたオブ |
\item $B%/%i%$%"%s%H$,%5!<%P$K!V7W;;$r9T$J$&F0:n$KBP1~$7$?%G!<%?!W$r(B |
ジェクトをスタックに積む。 |
$BAw$k$H!"%5!<%P$OI,MW$J$@$1%9%?%C%/$+$i%G!<%?$r<h$j=P$7!"(B |
\item |
$B<B9T$7$?7W;;$N7k2L$r%9%?%C%/$K@Q$`!#(B |
クライアントがサーバに計算の命令を送ると、 |
\item $B:G8e$K!V%G!<%?$r<h$j=P$7Aw?.$r9T$J$&L?Na$KBP1~$7$?%G!<%?!W$r(B |
サーバは必要なだけスタックからデータを取り出し、 |
$B%5!<%P$XAw$k$H!"%5!<%P$O%9%?%C%/$+$i7W;;7k2L$NF~$C$F$$$k(B |
実行した結果をスタックに積む。 |
$B%G!<%?$r<h$j=P$7!"%/%i%$%"%s%H$XAw=P$9$k!#(B |
%って書いてるけど、命令がSM\_popCMO とか SM\_shutdown の場合は? |
|
\item |
|
最後に SM\_popCMO もしくは SM\_popString をサーバへ送ると、 |
|
サーバはスタックから計算結果の入っているデータを取り出し、 |
|
クライアントへ送出する。 |
\end{enumerate} |
\end{enumerate} |
|
|
|
|
\section{CMO $B$N%G!<%?9=B$(B} |
\section{OpenXM スタックマシン}\label{sec:oxsm} |
|
|
OpenXM $B4V$G$d$j$H$j$5$l$k%a%C%;!<%8$r<B:]$K:n@.$9$k>l9g!"(B |
OpenXM 規約ではサーバはスタックマシンであると定義している。以下、OpenXM |
CMO $B7A<0$GDj5A$5$l$F$$$kB?G\D9@0?t$rM}2r$7$F$*$/$H!"(B |
スタックマシンと呼ぶ。この節ではOpenXM スタックマシンの構造について説明 |
CMO $B7A<0$NB>$N%G!<%?9=B$$@$1$G$J$/!"(B OX $B7A<0!"(B SM $B7A<0$N%G!<%?$r(B |
しよう。 |
$BM}2r$9$k=u$1$K$J$k$H;W$($k$N$G!"(B CMO $B7A<0$NB?G\D9@0?t$N(B |
|
$B%G!<%?9=B$$K$D$$$F@bL@$9$k!#(B |
|
|
|
CMO $B7A<0$GDj5A$5$l$F$$$k%G!<%?$OB?G\D9@0?t0J30$K$b(B |
まず、OpenXM 規約は通信時にやりとりされる共通のデータ形式については規定 |
$BJ8;zNs$d%j%9%H9=B$$J$I$,$"$k!#$I$N$h$&$J%G!<%?$G$"$k$+$O(B |
するが、OpenXM スタックマシンがスタックに積む、オブジェクトの構造までは |
$B%G!<%?$N@hF,$K$"$k(B tag $B$r8+$l$PH=JL$G$-$k$h$&$K$J$C$F$$$k!#(B |
規定しない。つまり、オブジェクトの構造は各数学システムごとに異なっている |
$B$3$l$O%a%C%;!<%8$N%G!<%?$NH=JL$N;EJ}$H$*$J$8$G$"$k!#(B |
ということである。このことは通信路からデータを受け取った際に、各数学シス |
$B$J$*!"(B tag $B$O3F%G!<%?Kh$K(B 32 bit $B$N@0?t$GI=$5$l$F$*$j!"(B |
テムが固有のデータ構造に変換してからスタックに積むことを意味する。この変 |
$BB?G\D9@0?t$O(B 20 $B$H$J$C$F$$$k!#(B |
換は1対1対応である必要はない。 |
$B$3$3$G(B 32 bit $B$N@0?t$NI=8=J}K!$K$D$$$F@bL@$9$kI,MW$,$"$k!#(B |
|
OpenXM $B$G$O%P%$%HNs$G(B 32 bit $B$N@0?t(B 20 $B$r(B |
|
{\tt 00 00 00 14} $B$HI=$9J}K!$H(B {\tt 14 00 00 00} $B$HI=$9J}K!$,$"$k!#(B |
|
$B$3$NI=8=J}K!$N0c$$$O%/%i%$%"%s%H$H%5!<%P$N:G=i$N@\B3;~$K(B |
|
$BAPJ}$N9g0U$G7hDj$9$k$3$H$K$J$C$F$$$k!#(B |
|
$B$J$*!"9g0U$,$J$$>l9g$K$O(B |
|
$BA0<T$NI=8=J}K!(B($B0J8e!"$3$NI=8=J}K!$r(B network byte order $B$H8F$V(B)$B$r(B |
|
$B;H$&$3$H$K$J$C$F$$$k!#(B |
|
$B$^$?!"Ii$N?t$rI=8=$9$kI,MW$,$"$k$H$-$K$O!"(B |
|
2 $B$NJd?tI=8=$r;H$&$3$H$K$J$C$F$$$k!#(B |
|
|
|
$BI=8=$7$?$$B?G\D9@0?t$N@dBPCM$r(B 2 $B?J?t$GI=$7$?>l9g$N7e?t$r(B $n$ $B$H(B |
次に OpenXM スタックマシンの命令コードについて説明する。OpenXM スタック |
$B$7$?$H$-!"<!$K$/$k%G!<%?$O(B $[(n+31)/32]$ $B$r(B 32 bit $B$N@0?t$H$J$k!#(B |
マシンにおけるすべての命令は4バイトの長さを持つ。OpenXM 規約の他の規定と |
$B$3$l$OB?G\D9@0?t$N@dBPCM$r(B $2^{32}$ $B?J?t$GI=$7$?>l9g$N7e?t$H$H$C$F$b$h$$!#(B |
同様に、4バイトのデータは32ビット整数と見なされるので、この論文でもその |
$B$?$@$7!"I=8=$7$?$$?t$,Ii$N>l9g$O(B $[(n+31)/32]$ $B$r(B 32 bit $B$N@0?t$GI=$7$?CM$r(B |
表記にしたがう。OpenXM スタックマシンに対する命令はスタックに積まれるこ |
2 $B$NJd?tI=8=$GIi$K$7$F!"@5$N>l9g$H6hJL$9$k!#(B |
とはない。現在のところ、OpenXM 規約では以下の命令が定義されている。 |
|
|
$BI=8=$7$?$$B?G\D9@0?t$N@dBPCM$,(B $2^{32}$ $B?J?t$G(B $(b_0 b_1 ... b_k)_{2^{32}}$ |
\begin{verbatim} |
$B$HI=$;$k$H$-!"<!$K$/$k%G!<%?$O(B $b_0$, $b_1$, $\cdots$, $b_k$ $B$r(B |
#define SM_popSerializedLocalObject 258 |
$B$=$l$>$l(B 32 bit $B$N@0?t$GI=8=$7$?CM$H$J$k!#(B |
#define SM_popCMO 262 |
%$B0J2<$O=q$-D>$7$NI,MW$,$"$k$+$b(B... |
#define SM_popString 263 |
$B$J$*!"(B GNU MP LIBRARY $B$rMQ$$$k$H!"(B |
|
C $B8@8l$+$iB?G\D9@0?t$dG$0U@:EYIbF0>.?t$r07$&$3$H$,$G$-$k!#(B |
|
$b_0$, $b_1$, $\cdots$, $b_k$ $B$r$=$l$>$l(B 32 bit $B@0?t$GI=8=$7$?CM$O(B |
|
$B$3$N(B GNU MP LIBRARY $B$GMQ$$$i$l$F$$$kB?G\D9@0?t$G;H$o$l$F$$$k7A<0$r(B |
|
$B;29M$K$7$F9g$o$;$F$"$k!#(B |
|
|
|
$B$3$3$G6qBNNc$r$@$=$&!#(B |
#define SM_mathcap 264 |
$4294967298 = 1 \times 2^{32} + 2$ $B$r(B network byte order $B$NB?G\D9@0?t$G(B |
#define SM_pops 265 |
$BI=8=$9$k$H!"(B |
#define SM_setName 266 |
\begin{center} |
#define SM_evalName 267 |
{\tt 00 00 00 14 00 00 00 02 00 00 00 02 00 00 00 01} |
#define SM_executeStringByLocalParser 268 |
\end{center} |
#define SM_executeFunction 269 |
$B$H$J$k!#$^$?!"F1$8I=8=J}K!$G(B $-1$ $B$rI=8=$9$k$H!"(B |
#define SM_beginBlock 270 |
\begin{center} |
#define SM_endBlock 271 |
{\tt 00 00 00 14 ff ff ff ff 00 00 00 01} |
#define SM_shutdown 272 |
\end{center} |
#define SM_setMathCap 273 |
$B$H$J$k!#(B |
#define SM_executeStringByLocalParserInBatchMode 274 |
|
#define SM_getsp 275 |
|
#define SM_dupErrors 276 |
|
|
|
#define SM_DUMMY_sendcmo 280 |
|
#define SM_sync_ball 281 |
|
|
\section{MathCap $B$K$D$$$F(B} |
#define SM_control_kill 1024 |
|
#define SM_control_to_debug_mode 1025 |
|
#define SM_control_exit_debug_mode 1026 |
|
#define SM_control_ping 1027 |
|
#define SM_control_start_watch_thread 1028 |
|
#define SM_control_stop_watch_thread 1029 |
|
#define SM_control_reset_connection 1030 |
|
\end{verbatim} |
|
|
$B%5!<%P$*$h$S%/%i%$%"%s%HAPJ}$H$b$K(B OpenXM $B$G5,Dj$5$l$F$$$k(B |
%以下、どういうときに結果をスタックに積むかエラーの場合どうするかの説明が |
$B%a%C%;!<%8$NCf$N%G!<%?7A<0$r$9$Y$F<u$1<h$l$k$o$1$G$O$J$$!#(B |
%必要であろう。 |
$B$7$+$b!"(B OpenXM $B5,Ls$G5,Dj$5$l$F$$$k%G!<%?7A<0$@$1$,(B |
|
$B<uEO$7$K;H$o$l$k$H$$$&$o$1$G$O$J$$!#(B |
|
$B$=$3$G!"(B OpenXM $B$G$OAj<jB&$,<u$1<h$k$3$H$,$G$-$k%G!<%?7A<0$r(B |
|
$B<}F@$9$kJ}K!$rMQ0U$7$F$$$k!#(B |
|
|
|
CMO $B7A<0$GDj5A$5$l$F$$$k(B MathCap $B%G!<%?$O(B |
スタックマシンに対する命令の中には実行によって結果が返ってくるものがある。 |
%$BM}2r2DG=$J%a%C%;!<%8$N(B |
結果が返ってくる命令を実行した場合、サーバはその結果をスタックに積む。 |
$B<u$1<h$k$3$H$,$G$-$k%G!<%?7A<0$rI=$9%G!<%?$G$"$j!"(B |
たとえば、 SM\_executeStringByLocalParser は |
$BMW5a$5$l$l$P%5!<%P$O%5!<%P<+?H$N(B MathCap $B%G!<%?$r%9%?%C%/$K@Q$`!#(B |
スタックに積まれているオブジェクトを |
$B$^$?!"%/%i%$%"%s%H$+$i(B MathCap $B%G!<%?$r%5!<%P$XAw$k$3$H$b$G$-!"(B |
サーバ側のローカル言語の文法に従った文字列とみなして計算を行なうが、 |
MathCap $B%G!<%?$r%5!<%P$H%/%i%$%"%s%H$N4V$G8r49$9$k$3$H$K$h$C$F!"(B |
行なった計算の結果はローカル言語で記述した文字列でスタックに積まれる。 |
$B$*8_$$$KAj<jB&$,<u$1<h$k$3$H$,$G$-$J$$%G!<%?7A<0$G(B |
なお、命令の実行中にエラーが起こり、結果が得られなかった場合には、 |
$B%a%C%;!<%8$rAw$C$F$7$^$&$N$rKI$0$3$H$,$G$-$k!#(B |
エラーオブジェクトがスタックに積まれる。 |
$B$J$*!"(B MathCap $B%G!<%?$NCf$G$O(B CMO $B7A<0$GDj5A$5$l$F$$$k(B |
|
32 bit $B@0?t!"J8;zNs!"%j%9%H9=B$$,;H$o$l$F$*$j!"(B |
|
MathCap $B%G!<%?$K4^$^$l$F$$$kFbMF$rM}2r$G$-$k$?$a$K$O(B |
|
$BI,A3E*$K$3$l$i$bM}2r$G$-$kI,MW$,$"$k!#(B |
|
|
|
OpenXM $BBP1~HG$N(B asir $B%5!<%P$G$"$k(B ox\_asir $B$,JV$9(B MathCap $B$r0J2<$K<($9!#(B |
|
|
|
%$B$J$*!"(B $a_1$, $a_2$, $\cdots$, $a_n$ $B$rMWAG$K(B |
\section{CMO のデータ構造}\label{sec:cmo} |
%$B;}$D%j%9%H9=B$$r(B {\tt [$a_1$, $a_2$, $\cdots$, $a_n$]} $B!"(B |
|
%$BJ8;zNs(B ``string'' $B$r(B {\tt "string"} $B!"(B 32 bit $B@0?t$r(B |
|
%$B$=$l$KBP1~$9$k(B 10 $B?J?t$N@0?t$G<($9!#(B |
|
|
|
%$B"-<j$G:n$C$?$N$G4V0c$($F$$$k2DG=@-$"$j!#(B |
OpenXM 規約では、数学的オブジェクトを表現する方法として CMO 形式(Common |
%%$B8E$$%P!<%8%g%s!#:9$7BX$($NI,MW$"$j!#(B |
Mathematical Object format)を定義している。この CMO 形式にしたがったデー |
|
タは、識別子が OX\_DATA であるようなメッセージのボディになることを想定し |
|
ている。 |
|
|
|
CMO 形式におけるデータ構造は次のような構造をもつ。 |
|
|
|
\begin{tabular}{|c|c|} \hline |
|
ヘッダ & \hspace{10mm} ボディ \hspace{10mm} \\ \hline |
|
\end{tabular} |
|
|
|
ヘッダは4バイトである。ボディの長さはそれぞれのデータによって異なるが、 |
|
0でもよい。 |
|
|
|
メッセージと同様にヘッダは4バイト単位に管理される。すなわち、CMO ではヘッ |
|
ダは一つだけの情報を含む。この4バイトのヘッダのことをタグともいう。さて、 |
|
CMO では、タグによってボディの論理的構造が決定する。すなわち、タグはそれ |
|
ぞれのデータ構造と1対1に対応する識別子である。それぞれの論理的構造は |
|
\cite{OpenXM-1999} に詳述されている。現在の OpenXM 規約では以下の CMO が |
|
定義されている。 |
|
|
\begin{verbatim} |
\begin{verbatim} |
[ [199901160,"ox_asir"], |
#define CMO_ERROR2 0x7f000002 |
[276,275,258,262,263,266,267,268,274 |
#define CMO_NULL 1 |
,269,272,265,264,273,300,270,271], |
#define CMO_INT32 2 |
[ [514,[1,2,3,4,5,2130706433,2130706434 |
#define CMO_DATUM 3 |
,17,19,20,21,22,24,25,26,31,27,33,60]], |
#define CMO_STRING 4 |
[2144202544,[0,1]] |
#define CMO_MATHCAP 5 |
] |
|
] |
#define CMO_START_SIGNATURE 0x7fabcd03 |
|
#define CMO_ARRAY 16 |
|
#define CMO_LIST 17 |
|
#define CMO_ATOM 18 |
|
#define CMO_MONOMIAL32 19 |
|
#define CMO_ZZ 20 |
|
#define CMO_QQ 21 |
|
#define CMO_ZERO 22 |
|
#define CMO_DMS_GENERIC 24 |
|
#define CMO_DMS_OF_N_VARIABLES 25 |
|
#define CMO_RING_BY_NAME 26 |
|
#define CMO_RECURSIVE_POLYNOMIAL 27 |
|
#define CMO_LIST_R 28 |
|
|
|
#define CMO_INT32COEFF 30 |
|
#define CMO_DISTRIBUTED_POLYNOMIAL 31 |
|
#define CMO_POLYNOMIAL_IN_ONE_VARIABLE 33 |
|
#define CMO_RATIONAL 34 |
|
|
|
#define CMO_64BIT_MACHINE_DOUBLE 40 |
|
#define CMO_ARRAY_OF_64BIT_MACHINE_DOUBLE 41 |
|
#define CMO_128BIT_MACHINE_DOUBLE 42 |
|
#define CMO_ARRAY_OF_128BIT_MACHINE_DOUBLE 43 |
|
|
|
#define CMO_BIGFLOAT 50 |
|
#define CMO_IEEE_DOUBLE_FLOAT 51 |
|
|
|
#define CMO_INDETERMINATE 60 |
|
#define CMO_TREE 61 |
|
#define CMO_LAMBDA 62 |
\end{verbatim} |
\end{verbatim} |
|
|
$B$3$N(B MathCap $B%G!<%?$N%j%9%H9=B$$OBg$-$/J,$1$F(B 3 $B$D$NItJ,$KJ,$+$l$k!#(B |
この中で CMO\_ERROR2, CMO\_NULL, CMO\_INT32, CMO\_DATUM, CMO\_STRING, |
$B:G=i$N(B {\tt [199901160,"ox\_asir"]} $B$NItJ,$K$O%5!<%P$N>pJs$,F~$C$F$$$k!#(B |
CMO\_MATHCAP, CMO\_LIST で識別されるオブジェクトは最も基本的なオブジェ |
%$B$3$N:G=i$NMWAG$,$^$?%j%9%H9=B$$H$J$C$F$*$j!"(B |
クトであって、すべての OpenXM 対応システムに実装されていなければならない。 |
$B:G=i$NMWAG$O%P!<%8%g%s%J%s%P!<$r!"<!$NMWAG$O%5!<%P$NL>A0$rI=$7$F$$$k!#(B |
|
|
|
$B<!$N(B {\tt [276,275,$\cdots$,271]} $B$NItJ,$O(B |
これらについての解説を行う前に記法について、少し説明しておく。 |
$B%5!<%P$KBP$9$kF0:n$KBP1~$7$?M}2r2DG=$J%G!<%?$N<oN`$rI=$7$F$$$k!#(B |
この論文では、大文字で CMO\_INT32 と書いた場合には、上記で定義した識別子 |
$B%5!<%P$NF0:n$KBP$9$k%G!<%?$O$9$Y$F(B 32 bit $B$N@0?t$GI=$7$F$*$j!"(B |
を表わす。また CMO\_INT32 で識別されるオブジェクトのクラス(あるいはデー |
$B$3$N%j%9%H$OM}2r2DG=$J%G!<%?$KBP1~$9$k(B 32 bit $B@0?t$N%j%9%H$H$J$C$F$$$k!#(B |
タ構造)を cmo\_int32 と小文字で表わすことにする。 |
|
|
$B:G8e$N(B {\tt [ [514,[1,2,3,$\cdots$,60]],[2144202544,[0,1]] ]} $B$NItJ,$O(B |
さて cmo を表現するための一つの記法を導入する。この記法は CMO expression |
$BM}2r2DG=$J%G!<%?$N7A<0$rI=$7$F$$$k!#(B |
と呼ばれている。その正確な形式的定義は \cite{OpenXM-1999} を参照すること。 |
$B$3$NItJ,$O$5$i$K(B {\tt [514,[1,2,3,$\cdots$,60]]} $B$H(B |
|
{\tt [2144202544,[0,1]]} $B$K$NItJ,$KJ,$1$k$3$H$,$G$-!"(B |
|
$B$=$l$>$l$,0l$D$N%G!<%?7A<0$K$D$$$F$N>pJs$H$J$C$F$$$k!#(B |
|
$B$I$N%G!<%?7A<0$K$D$$$F$N>pJs$+$O:G=i$NMWAG$K$"$k@0?tCM$r$_$l$P(B |
|
$BJ,$+$k$h$&$K$J$C$F$$$k!#(B |
|
$B$3$N@0?tCM$O(B CMO $B7A<0$G$O(B 514 $B$H$J$C$F$$$k!#(B |
|
$B:G=i$N%G!<%?7A<0$r6hJL$9$k@0?tCM0J8e$NMWAG$O(B |
|
$B3F%G!<%?7A<0$K$h$C$F$I$N$h$&$K;H$o$l$k$+Dj$^$C$F$$$k!#(B |
|
CMO $B7A<0$G$OM}2r2DG=$J%G!<%?$N(B tag $B$,%j%9%H$NCf$K<}$^$C$F$$$k!#(B |
|
$BA0@a$G(B CMO $B7A<0$G$OB?G\D9@0?t$rI=$9(B tag $B$,(B 20 $B$G$"$k$3$H$r=R$Y$?$,!"(B |
|
$B$3$N%j%9%H$K(B 20 $B$,4^$^$l$F$$$k$N$G!"(B |
|
ox\_asir $B$O(B CMO $B7A<0$NB?G\D9@0?t$r<u$1<h$l$k$3$H$,$o$+$k!#(B |
|
|
|
%%$B$3$N%j%9%H$NMWAG$O$^$?%j%9%H$H$J$C$F$*$j!"(B |
まず CMO expssion は Lisp 風表現の一種で、 cmo を括弧で囲んだリストとし |
%$B$3$N:G8e$NItJ,$b$^$?%j%9%H$H$J$C$F$*$j!"(B |
て表現する。それぞれの要素はカンマで区切る。 |
%$B$"$k%G!<%?7A<0$GM}2r2DG=$J$b$N$rI=8=$7$?%j%9%H$rMWAG$H$7$F$$$k!#(B |
例えば、 |
%{\tt [514,[1, 2, $\cdots$]]} $B$N:G=i$N(B 514 $B$O$3$N%j%9%H$,(B CMO $B7A<0(B |
\begin{quote} |
%$B$G$NM}2r2DG=$J%G!<%?$rI=$7$F$$$k$3$H$r<($7$F$*$j!"(B |
(17, {\sl int32}, (CMO\_NULL), (2, {\sl int32} $n$)) |
%$B$=$N8e$N%j%9%H$G$O(B CMO $BAX$GDj5A$5$l$F$$$k%G!<%?$N$&$A!"(B |
\end{quote} |
%$BM}2r2DG=$J%G!<%?$N(B tag $B$,JB$s$G$$$k!#(B |
は CMO expression である。ここで、小文字の斜体で表された``{\sl int32}'' |
|
は 4バイトの任意のデータを表す記号であり、``{\sl int32} $n$'' は同じく 4 |
|
バイトのデータであるが以下の説明で $n$ と表すことを示す。また数字 17, 2 |
|
などは 4バイトのデータで整数値としてみたときの値を意味する。CMO\_NULL は |
|
識別子(すなわち数字 1 と等価)である。この記法から上記のデータは 20 バイ |
|
トの大きさのデータであることが分かる。 |
|
なお、このデータは CMO ではないことに注意してほしい。 |
|
%なお、 CMO expression で表現できていても、 |
|
%それが CMO であることとは無関係である。 |
|
|
$B$J$*!"%G!<%?$,<u$1<h$l$k$3$H$H!"(B |
さて、この記法のもとで cmo\_int32 を次のデータ構造を持つと定義する。 |
$B%G!<%?$NO@M}9=B$$,M}2r$G$-$k$3$H$H$O$^$C$?$/JLJ*$G$"$k$N$G(B |
\begin{quote} |
$BCm0U$9$kI,MW$,$"$k!#(B |
cmo\_int32 := (CMO\_INT32, {\sl int32} $a$) |
|
\end{quote} |
|
|
|
%{\Huge 同様に cmo\_string, cmo\_list などを定義} |
|
|
\section{$B%;%-%e%j%F%#BP:v(B} |
これは CMO の 32 ビット整数 $a$ を表す。 |
|
他のオブジェクトも定義するために、 |
|
以後 ``{\sl string} $s$'' を文字列 $s$ 、 |
|
``{\sl cmo} $ob$'' を CMO の $ob$ とする。 |
|
これを用いて、 cmo\_string, cmo\_list を定義する。 |
|
|
OpenXM $B$G$O4v$i$+$N%;%-%e%j%F%#BP:v$r9M$($F$$$k!#(B |
\begin{quote} |
OpenXM $B$KBP1~$7$?%=%U%H%&%'%"$r%/%i%C%/$7$F$b(B |
cmo\_string := (CMO\_STRING, {\sl int32} $len$, {\sl string} $str$) \\ |
$BBg$7$?MxE@$O$J$$$H;W$($k$,!"$=$l$O@_7W>e$NOC$G$"$C$F!"(B |
cmo\_list := (CMO\_LIST, {\sl int32} $n$, {\sl cmo} $ob_1$, |
$BM=4|$;$L<jCJ$G967b$r<u$1$?>l9g$K$I$N$h$&$J;vBV$r(B |
{\sl cmo} $ob_2$, $\cdots$,{\sl cmo} $ob_n$) |
$B>7$/$+$OA[A|$7Fq$$!#(B |
\end{quote} |
|
|
|
これはそれぞれ長さ $len$ の文字列 $str$ と、 |
|
$ob_1$, $ob_2$, $\cdots$, $ob_n$ からなる長さ $n$ のリストを表す。 |
|
|
%$B$^$:!"@\B3$,I,MW$K$J$C$?;~$K!"%/%i%$%"%s%H$,%5!<%PB&$+$i$N(B |
|
%$B@\B3$rBT$D$h$&$K$J$C$F$$$k!#$3$l$O>o$K@\B3$rBT$D$3$H$K$h$C$F!"(B |
|
%$B?/F~<T$K@\B3$N5!2q$rM?$($F$7$^$&$3$H$r8:$i$9$?$a$G$"$k!#(B |
|
|
|
%$B<!$K!"%/%i%$%"%s%H$,@\B3Kh$K@\B3$rBT$D(B port $BHV9f$r%i%s%@%`$K7hDj$9$k(B |
% ここで 32 bit の整数の表現方法について触れておく。 |
%$B$3$H$K$J$C$F$$$k!#$3$l$K$h$j!"?/F~<T$,$I$3$K@\B3$r9T$J$($P$h$$$N$+(B |
% OpenXM 規約ではバイトストリームで 32 bit の整数 20 を |
%$B$o$+$j$K$/$/$7$F$$$k!#(B |
% {\tt 00 00 00 14} と表す方法と {\tt 14 00 00 00} と表す方法がある。 |
|
% この表現方法の違いはクライアントとサーバの最初の接続時に |
|
% 双方の合意で決定することになっている。 |
|
% なお、合意がない場合には前者の表現方法 |
|
% (以後、この表現方法をネットワークバイトオーダーと呼ぶ)を |
|
% 使うことになっている。 |
|
% また、負の数を表現する必要があるときには、 |
|
% 2 の補数表現を使うことになっている。 |
|
|
%$B:G8e$K!"@\B3;~$K%/%i%$%"%s%H$,(B 1 $B2s$N$_M-8z$J%Q%9%o!<%I$rH/9T$7!"(B |
% 先ほどの、 (CMO\_INT32, 123456789) をネットワークバイトオーダーで |
%$BG'>Z$r9T$J$&$h$&$K$J$C$F$$$k!#$3$N%Q%9%o!<%I$O0lC6;HMQ$5$l$k$H(B |
% バイト列に直すと、 |
%$BL58z$K$J$k$N$G!"$b$72>$K$J$s$i$+$N<jCJ$G%Q%9%o!<%I$,1L$l$?$H$7$F$b(B |
% \begin{center} |
%$B0BA4$G$"$k!#$J$*!"$3$N%Q%9%o!<%I$O0BA4$J<jCJ$GAw$i$l$F$$$J$$$H(B |
% {\tt 00 00 00 02 07 5b cd 15} |
%$B$$$1$J$$!#$^$?!"8=:_$N<BAu$G$O%5!<%P!"$*$h$S%/%i%$%"%s%H$NF0:n$7$F$$$k(B |
% \end{center} |
%$B%3%s%T%e!<%?>e$G$O$3$N%Q%9%o!<%I$,$o$+$C$F$7$^$&$?$a!"(B |
% となり、 |
%$BF10l$N%3%s%T%e!<%?>e$K0-0U$N$"$k%f!<%6$O$$$J$$$H2>Dj$7$F$$$k(B |
% (CMO\_STRING, 6, ``OpenXM'') は |
%$B$3$H$KCm0U$7$J$1$l$P$J$i$J$$!#(B |
% \begin{center} |
|
% {\tt 00 00 00 04 00 00 00 06 4f 70 65 6e 58 4d} |
|
% \end{center} |
|
% となる。 |
|
|
%$B$J$*!"@\B3$,3NN)$7$?8e$N%a%C%;!<%8$NAw<u?.$K4X$7$F$O!"(B |
% CMO 形式の多倍長整数は、 Gnu MPライブラリ等を参考にしており、 |
%$BFC$K0E9f2=$J$I$N=hCV$,9T$J$o$l$F$$$k$o$1$G$O$J$$!#(B |
% 符号付き絶対値表現を用いている。 |
%$B$b$7I,MW$,$"$l$P!"DL?.O)$N0E9f2=$r9T$J$&5!G=$,$"$k(B |
% タグ以降の形式は次のようになる。 |
%$B%=%U%H%&%'%"$r;H$&$3$H$r9M$($F$$$k!#(B |
|
|
|
|
% \begin{tabular}{|c|c|c|c|c|} \hline |
|
% $f$ & $b_0$ & $b_1$ & $\cdots$ & $b_{n-1}$ \\ \hline |
|
% \end{tabular} |
|
|
\section{$BB>$N%W%m%8%'%/%H(B} |
% ここで、 1 つの枠は 4 バイトを表し、 |
|
% $f$ は符号付き 32 ビット整数を、 |
|
% $b_0$, $b_1$, $\cdots$, $b_{n-1}$ は符号なし 32 ビット整数を表している。 |
|
% さらに、 $|f| = n$ が成り立たなければならない。 |
|
% このオブジェクトは |
|
% \[ \mbox{sgn}(f) \times \{ b_0 (2^{32})^0 + b_1 (2^{32})^1 + \cdots |
|
% + b_{n-1} (2^{32})^{n-1} \} \] |
|
% という整数であると定義されている。 |
|
% ただし、 |
|
% \[ \mbox{sgn}(f) = \left\{ \begin{array}{ll} |
|
% 1 & f>0 \\ |
|
% 0 & f=0 \\ |
|
% -1 & f<0 \\ \end{array} \right. \] |
|
% である。 |
|
|
$BB>$N%W%m%8%'%/%H$K$D$$$F4v$D$+>R2p$9$k!#(B |
% ここで具体例をだそう。 |
|
% $4294967298 = 1 \times 2^{32} + 2$ を CMO 形式の |
|
% ネットワークバイトオーダー、多倍長整数で表現すると、 |
|
% \begin{center} |
|
% {\tt 00 00 00 14 00 00 00 02 00 00 00 02 00 00 00 01} |
|
% \end{center} |
|
% となる。また、同じ表現方法で $-1$ を表現すると、 |
|
% \begin{center} |
|
% {\tt 00 00 00 14 ff ff ff ff 00 00 00 01} |
|
% \end{center} |
|
% となる。 |
|
|
OpenMath $B%W%m%8%'%/%H$O?t3XE*$J%*%V%8%'%/%H$r(B |
|
$B%3%s%T%e!<%?>e$GI=8=$9$kJ}K!$r7hDj$7$F$$$k!#(B |
|
$B3F%=%U%H%&%'%"4V$G%*%V%8%'%/%H$r8r49$9$k:]$N(B |
|
$B%*%V%8%'%/%H$NJQ49<j=g$K$D$$$F$b=R$Y$i$l$F$$$k!#(B |
|
$BI=8=J}K!$O0l$D$@$1$G$J$/!"(B XML $BI=8=$d(B binary $BI=8=$J$I$,(B |
|
$BMQ0U$5$l$F$$$k!#(B |
|
|
|
%$B0J2<!"D4$Y$kI,MW$"$j!#(B |
\section{mathcap について} |
%NetSolve |
|
|
|
%MP |
OpenXM 規約では、通信時に用いられるメッセージの種類を各ソフトウェアが制 |
|
限する方法を用意している。これは各ソフトウェアの実装によってはすべてのメッ |
|
セージをサポートするのが困難な場合があるからである。また、各ソフトウェア |
|
でメッセージの種類を拡張したい場合にも有効である。この制限(あるいは拡張) |
|
は mathcap と呼ばれるデータ構造によって行われる。この節では mathcap のデー |
|
タ構造と、具体的なメッセージの制限の手続きについて説明する。 |
|
|
%MCP |
では、手続きについて説明しよう。 |
|
|
\section{$B8=:_Ds6!$5$l$F$$$k%=%U%H%&%'%"(B} |
第一にサーバの機能を制限するには次のようにする。クライアントが mathcap |
|
オブジェクトをサーバへ送ると、サーバは受け取ったmathcap をスタックに積む。 |
|
次にクライアントが命令 SM\_setMathCap を送ると、サーバはスタックの最上位 |
|
に積まれている mathcap オブジェクトを取り出し、mathcap で設定されていな |
|
いメッセージをクライアントへ送らないように制限を行う。 |
|
|
$B8=:_(B OpenXM $B5,3J$KBP1~$7$F$$$k%/%i%$%"%s%H%=%U%H%&%'%"$K$O(B |
第二にクライアントを制限するには次のようにする。クライアントがサーバに命 |
asir, sm1, Mathematica $B$,$"$k!#(B |
令 SM\_mathcap を送ると、サーバは mathcap オブジェクトをスタックに積む。 |
$B$3$l$i$N%/%i%$%"%s%H%=%U%H%&%'%"$+$i(B |
さらに命令 SM\_popCMO を送ると、サーバはスタックの最上位のオブジェクト |
OpenXM $B5,3J$KBP1~$7$?%5!<%P$r8F$S=P$9$3$H$,$G$-$k!#(B |
(すなわち mathcap オブジェクト)をボディとするメッセージをクライアントに |
$B8=:_(B OpenXM $B5,Ls$KBP1~$7$F$$$k%5!<%P%=%U%H%&%'%"$K$O!"(B |
送付する。クライアントはそのオブジェクトを解析して、制限をかける。 |
asir, sm1, gnuplot, Mathematica $B$J$I$,$"$j!"(B |
|
$B$=$l$>$l(B ox\_asir, ox\_sm1, ox\_math $B$H$$$&L>A0$GDs6!$5$l$F$$$k!#(B |
次に mathcap のデータ構造について説明する。 |
$B$^$?!"(B OpenMath $B5,3J$N(B XML $BI=8=$GI=8=$5$l$?%G!<%?$H(B CMO $B7A<0$N(B |
mathcap は CMO の一種であるので、すでに説明したように \\ |
$B%G!<%?$rJQ49$9$k%=%U%H%&%'%"$,(B JAVA $B$K$h$C$F<BAu$5$l$F$*$j!"(B |
\begin{tabular}{|c|c|} \hline |
OMproxy $B$H$$$&L>A0$GDs6!$5$l$F$$$k!#(B |
ヘッダ & \hspace{10mm} ボディ \hspace{10mm} \\ \hline |
|
\end{tabular} \\ |
|
の構造を持ちヘッダの値は 5 である(\ref{sec:cmo} 節を参照のこと)。 |
|
ボディは cmo\_list オブジェクトでなければならない。 |
|
|
|
%\begin{quote} |
|
% cmo\_mathcap := (CMO\_MATHCAP,{\sl cmo} obj) |
|
%\end{quote} |
|
|
|
さて、mathcap オブジェクトのボディの cmo\_list オブジェクトは以下の条件を |
|
満たすことを要求される。 |
|
|
|
まず、その cmo\_list オブジェクトは少なくともリスト長が 3 以上でなければ |
|
ならない。 |
|
|
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $3$, |
|
{\sl cmo} $A$, {\sl cmo} $B$, {\sl cmo} $C$) |
|
\end{quote} |
|
%\[ \begin{tabular}{|c|c|c|} \hline |
|
% $A$ & $B$ & $C$ \\ \hline |
|
% \end{tabular} \] |
|
|
|
第一要素 $A$ はまた cmo\_list であり、リスト長は 4 以上、 |
|
$a_1$ は 32 ビット整数でバージョンナンバーを、 |
|
$a_2$, $a_3$, $a_4$ は文字列で |
|
それぞれシステムの名前、、 CPU の種類を表すことになっている。 |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $4$, |
|
{\sl cmo\_int32} $a_1$, {\sl cmo\_string} $a_2$, |
|
{\sl cmo\_string} $a_3$, {\sl cmo\_string} $a_4$) |
|
\end{quote} |
|
|
|
第二要素 $B$ は次のようなリスト構造をしている。 |
|
この $b_1$, $b_2$, $\cdots$, $b_n$ はすべて 32 ビットの整数である。 |
|
\ref{sec:oxsm} 節でみたように、 |
|
スタックマシンへの命令はすべて 32 ビットの整数で表しており、 |
|
各 $b_i$ は利用可能な命令に対応する 32 ビットの整数となっている。 |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $n$, |
|
{\sl cmo\_int32} $b_1$, {\sl cmo\_int32} $b_2$, |
|
$\cdots$, {\sl cmo\_int32} $b_n$) |
|
\end{quote} |
|
|
|
第三要素 $C$ は以下のようなリスト構造をしている。 |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $m$, \\ |
|
\hspace{10mm} (CMO\_LIST, {\sl int32} $l_1$, {\sl cmo\_int32} $c_{11}$, |
|
{\sl cmo} $c_{12}$, $\cdots$, {\sl cmo} $c_{1l_1}$) \\ |
|
\hspace{10mm} (CMO\_LIST, {\sl int32} $l_2$, {\sl cmo\_int32} $c_{21}$, |
|
{\sl cmo} $c_{22}$, $\cdots$, {\sl cmo} $c_{1l_2}$) \\ |
|
\hspace{10mm} $\vdots$ \\ |
|
\hspace{10mm} (CMO\_LIST, {\sl int32} $l_m$, {\sl cmo\_int32} $c_{m1}$, |
|
{\sl cmo} $c_{m2}$, $\cdots$, {\sl cmo} $c_{1l_m}$)) |
|
\end{quote} |
|
%%$n$ は OX\_COMMAND 以外の受け取れるメッセージのタグの種類の数に等しい。 |
|
%%要素数は 1 でももちろん構わない。 |
|
どの $c_{i1}$ にも 32 ビットの整数が入っており、 |
|
OX\_COMMAND 以外の、受け取れるメッセージのタグが入っている。 |
|
$c_{i2}$ 以降については最初の $c_{i1}$ の値によってそれぞれ異なる。 |
|
ここでは、最初の要素が OX\_DATA の場合についてのみ説明する。 |
|
この $c_{i1}$ が OX\_DATA の場合、 |
|
$c_{i1}$, $c_{i2}$, $\cdots$, $c_{il_i}$ を要素とする cmo\_list は |
|
CMO 形式についての情報を表しており、 $l_i=2$ と決められている。 |
|
$c_{i1}$ にはもちろんのこと OX\_DATA が入っており、 |
|
$c_{i2}$ は以下の図のような cmo\_list になっている。 |
|
各要素は 32 ビットの整数であり、 |
|
受け取ることが可能な CMO 形式のタグが入る。 |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $k$, |
|
{\sl cmo\_int32} $c_{i21}$, {\sl cmo\_int32} $c_{i22}$, |
|
$\cdots$, {\sl cmo\_int32} $c_{i2k}$) |
|
\end{quote} |
|
%\[ \overbrace{ |
|
% \begin{tabular}{|c|c|c|c|c|} \hline |
|
% $c_{i21}$ & $c_{i22}$ & $\cdots$ & $c_{i2l}$ \\ \hline |
|
% \end{tabular} |
|
% }^{c_{i2}} \] |
|
|
|
%なお、 mathcap データの中では CMO 形式で定義されている |
|
%32 bit 整数、文字列、リスト構造が使われており、 |
|
%mathcap データに含まれている内容を理解できるためには |
|
%必然的にこれらも理解できる必要がある |
|
%(ってことは CMO 形式のところでこれらを |
|
%説明しなければならないってことです)。 |
|
|
|
具体的な mathcap の例をあげよう。 |
|
名前が ``ox\_test''、バージョンナンバーが 199911250 のサーバで、 |
|
PC-UNIX 上で動いていれば、 |
|
$A$ の部分は |
|
\begin{quote} |
|
(CMO\_LIST, 4, {\sl cmo\_int32} $199911250$, {\sl cmo\_string} "ox\_test", |
|
{\sl cmo\_string} "Version=199911250", |
|
{\sl cmo\_string} "HOSTTYPE=i386") |
|
\end{quote} |
|
となる。 |
|
さらに、このサーバのスタックマシンが |
|
命令コード 2, 3, 5, 7, 11 番を利用可能 |
|
(実際にはこのような命令コードは存在しない)であれば、 $B$ の部分は |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $5$, |
|
{\sl cmo\_int32} $2$, {\sl cmo\_int32} $3$, |
|
{\sl cmo\_int32} $5$, {\sl cmo\_int32} $7$, |
|
{\sl cmo\_int32} $11$) |
|
\end{quote} |
|
となり、 |
|
CMO 形式の 32 ビット整数、文字列、 mathcap 、リスト構造のみが |
|
受け取れるときには、 $C$ の部分は |
|
\begin{quote} |
|
(CMO\_LIST, {\sl int32} $1$, \\ |
|
\ \ (CMO\_LIST, {\sl int32} $4$, |
|
{\sl cmo\_int32} $2$, {\sl cmo\_int32} $4$, |
|
{\sl cmo\_int32} $5$, {\sl cmo\_int32} $17$)) |
|
\end{quote} |
|
となる。 |
|
%CMO\_ZZ がないので、このサーバは多倍長整数が送られてこないことを |
|
%期待している。 |
|
|
|
%{\Huge ってなんででしょうか? データの論理構造を知らないと受け取れないと |
|
%思うんですが$\ldots$} |
|
|
|
なお、この mathcap では、データの論理構造が理解できるかどうか |
|
までは分からないので注意する必要がある。 |
|
|
|
|
|
\section{セキュリティ対策} |
|
|
|
OpenXM 規約は TCP/IP を用いて通信を行うことを考慮している。ネットワーク |
|
によって接続される現代の多くのソフトウェアと同様、OpenXM 規約もまた通信 |
|
時のセキュリティについて注意している。以下、このことについて説明しよう。 |
|
|
|
OpenXM では侵入者に攻撃の機会をできるだけ与えないようにするため、 |
|
接続が必要になった時のみ接続を待つようにし、 |
|
常に接続に関与するといったことは避けている。 |
|
|
|
しかし、これだけでは侵入者が接続を行なう一瞬のすきを |
|
狙ってくる可能性もある。 |
|
そこで接続を行なう時に、 |
|
接続を待つ port 番号をランダムに決めている。 |
|
こうすることで、特定の port 番号を狙って接続を行なう |
|
瞬間を待つ手口を幾らか防ぐことができる。 |
|
|
|
さらにもう一段安全性を高めるために、 |
|
接続時に 1 回だけ使用可能なパスワードを作成し、 |
|
そのパスワードを使って認証を行なう。 |
|
このパスワードは一旦使用されれば無効にするので、 |
|
もし仮になんらかの手段でパスワードが洩れたとしても安全である。 |
|
|
|
なお、上記の port 番号とパスワードは安全な手段で送られて |
|
いると仮定している。 |
|
また、同一のコンピュータ上に悪意のあるユーザはいないと仮定している |
|
ことに注意しなければならない。 |
|
なぜなら、現在の実装ではサーバ、およびクライアントの動作している |
|
コンピュータ上ではこの port 番号とパスワードがわかってしまうためである。 |
|
|
|
なお、接続が確立した後のメッセージの送受信に関しては、 |
|
特に暗号化などの処置を行っているわけではない。 |
|
もし必要があれば、通信路の暗号化を行なう機能がある |
|
ソフトウェア ssh を使うことを考えている。 |
|
|
|
|
|
\section{他のプロジェクト} |
|
|
|
他のプロジェクトについても触れておこう。 |
|
|
|
\begin{itemize} |
|
\item OpenMath\\ |
|
OpenMath プロジェクトは数学的なオブジェクトをコンピュータ上で表現する方 |
|
法を規定している。各ソフトウェア間でオブジェクトを交換する際のオブジェク |
|
トの変換手順につても定められている。表現方法は幾つかの段階で定められて |
|
いて、XML 表現やバイナリ表現などが用意されている。詳細は |
|
|
|
http://www.openmath.org/omsoc/ A.M.Cohen |
|
|
|
\item NetSolve |
|
|
|
http://www.cs.utk.edu/netsolve/ |
|
|
|
\item MP |
|
|
|
http://symbolicNet.mcs.kent.edu/SN/areas/protocols/mp.html |
|
|
|
\item MCP |
|
|
|
http://horse.mcs.kent.edu/~pwang/ |
|
\end{itemize} |
|
|
|
|
|
\section{現在提供されているソフトウェア} |
|
|
|
現在 OpenXM 規約に対応しているクライアントにはasir, sm1, Mathematica が |
|
ある。これらのクライアントから OpenXM 規約に対応したサーバを呼び出すこと |
|
ができる。現在 OpenXM 規約に対応しているサーバソフトウェアには、asir, |
|
sm1, gnuplot, Mathematica などがあり、それぞれ ox\_asir, ox\_sm1, |
|
ox\_sm1\_gnuplot, ox\_math という名前で提供されている。また、 OpenMath |
|
規約の XML 表現で表現されたオブジェクトと CMO 形式のオブジェクトを変換す |
|
るソフトウェアが JAVA によって実装されており、OMproxy という名前で提供さ |
|
れている。 |
|
|
|
\begin{thebibliography}{99} |
|
\bibitem{Ohara-Takayama-Noro-1999} |
|
小原功任, 高山信毅, 野呂正行: |
|
{Open asir 入門}, 1999, 数式処理, Vol 7, No 2, 2--17. (ISBN4-87243-086-7, SEG 出版, Tokyo). |
|
\bibitem{OpenXM-1999} |
|
野呂正行, 高山信毅: |
|
{Open XM の設計と実装 --- Open message eXchange protocol for Mathematics}, |
|
1999/11/22 |
|
\end{thebibliography} |
|
|
\end{document} |
\end{document} |