このブログは、株式会社フィックスターズのエンジニアが、あらゆるテーマについて自由に書いているブログです。
皆さんは大規模計算のアーキテクチャといえば何を思い浮かべますか?
現在世界中のAI の学習の中心を担っている GPU、スーパーコンピュータ『富岳』、Preferred Networks 社が発表している『MN-Core』と人により様々だと思います。
世はまさに AI 戦国時代、世界中の半導体企業がより高速で、より省電力で、より扱いやすいアーキテクチャの開発に切磋琢磨しています。
そのような中でもチップ一つを製造するのにウエハを一枚使い切ってしまうというプロセッサが 2021年8月に発表されました。
Cerebras Systems 社 の 第二世代 Wafer-Scale Engine (WSE-2)です。1
さらに、Cerebras Systems 社の WSE 向けにプログラミングする開発環境『Cerebras SDK』が一般公開が開始されました。
今回はそのWSE-2へ、Fixstars の有志でSimulated Annealingを実装しました。
この記事はその解説記事です。
実装内容の説明の前に、Cerebras WSEについて簡単に紹介します。
ウエハースケールプロセッサ(WSE)ということで、とにかく大きいことが特徴です。
Cerebras が公開している Whitepaper をもとに、 NVIDIA の H100 との比較すると次のようになります。
WSE-2 | H100 | Cerebras Advantage | |
---|---|---|---|
Chip size | 46,225 mm^2 | 826 mm^2 | 56x |
Cores | 850,000 | 16,896 FP32 + 528 Tensor | 48x |
On chip memory | 40 GB | 0.05 GB | 800x |
Memory bandwidth | 20 PB/s | 0.003 PB/s | 6666x |
Fabric bandwidth | 220 Pb/s | 0.0576 Pb/s | 3819x |
チップサイズが56倍あり、その分コアがたくさんあります。 各コアがそれぞれのオンチップメモリを持っており、メッシュ状に接続されています。 そのため、巨大な分散メモリシステムがチップ内に形成されています。
WSE-2 はデータフローアーキテクチャと呼ばれるアーキテクチャです。
以下のように二次元に Processing Element (以下、PE ) が配置されています。
この PE 一つひとつに 48 KB のメモリと演算器が積まれており、『あるコアで計算した値を隣のコアに渡す』という処理が可能です。
このように複数のコアを同時に動かすことで高い並列性やパイプライン性を実現しています。
WSE-2 上で任意の処理を実行するために Cerebras Systems 社が用意している、 Cerebras SDK について紹介します。
WSE-2を搭載したシステムであるCS-2は、制御を行うホスト部とWSE-2部に分かれています。 ホスト部とWSE-2部で使用するプログラミング言語は次のようになっています。
プログラミング対象 | 言語 |
---|---|
ホスト | Python |
WSE-2 | CSL (Zig言語ベースの専用プログラミング言語) |
CSL で書かれたプログラムをコンパイルし、ホスト部のプログラムでロードし、WSE-2 でプログラムを実行します。
CSL での WSE-2 のプログラミングには、color, route の 2つの概念があります。これにタスクベースのプログラミングモデルを組み合わせることで処理を記述します。
概念 | 概要 |
---|---|
color |
|
route |
|
colorによりデータに色を付け、routeによりどの通信経路からデータが来てどの通信経路へデータを送るのかを制御します。 ここで、RAMPは受け取ったPE自身のメモリへの転送を指します。
さらに、タスクには2種類あります。
task の種類 | 概要 |
---|---|
local_task | 他の task からの activate によって実行されるタスク |
data_task | 指定した color の data が入力された際に実行されるタスク |
タスクはCSL上の関数とは異なり、呼び出し元には戻りません。
local_taskは処理をタスクとして記述することで、非同期な操作が完了した時に特定のタスクを実行するといったことが可能になります。
また、data_taskを用いることで、『ある色のデータが流れてきた際に、流れてきたデータの総和を取る』といった処理が可能となります。
SA は NP困難 である組合せ最適化問題に対して、ヒューリスティックに近似解を求めるアルゴリズムです。
基本的にはエネルギーが減少する向きに状態を遷移させますが、一定確率でエネルギーを増加させる向きにも遷移させるため、局所解に陥りにくいという特徴があります。
詳しくは Fixstars Amplify が公開している記事をご参照ください。
今回はスピン変数を0,1で表現したQUBOモデルを対象として実装を行うことにしました。 QUBO – Wikipedia
Simulated Annealingを並列化する方法は様々あると思います。 今回は最も単純な方法として、複数のPEで乱数のシード値を変えて各々が独立にアニーリングをした後に、各PEの中から最も良い結果を集めてホストに送るという方針をとります。
実装は、以下のようなフローとなります。
ホストからデバイスへの転送は、SDKで用意されているストリーミングを使用します。
ストリーミングにより、左上の(0,0)のPEへ入力を送信します。
入力を(x,y)=(0,0)から縦方向に(0,h-1)までデータを配った後、横方向にデータを広げていくことで計算を行うPEに入力データを配ります。
次の図はその様子を示しています。
各PEでSimulated Annealingを行います。
ここでは、各PEが並列でSimulated Annealingによる探索計算を繰り返している状態になります。
最終的に最小のエネルギーを得るために、各PEは『送られてきたデータと自分のデータを比較し、より小さいエネルギーを持つデータを指定方向に流す』ことで右下に最小のエネルギーを集めます。
まずは横方向にこの処理を行い、最終的に一番右端にいるPEが縦方向に処理を行うことで実現しています。
次の図はその様子を示しています。
デバイスからホストへの転送も、SDKで用意されているストリーミングを使用します。
ストリーミングにより、右下の(w-1,h-1)のPEからホストへ最終的なデータを転送します。
実装したSimulated AnnealingをWSE-2の実機で試したいところですが、シミュレータを使用して実行してみます。 SDKはv1.0.0を使用しました。
サイズは4×8として、N=6の問題を解いてみましょう。入力は乱数で与えます。
シミュレータはSDKに同梱されており、環境を設定して実行をすると次のような結果となります。
+ width=4
+ height=8
+ cslc ./layout.csl --fabric-dims=11,10 --fabric-offsets=4,1 --params=Num:6,width:4,height:8,max_iters:16,T:1000 --params=MEMCPYH2D_DATA_1_ID:0,MEMCPYD2H_DATA_1_ID:1 -o out --memcpy --channels 1
INFO: Using SIF: /home/Owner/Cerebras-SDK-1.0.0-202311111408-11-4ef8698c/cbcore_sdk-202311111408-10-4a54bce5.sif
INFO: User's specified CSL_IMPORT_PATH=
NOTE: CSL_IMPORT_PATH accepts colon separated list of paths generated by 'realpath <path>'
compile successful
+ cs_python run.py --name out
INFO: Using SIF: /home/Owner/Cerebras-SDK-1.0.0-202311111408-11-4ef8698c/cbcore_sdk-202311111408-10-4a54bce5.sif
Reading file out/out.json
Reading file out/west/out.json
Reading file out/east/out.json
fab w,h = 11,10
Kernel x,y w,h = 4,1 4,8
memcpy x,y w,h = 1,1 9,8
inputs = [ 0.5479121 -0.12224312 0.71719587 0.39473605 -0.8116453 0.9512447
0.5222794 0.5721286 -0.74377275 -0.09922812 -0.25840396 0.85353
0.28773025 0.64552325 -0.1131716 -0.5455226 0.10916957 -0.8723655
0.65526235 0.2633288 0.5161755 ]
18.179776396s
best_s = [0 1 0 1 0 1]
energy = -1.3816108703613281
この結果は、同様のアルゴリズムをCPUへ実装したものと一致します。
今後も有志メンバーで時間を作りながら開発を進めていきます。 現在考えている展望は次のようなものとなります。
本記事にて解説している内容は、日本でCerebras Systems 社の販売代理店である、東京エレクトロンデバイス様が開催したハッカソンにおいて発表した内容です。
ハッカソンを開催し、実機での動作確認をしていただけたことに感謝申し上げます。
本記事で紹介した実装内容は、弊社GitHub上でOSSとして公開しています。 興味がある方はぜひ見てください。
2024年3月13日にさらに最新のWSE-3が発表されました。↩︎
コンピュータビジョンセミナー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デバイスメモリもスマートポインタで管理したい
ありがとうございます。別に型にこだわる必要がないので、ユニバーサル参照を受けるよ...