Japanese | English
OpenMHD は Fortran 90 で書かれた 磁気流体 (MHD) シミュレーションコードです。 シンプルな実装で動作が軽く、 Python/IDL の解析・可視化ルーチンを同梱しているため、 計算実行から可視化までを手軽に体験できるようになっています。 シリアル版、MPI/OpenMP並列版、GPU版、MPI/GPU版を同梱していますので、 PCでもスーパーコンピューターでも、CPU/GPUを問わず 同じコードを走らせることができます。
OpenMHD コードは、作者の MHD リコネクション研究 [1,2] の並列コードをパッケージ化したものです。 そして、その後の改良を取り入れて、随時更新を続けています。
tar.gz ファイルを直接ダウンロードしてください。
開発中のコードは GitHub で管理しています。以下のように git
コマンドでリポジトリを複製します。
$ git clone https://github.com/zenitani/OpenMHD.git
動作を確認している環境は以下の通りです。 あまり凝ったことをしていないので、ほとんどの環境で動作するはずです。 MPI 並列計算には、MPI-3.1 規格に準拠したMPIライブラリが必要です。
次のような保存形の磁気流体方程式を採用しています。 \begin{align} \frac{\partial \rho}{\partial t} &+ \nabla \cdot ( \rho \vec{v} ) = 0, \\ \frac{\partial \rho \vec{v}}{\partial t} &+ \nabla \cdot ( \rho\vec{v}\vec{v} + p_T\overleftrightarrow{I} - \vec{B}\vec{B} ) = 0, \\ \frac{\partial \mathcal{E}}{\partial t} &+ \nabla \cdot \Big( (\mathcal{E}+p_T )\vec{v} - (\vec{v}\cdot\vec{B}) \vec{B} + \eta \vec{j} \times \vec{B} \Big) = 0, \\ \frac{\partial \vec{B}}{\partial t} &+ \nabla \cdot ( \vec{v}\vec{B} - \vec{B}\vec{v} ) + \nabla \times (\eta \vec{j}) + \nabla \psi = 0, \\ \frac{\partial \psi}{\partial t} &+ c_h^2 \nabla \cdot \vec{B} = - \Big(\frac{c_h^2}{c_p^2}\Big) \psi, \end{align} このうち、$p_T=p+B^2/2$ は磁気圧を含めた総圧、 $\overleftrightarrow{I}$ は単位テンソル、 $\mathcal{E}=p/(\Gamma-1) + \rho v^2/2 + B^2/2$ はエネルギー密度、 $\Gamma=5/3$ は断熱定数です。 これらの式は $4\pi$ や $\mu_0$ などの係数が消える Lorentz-Heaviside 風の表式を使っています。 音速は $c_s \equiv \sqrt{\Gamma p / \rho }$、アルヴェン速度は $c_A \equiv B / \sqrt{\rho}$ になります。
$\psi$ は計算結果を補正するための仮想ポテンシャルです。 本来、磁場はソレノイダル条件 $\nabla \cdot \vec{B} = 0$ を満たしますが、 シミュレーションでは数値エラーで $\nabla \cdot \vec{B} \ne 0$ となることがあります。 そうすると、磁気トポロジーが変わるだけでなく 磁気単極子に対して非物理的な力が働いて計算結果を壊してしまいます。 そこで、仮想ポテンシャル $\psi$ を介して $\nabla \cdot \vec{B}$ の数値エラーを 拡散・減衰させます(Hyperbolic divergence cleaning 法 [3])。 $c_h,c_p$ は拡散・減衰度合いを調整するパラメーターです。
OpenMHD コードは、数値流束を評価する際に、 隣接するセル境界での一次元リーマン問題を考える近似リーマン解法を使っています。 その中でも5種類の不連続面に対応した HLLD 法 [4] を採用しています。 HLLD 法は、リーマン問題を適度な計算コストで精度良く近似するため、 MHD シミュレーションでは非常によく使われています。 本稿執筆(2024年)時点では業界標準と言って良いでしょう。
コードの計算精度は時間・空間ともに2次精度です。 2段の Runge=Kutta 法あるいは TVD Runge=Kutta 法で時間発展を解くとともに、 物理量(基本量)を MUSCL 補間して2次の空間精度を達成しています。 ψのソース項は、Runge=Kutta 部分の前後で 0.5Δt ずつ、 解析解 $\psi = \psi_0 \exp [ - ({c_h^2}/{c_p^2}) t ]$ を解いています。 他の技術的な詳細は、文献 [1,2] 及びその参考文献を参照してください。 OpenMHD を取り巻く研究状況については「生存圏研究」の記事 [5] を参照してください。
OpenMHD の使い方は以下の通りです。
Google Colaboratoryのアカウントをお持ちの方は、クラウド上で計算・データ解析を試すことができます。
OpenMHD ウェブページのトップから tar.gz ファイルをダウンロードしてください。 そして、ダウンロードしたファイルを tar コマンドで解凍すると、 たくさんのディレクトリと関連ファイルが展開されます。 (以下の操作には、Linux/macOS のターミナル環境が必要です。 Windows では、ディレクトリの中に含まれているUNIXのシンボリックリンクをうまく扱うことができません。)
$ gunzip openmhd-20241124.tar.gz $ tar xvf openmhd-20241124.tar
しかるべきディレクトリに移動します。
手始めに 2D_basic
ディレクトリに移動してみましょう。
$ cd openmhd-20241124/2D_basic/
OpenMHD は UNIX の
make コマンドを使って、
ソースコードをコンパイルします。
まず、テキストエディタで Makefile
というファイルを開いて、
F90 という変数を書き換えてコンパイラを指定してください。
MPI並列計算を行う場合は、MPI用の fortran コンパイラを指定します。
シリアル版のみを利用する場合は、普通の fortran コンパイラを指定するだけで良いでしょう。
F90 = gfortran
そして、make
コマンドでプログラムをコンパイルします。
$ make
OpenMHD の2次元問題には MPI 並列コードが同梱されています。 デフォルトでは、シリアル版と並列版の両方をコンパイルします。 シリアル版だけをコンパイルするのであれば、
$ make run
だけでも OK です。
このあたりのルールは Makefile の中身を参照してください。
無事コンパイルが終われば、
シリアル版の実行ファイル(a.out
)および
MPI 並列版の実行ファイル(ap.out
)ができているはずです。
次にプログラムを走らせてみましょう。 シリアル版の場合は、コマンドラインで直接実行ファイルを指定します。
$ ./a.out
あるいは
$ ./a.out | tee output.log
とします。 UNIXでは自分が今いるディレクトリ(カレントディレクトリ)を "." と表しますが、 "a.out" ではなく "./a.out" と書くことで、 カレントディレクトリの a.out ファイルを実行することを明示しています。 UNIXの tee コマンドは、プログラムの(標準)出力を画面に表示しつつ、ファイル(output.log)に記録します。
MPI 並列版を使う場合は、メインファイル(例:mainp.f90
)先頭付近の
mpi_nums
変数を編集してプロセス数を指定してください。
計算領域を 4 x 1 に分割する場合は (/4, 1/)
、
2 x 2 に分割する場合は (/2, 2/)
のようにします。
mpi_nums(2)
の (2) の部分は
2次元を意味していますので変える必要はありません。
integer, parameter :: mpi_nums(2) = (/4, 1/) ! MPI numbers
make
コマンドでプログラムをコンパイルした後、
mpirun
コマンドを使って実行ファイルを走らせます。
その際、-np
オプションで並列プロセス数 4 を指定します。
このプロセス数は mpi_nums
変数で指定したプロセス数と一致させてください。
$ mpirun -np 4 ./ap.out $ mpirun -np 4 ./ap.out | tee output.log
計算が終了すると、data/
ディレクトリに結果が出力されています。
$ ls data/ field-00000.dat field-00007.dat field-00014.dat field-00021.dat field-00028.dat field-00035.dat field-00001.dat field-00008.dat field-00015.dat field-00022.dat field-00029.dat field-00036.dat field-00002.dat field-00009.dat field-00016.dat field-00023.dat field-00030.dat field-00037.dat field-00003.dat field-00010.dat field-00017.dat field-00024.dat field-00031.dat field-00038.dat field-00004.dat field-00011.dat field-00018.dat field-00025.dat field-00032.dat field-00039.dat field-00005.dat field-00012.dat field-00019.dat field-00026.dat field-00033.dat field-00040.dat field-00006.dat field-00013.dat field-00020.dat field-00027.dat field-00034.dat
スーパーコンピューターでは、
qsub
, qstat
などのコマンドを使って、
計算ノードにジョブを投入することが多いと思います。
こうしたジョブ投入系のコマンドの使い方は、
それぞれのスパコンシステムの利用マニュアルを参照してください。
無事、計算が実行できたら、計算結果をプロットして確認してみましょう。 OpenMHD では、各物理課題ごとに Python (Matplotlib) と IDL の可視化ルーチンを用意しています。 Python は、IPython・Jupyter の2通りの解析・可視化方法に対応しています。
Python で解析・可視化する方法はいくつかありますが、
我々は IPython を使うことをお勧めします。
まず、ipython3
コマンドで作業用の IPython シェルを起動します。
すると、In [番号]
の形式のプロンプトが表れます。
$ ipython3 --pylab
Python 3.12.8 (v3.12.8:2dc476bcb91, Dec 3 2024, 14:43:20) [Clang 13.0.0 (clang-1300.0.29.30)]
Type 'copyright', 'credits' or 'license' for more information
IPython 8.31.0 -- An enhanced Interactive Python. Type '?' for help.
Using matplotlib backend: MacOSX
In [1]:
次にサンプルスクリプト(plot.py
)を実行してみます。
%run -i
は IPython のコマンドで、"%" から入力します。
-i
オプションはファイルを interactive モードで実行することを意味しています
(参考:IPython tutorial)。
なお、plot.py
を途中まで入力してタブキーを押すと、
IPython が plot.py
を含む候補名を補完してくれます。
このようなファイル名補完は IPython の便利な機能の1つです。
In [2]: %run -i plot.py
これを実行すると、下のようなプロット画面が現れるはずです。
もし画面が現れなければ、plt.show()
で明示的に表示させてください。
In [3]: plt.show()
プロットの作成方法は plot.py
ファイルに
Python 言語で記述されていますので、
興味のある方は見てみてください。
このうち、ファイル冒頭の
import openmhd
のところで、OpenMHD データファイル用のライブラリを読み込んでいます。
このライブラリの本体はシンボリックリンクを辿った先の
visualization/python/openmhd.py
にあり、
以下の2種類のサブルーチンを提供しています。
data_read
data_read_from_bigendian
そして、以下の箇所で
データファイル(field/data-00020.dat)を指定して読み込んでいます。
x
, y
は位置座標の配列、
t
は時間、そして、
data
が物理量の配列です。
# reading the data ... x,y,t,data = openmhd.data_read("data/field-00020.dat")
同じルーチンに ix1, ix2, jx1, jx2
(配列の添字番号)や
xrange, yrange
(座標の範囲)などのオプション引数を渡すと、
データの読み込み範囲を絞り込むことができます。
# reading the data (partial domain: [ix1,ix2] x [jx1,jx2]) x,y,t,data = openmhd.data_read("data/field-00020.dat",ix1=0,ix2=100,jx1=11) x,y,t,data = openmhd.data_read("data/field-00020.dat",xrange=[0.0,3.14159],yrange=[0.0,3.14159])
実行環境(スパコン)と解析環境(PC等)でエンディアンが違う場合は、
Python側でエンディアンを変換することができます。
Little endian 環境で Big endian データを読み込む場合は、
data_read_from_bigendian
ルーチンを使ってください。
エンディアン変換以外の動作は data_read
と同じです。
読み込んだデータの配列(data[ix,jx,9]
)は、サイズが ix, jx, 9 の3次元配列になっていて
最後の次元が物理量を表しています。
例えばプラズマ圧力は data[:,:,3]
に入っていますが、
ファイル冒頭のダミー変数を使って data[:,:,pr]
とするとわかりやすいでしょう。
ダミー変数の最後の ps
は、$\nabla \cdot \vec{B}$ 問題対策の仮想ポテンシャルψです。
(「はじめに > 基礎方程式と計算手法」を参照)
IPython シェルでは、引き続きこれらのデータとプロットを操作することができます。 例えば、以下のようにするとカラーバーが赤=青色に変わります。
In [4]: myimg.set_cmap('RdBu_r')
よく使いそうなコマンドは、サンプルファイルにコメント形式で残していますので 試してみてください。
画像をたくさん出力して、1つのムービーにまとめることもできます。
2次元渦問題(2D_basic
)には、
ムービー作成用のサンプルファイル(plot_movie.py)を用意しています。
このサンプルでは、以下のループをまわして連番ファイルを作ります。
その際、movie
というディレクトリを自動生成しますがご了承ください。
In [5]: %run -i plot_movie.py
他の物理問題の場合は、このサンプルを参考にスクリプトを用意してください。
# ---- loop for movies --------------- for ii in range(0,41): # reading the data ... x,y,t,data = openmhd.data_read(ii)
Python の range(0,41) は 0,1,...40 を表すことに注意しましょう。
実行し終えたら movie/
ディレクトリの中を確認してみましょう。
$ ls movie/ output-00000.png output-00011.png output-00022.png output-00033.png output-00001.png output-00012.png output-00023.png output-00034.png output-00002.png output-00013.png output-00024.png output-00035.png output-00003.png output-00014.png output-00025.png output-00036.png output-00004.png output-00015.png output-00026.png output-00037.png output-00005.png output-00016.png output-00027.png output-00038.png output-00006.png output-00017.png output-00028.png output-00039.png output-00007.png output-00018.png output-00029.png output-00040.png output-00008.png output-00019.png output-00030.png output-00009.png output-00020.png output-00031.png output-00010.png output-00021.png output-00032.png
そして、次の4つの方法のどれかを使って、 バラバラの画像ファイルを1つのムービーにまとめます。
最近では Jupyter ノートブック で Python の解析を行うことができます。 Jupyter がインストールされていない場合は、 pip3 で jupyter および ipywidgets ライブラリをインストールしてください。 (ipywidgets は後述のインタラクティブ機能を使うために必要です。これがなくてもその他の基本機能を使うことができます)
$ pip3 install jupyter ipywidgets [ $ pip3 install -U jupyter ipywidgets ]
コマンドラインで次のコマンドを実行すると、 ブラウザの画面で Jupyter ノートブックを見ることができます。
$ jupyter-notebook plot.ipynb
そして、セルを選択して Shift + Enter キーを押すと、
セル内の Python プログラムを実行することができます。
プロット画像はノートブックの中にインライン表示されます。
Python プログラムの中身は、
解析・可視化(Python/IPython) で紹介したものと同じものです。
2次元渦問題(2D_basic
)のノートブックには、
ムービー作成用のサンプルスクリプトも用意しています。
最後に jupyter-notebook
コマンドを実行したターミナルで Ctrl + C キーを押すと、Jupyter を終了させることができます。
idl コマンドで IDL を立ち上げます。
IDL>
が IDL のプロンプトです。
$ idl IDL 8.7.2 (darwin x86_64 m64). (c) 2019, Harris Geospatial Solutions, Inc. Licensed for use by: Kobe University - Zenitani License: ***-******* IDL>
以下のように .r
コマンドでバッチファイル(batch.pro
)を実行してみましょう。
IDL> .r batch
下のようなプロット画面が現れるはずです。
プロットの生成方法は batch.pro ファイルの中に IDL 言語で記述されています。
興味のある方はぜひ見てみてください。
ファイル冒頭の以下の部分で、OpenMHD 専用のルーチンを読み込んでいます。
サブルーチンファイルはシンボリックリンクを辿った先の visualization/idl/data_read.pro
にあります。
;; find and compile data_read routine resolve_routine, "data_read"
そして、適当なデータファイル(field/data-00020.dat)を読み込みます。
x
, y
は位置座標の配列、
t
は時間、そして、
data
が物理量の配列です。
;; reading the data ... data_read,data,x,y,t,'data/field-00020.dat'
同じルーチンに ix1, ix2, jx1, jx2
(配列の添字番号)や
xrange, yrange
(座標の範囲)などのオプション引数を渡すと、
データの読み込み範囲を絞り込むことができます。
;; reading the data (partial domain: [ix1,ix2] x [jx1,jx2]) data_read,data,x,y,t,'data/field-00020.dat',ix1=0,ix2=100,jx1=11 data_read,data,x,y,t,'data/field-00020.dat',xrange=[0.0,3.14159],yrange=[0.0,3.14159])
実行環境(スパコン)と解析環境(PC等)でエンディアンが違う場合は、
data_read.pro
ファイルの中の openr
関数に
/swap_endian
キーワードを追加して、
データ読み込み時にエンディアンを変換して下さい。
; no record marker
openr,/get_lun,unit,filename,/swap_endian
読み込んだデータの配列は3次元配列になっていて、
最後の次元が物理量を表しています。
例えばプラズマ圧力は data[*,*,3]
に入っていますが、
冒頭のダミー変数を使って data[*,*,pr]
のように書くとわかりやすいです。
ムービーを作る場合は、サンプルファイル(batch_movie.pro)を参考に大量の画像を出力すれば良いでしょう。
IDL> .r batch_movie
このファイルを実行し終えると、Python の場合と同様、movie/
ディレクトリの中に画像ファイルができています。
$ ls movie/ output-00000.png output-00011.png output-00022.png output-00033.png output-00001.png output-00012.png output-00023.png output-00034.png output-00002.png output-00013.png output-00024.png output-00035.png output-00003.png output-00014.png output-00025.png output-00036.png output-00004.png output-00015.png output-00026.png output-00037.png output-00005.png output-00016.png output-00027.png output-00038.png output-00006.png output-00017.png output-00028.png output-00039.png output-00007.png output-00018.png output-00029.png output-00040.png output-00008.png output-00019.png output-00030.png output-00009.png output-00020.png output-00031.png output-00010.png output-00021.png output-00032.png
そして、次のどれかの方法で、 バラバラの画像ファイルを1つのムービーにまとめます。
OpenMHD の計算結果は、データ形式さえ合えば Python/IDL 以外の言語・ツールでも可視化できるはずです。可視化ルーチンの開発に役立ちそうな情報を挙げておきます。
visualization/gnuplot
に収録しています。
Python の画像処理ライブラリ Pillow を使うと、
上記の2次元課題で生成した画像ファイルを
1つのアニメーション GIF ファイルに変換することができます。
まず、pip
で Pillow をインストールします。
$ pip3 install pillow [あるいは $ pip install pillow]
そして、以下のスクリプトを IPython などで実行します。 インポート先のパッケージ名が Pillow でなく PIL になっていますので、ご注意ください。
import glob from PIL import Image images = [] for png in sorted(glob.glob("movie/output-*.png")): im = Image.open(png) images.append(im) images[0].save('animation_pillow.gif', save_all=True, append_images=images[1:], optimize=False, duration=100, loop=0)
最後の save() メソッドの duration オプションの数字は、各コマの表示時間(ミリ秒)です。 これを変更すると、動画の再生速度を変更することができます。
その他の詳細は、以下のページを参考にしてください。
コンピュータビジョンライブラリ OpenCV の Python インターフェースを使うと、
バラバラの画像ファイルを MP4 ムービーに変換することができます。
インストール方法はいろいろありますが、
例えば、pip
で opencv-python をインストールすれば良いでしょう。
$ pip3 install opencv-python [あるいは $ pip install opencv-python]
そして、以下のスクリプトを IPython などで実行します。
このサンプルはノートブックファイル(plot.ipynb)に含まれているはずです。
OpenCV のパッケージ名は cv2
です。
import glob import cv2 images = [] for png in sorted(glob.glob("movie/output-*.png")): im = cv2.imread(png) height, width, layers = im.shape images.append(im) out = cv2.VideoWriter("animation_cv2.mp4", cv2.VideoWriter_fourcc(*'MP4V'), 10.0, (width, height)) for i in range(len(images)): out.write(images[i]) out.release()
cv2.VideoWriter
の2つめの引数はコーデック、
3つめの引数は動画の再生レート(fps)、
4つめは動画の大きさ(横・縦)です。
fps の数値を変えて、動画の再生速度を調整することができます。
その他の詳細は、以下のページを参考にしてください。
ImageMagick で
画像ファイルを1つにまとめることができます。
次のコマンドは、上記の2次元課題で生成した movie/
ディレクトリの画像ファイルをアニメーション GIF に出力します。
$ convert movie/output-000*.png result.gif
出力するムービーの形式は、ファイル名の拡張子によって自動判別されます。 例えば
$ convert movie/output-000*.png result.mp4
とすると MPEG4 ムービーを生成できます。 後者の MPEG4 変換機能は後述の FFmpeg を呼び出しているため、FFmpeg のある環境で使えます。
Ubuntu Linux では、これらのツールを以下のようにしてインストールできます。
$ sudo apt install imagemagick $ sudo apt install imagemagick ffmpeg
参考ページ:
FFmpeg でも同様に、
バラバラの画像ファイルを1つにまとめることができます。
次のコマンドは、movie/
ディレクトリの画像ファイルを MPEG4 ムービーに出力します。
GIF アニメーションの場合と比べるとファイルサイズはかなり小さくなります。
(一方、少し画質も荒くなります)
$ ffmpeg -i movie/output-%05d.png result.mp4
-r
オプションで
ムービーの再生レートを調整することができます。
$ ffmpeg -r 10 -i movie/output-%05d.png -r 10 result.mp4
Ubuntu Linux では、以下のようにして FFmpeg をインストールします。
$ sudo apt install ffmpeg
参考ページ:
いちばん利用者の多い磁気リコネクション課題を例に説明します。
以下は 2D_reconnection
ディレクトリの下のファイル一覧です。
一部のファイルはシンボリックリンクになっていますが、
基本的に必要なファイルは同じディレクトリの中から見えるはずです。
ユーザーは * のファイルを編集してプログラムをコンパイル・計算を実行します。
黄色・青色背景のファイルは可視化・解析用のファイル群です。
同様に、ユーザー側で * のファイルを編集してください。
磁気リコネクション課題にはシリアル版・並列版の2つのサンプルが収録されています。 それぞれについて、メインファイル、初期条件、境界条件ファイルと Makefile のコンパイル手順を用意しています。 コードをコンパイルする場合は、以下のようにしてください。
$ make run (シリアル版) $ make runp (並列版)
ファイル名 | 内容 | |
Makefile | * | コンパイルの手順書 |
batch.pro | * | IDL の可視化ルーチン |
bc.f90 | * | 境界条件(シリアル版) |
data/ | データディレクトリ | |
data_read.pro | IDL のサブルーチン(データ読み込み部) | |
fileio.f90 | ファイル入出力(シリアル版) | |
flux_resistive.f90 | * | 電気抵抗の設定と数値流束の計算(その2) |
flux_solver.f90 | 数値流束の計算(その1:LLF/HLL/HLLC/HLLD 法による主要部分) | |
glm_ss2.f90 | ソース項(div B 対策の仮想ポテンシャル ψ) | |
limiter.f90 | Slope limiter:空間方向に物理量を補間する | |
main.f90 | * | メインファイル(シリアル版) |
mainp.f90 | * | メインファイル(並列版) |
model.f90 | * | 初期条件(シリアル版) |
modelp.f90 | * | 初期条件(並列版) |
mpibc.f90 | * | 境界条件(並列版) |
mpiio.f90 | ファイル入出力(並列版) | |
openmhd.py | Python3 のサブルーチン(データ読み込み部) | |
parallel.f90 | 並列通信モジュール | |
param.h | 比熱比、配列番号 | |
plot.ipynb | * | Python3 可視化ルーチンの Jupyter notebook |
plot.py | * | Python3 可視化ルーチン |
rk.f90 | Runge-Kutta 法コード | |
set_dt.f90 | 時間ステップの設定 | |
u2v.f90 | 保存量 (U) から基本量 (V) への変換 |
配布ファイルには、以下の物理課題を収録しています。 対応するディレクトリ名を(括弧)で示しています。
(1D_basic)
(1D_relativisticJet)
(2D_basic)
(2D_KH)
(2D_reconnection,3D_reconnection)
(3D_basic)
1次元リーマン問題は、ある境界(x=0.0)の左右に別々の状態を設定した後で、
次の瞬間に手を離し、系の時間発展を見る基本問題です。
このとき、系は複雑な構造を伴いながら、自己相似的に広がっていきます。
そもそも OpenMHD は、セル境界でリーマン問題を考慮するリーマンソルバーを採用しているわけですから、リーマン問題はコードの精度に直結する最重要テスト問題です。
初期設定ファイル(model.f90
)には、典型的なテスト問題のパラメーターが収録されています。下の図は、Brio & Wu (1988) で議論されている複合波問題です。
基本問題のいくつかは比熱比 Γ=2 で計算します。必要であれば、パラメーターファイル(param.h
)の変数 gamma
を変更してください。また、param.h
ファイルはコード全体で共有されていますので、後で Γ=5/3 に戻すのを忘れないでください。
活動銀河核やガンマ線バースト (GRB) では、相対論的速度でジェットが吹き出していると考えられています。 こうしたジェットと外側の静止プラズマとの境界領域で、 ジェットの膨張に伴う希薄波が発生すると、希薄波領域の内部で、 新しいタイプの流体加速(異常加速)が起きることが Aloy & Rezzolla (2006) によって報告されました。一見不思議なこの現象は、相対論完全流体近似のもとで現れる断熱加速です。 Zenitani et al. (2010) はこの現象を磁気流体の相対論 HLLD シミュレーションで確認・検証し、 運動論的に見るとプラズマ熱速度の広がりに埋もれた バルク速度を見ていることを指摘しています。
Orszag & Tang (1979) の渦問題を収録しています。渦が作り出した衝撃波が飛び交って複雑な模様を作り出すことや、磁場のソレノイダル性($\nabla\cdot\vec{B}=0$)が破れると計算が発散しやすいこともあり、2次元 MHD の標準問題として広く使われています。
磁場のソレノイダル問題の重要性を確認するために、
例えば main.f90
コードの102行目をコメントアウトして、
その後4箇所の flux_solver
の引数 ch をゼロにしてみてください。
こうすると、ソレノイダル問題の補正項の働きがオフになって
非物理的な効果が累積するため、途中で計算が破綻します。
速度差がある2つの領域の境界層では、大きな渦が成長することが知られています。
これは Kelvin-Helmoholtz(KH)不安定 という流体力学の不安定です。
ここでは、Matsumoto & Hoshino (2004) に倣って、磁気流体での KH の渦の成長を見てみましょう。
初期条件ファイル(model.f90
)の中では、以下のような設定を用いています。
上下 (y) 方向には、上側が左向き、下側が右向きの速度勾配を tanh 関数で設定しています。
さらに、下側密度に相当するパラメーターαを使って、上下に密度差を入れています。
α=1 のとき密度は一様で、α=0.2 のとき上下に 5 倍の密度差ができます。
磁場は面に垂直な方向を向いています。
解像度は、最初に計算を試す場合は 160 [X] x 200 [Y] で良いと思います。 その後、細かな構造を見る場合は 320 [X] x 400 [Y] 以上に設定してください。 下図は、解像度 480 [X] x 600 [Y], α=0.2 の計算結果です。 巻き込んだ渦の境界面に、2次的な構造(K-H 不安定あるいは R-T 不安定)が出来ています。
磁気流体中の KH 不安定は、磁場の向きと不安定の波数方向の関係によって、 成長の仕方が大きく変わります。例えば、初期条件に以下のように変えてみましょう。 面内にある磁場は張力を生じて、渦の上下方向の運動にブレーキをかけようとします。 渦の成長がどう変わるのか、計算して確かめてみましょう。
U(i,j,bx) = 0.d0 U(i,j,bz) = 0.d0
U(i,j,bx) = 0.2d0 U(i,j,bz) = 0.d0
この $B_x$ の値を大きく/小さくすると、磁場によるブレーキ効果も強く/弱くなります。 渦を止めるのに必要な $B_x$ の大きさは何で決まるでしょうか?
宇宙空間プラズマ中で磁場が反転する電流層では、 磁力線が激しくつなぎ変わる「磁気リコネクション」が起きます。 磁気リコネクションは、理想 MHD 近似では起きませんが、 オームの法則に電気抵抗を取り入れた抵抗性 MHD 近似では扱うことができます。 OpenMHD には、Zenitani & Miyoshi (2011, 2015) で使った磁気リコネクション問題のコードの改良版を収録しています。 初期条件は、以下のような磁場反転構造です。
\begin{align} \vec{B}(y) &= B_0 \tanh(y/L) {\hat{x}},\\ \vec{v} &= 0,\\ \rho(y) &= \rho_0 \big[ 1 + \cosh^{-2}(y/L)/ \beta_{\rm in} \big], \\ p(y) &= p_0 \big[ \beta_{\rm in} + \cosh^{-2}(y/L) \big], \end{align}この $\beta_{\rm in}$ は、プラズマベータ(磁気圧に対するプラズマ圧力の比)を上流のインフロー領域で評価したものです。簡単のため、系全体で温度($T=p/\rho$)を一定にしています。磁場が弱い真ん中(y ~ 0)付近では磁気圧が働かないため、代わりにプラズマの圧力を増やして全体構造を支えています。このような構造を「プラズマシート」あるいは「電流シート」と呼びます。この課題では (1) 原点付近の電気抵抗を強くしたうえで (2) 擾乱ポテンシャルを用いて 3% 程度の弱い磁場を追加して、リコネクションを起こしています。
\begin{align} \eta(x,y) &= \eta_0 + (\eta_1-\eta_0)\cosh^{-2} ( \sqrt{x^2+y^2} ) \\ \delta A_z &= 0.03 \times 2LB_0 \exp[-(x^2+y^2)/(2L)^2 ] \end{align}
いくつかファイルがあるうち、main.f90
がシリアルコード、mainp.f90
が並列コードです。系の対称性を利用して、第一象限のみを計算して全体の計算量を節約しています。
上の図は、デフォルトの設定で計算した磁気リコネクションの横方向の速度($V_x$)です。 リコネクション点からは高速のプラズマジェットが吹き出していて、その速度は簡単な議論から上流プラズマの Alfven 速度程度であることがわかっています。 ジェットの(音速に対する)マッハ数が1を超えた場合、下流側に複雑な衝撃波や不連続面を作ります。 こうした不連続面構造をまとめたものが、下の2016年版「プラズモイド・ダイアグラム」(解説記事)です。
これらは (1) リコネクションが作り出す固有の構造と (2) 周辺環境との相互作用が作り出す構造に分けられると思いますが、 両者を切り分けるにはどうしたら良いでしょうか? さらに面外磁場成分 $B_z$ があるとき、これらの構造はどう変わるでしょうか?
MHD リコネクションは40年以上にわたって研究されている分野ですが、 未解明の基本問題が多く残っています。 MHD コードがあれば少なくとも数値結果が出ますから、 著者の腕次第で研究成果を出すことは可能です。 OpenMHD コードが皆様の研究をアシストできれば幸いです。
Gardiner & Stone (2008) で提案された、斜めの磁気ループを移流させるテスト問題です。 多次元でのコードの性能、特に接線不連続面(tangential discontinuity)の分解能力と安定性を 調べるためによく使われます。以下のような初期条件を用います。
\begin{align} \rho &= 1, \\ p &= 1, \\ \vec{v} &= (1,1,2) \\ A_3(r) &= 0.001 ~ {\rm max}( 0.3 - r, 0 ) \\ r &\equiv \sqrt{x^2_1+x_2^2} \end{align}A はベクトルポテンシャルで、添え字の 1,2,3 は斜めに傾いた系での座標を表しています。 磁場 B は $\vec{B} = \nabla \times \vec{A}$ から計算します。 詳細は原論文を参照してください。
この設定では、プラズマベータが高く磁場自体はほとんど圧力を持っていません。 磁場は全体の流れに乗って流されるだけになりますから、 元のループ構造が良く保たれるかどうかは数値解法の性能で決まります。 下の図は、一定時間後の磁気エネルギーの大きさの空間分布です。
簡単のため、Δx = Δy = Δz に固定しています。 比率を変えるためには、Runge=Kutta ソルバーやタイムステップ部分など、 何箇所かコードを修正する必要があります。
Lorentz-Heaviside 系(Heaviside-Lorentz 系)を使っています。馴染みのない方は、東北工業大学 中川先生の解説を参考にしてください。
$\mu_0$ や $4\pi$ などが含まれない Lorentz-Heaviside スタイルの表式を使っています。 MHD方程式には光速cが含まれないので、$c=1$となるLorentz-Heaviside単位系とは少し異なります。
X ウィンドウをリモートで表示すると、 OpenGL のハードウェア加速機能が効かないために 画面が真っ黒になってしまうことがあります。 この場合は、IDL をソフトウェアレンダリングモードで動作させる必要があります。
IDLを起動するときに、以下のオプションを追加するか、
$ idl -IDL_GR_X_RENDERER 1
あるいは、IDL の image
関数に
renderer=1
というオプションを追加すると良いでしょう。
このオプションは、付属プログラムのコメント部分に記述しています。
;; 2D image myimg = image(data[*,*,vx],x,y,axis_style=2,xtitle='$X$',ytitle='$Y$',xtickdir=1,xticklen=0.02,ytickdir=1,yticklen=0.01,font_size=16,rgb_table=13,dimensions=[1000,500]) ;; ,renderer=1) ;; use software rendering over a remote connection.
Ubuntu Linux であれば、以下のライブラリを追加インストールしてください。 CentOS の場合は、西田さんのメモが参考になると思います。
sudo apt install libxp6
ノートブックの先頭付近に以下の行を追加して グラフのデフォルトサイズを変更することができます。 (横、縦)をインチ数で指定します。 この plt は matplotlib.pyplot のインポート名です。
plt.rcParams['figure.figsize'] = (6.4, 4.8)
Matplotlib 公式サイトにカラーマップ一覧があります。
シリアル版と同様、1つのファイルにデータを書き出します。
ファイル先頭付近の変数を io_type = 0
に変更すると、
並列プロセス毎にファイルを書き出します。
自作プログラムでデータを加工する場合は、
各ファイルに1グリッドずつののりしろが含まれていることを考慮してください。
ぜひ自分の手でプログラミングしてみてください。 PythonもIDLも汎用言語ですから、思いのままの解析ができるはずです。
はい。のりしろデータをノード間で転送したあと、 MUSCL補間した基本量をさらにノード間転送しています。 この過程で2グリッド先の情報が伝わっています。
以下の箇所です。
main.f90, mainp.f90, main.cuf, mainp.cuf
)
---
グリッド数、並列数、数値解法の選択などを計算全体の設定を定義しています。
model.f90, modelp.f90
)
---
空間座標や物理量の値を定義しています。
以下の変数で、X方向の始点・終点と、Y方向の始点の位置を指定します。
Y方向の終点はグリッド数から自動計算されます。
! x & y positions (Note: domain_y(2) is automatically calculated) real(8), parameter :: domain_x(2) = (/0.d0, 200.d0/) real(8), parameter :: domain_y(1) = (/0.d0/)
また、磁気リコネクション問題の電気抵抗 $\eta$ は
resistive flux ルーチン(flux_resistive.f90
)で
定義しています。
eta
関数:電気抵抗の関数形を定義しています。
set_dt_resistive
ルーチン:時間ステップを制限します。
過去のバージョンでは、以下の箇所に分散して電気抵抗 $\eta$ を設定していました。
set_eta.f90
)
main.f90, mainp.f90, main.cuf, mainp.cuf
)
冒頭の Rm1, Rm0
パラメーター
シリアル版では bc*.f90
、並列版では mpibc*.f90
で
境界条件を設定しています。
(mpi)bc_for_U
ルーチン
---
保存量(U)の境界条件を設定します。
(mpi)bc_for_F
ルーチン
---
X方向にMUSCL補間した基本量(VL, VR)の境界条件を設定します。
これをX方向の数値流束 F の計算に利用します。
(mpi)bc_for_G
ルーチン
---
Y方向にMUSCL補間した基本量(VL, VR)の境界条件を設定します。
これをY方向の数値流束 G の計算に利用します。
(mpi)bc_for_H
ルーチン
---
Z方向にMUSCL補間した基本量(VL, VR)の境界条件を設定します。
これをZ方向の数値流束 H の計算に利用します。
並列版では、メインファイル(mainp.f90, mainp.cuf
)先頭付近の
bc_periodicity
変数を、
周期境界では .true.
、そうでない場合は .false.
にセットして下さい。
以下は、X方向を周期境界、Y方向を別の境界条件を利用する例です。
これを .true.
にした場合、その方向の周期境界の処理は通信モジュールが行います。
境界条件ファイル(mpibc.f90
)の mpibc_for_U, mpibc_for_F, mpibc_for_G
の中で
同じ処理を書く必要はありません。
logical, parameter :: bc_periodicity(2) = (/.true., .false./)
仮想ポテンシャル $\psi$ の境界条件は特に注意してください。 値の小さな補正項なので符号を間違えても動いてしまいますが、 エラーを累積した後で結果を変えるので厄介です。
並列通信モジュール(parallel.f90
)の以下の箇所を
use_shm = .true.
に変更すると、
ノード内通信を共有メモリのデータコピーで代用します。
これは実験的な機能です。
! ----- MPI-3 shared memory communication ------------------------ logical, parameter, private :: use_shm = .false. ! MPI communication ! logical, parameter, private :: use_shm = .true. ! MPI-3 SHM model (experimental)
はい、波に乗って運ばれるのは特性変数ですから、 特性変数を補間した方が良いでしょう。 しかし、物理量を特性変数に変換する計算コストがかなり大きいため、 OpenMHDでは基本変数の補間で代用しています。 2次精度では特性変数 vs 基本量の違いはあまり気にならないと思います。
なるべく計算を軽くし、代わりにグリッド数を増やして 実質精度を上げることを目指しています。
デフォルトでは最も無難な minmod リミターを採用しています。 minmod, van Leer, MC, Koren のうち、 下線 のものを条件文(if 文)を使わずに実装しています。 GPU 環境では下線のものを選んだ方が良いでしょう。
HLLD 法は、リーマンファンの中に Alfven speed で広がる回転不連続面(RD)を考え、 さらにその外側に fast speed で広がる外側境界を考えます。 低ベータ(β≦1.2)プラズマでは、 教科書やこちらのプロットで確認できるように、 磁力線方向の Alfven speed は fast speed とほぼ同じです。 そのため沿磁力線方向では、片方の RD が外側境界を追い越してしまって、 HLLD 法の4状態の仮定が成立しなくなることが稀にあります。 これは、HLLD 論文 [1] の p.326 冒頭で言及されている 0/0 問題に関係しています。
OpenMHD は、RD と外側境界が接近して HLLD 法の仮定が破れそうになると、 RD を取り払って HLLC 法にスイッチします。
HLLD ソルバーは非線形ソルバーなので、ガリレイ不変ではありません。
できません。計算が止まります。
はい。例えば U(i,j,en)
と U(i,j,4)
は全く同じです。
この en
は、コードを読みやすくするために使っている定数で
param.h ファイルで定義しています。
磁場の発散部分($\nabla \cdot \vec{B}$)に溜まる数値誤差を 拡散・減衰させるための仮想ポテンシャル($\psi$: psi)です。 物理的な意味はありません。 (「はじめに > 基礎方程式と計算手法」参照)
デフォルトのリコネクション問題(2D_reconnection/mainp.f90
)では、
Intelコンパイラ(ifort)でコンパイルしたプログラムを
ブーストクロック 5.0 [GHz] の Intel i9 プロセッサ4並列(4コア)で走らせると、
23分(HLL)〜28分(HLLD)かかりました。
他の CPU/GPU での計測結果は以下のとおりです。
2次元シミュレーションの計算コストは解像度の3乗に比例しますから、
PCやGPUの性能に応じて計算サイズと解像度を調整すれば良いでしょう。
Intel i9-9900K プロセッサ・MPI 4並列 | 28分 |
AMD EPYC 7742 プロセッサ・MPI 4並列 | 24分50秒 |
AMD EPYC 7742 プロセッサ・MPI 64並列 | 1分50秒 |
NVIDIA Tesla T4 グラフィックスカード(Google Colab) | 5分 |
NVIDIA GeForce RTX 2080 グラフィックスカード | 2分10秒 |
NVIDIA RTX A6000 グラフィックスカード | 1分50秒 |
NVIDIA GV100 グラフィックスカード | 1分20秒 |
NVIDIA A100 グラフィックスカード | 35秒 |
HLLDコードをお使いの場合は、文献 [1] と、GitHub の URL (https://github.com/zenitani/OpenMHD) あるいは Astrophysics Source Code Library のエントリー番号 (ascl:1604.001) を紹介してくださると嬉しいです。 ドキュメントの多くを [1] の一環で整備していますが、 [1] にはコードの入手先(特にURL)の情報がありません。 そこで、別途、GitHub or ASCL の情報を追加してくださると、 読者がこのページに到達することができるようになります。 文例を以下に示します。 GitHub の場合は、本文中に URL があるだけでも良いと思います。
Physics of Plasmas スタイルの文例:
S. Zenitani, Phys. Plasmas, 22, 032114 (2015). S. Zenitani, Astrophysics Source Code Library, record ascl:1604.001 (2016).
ApJ スタイルの文例:
Zenitani, S. 2015, PhPl, 22, 032114 Zenitani, S. 2016, OpenMHD: Godunov-type code for ideal/resistive magnetohydrodynamics, Astrophysics Source Code Library, ascl:1604.001
このページの URL (https://sci.nao.ac.jp/MEMBER/zenitani/openmhd-j.html) は 作者の異動やサーバー環境の都合で変わる可能性があります。 変更の可能性が少ない GitHub (https://github.com/zenitani/OpenMHD) あるいは Astrophysics Source Code Library (ASCL) (https://ascl.net/1604.001) の URL を紹介してくださればと思います。
作者の好みです。Δt/Δx > 0.5 に設定すると、 両側のセル境界からやってきた波が真ん中でぶつかり、直感に合わないと思うのです。 Toth (2010) の教科書の pp.216--218 によると Δt/Δx<1.0 で良いそうですが、私はどうも納得がいきません。
一部のコードには、メインファイル先頭近くに n_start
という変数があります。
この変数を
0
にすると計算を最初から実行、
1
以上にすると1番目のデータファイルを読み込んで計算を再開、
-1
にするとファイル番号を予想して計算を再開します。
! If non-zero, restart from a previous file. If negative, find a restart file backword in time. integer :: n_start = 0
HLLD フラックスの部分です。 (ベンチマーク結果)
後述の OpenMP 並列が思いがけず有効になっていることがあります。 コンパイラのオプションをよく確認してください。
Makefile のコンパイラ引数に
Intel fortran では -qopenmp
、
gfortran では -fopenmp
オプションを追加してください。
そして、プログラムを実行する際に、環境変数 OMP_NUM_THREADS
を使って並列スレッド数を設定します。
$ export OMP_NUM_THREADS=4 $ ./a.out
全体の環境変数を変えたくなければ、以下でも良いでしょう。
$ OMP_NUM_THREADS=4 ./a.out
これに関連して、Cray fortran でフラット MPI を使う場合、
ユーザーの意図に反して OpenMP がオンになってしまうことがあります。
その場合は、Makefile でコンパイラに -hnoomp
オプションを渡して
OpenMP を明示的にオフにしてください。
F90 = ftn -hnoomp
segmentation fault...
でプログラムが異常終了する。
以下のどちらかのコマンドを使って、 シェルのスタックサイズの制限を解除してください。
$ ulimit -s unlimited % unlimit stacksize
Intel Fortran では -heap-arrays
オプションを
使うことができるそうです。
ログファイルを参考にしてください。 スーパーコンピューターで計算ジョブが実行されると、 プログラムの出力とエラー出力を記録したログファイル(テキストファイル)が、 それぞれ、同じディレクトリ内に生成されます。 ファイル名はスパコンによって違います。
UNIX の time コマンド を使っています。 並列計算では、CPU 時間に含まれない ネットワークの遅延時間がかかりますので、 全てを含めた real タイムを見てください。 MPI 並列と OpenMP 並列を併用する場合、 最適のノード数・スレッド数はケース・バイ・ケースで変わりますから、 自分で検証してみると良いと思います。
$ time ./a.out $ time mpirun -np 16 ./ap.out $ OMP_NUM_THREADS=16 time ./a.out
Cray コンパイラでMPIを使う場合は、
以下のように -hnoomp
オプションを追加します。
F90 = ftn -hnoomp
例えば、お試し用の XC-trial カテゴリで 120 コアを使う場合は、
次のようなジョブスクリプト(test.sh
)を用意します。
#PBS -N jobname #PBS -l nodes=3 # 使用ノード数 #PBS -j oe #PBS -q large-t cd ${PBS_O_WORKDIR} aprun -n 120 -N 40 ./ap.out > output.log # end
そして、qsubコマンドとジョブスクリプト(test.sh
)を使って
ジョブを投入してください。
$ qsub test.sh
WikiBookや UNIX 標準テキスト、 UNIX/Linux の基礎知識(京大) などを参考にして、少しずつ慣れていってください。
Linux, macOS のターミナル、Windows 10(バージョン1803以降)のPowerShellで ssh コマンド を使ってください。
Linux, macOS, Windows 10(バージョン1803以降)では scp
コマンドを使うことができます。
Linux, macOS では rsync
コマンドを使うこともできます。
macOS の Fetch(学生・教育関係者は無料)、Windows の WinSCP、
Linux/macOS/WindowsのFileZillaなど、GUIの sftp クライアントソフトも便利です。
Emacs, Vim, Smultron なら全ての言語に対応しています。Visual Studio Code (VS Code) は Python extension に加えて、Marketplaceで Modern Fortran および IDL: Interactive Data Language 対応モジュールを追加することができます。macOS では BBEdit と CotEditor が Fortran 90, Python に対応しています。
screen コマンド を使うと、
途中画面を出力したままログアウト(デタッチ)して、後で再び接続(アタッチ)することができます。
screenは、デフォルトでは少し使いにくいので、以下のサイトを参考にカスタマイズしても良いと思います。
screenの設定は .screenrc
ファイルを編集して行います。
Gravity wave と gravitational wave と同じぐらい紛らわしいのは承知しています。 最後まで目を通してくださり、ありがとうございます。
Ubuntu / Debian では、以下のような apt コマンドで fortran コンパイラ(gfortran)やMPIライブラリ(Open MPI, MPICH)を インストールすることができます。
$ sudo apt install gfortran $ sudo apt install openmpi-bin libopenmpi-dev # MPIライブラリに Open MPI を使う場合 $ sudo apt install mpich libmpich-dev # MPICH を使う場合
とします。
CentOS / RedHat をお使いの方は、代わりに yum を使います。 gfortran や MPI も以下の手順でインストールできます。
# yum install gcc-gfortran # yum install openmpi openmpi-devel # MPIライブラリに Open MPI を使う場合 # yum install mpich mpich-devel # MPICH を使う場合
また、これらの Ubuntu(CentOS も?)標準の方法では、 gfortran, gcc にバインドされたMPIライブラリがインストールされます。 例えば Intel fortran など、別のコンパイラにバインドする場合は、 後述の「MPIライブラリのインストール」を参考に自力でMPIライブラリをインストールしてください。
Intel の Web サイト から、 Intel oneAPI Base Toolkit と HPC Toolkit をダウンロードしてインストールします。
そして、ターミナルで setvars.sh スクリプトを走らせれば、
コンパイラ(ifort
)や並列コンパイラ(mpiifort
)を使う準備が整います。
$ . /opt/intel/oneapi/setvars.sh :: initializing oneAPI environment ... bash: BASH_VERSION = 5.2.21(1)-release args: Using "$@" for setvars.sh arguments: :: advisor -- latest ... :: oneAPI environment initialized ::
NVIDIA の Web サイト から、 NVIDIA HPC SDK をダウンロードしてインストールします。
$ tar xpzf nvhpc_2024_2411_Linux_x86_64_cuda_12.6.tar.gz $ cd nvhpc_2024_2411_Linux_x86_64_cuda_12.6/ $ sudo ./install
そのうえで、/opt/nvidia/hpc_sdk/Linux_x86_64/2024/compilers/bin, /usr/local/cuda/bin
などに PATH を通せば良いでしょう。これで、コンパイラ(nvfortran
)を使う準備が整います。
$ nvfortran -V nvfortran 24.11-0 64-bit target on x86-64 Linux -tp znver3 NVIDIA Compilers and Tools Copyright (c) 2024, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
最近の Linux には Python 3 系が 最初からインストールされています。 これに加えて matplotlib, ipython3 などをインストールすれば良いと思います。 Ubuntu / Debian では、以下のように apt コマンドを使えば良いようです。
$ sudo apt install python3-dev python3-matplotlib ipython3
また、日本 Python ユーザー会による 環境構築ガイドも 参考になると思います。
Windows Subsystem for Linux をインストールすると、Linux 環境を作ることができます。Linux で OpenMHD が動作します。
未確認情報ですが、CUDA on WSL 2 と NVIDIA HPC SDK を使うと、GPU計算もできるかもしれません。
XCode は Apple 純正の開発環境で、 C コンパイラや make などの開発ツールが一通り入っています。 OS 付属のディスクあるいは AppStore からダウンロードして インストールしてください。 XCode 付属の C コンパイラや XCode がインストールするライブラリのバージョンが古いと、 gfortran を使ったプログラムのコンパイル・実行に影響が出ることがあります。 既に XCode をインストールしている方も、 いちどバージョンを確認して、なるべく最新版のものをインストールして下さい。 (例えば本稿執筆時点の XCode の最新バージョンは 16.2 です。)
XCode をインストール後、開発のためのコマンドラインツール(command line tools)を追加インストールする必要があります。次のコマンドを試してみてください。
$ xcode-select --install
コマンドラインツールのインストールが始まりますので、 ダイアログの指示に従ってください。 以下のようなメッセージが出る場合は、 既にコマンドラインツールがインストールされていますので それ以上の操作は必要ありません。
xcode-select: error: command line tools are already installed, use "Software Update" to install updates
gfortran は、Homebrew や MacPorts といった UNIX パッケージ集でインストールすることができます。 gfortran を単体でインストールするには、 以下のどちらかのサイトからファイルを入手することをお勧めします。
最近の macOS では、前者のパッケージを使うほうが無難だと思います。 後者を使う場合は、アーカイブファイルを以下のように手動で展開します。
$ sudo tar xvf ~/Downloads/gfortran-9.2-bin.tar.gz -C /
gfortran でプロダクト・ラン(本計算)を走らせる場合は、
必ず -O2
以上の最適化オプションを使うようにしてください。
このオプションをオンにすることで、市販のコンパイラと同程度まで
プログラムが速く動作するようになります。
また、gfortran は OpenMP に対応しています。
2〜8 程度のコア数であれば、後述の Open MPI をインストールしなくても
並列計算を行うことができます。
OpenMP を使う場合は、Q & A でも説明しているように
コンパイラオプション -fopenmp
を使ってください。
OpenMHD では可視化に Python 3 + matplotlib を用います。 macOSのデフォルトのpythonはバージョンが古いことが多いので、 Homebrew や MacPorts といった UNIX パッケージ集、あるいは Anaconda などのディストリビューションを使って、 Pythonや周辺ライブラリをインストールした方が良いでしょう。
ここではPython公式サイトのパッケージを使う方法を紹介します。 以下のサイトから Python 3.12 パッケージをダウンロードしてインストールしてください。
次にPython標準のパッケージ管理システム pip
コマンドを使って
Python のライブラリをインストールしていきます。
例えば matplotlib をインストールしてみましょう。
$ pip3 install matplotlib
システムによっては、コマンド名は pip
かもしれません。
適宜、pip3
を pip
と読み替えてください。
さらに追加して、IPython という高機能シェルをインストールすることをお勧めします。 IPython では、タブキーによる変数・ファイル名補完などが効きますので、 便利に作業できるようになります。
$ pip3 install ipython
可視化に Jupyter, Pillow, OpenCV を使う場合は、 pip でインストールしてください。
$ pip3 install jupyter $ pip3 install pillow opencv-python
インストール済みのライブラリをアップデートする場合は、 -U オプションを使います。
$ pip3 install -U matplotlib
以下のように IPython シェルを立ち上げてみましょう。 matplotlib がインストールできているかどうか、確認してみてください。 最後に exit で終了しています。
$ ipython3 --pylab Python 3.12.8 (v3.12.8:2dc476bcb91, Dec 3 2024, 14:43:20) [Clang 13.0.0 (clang-1300.0.29.30)] Type 'copyright', 'credits' or 'license' for more information IPython 8.31.0 -- An enhanced Interactive Python. Type '?' for help. Using matplotlib backend: MacOSX In [1]: print(matplotlib.__version__) 3.10.0 In [2]: exit
コア数の多いスーパーコンピューターでは、 MPI並列(フラット MPI)、あるいは MPI と OpenMP を併用したハイブリッド並列が有効です。 PC・ワークステーションにもMPIライブラリをインストールすれば、 スパコンと同じ並列条件でプログラムを開発・実行することができます。 MPIライブラリは、使用するコンパイラにあわせてインストールする必要があります。 そこで、以下のサイトからMPIライブラリのソースコードをダウンロードして、 自分でコンパイル&インストールすることをお勧めします。
ここでは、gfortran用の Open MPI を /opt/openmpi64g
、
Intel fortran 用の Open MPI を /opt/openmpi64i
... のように
それぞれ別のディレクトリにインストールすることを考えます。
コンパイラが1種類だけであれば、以下のコマンド例の
--prefix=
オプションを指定しなくても結構です。
その場合は、/usr/local
にファイルがインストールされます。
/opt/openmpi5
ディレクトリに
gfortran用の Open MPI をインストールするには以下のようにします。
configure と make には時間がかかりますが、気長に待ってください。
make は並列コンパイルに対応しています。
作業を早く終わらせたい場合は、-j
オプションを使ってください。
$ tar jxvf openmpi-5.0.6.tar.bz2 $ cd openmpi-5.0.6 $ ./configure --prefix=/opt/openmpi5 $ make [ $ make -j 8 ] $ sudo make install
以下は、/opt/mpich
ディレクトリに MPICH をインストールした手順です。
必要に応じて ./configure --prefix=/opt/mpich FC=__my_compiler__
のようにして
FC オプションでコンパイラ名を調整してください。
(私の環境では mpich 4.1 ではプログラムをコンパイルできませんでしたが、4.2 では問題ありませんでした。)
$ tar zxvf mpich-4.2.3.tar.gz $ cd mpich-4.2.3 $ ./configure --prefix=/opt/mpich $ make [ $ make -j 4 ] $ sudo make install
最後に、mpirunコマンドを実行して、 MPIライブラリのバージョン情報を確認してください。
$ /opt/openmpi5/bin/mpirun --version mpirun (Open MPI) 5.0.6 Report bugs to http://www.open-mpi.org/community/help/
$ /opt/mpich/bin/mpirun --version HYDRA build details: Version: 4.2.3 Release Date: Wed Oct 2 09:35:21 AM CDT 2024
ifort, mpiifort
から ifx, mpiifort -fc=ifc
に変更しました。flux_resistive_uni
)を追加しました。flux_glm
)をHLLDソルバー(flux_solver
)に統合しました。data_read
)に xrange, yrange
キーワードを追加しました。flux_resistive
)に重大なミスがありました。(名下宥佑さんに感謝)glm_ss2, flux_resistive
)の引数を変更。ipywidgets
によるインタラクティブ操作にも対応。(和田智秀さんに感謝)data_read
で、ファイル番号に加えてファイル名を指定できるようにしました。n_start
< 0 のとき、ファイルを自動検索します。OpenMHD コードのノウハウを共有したり、諸々の問題を解決するための メーリングリストを設けています。 良かったらこちらに問い合わせてみてください。
また、作者宛に直接、メールをいただいても結構です。 サポート・相談の必要度合いによっては、共同研究という形で進めた方が良い場合もあります。 その場合は、共同研究を申し込んでくだされば幸いです。
2024年12月時点では、国内外の利用者グループから 以下の研究論文(査読付き)が出版されています。
文献 [7,8,11] は OpenMHD をベースに開発した double-polytropic MHD コードを使っています。コードを使ってみたい方は個別にご連絡ください。
以下の修士/卒業研究に OpenMHD を利用していただいています。