もくじ
はじめに
RISC-V の注目度は年々増加しており、多くのメーカーで RISC-V をベースとした開発が進んでいる印象です。
今回私が使用する Nios V も RISC-V ベースのプロセッサーの一つであり、こちらはインテルにより開発されたソフトコア・プロセッサーです。
この記事は、RISC-V を使用して何か面白ことができないかと考えた結果、インテル FPGA の Agilex で Nios V を限界まで実装してみた、実験系の記事となっております。
環境
今回はインテルの FPGA を使用するということと Nios V を使用するということから。。
HW 側の開発は
SW 側の開発は
上記を使用します。
また、FPGA (Agilex) については下記を使用します。
構成(HW)
今回は Nios V をたくさん実装するということで、Nios V が入ったサブモジュールを作成し、そのモジュールをトップ階層にインスタンスしていく構成にしています。
まず、サブモジュールの構成としては。
- Nios V/m
- On Chip RAM
- JTAG-UART
動作確認するうえで最低限必要なこの 3 つです。
トップ階層には。
- CLK
- Reset
- サブモジュール (Nios V)
- ISSP 用 Reset
を使用します。
ISSP については見慣れないかと思いますが、今回使用する開発 Kit に FPGA Reset 用のボタンがないため、ISSP により HW ロジックを使用して、Reset を toggle できるようにしていると理解していただければ大丈夫です。
ではこの構成をブロック図にするとこのような感じです。
(オレンジ色は Birdge、紫色は IP で色分けしています。)
今回の構成では、Nios V 用の実行メモリの On chip RAM は 128kByte で生成しています。
詳細については『たくさん実装してみた』にて Platofomr Deisgner のキャプチャを紹介していますのでそちらも併せてご参照ください。
また、今回はスタンドアロンでの動作確認はしないため、各 Nios V の Reset Vector や、Boot 方法など細かい設定は特に意識せずデバッガ で elf ファイルを RAM に流して動作させるためだけの構成です。
構成(SW)
SW は JTAG コンソールに出力するだけの簡易的なプログラムにしており、
複数実装するため、それぞれのコンソールでどの Nios V が動いているか確認できるよう差分を作っています。
以下今回実装する SW です。
#include "system.h"
#include <stdio.h>
#include <unistd.h>
int main() {
for (int i = 0; i < 10000; ++i) {
usleep(500000);
printf("Hello world count %d, Nios V/m cpu0 \n",i);
usleep(500000);
}
printf("Bye world!\n");
return 0;
}
たくさん実装してみた。
今回の構成は Platform Designer にて作成しております。トップ階層にはサブモジュールをインスタンスすればいいため、それぞれの IP を一つ一つ実装する手間はなく、幸いトップ階層は綺麗に保てています。出来なかった場合大変だなと考えていたためこの点に関しては安心しました。( Generate に時間はかかりましたが。。)
まず Nios V 一つだけで確認してみましょう。
先程説明した通りサブモジュール化した Nios V を 1 つだけ Platform Designer システムに実装しているように見えています。
下図が トップ階層システム。(赤枠が Sub module)
こちらが Sub Module (赤枠が Nios V/m)
コンパイル結果は下図です。
128kB Onchip RAM に使用しておりますが、まだ 1% と余裕ですね。
では次は 10 個で試してみます。
こちらも後々楽をするために Sub module を 10 個実装した Sub module を作成し、トップ階層にインスタンスしています。
下図がコンパイル結果です。
おおよそ、RAM Block については Sub Module 一つにつき 1 % といったところでしょうか?
では、思い切って 100 個実装してみましょう。
ギリギリ実装できました!
100 個実装しても大丈夫です!!
もう少し実装できるかとは思いますが、RAM リソースが 9 割以上となったため、今回は 100 個で一区切りとしたいと思います。。
マルチコアデバッグ
実際に複数実装した際のデバッグについても最後に書きたいと思います。
まず Nios V の開発環境ですが 2 章でも紹介した RiscFree を使用します。こちらは Quartus Prime のダウンロード時にインストーラーを同時にダウンロードできるため、併せてインストールしています。
ではマルチコアデバッグの手順ですが RiscFree の起動や、プロジェクトのインポート等の手順はこの記事では割愛しています。
(Build までの手順は下記記事を参考にしました。)
Ashling* RiscFree* IDE を使用した Nios®V プロジェクト開発手順
下図はプロジェクトをインポートし Build が完了した状態です。
(画面サイズの都合上。今回は 5 つ同時デバッグにしています。)
この状態から手順の紹介をします。
まず CPU それぞれの Debug configuration を作成します。
どの CPU に対して作成するかは、下図のように Deugger タブ内の Core selection から選択できます。
設定後 Apply によるそれぞれの Debug configuration を作成後、Launch Group により作成した Debug configuration を Group 化し同時実行できるようにします。
これで、Debug configuration は完成です。
次に、コンソール出力先を準備します。
今回は画面表示の都合上、Nios V command shell にそれぞれの出力を表示させます。
Nios V command shell を実行する CPU の数だけ起動し、下記コマンドを実行します。
#juart-terminal -c <CPU 毎に変更> -d <デバイス番号> -i <インスタンス番号>
juart-terminal -c 1 -d 0 -i 0
このコマンドにより、それぞれの Command shell が JTAG コンソールと紐づきます。
( Core selection より設定した CPU の末番と -i の引数が対応しています)
それでは、先程作成した Group を選択し Debug を押下しましょう。
デフォルトで main の最初で Break するようになっているので、ユーザー側で それぞれのソースに対して Break point を増やしたり、レジスタや変数の値を確認しながらデバッグすることができます。
今回の実行結果は下図です。
JTAG コンソールが同時に動作している状況をキャプチャしています。
おわりに
今回は面白そうという理由だけで Nios V をたくさん実装してみましたが、コンパイル時間、Platform Designer の階層設計など時間がかかる作業が多く、軽い気持ちで始めた記事でしたがなかなか大変でした。ただ、この構成を実際にされる方は少ないかと思いますので、マルチコアデバッグのところを参考にしていただければと思います。。