このブログは、株式会社フィックスターズのエンジニアが、あらゆるテーマについて自由に書いているブログです。
みなさん、今日も元気にGPGPUしていますか?
去年(SC15)の話ですが、「RadeonでCUDAが使えるようにするよ!」とAMDが発表したニュースを覚えている方いらっしゃいますでしょうか。Boltzmann Initiativeの話です。
あれからしばらく時が流れ、機は熟しました。「CUDAはNVIDIA専用」そんな時期は今は昔。そう、CUDAをAMDのGPUであるRadeonで動かすことに成功しました!!
ので、ここではその方法と、その時に使ったコードを紹介したいと思います。
予めお断りしておくと、表題にもある通り、今回は導入編ということで「動くことを確認する」までになります。
そのうち時間を見つけて「性能測定編」もやってみたいと思いますが、ぜひこれを読んだ方、ぜひ性能測定してみてください&結果を教えて下さい!
また、この情報は、本記事執筆時点(つまり2016年7月21日現在)の話です。
Boltzmann Initiativeは現在も活発に進行中で、未来では手順が変わったりする可能性が十分にありますので、ご了承ください。
説明に入る前に、最初に参考文献を挙げておきます。
本記事はこれらの文献のまとめになりますので、手順書さえあればできるという方・最新かつ正確な情報がほしい方は、以下を参照してください。
使うのはROCmというプロジェクトです。
ROCmとは、Radeon Open Computeの略のようです。
ROCmはいくつかのプロジェクトのプロジェクトの集合体となっていますが、今回直接的に使うのは以下の3つです
Kernel Fusion Driver(KFD)とは、ROCmの各種プロジェクトを動かすための「Linuxカーネルと一体となったドライバー」のことです。
なので、後述の手順で説明しますが、カーネルを入れ替える必要があります。
個人的にはkmodでどうにかしてほしいんですが・・・。
ともかく、このKFDを入れることで、AMD Radeonの上で様々なことができるようになります。
HCC compilerは、KFDの上で動く、異種混合実行環境向けのC++コンパイラーです。
HCCとは、Heterogeneous Compute C++の略のようです。
現在のところ、バックエンドとしてはAMD独自のAMD GCN ISAを生で出力するモードと、HSA財団の策定したHSAILという中間言語を出力するモードがあります。
ただし、今回はRadeonを使うので前者で十分なことと、既定で前者が使われること、加えてそもそも後者はこの先開発はされないようなので、ここではAMD GCN ISAを使うモードを前提として話を進めます
(とは言っても、バックエンドが何であるかはアプリケーション層ではあまり気にする必要はなさそうですので、よく分からない方は飛ばしても大丈夫です)。
また、フロントエンドにはC++17 Parallel STL、C++ AMP、OpenMP offload、HC C++ APIなどが用意されています。
今回は説明しませんが、C++ AMPをLinuxで動かしたい場合は、このHCC compilerを使うと動かせるようになります。
HIPは、CUDAをHCCでコンパイルできるようにするプラットフォームです。今回の話の主役になります。何の略称かは分かりません・・・。
HIP自体は、ほぼCUDAと同じ形式のRuntime APIやカーネル言語を既定するもので、基本的にcuXXXXをhipXXXXに置き換えたものです(例えばcuMemcpy
はhipMemcpy
に対応します)。
HIPで書かれたものは、hipccというコンパイラによって解釈され、バックエンドであるnvccまたはhccに渡すことで、NVIDIAでもAMDでも動かせるようにできます。
このhipccは、CUDAあるいはHC言語そのものへの直接的な翻訳機に近い動きをするので、HIPで書くことにオーバーヘッドなどはなく、性能劣化することは全くないらしいです。
また、hipifyという、CUDAからHIPへ(逆)変換するツールが付属しており、これを用いることで既存のCUDAコードを簡単にHIPが解釈できるようになります。
ということで、このHIPを信じるなら、CUDAで書いたコードを、性能劣化させることなくRadeonでも動く形にすることができる、ということになります。
ということで、実際に動かしてみましょう。
ROCmのREADMEやROCm Hardware Requirementsに情報が散乱しているのですが、ROCmを動かすには、現在のところ以下の環境が必須です。
今回は、Ubuntu 14.04、Intel Core i7-4790、AMD Radeon R9 FuryXの環境で試しました。
また、カーネルを入れ替えるという性質上、クリーンインストールすることにしました。
既存のシステムの上に入れても動かないわけではないと思いますが、何かあった時にサルベージが大変なので・・・。
あとこれ重要なのですが既存のAMDのプロプライエタリ・ドライバーとは衝突してまともに動かなくなるっぽいです。
公式ブログのコメント欄でしれっと書かれているのですが、
KFDでドライバーも提供されているので、ROCm導入時には既存のプロプライエタリ・ドライバーは除去する必要があります。
つまり、ROCmを入れるとOpenCLは使えなくなるということです(Issue参照)。
将来的にはサポートされるかもしれませんが・・・。
ということで、既存の環境をかなり破壊することになるので、試される方は新しいストレージを持ってきて、クリーンインストールして始めることをオススメします
(これで安全にはなりますが、カーネルの入れ替え等に限らず、ここに書いてあることは全て自己責任でお願いします)。
計算機を用意して、OSをクリーンインストールしたら、まずは普通に起動します。
起動したら、とりあえずまずインストールされているパッケージを最新に更新します。
これはセキュリティーリスクもそうですが、今回用意した環境だと後でバージョン違いでやり直しという羽目になったので、ここで一度更新しておくことをオススメします。
Ubuntuならapt-get upgrade、Fedoraならdnf updateですね。
その後、ビルド済みバイナリをAMDのレポジトリから持ってきます。
Ubuntuの場合と
Fedoraの場合の両方が用意されています。
今回はUbuntuを使ったので、以下のコマンドを実行しました。
$ wget -qO - http://packages.amd.com/rocm/apt/debian/rocm.gpg.key | sudo apt-key add -
$ sudo sh -c 'echo deb [arch=amd64] http://packages.amd.com/rocm/apt/debian/ trusty main > /etc/apt/sources.list.d/rocm.list'
$ sudo apt-get update
$ sudo apt-get install rocm
なお、AMDのサーバーがとても遅いのか、恐ろしく時間がかかります。
具体的には、一晩放置しても終わってませんでした。半日まるっとかかるぐらいは覚悟して放置しましょう。
インストールが無事に終わったら再起動します。
再起動後、uname -r
とかでカーネル名が4.4.0-kfd-compute-rocm-rel-1.1.1-10
などになっていれば新しいカーネルに切り替わっています。
その後、手順に従って、HSAILが動くことを確認しましょう。
ちゃんと動けばインストール成功です。
これで、ROCmの一式が全て入っているので、HCC compilerもHIPも全て/opt/rocmの下にインストールされました。
先のレポジトリから持ってくる手順で、CUDAからHIPに変換するツールであるhipifyは/opt/rocm/bin/hipifyにインストールされました。
が、実はこの標準のhipifyだけでは変換が不十分になります。
これは、このhipifyがただの文字列置換Perlスクリプトであり、C++の構文解析を一切していないからです。
なので、hipify-clangと言う、clangベースのツールを別に入れる必要があります。
別のツールといっても、clang-hipifyもHIPの中の1つです(↑の手順で入らないのはまだβ版だからのようですね)。
手順は、HIPのmasterレポジトリをダウンロードして、
clang-hipifyのREADMEに従ってビルド&インストールしましょう。
今回は以下のコマンドを実行しました。
$ cd ~
$ git clone https://github.com/GPUOpen-ProfessionalCompute-Tools/HIP.git
$ cd HIP
$ wget http://llvm.org/releases/3.8.0/clang+llvm-3.8.0-x86_64-linux-gnu-ubuntu-14.04.tar.xz
$ tar xvfJ clang+llvm-3.8.0-x86_64-linux-gnu-ubuntu-14.04.tar.xz
$ mkdir build
$ cd build
$ cmake -DBUILD_CLANG_HIPIFY=1 -DLLVM_DIR=~/HIP/clang+llvm-3.8.0-x86_64-linux-gnu-ubuntu-14.04/ -DCMAKE_BUILD_TYPE=Release ..
$ make
$ sudo make install
$ wget http://developer.download.nvidia.com/compute/cuda/repos/ubuntu1404/x86_64/cuda-repo-ubuntu1404_7.5-18_amd64.deb
$ sudo gdebi cuda-repo-ubuntu1404_7.5-18_amd64.deb
$ sudo apt-get update
$ sudo apt-get install cuda-minimal-build-7-5 cuda-curand-dev-7-5
これで、/opt/rocm/hip/bin/clang-hipify
がインストールされます。
ということで準備が終わったので、実際にCUDAコードを動かす手順に移ります。
ということで手始めに元にするコードを用意します。今回は簡単なベクトルの加算を用意しました。
__global__
void kernel(double z[], const double x[], const double y[])
{
const auto i = (blockIdx.x * blockDim.x + threadIdx.x);
z[i] = x[i] + y[i];
}
せっかくなので、これをまずは普通にCUDAの動く環境で動かしてみてください(vectorAddに移動してmakeで動くはずです)。
CUDAコードを用意したら、いよいよ次はCUDAをHIPに変換する時です!
ということで、とりあえず普通についてくるhipifyを使ってみましょう。
hipify [yoursource.code]
とやると、標準出力に変換結果が表示されます。
これをやると、例えば
#include "kernel.hpp"
__global__
void kernel(double z[], const double x[], const double y[])
{
const auto i = (blockIdx.x * blockDim.x + threadIdx.x);
z[i] = x[i] + y[i];
}
void Kernel(double z[], const double x[], const double y[],
const std::size_t block, const std::size_t thread)
{
kernel<<<block, thread>>>(z, x, y);
}
は、
#include <hip/hip_runtime.h>
#include "kernel.hpp"
__global__
void kernel(double z[], const double x[], const double y[])
{
const auto i = (hipBlockIdx_x * hipBlockDim_x + hipThreadIdx_x);
z[i] = x[i] + y[i];
}
void Kernel(double z[], const double x[], const double y[],
const std::size_t block, const std::size_t thread)
{
hipLaunchKernel(HIP_KERNEL_NAME(kernel), dim3(block), dim3(thread), 0, 0, z, x, y);
}
と変換されます。概ね一対一になっていて対応関係はそんなに難しいものではないと思います。
ただ、これは実は不完全です。これを後述のhipccに食わせてもコンパイルできません。
やってみると分かるのですが、カーネル関数の第一引数に必要な物がない、と言われます。
実はHIPの言語仕様として、カーネル関数の最初にはhipLaunchParm lp
というものが必要ということになっています。
そのため、通常のhipifyを使っただけでは、変換後に手動でhipLaunchParm lp
を挿入する必要があります。面倒ですね。
そこで、先にインストールしたhipify-clangの出番です。
これを使ってhipify-clang [yoursource.code] -o [yourhip.code]
とすると、無事に全部変換してくれます。
しかしhipify-clangだけだとまだ不完全なようで、ひとつは、hip_runtime.hのincludeを追加してくれないんですね・・・。
ということで、現時点では正しく自動変換させるには、hipify-clangを書けた後に、通常のhipifyをする必要があるようです。
HIPに変換できたら、あとはhipcc [yourhip.code]
とするだけでビルドできます。
あとはビルドされた実行ファイルを普通に実行できるはずです。
というわけで、HIPへの変換からビルドまでを一括でやってくれるMakefileを書きました。
ご覧のとおり、このMakefileでは以下のことをやってくれます。
先述のベクトル加算の例にこのMakefileを置いたものも用意しました。
make
とやると、vectorAddという実行ファイルができます。
$ git clone git@bitbucket.org:LWisteria/hipvectoradd.git
$ cd hipvectoradd/vectorAdd
$ make
/opt/rocm/hip/bin/hipify-clang main.cpp -o main.cpp.hip
hipify main.cpp.hip -inplace
rm main.cpp.hip.prehip -f
mv main.cpp.hip obj/main.hip.cpp
hipcc -g -O3 -Wall -I/opt/rocm/include/hip -std=c++11 -I. -c obj/main.hip.cpp -o obj/main.cpp.o
/opt/rocm/hip/bin/hipify-clang kernel.cu -o kernel.cu.hip
hipify kernel.cu.hip -inplace
rm kernel.cu.hip.prehip -f
mv kernel.cu.hip obj/kernel.hip.cu
hipcc -g -O3 -Wall -I/opt/rocm/include/hip -std=c++11 -I. -c obj/kernel.hip.cu -o obj/kernel.cu.o
hipcc -o vectorAdd obj/main.cpp.o obj/kernel.cu.o
$ ./vectorAdd
Device: Fiji
0: -1.38309, -1.38309
1: -1.91112, -1.91112
2: -0.682073, -0.682073
3: 1.17407, 1.17407
4: 1.95078, 1.95078
5: 0.93395, 0.93395
6: -0.941547, -0.941547
7: -1.95139, -1.95139
8: -1.16713, -1.16713
9: 0.69018, 0.69018
実行結果はあまりちゃんと書いてませんが、main.cpp
を読んでもらうと分かる通り、左側の数値がCPUの実行結果で、右側がGPUでの実行結果です。
ちゃんと実行できたことが分かりますね!
ということで、CUDAのコードをRadeonで動かすことに成功しました!
ROCmあるいはHIPに関しての日本語での情報がほぼ皆無だったので、この記事が誰かの役に立てばとても嬉しく思います。
なお、最初に書いた通り、今回は「動かす」ことを目的にしたので、特に性能評価はしていません。
そのうち何か良い題材を見つけて性能評価をしてみたいと思います。
コンピュータビジョンセミナーvol.2 開催のお知らせ - ニュース一覧 - 株式会社フィックスターズ in Realizing Self-Driving Cars with General-Purpose Processors 日本語版
[…] バージョンアップに伴い、オンラインセミナーを開催します。 本セミナーでは、...
【Docker】NVIDIA SDK Managerでエラー無く環境構築する【Jetson】 | マサキノート in NVIDIA SDK Manager on Dockerで快適なJetsonライフ
[…] 参考:https://proc-cpuinfo.fixstars.com/2019/06/nvidia-sdk-manager-on-docker/ […]...
Windowsカーネルドライバを自作してWinDbgで解析してみる① - かえるのほんだな in Windowsデバイスドライバの基本動作を確認する (1)
[…] 参考:Windowsデバイスドライバの基本動作を確認する (1) - Fixstars Tech Blog /proc/cpuinfo ...
2021年版G検定チートシート | エビワークス in ニューラルネットの共通フォーマット対決! NNEF vs ONNX
[…] ONNX(オニキス):Open Neural Network Exchange formatフレームワーク間のモデル変換ツー...
YOSHIFUJI Naoki in CUDAデバイスメモリもスマートポインタで管理したい
ありがとうございます。別に型にこだわる必要がないので、ユニバーサル参照を受けるよ...