

## 株式会社ウーノラボ

https://www.unolabo.co.jp

## Efinix FPGA への Trinita 1stage Core の実装手順

Rev.4 2025.04.09



Efinix FPGA に実装されている Sapphire SoC 内部の Vex RISC-V を Trinita 1stage Core に置き換える手順を説明します。Sapphire SoC の Peripheral はそのまま使えるため、置き換え後も Sapphire SoC 用のソフトウェアコードをそのまま流用できます。









### テンプレートコピー

Efinity で Sapphire SoC プロジェクトを作る



Sapphire SoC の VexRISC-V を 1stage Core に置換 (スクリプト)



1stage Core で使う クロックを PLL で生成・ タイミング制約追加



1stage Core の パラメータを定義 (.vh ファイル)



トップデザインのポート宣言にクロックを追加し、.vhをインクルード



ソフトウェアをビルドし、 RAM 初期化ファイル生成 (RISC-V IDE とスクリプト)



FPGA デザインを コンパイル (Efinity)



この資料で紹介する手順は GitHub で公開されています。 チュートリアル動画も YouTube で公開していますので、合わせて参照下さい。







## 置き換え手順



置き換え作業を始める前に、Sapphire SoC を実装済みのプロジェクトを用意します。 ※SoC の Peripheral 構成をこの時点で決定することを推奨します。







### ニューテンプレートをプロジェクトフォルダにコピー

#### ./template フォルダの内容をプロジェクトフォルダにコピーします。



※ template の embedded\_sw 配下の "sap" は Sapphire SoC のインスタンス名を表してい ます。所望のプロジェクトの Sapphire SoC の インスタンス名に変更してからコピーして下 さい。

#### プロジェクトフォルダ





### 追加されるファイル (テンプレートからコピー)

© UNO Laboratories, Ltd.



### Yex RISC-V を Trinita 1stage Core に置き換える

© UNO Laboratories, Ltd.

コマンドプロンプトで Efinity プロジェクトのフォルダ に移動し、Pythonスクリプトを実行します。

これで Trinita 1stage Core が sap.v に実装されます。



>python sap2tri.py ./ip/sap/sap.v

Trinita 1stage Core では位相をずらしたクロックを必要とします。 Efinity の Interface Designer で下記のように PLL を設定します。

- •Trion T20 最大 25 MHz
  - io systemClk: 0 deg
  - io\_systemClk2: 270 deg (io\_systemClk3 の反転クロック)
  - io\_systemClk3 : 90 deg
- •Titanium Ti60 最大 75MHz
  - io systemClk: 0 deg
  - io\_systemClk2 : 225 deg
  - io\_systemClk3 : 90 deg

T8 の PLL では、位相をずらした 12.5 MHz のクロックを生成できないため、ロジック領域で分周・位相設定してクロックを生成します。

#### •Trion T8 最大 12.5 MHz

io\_systemClk : 0 deg

io\_systemClk2 : 252 deg

io\_systemClk3: 108 deg



```
create_clock -period 8 CLK125M
create_clock -name io_systemClk -period 80 -waveform { 0 40} io_systemClk
create_clock -name io_systemClk2 -period 80 -waveform { 56 16} io_systemClk2
create_clock -name io_systemClk3 -period 80 -waveform { 24 64} io_systemClk3
```

```
114
115
          reg io_systemClk;
116
          reg io_systemClk2;
117
          reg io_systemClk3;
118
          reg [9:0] cntdiv;
119
120
          (* syn preserve = "true" *) reg tri clk 1;
121
          (* syn_preserve = "true" *) reg tri_clk_2;
122
          (* syn_preserve = "true" *) reg tri_clk_3;
123
124
          always@(posedge CLK125M)
125 (-)
         begin
                                                    10分周
126
            if (~io_pllResetn)
127
              cntdiv <= 10'b0000011111:
128
129
              cntdiv <= {cntdiv[8:0], cntdiv[9]};</pre>
130
131
132
          always@(posedge CLK125M)
133 (-)
                                                  位相設定
134
              io systemClk <= cntdiv[0];
              io_systemClk2 <= cntdiv[7];</pre>
              io_systemClk3 <= cntdiv[3];
136
137
          end
138
```

#### PLL で追加したクロックを .sdc で定義します。

T20 では io\_systemClk3 を LogicBlock で not 反転して io\_systemClk2 を生成しています。 そのため、下記の例では io\_systemClk2 を定義していません。 ※Efinity では not 反転クロックは .sdc で定義しなくても自動的にタイミング解析されるためです。

```
# PLL Constraints
################

create_clock -period 40.0000 io_systemClk

create_clock -waveform {10.0000 30.0000} -period 40.0000 io_systemClk3
```

#### Trinita 1stage Core のパラメータ(マクロ)を.vhファイルで定義します。

```
define EFINIX 1
                     動作周波数 [MHz]
define FREQ 25
define START ADDRESS 32'hF9000000
                           IMEM の深さ 2^15=32KB
define IMEM AWIDTH
define DMEM AWIDTH
                          ┙DMEM の深さ 2^15=32KB
'define FILE IMEM "./romdata/imem.hex"
define FILE IMEMO "./romdata/imem0.hex"
'define FILE IMEM1 "./romdata/imem1.hex"
'define FILE IMEM2 "./romdata/imem2.hex"
`define FILE IMEM3 "./romdata/imem3.hex"
define FILE DMEM
                   "./romdata/dmem.hex"
'define FILE DMEMO "./romdata/dmem0.hex"
`define FILE DMEM1 "./romdata/dmem1.hex"
define FILE DMEM2 "./romdata/dmem2.hex"
`define FILE DMEM3
                    Titanium FPGA の場合は、
define TRION 1
                    `define TITANIUM 1 に変更する
define SAPPHIRE 1
```



先ほど PLL で追加した io\_systemClk2, 3 をトップデザインの port 宣言に追加し、Sapphire SoC のインスタンスにも接続します。

```
module top

input io_asyncResetn,
input io_systemClk.

//input io_systemClk2,
input io_systemClk3,

output io_systemClk3,

output io_pllResetn,
input io_pllLocked,

input io_pllLocked,
```

```
assign io_systemClk2 = ~io_systemClk3;
```

```
sap u_sap(
    .io_svstemClk ( io_svstemClk ),
    .io_systemClk2 ( io_systemClk2 ),
    .io_systemClk3 ( io_systemClk3 ),
    .jtagCtrl_enable ( jtagCtrl_enable ),
    .jtagCtrl_tdi ( jtagCtrl_tdi ),
    .jtagCtrl_capture ( jtagCtrl_capture ),
    .jtagCtrl_shift ( jtagCtrl_shift ),
    .jtagCtrl_update ( jtagCtrl_update ),
    .jtagCtrl_reset ( jtagCtrl_reset ),
    .jtagCtrl_tdo ( jtagCtrl_tdo ),
    .jtagCtrl_tck ( jtagCtrl_tck ),
    .system_spi_0_io_data_0_read ( system_spi_0_io_dat
```

T20 では io\_systemClk3 を LogicBlock で not 反転して io\_systemClk2 を生成しています。 そのため、top のポート宣言で io\_systemClk2 がコメントアウトされています。

先ほど定義したパラメータ (.vh ファイル) をプロジェクトのトップデザインにインクルード します。下記のように `include 文を追加して下さい。

```
output jtagCtrl_tdo,
input jtagCtrl_tck
);

include "trinita_define.vh"

wire io_systemReset;
```

## ソフトウェアのビルド

RISC-V IDE で ./embedded\_sw/sap をワークスペースとして開きます。 サンプルソフトウェアである "gpioDemo\_Trinita" プロジェクトをインポートします。 このプロジェクトをビルドすると gpioDemo\_Trinita/build 配下に dmem.bin と imem.bin が生成されます。



コマンドプロンプトで Efinity プロジェクトのフォルダに移動し、Python スクリプトを実行します。

このスクリプトによって、先程生成した imem.bin と dmem.bin が FPGA の Block RAM に埋め込まれます。

>python conv\_tri\_mem.py ./embedded\_sw/sap/software/standalone/gpioDemo\_Trinita/build

## Efinity でコンパイル

DMEM / IMEM の生成が終わったら Efinity でプロジェクトをコンパイルします。 ビットストリームファイル(.hex)が生成されたら、これを Efinity Programmer でフラッシュ メモリに書き込みます。









# 補足

### Bootloader を使ったソフトウェアの起動

以上の手順で、ソフトウェアのバイナリ データを FPGA ビットストリームファイル に埋め込んだので、FPGA の電源起動 の後、自動的にソフトウェアも起動しま す。

もしソフトウェアを外部フラッシュメモリ に格納して、Bootloader でソフトウェア を起動させたい場合は、下記の GitHub のページを参照して下さい。

https://github.com/unolabo/efx-trinitaexa/blob/main/README Bootloader.m d





## 株式会社ウーノラボ

https://www.unolabo.co.jp

ありがとうございました