# Vivado Design Suite ユーザー ガイド

Tclスクリプト機能の使用

UG894 (v2012.3) 2012 年 11 月 16 日





#### **Notice of Disclaimer**

The information disclosed to you hereunder (the "Materials") is provided solely for the selection and use of Xilinx products. To the maximum extent permitted by applicable law:(1) Materials are made available "AS IS" and with all faults, Xilinx hereby DISCLAIMS ALL WARRANTIES AND CONDITIONS, EXPRESS, IMPLIED, OR STATUTORY, INCLUDING BUT NOT LIMITED TO WARRANTIES OF MERCHANTABILITY, NON-INFRINGEMENT, OR FITNESS FOR ANY PARTICULAR PURPOSE; and (2) Xilinx shall not be liable (whether in contract or tort, including negligence, or under any other theory of liability) for any loss or damage of any kind or nature related to, arising under, or in connection with, the Materials (including your use of the Materials), including for any direct, indirect, special, incidental, or consequential loss or damage (including loss of data, profits, goodwill, or any type of loss or damage suffered as a result of any action brought by a third party) even if such damage or loss was reasonably foreseeable or Xilinx had been advised of the possibility of the same.Xilinx assumes no obligation to correct any errors contained in the Materials or to notify you of updates to the Materials or to product specifications. You may not reproduce, modify, distribute, or publicly display the Materials without prior written consent. Certain products are subject to the terms and conditions of the Limited Warranties which can be viewed at <a href="http://www.xilinx.com/warranty.htm">http://www.xilinx.com/warranty.htm</a>; IP cores may be subject to warranty and support terms contained in a license issued to you by Xilinx. Xilinx products are not designed or intended to be fail-safe or for use in any application requiring fail-safe performance; you assume sole risk and liability for use of Xilinx products in Critical Applications: <a href="https://www.xilinx.com/warranty.htm#critapps">https://www.xilinx.com/warranty.htm#critapps</a>.

© Copyright 2012 Xilinx, Inc. Xilinx, the Xilinx logo, Artix, ISE, Kintex, Spartan, Virtex, Vivado, Zynq, and other designated brands included herein are trademarks of Xilinx in the United States and other countries. All other trademarks are the property of their respective owners.

本資料は英語版 (v2012.3) を翻訳したもので、内容に相違が生じる場合には原文を優先します。 資料によっては英語版の更新に対応していないものがあります。 日本語版は参考用としてご使用の上、最新情報につきましては、必ず最新英語版をご参照ください。

この資料に関するフィードバックおよびリンクなどの問題につきましては、jpn trans feedback@xilinx.com までお知らせください。いただきましたご意見を参考に早急に対応させていただきます。なお、このメール アドレスへのお問い合わせは受け付けておりません。あらかじめご了承ください。

#### 改訂履歴

次の表に、この文書の改訂履歴を示します。

| 日付          | バージョン  | 改訂内容 |
|-------------|--------|------|
| 2012年11月16日 | 2012.3 | 初版   |



# 目次

| 第1章:Vivado での Tcl スクリプト    |    |
|----------------------------|----|
| 概要                         | 4  |
| Tcl の概要                    | 5  |
| サンプル スクリプト : 非プロジェクト コンパイル |    |
| フロー                        | 6  |
| サンプル スクリプトの詳細              | 8  |
| Tcl プロシージャの定義              | 9  |
| デザイン オブジェクトへのアクセス          | 11 |
| デザイン階層での検索                 |    |
| 関連性を使用したオブジェクトの検索          | 16 |
| オブジェクトのリストの処理              | 18 |
| 出力先の指定                     | 19 |
| ファイルへのアクセス                 | 20 |
| 文字列の操作                     | 22 |
| カスタム DRC の作成               | 23 |
| Tcl DRC チェッカーの記述           | 24 |
| Vivado Tcl DRC コマンド        | 25 |
| Tcl スクリプトの読み込みと実行          | 25 |
| Tcl スクリプト記述のヒント            | 27 |
| オブジェクトのキャッシュ               | 27 |
| オブジェクト名と NAME プロパティ        |    |
| オブジェクトのリストのフォーマット          |    |
| Vivado Tcl コマンドをオプションで検索   | 28 |
| 付録 A: その他のリソース             |    |
|                            | 20 |
| ザイリンクス リソース                |    |
| ソリューション センター               | 29 |



# Vivado での Tcl スクリプト

### 概要

Tcl (ツール コマンド言語) は、さまざまなデザイン ツールおよびデザイン データにアクセスするための、変数、プロシージャ、制御構造を含むインタープリター型プログラミング言語です。

この言語は新しい関数呼び出しで簡単に拡張することができ、1990年代初期に開発されてから、新しいツールやテクノロジをサポートするため拡張されてきています。簡単に拡張できることから、多くの EDA ベンダーが標準の API (アプリケーション プログラミング インターフェイス) としてアプリケーションを制御および拡張するために導入しています。

ザイリンクスでは、Vivado™ Design Suite のネイティブ プログラミング言語として Tcl を導入しているので、この業界標準言語に精通している設計者であれば簡単に取り入れ、理解することができます。Vivado Design Suite の Tcl インタープリターは、アプリケーションの制御、デザインオブジェクトおよびプロパティへのアクセス、カスタムレポートの作成を実行するための、Tcl 言語の機能と柔軟性を提供しています。Tcl を使用すると、デザインの特定の要件に合わせてデザインフローを変更できます。

Tcl 言語には、ローカル ファイル システムのファイルに対して読み出しおよび書き込みを実行するビルトイン コマンドが含まれます。これにより、動的にディレクトリを作成し、FPGA デザイン プロジェクトを開始して、プロジェクトにファイルを追加したり、合成およびインプリメンテーションを実行できます。デザイン プロジェクトからデバイス リソースの使用率や QoR (結果の質) に関するカスタマイズ レポートを生成し、企業内で共有できます。

また、Tcl 言語を使用して、新しい設計手法を試したり、既存の問題を回避したり、必要に応じてデザインオブジェクトの挿入および削除、プロパティの変更を実行できます。デザインフローの確立された部分を再実行するためのスクリプトを記述し、プロセスを標準化できます。

このガイドで説明する Tcl コマンドおよびスクリプト例のほとんどは、Vivado Design Suite 特定のものです。Vivado 特定の Tcl コマンドの詳細は、『Vivado Design Suite Tcl コマンド リファレンス ガイド』(UG835) を参照するか、または Vivado のヘルプ機能を使用してください。

Vivado ツールでは、vivado.jou というジャーナル ファイルが Vivado を起動したディレクトリに作成されます。 ジャーナル ファイルにはセッション中に実行された Tcl コマンドが記録されるので、このファイルから新しい Tcl ス クリプトを作成できます。

Vivado Design Suite にビルトインされている Tcl インタープリターにより、追加の Tcl コマンドが提供されています。 Tcl ビルトイン コマンドについては、Tcl のオープン ソース ベースおよび資料を管理している Tcl Developer Xchange サイト (<a href="http://www.tcl.tk">http://www.tcl.tk</a>) を参照してください。

Tcl プログラミング言語の入門チュートリアルは、http://www.tcl.tk/man/tcl/tutorial/tcltutorial.html を参照してください。

このガイドでは、Tcl コマンドの例、Tcl スクリプト、および Vivado Design Suite での戻り値が示されています。これらのコマンド例とその戻り値は、次の形式で記述されています。

- Tcl コマンドおよびスクリプト例 puts \$outputDir
- Tcl コンソールへの出力または Tcl コマンドの結果
  - ./Tutorial Created Data/cpu output



### Tcl の概要

Tcl スクリプトは、改行またはセミコロンで区切られた一連の Tcl コマンドです。Tcl コマンドは、スペースまたはタブで区切られた単語の文字列です。Tcl インタープリターはコマンド ラインを単語に分割し、必要に応じてコマンドおよび変数置換を実行します。コマンドラインは左から右に読み込まれ、各単語が完全に評価されてから次の単語が評価されます。コマンドおよび変数置換は、左から右に実行されます。

単語は、1つの単語または中かっこ({})またはダブルクォーテーション("")で囲まれた複数の単語です。中かっこまたはダブルクォーテーション内のセミコロン、中かっこ、タブ、スペース、改行は、通常の文字として処理されますが、バックスラッシュ()はこの後説明するように、中かっこまたはダブルクォーテーション内でも特殊文字として処理されます。最初の単語はコマンドとして扱われ、その後の単語は引数としてコマンドに渡されます。

set outputDir ./Tutorial\_Created\_Data/cpu\_output

この例では、最初の単語は Tcl set コマンドで、値を割り当てるために使用します。2 番目の単語は変数名 (outputDir)、3 番目の単語は変数値 (./Tutorial\_Created\_Data/cpu\_output) として set コマンドに渡されます。

単語にバックスラッシュ()が使用されている場合、Tcl インタープリターによりバックスラッシュ置換が実行されます。ほとんどの場合、バックスラッシュの次の文字は標準文字として処理されます。これを使用して、ダブルクォーテーション、中かっこ、ドル記号などの特殊文字を文字列に追加できます。Tcl インタープリターでバックスラッシュ文字がどのように処理されるかは、Tcl/Tk のリファレンスを参照してください。

中かっことダブルクォーテーションマークの使用法も異なります。中かっこ内の文字に対しては、置換は実行されません。中かっこ内の単語や文字列はそのまま処理され、変数またはコマンド置換のために評価されません。次の例に示すように、単語は中かっこに囲まれたそのままの文字列(中かっこは含まない)となります。ダブルクォーテーションに囲まれた文字列は評価され、変数およびコマンド置換が必要に応じて実行されます。ダブルクォーテーションに囲まれた文字列に対してコマンド置換、変数置換、およびバックスラッシュ置換が実行されます。

puts {The version of Vivado Design Suite is [version -short]}
The version of Vivado Design Suite is [version -short]

puts "The version of Vivado Design Suite is [version -short]"
The version of Vivado Design Suite is 2012.3

上記の例で、ダブルクォーテーションを使用した2番目の例では[version -short]コマンドが戻り値で置換されていますが、中かっこを使用した1番目の例では置換が実行されていないことに注目してください。文字列を囲む場合には、このことに注意してダブルクォーテーションまたは中かっこを選択してください。

変数に値を代入するには、set コマンドを使用します。変数を参照するには、変数名の前にドル記号 (\$) を付けて指定します。単語がドル記号で開始している場合、Tcl インタープリターで変数置換が実行され、変数が現在その変数に保存されている値に置換されます。Tcl 言語では、\$ は予約語です。

set outputDir ./Tutorial\_Created\_Data/cpu\_output
puts \$outputDir

./Tutorial\_Created\_Data/cpu\_output

角かっこ([])を使用すると、コマンド内にコマンドをネストできます。ネストされたコマンドは、ボトムアップに評価されます。角かっこ内の文字列が新しい Tcl スクリプトとして反復的に処理されます。ネストされたコマンドに、さらにコマンドをネストさせることもできます。ネストされたコマンドの結果がその上位のコマンドに渡されてから、その上位のコマンドが処理されます。

set listCells [lsort [get\_cells]]

上記の例では、現在のデザインの最上位にあるセルオブジェクトがアルファベット順に並べ替えられ、そのリストが \$listCells 変数に代入されます。まず get\_cells コマンドが実行され、返されたオブジェクトが lsort コマンドで並べ替えられて、並べ替えが終了したリストが変数に代入されます。



ただし Vivado Design Suite では、角かっこの処理は標準の Tcl と異なります。角かっこは Verilog および VHDL では標準文字として処理され、通常はバスやインスタンスの配列など、ベクターの 1 つまたは複数の要素を示します。Vivado ツールでは、角かっこがネットリスト オブジェクト名の一部である場合はボトムアップに評価されません。

次の3つのコマンドは同等です。

- 1.) set list\_of\_pins [get\_pins transformLoop[0].ct/xOutReg\_reg/CARRYOUT[\*] ]
- 2.) set list\_of\_pins [get\_pins {transformLoop[0].ct/xOutReg\_reg/CARRYOUT[\*] } ]
- 3.) set list\_of\_pins [get\_pins transformLoop\[0\].ct/xOutReg\_reg/CARRYOUT\[\*\]]

1) では、外側の角かっこは標準の Tcl と同様にコマンドのネスト ([get\_pins]) を表しますが、内側の角かっこは Vivado ツールでは指定したオブジェクト名の一部として処理されます (transformLoop [0])。 Vivado Design Suite ではこれが自動的に処理されますが、一部の文字に限られ、それ以外の場合は角かっこは標準の Tcl と同様に評価されます。

- アスタリスク[\*]:任意の数のビットまたはインスタンスを示すワイルドカードです。
- 整数 [12]: 特定のビットまたはインスタンスを指定します。
- ベクター [31:0]: 特定の範囲のビットまたはインスタンスのグループを指定します。
- 2)では、中かっこを使用して内側の角かっこ内の文字列がコマンド置換されないようにしており、オブジェクト名の一部として処理されます(transformLoop[0])。
- 3)では、バックスラッシュを使用して角かっこを特殊文字でなく標準文字として評価するよう指定しており、コマンド置換は実行されません。
- 2) および3) は角かっこが適切に処理されるようにする方法を示していますが、中かっこまたはバックスラッシュを手動で追加する必要があります。1) は、これが Vivado Design Suite で自動的に処理されることを示しています。

Tel スクリプトにコメントを追加するには、行を#で開始します。#の後に続く次の改行までの文字は、無視されます。行の最後にコメントを追加するには、次の例に示すように、コマンドの最後にセミコロン(;)を記述し、その後に#を追加してコメントを記述します。

# This is a comment
puts "This is a command"; # followed by a comment

# サンプル スクリプト: 非プロジェクト コンパイル フロー

次に、非プロジェクトデザインフローを定義する Tcl スクリプトの例を示します。

このサンプル スクリプトでは reportCriticalPaths というカスタム コマンドが使用されており、Vivado Design Suite にカスタム コマンドやプロシージャを追加できることを示しています。reportCriticalPaths の内容は、9ページの「Tcl プロシージャの定義」を参照してください。



```
# STEP#1: define the output directory area.
set outputDir ./Tutorial Created Data/cpu output
file mkdir $outputDir
# STEP#2: setup design sources and constraints
read_vhdl -library bftLib [ glob ./Sources/hdl/bftLib/*.vhdl ]
read vhdl ./Sources/hdl/bft.vhdl
read verilog [ glob ./Sources/hdl/*.v ]
read_verilog [ glob ./Sources/hdl/mgt/*.v ]
read_verilog [ glob ./Sources/hdl/or1200/*.v ]
read_verilog [ glob ./Sources/hdl/usbf/*.v ]
read_verilog [ glob ./Sources/hdl/wb_conmax/*.v ]
read_xdc ./Sources/top_full.xdc
# STEP#3: run synthesis, write design checkpoint, report timing,
# and utilization estimates
synth_design -top top -part xc7k70tfbg676-2
write checkpoint -force $outputDir/post synth.dcp
report timing summary -file $outputDir/post synth timing summary.rpt
report_utilization -file $outputDir/post_synth_util.rpt
# Run custom script to report critical timing paths
reportCriticalPaths post_synth_critpath_report.csv
# STEP#4: run logic optimization, placement and physical logic optimization,
# write design checkpoint, report utilization and timing estimates
opt design
reportCriticalPaths post_opt_critpath_report.csv
place design
report clock utilization -file $outputDir/clock util.rpt
# Optionally run optimization if there are timing violations after placement
if { [get property SLACK [get timing paths -max paths 1 -nworst 1 -setup] ] < 0 }
    puts "Found setup timing violations => running physical optimization"
    phys_opt_design
write_checkpoint -force $outputDir/post_place.dcp
report utilization -file $outputDir/post place util.rpt
report_timing_summary -file $outputDir/post_place_timing_summary.rpt
# STEP#5: run the router, write the post-route design checkpoint, report the routing
# status, report timing, power, and DRC, and finally save the Verilog netlist.
route design
write_checkpoint -force $outputDir/post_route.dcp
report route status -file $outputDir/post route status.rpt
report timing summary -file $outputDir/post route timing summary.rpt
report_power -file $outputDir/post_route_power.rpt
report_drc -file $outputDir/post_imp_drc.rpt
write verilog -force $outputDir/cpu impl netlist.v -mode timesim -sdf anno true
# STEP#6: generate a bitstream
write_bitstream -force $outputDir/cpu.bit
```



#### サンプル スクリプトの詳細

上記のサンプルスクリプトは、次の段階から構成されています。

- **手順1**:変数 \$outputDir を定義して出力ディレクトリを指定し、そのディレクトリを実際に作成します。 \$outputDir 変数は、スクリプトで必要に応じて参照されます。
- **手順**2: デザインを記述する VHDL および Verilog ファイルと、デザインの物理制約およびタイミング制約を含む XDC ファイルを読み込みます。合成済みネットリストを読み込む場合は、read edif コマンドを使用します。

Vivado Design Suite では、デザイン制約を使用してデザインの物理特性およびタイミング特性を定義します。 read\_xdc コマンドは XDC 制約ファイルを読み込み、読み込まれた制約ファイルが合成およびインプリメンテーションに適用されます。



**重要: Vivado Design Suite** では、UCF フォーマットはサポートされません。UCF 制約を XDC コマンドに移行する方法は、『Vivado Design Suite 移行手法ガイド』(UG911)を参照してください。

read\_\* Tcl コマンドは、非プロジェクトモードで使用し、Vivado Design Suite でディスク上のファイルを読み込んでメモリ内にデザインデータベースを構築します。ファイルがコピーされたり、プロジェクトモードでのようにファイルの依存関係が作成されることはありません。非プロジェクトモードでのすべての操作は、Vivadoツール内のインメモリデータベースに対して実行されます。そのため、非プロジェクトモードは非常に柔軟ですが、ユーザーがソースデザインファイルの変更を管理し、それに応じてデザインをアップデートする必要があります。プロジェクトモードまたは非プロジェクトモードを使用した Vivado Design Suite の実行に関する詳細は、『Vivado Design Suite ユーザーガイド:デザインフローの概要』(UG892)を参照してください。

手順3:デザインを特定のパッケージ用に合成します。

HDL デザイン ファイルをコンパイルし、XDC ファイルに含まれるタイミング制約を適用し、ロジックをザイリンクスプリミティブにマップして、メモリ内にデザイン データベースを作成します。Vivado ツールをバッチ モードで実行している場合でも、Tcl シェル モードで対話的に Tcl コマンドを実行している場合でも、グラフィカルモードでデザイン データを Vivado 統合設計環境 (IDE) で表示している場合でも、メモリ内のデザインは Vivado ツール内に存在します。

合成が終了したら、チェックポイントを保存します。この時点では、デザインはタイミング制約および物理制約が適用された未配置の合成済みネットリストです。タイミングやリソース使用率など、さまざまなレポートを作成すると、デザインを理解するのに有益です。

このサンプル スクリプトでは、reportCriticalPaths というカスタム コマンドを使用して、TNS、WNS、違反を CSV ファイルにレポートします。これにより、クリティカルなパスをすばやく特定できます。

合成後に read\_xdc または source コマンドを使用して読み込まれた XDC ファイルは、インプリメンテーションにのみ適用されます。それらのファイルは、その後デザイン チェックポイントを保存した場合にネットリストと共に保存されます。

手順4:配置配線の準備としてロジック最適化を実行します。最適化の目的は、ターゲットパーツの物理リソースに配置する前にロジックデザインを簡略化することです。最適化後、タイミングドリブン配置を実行します。

各手順の後、reportCriticalPaths コマンドを実行して新しい CSV ファイルを生成します。デザインの異なる段階からの複数の CSV ファイルを使用すると、カスタム タイミング サマリ スプレッドシートを作成でき、インプリメンテーションの各段階でタイミングがどのように向上したかを理解するのに役立ちます。

配置が完了したら、get\_timing\_paths コマンドを使用して配置済みデザインのワースト タイミング パスの SLACK プロパティを取得します。report\_timing コマンドを使用すると、ワースト スラックを含むタイミング パスの詳細なテキスト形式レポートが生成されますが、get\_timing\_paths コマンドを使用すると、同じタイミング パスが Tcl オブジェクトとして、パスの主なタイミング特性に対応するプロパティと共に返されます。 SLACK プロパティは指定したタイミング パス (この例の場合はワースト パス) のスラックを返します。スラックが負の場合、物理最適化を実行して、配置タイミング違反の解決を試みます。

手順4の最後にチェックポイントを保存し、デザインのタイミング サマリとデバイス使用率をレポートします。これにより、配線前と配線後のタイミングを比較し、配線のタイミングへの影響を評価できます。



- 手順5:タイミングドリブン配線を実行し、チェックポイントを保存します。これでメモリ内のデザインが配線されたので、追加のレポートを生成して、消費電力、デザインルール違反、最終的なタイミングに関する重要な情報を入手できます。レポートはファイルに出力するか、Vivado IDE に表示して確認できます。その後、タイミングシミュレーション用に Verilog ネットリストをエクスポートします。
- **手順** 6: デザインをザイリンクス FPGA にプログラムするビットストリームを生成します。

# Tcl プロシージャの定義

Vivado Design Suite には、完全な Tcl インタープリターがビルトインされており、新しいカスタム コマンドやプロシージャを簡単に作成できます。読み込む Tcl スクリプトを記述し、Vivado IDE 内から実行したり、プロシージャを記述して、引数を取り、エラーをチェックして、結果を返すデザイン コマンドとして使用できます。

Tcl プロシージャは proc コマンドで指定します。プロシージャ名、引数のリスト、実行するコードの本文を引数として指定します。次に、プロシージャ定義の簡単な例を示します。

```
proc helloProc { arg1 } {
    # This is a comment inside the body of the procedure
   puts "Hello World!Arg1 is $arg1"
}
```



**ヒント**: このプロシージャの定義では引数は1つなので中かっこで囲む必要はありませんが、中かっこを使用することでプロシージャ定義がわかりやすくなります。引数が複数ある場合は、中かっこは必須です。

通常プロシージャでは、定義済みの引数と、オプションでデフォルト値を指定します。return コマンドを使用して 値を返すよう指定していない場合は、空のリストが返されます。次の例では、3 つの定義済み引数を持つ reportWorstViolations というプロシージャを定義しています。

```
proc reportWorstViolations { nbrPaths corner delayType } {
   report_timing -max_paths $nbrPaths -corner $corner -delay_type $delayType -nworst 1
}
```

プロシージャを実行する際、コマンドを完了するには、次の例に示すようにすべての引数を指定する必要があります。

%> reportWorstViolations 2 Slow max
%> reportWorstViolations 10 Fast min

次の例では、同じプロシージャで3つの引数のうち2つのデフォルト値を設定しています。cornerのデフォルト値はSlow、delayTypeのデフォルト値はMaxです。デフォルト値が設定されているので、プロシージャを呼び出す際はcornerおよびdelayType引数の指定はオプションです。

```
proc reportWorstViolations { nbrPaths { corner Slow } { delayType Max } } {
  report_timing -max_paths $nbrPaths -corner $corner -delay_type $delayType -nworst 1
}
```

プロシージャを実行する際は、次のいずれの形式でも機能します。

```
%> reportWorstViolations 2
%> reportWorstViolations 10 Fast
%> reportWorstViolations 10 Slow Min
```

次のプロシージャの例には定義済みの引数 nbrPath がありますが、それ以外にも追加の引数を指定できます。この場合、プロシージャを定義する際に引数のリストとして Tcl キーワード args を使用します。args キーワードは、任意の数の要素 (0 を含む) を含む Tcl リストを示します。

```
proc reportWorstViolations { nbrPaths args } {
  eval report_timing -max_paths $nbrPaths $args
}
```



Tcl コマンドを実行する際、Tcl コマンドで使用可能なまたは必須のコマンド ライン引数の代わりに変数置換を使用できます。この場合、Tcl eval コマンドを使用してコマンドの一部として Tcl 変数を含めたコマンド ラインを評価する必要があります。上記の例では、引数のリスト変数 (\$args) が report\_timing コマンドに変数として渡されるので、eval コマンドが必要です。

プロシージャを実行する際は、次のいずれの形式でも機能します。

```
%> reportWorstViolations 2
%> reportWorstViolations 1 -to [get_ports]
%> reportWorstViolations 10 -delay_type min_max -nworst 2
```

最初の例では、値2が \$nbrPaths 引数に渡され、-max\_paths に適用されます。2番目と3番目の例では、それぞれ1と10が -max paths に適用され、その後の文字列は \$args に代入されます。

次の例は、非プロジェクト モードのサンプル スクリプトで使用されていた reportCriticalPaths コマンドを示します。このプロシージャでは1つの引数 \$filename が使用され、コメントで各セクションを説明しています。

```
#------
# reportCriticalPaths
#-----
# This function generates a CSV file that provides a summary of the first
# 50 violations for both Setup and Hold analysis. So a maximum number of
# 100 paths are reported.
#------
proc reportCriticalPaths { fileName } {
 # Open the specified output file in write mode
 set FH [open $fileName w]
 # Write the current date and CSV format to a file header
 puts $FH "#\n# File created on [clock format [clock seconds]]\n#\n"
 puts $FH "Startpoint, Endpoint, DelayType, Slack, #Levels, #LUTs"
 # Iterate through both Min and Max delay types
 foreach delayType {max min} {
   # Collect details from the 50 worst timing paths for the current analysis
   # (max = setup/recovery, min = hold/removal)
   # The $path variable contains a Timing Path object.
   foreach path [get timing paths -delay type $delayType -max paths 50 -nworst 1] {
    # Get the LUT cells of the timing paths
    set luts [get_cells -filter {REF_NAME =~ LUT*} -of_object $path]
    # Get the startpoint of the Timing Path object
    set startpoint [get_property STARTPOINT_PIN $path]
    # Get the endpoint of the Timing Path object
    set endpoint [get_property ENDPOINT_PIN $path]
    # Get the slack on the Timing Path object
    set slack [get_property SLACK $path]
    # Get the number of logic levels between startpoint and endpoint
    set levels [get_property LOGIC_LEVELS $path]
    # Save the collected path details to the CSV file
    puts $FH "$startpoint,$endpoint,$delayType,$slack,$levels,[llength $luts]"
 # Close the output file
 puts "CSV file $fileName has been created.\n"
 return 0
}; # End PROC
```

# デザイン オブジェクトへのアクセス

Vivado Design Suite では、プロジェクト、デザイン、デバイス情報がインメモリ データベースに読み込まれ、合成、インプリメンテーション、タイミング解析、およびビットストリームの生成に使用されます。このデータベースは、プロジェクト モードでも非プロジェクト モードでも同じです。FPGA デザイン フローを実行していくと、それに応じてデータベースがアップデートされます。デザイン フローのどの段階でも、データベースの内容をチェックポイント ファイル (.dcp) に保存できます。Vivado ツールで Tcl コマンドを使用すると、デザイン データベースにアクセスし、Tcl オブジェクトをクエリしたり、プロパティを読み出しまたは設定したりして、その結果を Tcl スクリプトでさまざまな目的で使用できます。データベースの内容を理解し、それに対してスクリプトをいかに効率的に記述できるかを理解しておくと有益です。

Vivado Design Suite の Tcl インタープリターでは、プロジェクト、デバイス、ネット、セル、ピンなど、多数のファースト クラス オブジェクトにアクセスできます。 Vivado Design Suite では、プロジェクト モードでも非プロジェクトモードでも、デザインの進行に応じてこれらのデザイン オブジェクトが随時アップデートされ、インメモリ データベースに読み込まれます。

対話的にデザイン オブジェクトのクエリ、プロジェクトの状態の解析、インメモリ デザインにアクセスするスクリプトの記述、カスタム レポートの生成、オプションのデザイン フロー手順などを実行できます。各オブジェクトには複数のプロパティがあり、いつでも読み出すことができ、また一部のプロパティは設定もできます。ほとんどのデザイン オブジェクトはほかのデザイン オブジェクトに関連付けられており、その関連性をたどって関連オブジェクトやその情報を取得できます。

デザインオブジェクトのクエリには、get\_\* Tcl コマンドを使用します。結果取得されたデザインオブジェクトは、直接処理するか、Tcl 変数に代入できます。オブジェクトを変数に代入すると、デザインデータベースに対するクエリの回数を削減でき、実行時間を短縮できます。ネットやピンのリストのクエリは時間のかかるプロセスであり、結果を保存しておくことで、同じ情報に繰り返しアクセスする必要がある場合にデザインフローを高速化できます。詳細は、27ページの「オブジェクトのキャッシュ」を参照してください。

デザイン オブジェクトの各クラス (ネット、ピン、ポートなど) には標準のプロパティがあり、読み出したり、一部のプロパティは値を変更できます。また、RTL ソース ファイルで指定されているデザイン属性、Verilog パラメーター、VHDL ジェネリックも、それらが設定されているネットリスト オブジェクトのプロパティとして保存されます。たとえば、ポート オブジェクトには方向を指定するプロパティがあり、ネット オブジェクトにはファンアウトを指定するプロパティがあります。Vivado ツールでは、これらのプロパティを追加、変更、およびレポートする多数のコマンドがあります。get\_\* -filter オプションを使用すると、デザイン オブジェクトのリストにフィルターを適用し、特定のプロパティ値のオブジェクトのみを取得できます。詳細は、15ページの「フィルター結果」を参照してください。

すべてのオブジェクトには、NAME および CLASS プロパティがあります。オブジェクトを変数に代入すると、そのオブジェクトへのポインターが変数に保存されます。オブジェクトを変数によりほかの Tcl コマンドや Tcl プロシージャに渡すことができます。ただし、デザインオブジェクトを含む変数が文字列を必要とする Tcl コマンドに渡された場合は、オブジェクトそのものではなくオブジェクトの NAME プロパティが渡されます。Vivado Tcl コンソールにデザインオブジェクトのみを返すコマンドで返されたオブジェクトの名前が表示されるのは、このためです。次の例に、タイミング パス オブジェクトを変数 \$path1 に代入し、その変数に対して puts コマンドおよびreport\_propertyコマンドを実行た結果を示します。putsコマンドではオブジェクトの名前のみが表示され、report propertyコマンドではオブジェクトのプロパティとその値が返されていることに注目してください。

set path1 [get\_timing\_paths -delay\_type max]

{usbEngine1/u4/inta\_reg/C -->
cpuEngine/or1200 du/tbar ram/ramb16 s36 s36/DIBDI[12]}

puts \$path1

{usbEngine1/u4/inta\_reg/C --> cpuEngine/or1200 du/tbar ram/ramb16 s36 s36/DIBDI[12]}



report property -all \$path1

| Property                                               | Type   | Read-only | Visible | Value                      |  |  |
|--------------------------------------------------------|--------|-----------|---------|----------------------------|--|--|
| CLASS                                                  | string | true      | true    | timing_path                |  |  |
| $DATAPATH\_DELAY$                                      | double | true      | true    | 7.972                      |  |  |
| DELAY_TYPE                                             | string | true      | true    | max                        |  |  |
| ENDPOINT_CLOCK                                         | clock  | true      | true    | cpuClk                     |  |  |
| ${\tt ENDPOINT\_PIN}$                                  | pin    | true      | true    |                            |  |  |
| cpuEngine/or1200_du/tbar_ram/ramb16_s36_s36/DIBDI[12]  |        |           |         |                            |  |  |
| GROUP                                                  | string | true      | true    | cpuClk                     |  |  |
| $LOGIC\_LEVELS$                                        | int    | true      | true    | 11                         |  |  |
| NAME                                                   | string | true      | true    | {usbEngine1/u4/inta_reg/C> |  |  |
| cpuEngine/or1200_du/tbar_ram/ramb16_s36_s36/DIBDI[12]} |        |           |         |                            |  |  |
| REQUIREMENT                                            | double | true      | true    | 10.000                     |  |  |
| SKEW                                                   | double | true      | true    | -0.010                     |  |  |
| SLACK                                                  | double | true      | true    | 1.816                      |  |  |
| $STARTPOINT\_CLOCK$                                    | clock  | true      | true    | usbClk                     |  |  |
| $STARTPOINT\_PIN$                                      | pin    | true      | true    | usbEngine1/u4/inta_reg/C   |  |  |
| UNCERTAINTY                                            | double | true      | true    | -0.202                     |  |  |

どのクラスのデザインオブジェクトに対しても、カスタムプロパティを作成できます。これは、デザインオブジェクトにスクリプトからの情報を追記する場合に有益です。次の例では、セルオブジェクトに対して SELECTED というプロパティを作成しています。プロパティ値は整数として定義されます。

create property SELECTED cell -type int

オブジェクトのクラスにプロパティを作成したら、set\_property および get\_property コマンドを使用して管理し、list\_property および report\_property コマンドを使用してレポートできます。次の例では、名前が \*aurora 64b66b\* というパターンに一致するすべてのセルの SELECTED プロパティを 1 に設定しています。

set\_property SELECTED 1 [get\_cells -hier \*aurora\_64b66b\*]

### デザイン階層での検索

ほとんどのデザインは、階層的に接続されたブロックまたはモジュールで構成されています。ボトムアップ、トップダウン、またはミドルアウトで構築されたデザインのいずれでも、デザイン階層で特定のオブジェクトを検索するのは一般的なタスクです。



図 1-1: デザイン階層の検索



get\_\* コマンドでは、デフォルトではデザイン階層の最上位のオブジェクトのみが返されます。get\_\* コマンドを使用する前に current\_instance コマンドを使用すると、デザインの特定の階層インスタンスでデザイン オブジェクトを検索できます。検索範囲をデザインの最上位に戻すには、current\_instance コマンドを引数を指定せずに実行します。

図 1-1 に、最上位にモジュール A および B がインスタンシエートされている例を示します。モジュール A には a1 および a2 階層インスタンスが含まれ、モジュール B には b1 および b2 階層インスタンスが含まれます。a1、a2、b1、および b2 には、それぞれ最下位セルが含まれます。

```
# Set the current instance of the design to module B.
current_instance B
get_cells * ; # Returns b1 and b2, cells found in the level of the current instance.
get_nets * ; # Returns nets from module B, the current instance.
# Reset the current instance to the top-level of the design.
current_instance
get_cells * ; # Returns A and B, located at the top-level of the design.
```

get\_\* コマンドでは最上位または current\_instance で指定した現在のインスタンスのレベルでのみ検索が実行されますが、現在のインスタンスに対する階層インスタンス名を含む検索パターンを指定できます。デフォルトでは、現在のインスタンスはデザインの最上位に設定されています。最上位からインスタンス b1 を参照するには、次のように指定します。

get cells B/b1; # Search the top-level for an instance with a hierarchical name.

#### -hierarchical オプションの使用

get\_\* コマンドでは、デフォルトでは現在のインスタンスのレベルでのみオブジェクトが検索されますが、-hierarchical オプションを使用すると、現在のインスタンスのレベルから各デザイン階層を検索できます。

```
get_cells -hierarchical * ; # Returns all cells in the hierarchy.
get_nets -hier *nt* ; # Returns all hierarchical nets that match *nt*.
```

-hierarchical オプションでは、オブジェクトの完全な階層名に対してではなく、デザイン階層の各レベルで指定された名前のパターンが検索されます。名前の検索パターンを指定する場合は、12ページの図 1-1 を使用した次の例に示すように、階層区切り文字を含めないでください。

```
get_cells -hierarchical B/* ; # No cell is returned.
get cells -hierarchical b* ; # B/b1 and B/b2 are returned.
```



**重要**: -hierarchical を -regexp と共に使用する場合、検索パターンは完全な階層名と比較され、検索パターンとして「B/\*」を指定した場合にこのパターンに一致するセル名が返されます。 -regexp の詳細は、『Vivado Design Suite Tcl コマンド リファレンス ガイド』(UG835) を参照してください。

-hierarchical を使用した検索は、current\_instance コマンドを使用して各階層レベルを指定し、指定の名前のパターンを手動で検索するのと同じです。次の例では、12ページの図 1-1 を使用してこの手動検索を実行しています。

```
set result {}
foreach hcell [list "" A B A/a1 A/a2 B/b1 B/b2] {
  current_instance $hcell ;# Move scope to $hcell
  set result [concat $result [get_cells <pattern>]]
  current_instance ;# Return scope to design top-level
}
```



#### ピンの検索



図 1-2:ピン名の検索

ピンの名前は、そのピンが属するインスタンスに基づいています。ピンを検索する場合、階層区切り文字を使用し、インスタンス名とピン名を区切る必要があります。次の例は、図 1-2 に示されています。

```
# Current instance is set to design top-level
get_pins B/* ; # Returns B/clk B/din0 B/din1 B/dout0 B/dout1
get_pins B/b2/*/0 ; # Returns B/b2/data_reg_i_1/0
current_instance B/b2 ; # Change scope to B/b2
get_pins *_reg/D ; # Returns B/b2/data_reg/D
```

ピンを検索する際、-hierarchical も使用できます。

```
current_instance ; # Reset to the top-level of the hierarchy
get_pins -hier */D # Returns pin objects for all D pins in the design
```



#### フィルター結果

get\_\* を使用してデザイン オブジェクトを検索する場合、通常必要なのは一部のオブジェクトのみです。デザイン のすべてのネットリスト オブジェクトは必要なく、特定のタイプのセルや特定の名前のネットのみなどが必要です。 要素の一部のみが返されるようにする必要があることもあります。



図 1-3: 階層デザインの検索

ワイルドカード\*および?を使用したり、-regexpを使用したりして検索パターンを指定し、返される検索結果を制限できます。検索する階層範囲を指定するには、current\_instance コマンドまたは -hierarchy オプションを使用します。

次の例では、図 1-3 に示すデザインに対する異なる結果を示します。

```
get_cells * ; # Returns 2 cells
get_cells -hier * ; # Returns 12 cells
get_cells -hier * -filter {NAME =~ */?1/*} ; # Returns 3 cells
```



-filter オプションを使用すると、 $get_*$  コマンドの結果を特定のプロパティに基づいてフィルターできます。たとえば次のコマンドでは、完全な階層名が「B/b\*」と一致するすべてのセルのうち、ユーザーにより配置されていないもの (IS LOC FIXED が FALSE または 0) のものが返されます。

set unLoced [ get cells -hier -filter {NAME =~ B/b\* && !IS LOC FIXED} ]

-filter オプションにより、結果がフィルターされてから返されます。ただし、フィルターを適用する前の検索結果を変数に代入している場合は、それがメモリに保存されます。filter コマンドを使用すると、変数として保存されているリストも含め、オブジェクトの任意のリストの内容をフィルターできます。先ほどの例の場合、\$unLocedに保存されているリストを次のようにフィルターできます。

filter \$unLoced {IS PRIMITIVE}

この例では、\$unLoced に保存されている結果をフィルターし、デザインのプリミティブ インスタンスのみを返しています。



**ヒント**:上記の例で、ブール型プロパティ!IS\_LOC\_FIXED および IS\_PRIMITIVE が直接使用されていることに注目してください。ブール型 (bool) プロパティでは、フィルター式が True か False かを直接評価できます。

フィルター パターンに使用できる演算子は等価 (==)、不等価 (!=)、含める (=~)、含めない (!~) です。数値比較演算子 <、>、<=、および >= も使用できます。複数のフィルター式を AND (&&) および OR (||) で組み合わせることもできます。

#### 関連性を使用したオブジェクトの検索

デザインのほかのオブジェクトに関連するオブジェクトを検索する必要がある場合があります。たとえば、特定のセルのピンに接続されているすべてのネットや、特定のネットに接続されているすべてのセルを選択する場合などです。 Vivado Design Suite では、デザインのエレメントをそれらの関連性を利用して検索できます。これには、get\_\*コマンドで -of\_objects オプションを使用します。図 1-4 に、インメモリ デザインのオブジェクト間の関連性を示します。



図 1-4: Vivado Design Suite でのオブジェクト間の関連性



注記:これは概念的に図示したものであり、オブジェクトとその関連性をすべて表すものではありません。

-of objects オプションをサポートする get \* コマンドのヘルプに、関連性のあるオブジェクトがリストされます。

```
get_cells -of_objects {pins, timing paths, nets, bels or sites}
get_clocks -of_objects {nets, ports, or pins}
get_nets -of_objects {pins, ports, cells, timing paths or clocks}
get_pins -of_objects {cells, nets, bel pins, timing paths or clocks}
get_ports -of_objects {nets, instances, sites, clocks, timing paths, io standards, io banks, package pins}
```

-of\_objects オプションを使用すると、ネット オブジェクトのリストに接続されているピン オブジェクトのリストを簡単に取得できます。

```
get pins -of objects [get nets -hier]
```

これらのネットのドライバーのリストのみを取得する場合は、-filterオプションを使用します。

```
qet pins -of [get nets -hier] -filter {DIRECTION == OUT}
```

また、セルのリストからピンのリストを取得したり、ネットのリストからセルのリストを取得したりできます。



図 1-5: 関連性を使用したオブジェクトの検索



次の例では、図 1-5 に示すように、instance al からクロックピンを取得し、そのクロックピンに接続されているネットを取得して、そのネットに接続されているピンを取得して、さらにそのピンに接続されているネットを取得して、最後にそれらのネットに接続されているピンを取得しています。

get\_pins -of [get\_nets -of [get\_pins -of [get\_nets -of [get\_pins A/a1/clk]]]]
A/a2/clk A/clk A/a1/clk B/clk

最後の get\_pins コマンドにより、それまでに返されたピンに加え、階層モジュール B のクロック ピン B/clk が返されます。階層をまたいでクロック ネット オブジェクトのプリミティブ ピンを取得するには、get\_pins コマンドの -leaf オプションを使用できます。次の例では、-leaf を使用した場合の結果を示します。

get\_pins -leaf -of [get\_nets -of [get\_pins -of [get\_nets -of [get\_pins A/a1/clk]]]]
B/b1/data\_reg/C A/a2/data\_reg/C A/a1/data\_reg/C B/b2/data\_reg/C

# オブジェクトのリストの処理

get\_\*コマンドを使用すると、返されたオブジェクトは標準の Tcl リストと同様で、同じよう機能しますが、Vivado Design Suite ではオブジェクトの1つのクラスのコンテナー (セル、ネット、ピン、ポートなど) が返され、標準の Tcl リストとは異なります。ただし、このコンテナーは通常、標準の Tcl リストと同様で、同じように機能します。オブジェクトのコンテナーは Vivado Design Suite で自動的に処理されるので、ユーザーが意識する必要はありません。たとえば、標準 Tcl コマンド 1length をオブジェクトのコンテナーに対して使用し (get\_cells の結果など)、通常の Tcl リストでの場合と同様に、コンテナーに含まれるエレメントの数を取得できます。

Vivado Design Suite のリストを処理するビルトインの Tcl コマンドは、オブジェクトおよびオブジェクトのコンテナーを完全にサポートするため拡張されています。 たとえば、1sort、1append、1index、および 1length は、オブジェクトの NAME プロパティに基づいてコンテナーを制御するよう拡張されています。 これらのコマンドは、オブジェクトのコンテナーが渡された場合、オブジェクトのコンテナーを返します。

たとえば、1sort は、get\_cellsで取得したセルのコンテナーを、オブジェクトの階層名に基づいて並べ替えます。

lappend を使用するなどしてコンテナーにオブジェクトを追加できますが、現在コンテナーに含まれるオブジェクトと同じタイプのオブジェクトしか追加できません。異なるタイプのオブジェクトや文字列を追加しようとすると、Tcl エラーが返されます。

次の例では、Vivado のコンテナーを降順に並べ替え、puts コマンドと foreach ループを使用して、オブジェクトを1行に1つずつ表示しています。

foreach X [lsort -decreasing [get\_cells]] {puts \$X}
wbArbEngine
usb\_vbus\_pad\_1\_i\_IBUF\_inst
usb\_vbus\_pad\_0\_i\_IBUF\_inst
usbEngine1
usbEngine0



# 出力先の指定

Vivado Design Suite の多くの Tcl コマンドでは、コマンドから返される情報を、-file オプションを使用して印刷やツール外での処理用にファイルに保存したり、-return\_string オプションを使用して Vivado ツールでの処理用に文字列として変数に保存できます。

すべてのレポート コマンドで、-file オプションがサポートされています。大量の情報を出力するレポート コマンドでは、その後の解析、デザイン プロジェクトの文書サポート、またはほかの部署にダウンストリーム処理用に渡す場合などに、ファイルに出力すると有益です。次に、ファイル出力をサポートするコマンドの一部を示します。

report\_datasheet
report\_drc
report\_power
report\_timing
report\_timing\_summary
report\_utilization

たとえば、report timingコマンドの結果をファイルに記述するには、次のコマンドを使用します。

```
report_timing -delay_type max -file setup_violations.rpt
report timing -delay type min -file hold violations.rpt
```

ファイル名の一部として、相対パスまたは絶対パスを指定できます。相対パスは、Vivado ツールを起動したディレクトリまたは pwd コマンドで返される現在の作業ディレクトリを基準として指定します。



**ヒント**: パスをファイル名の一部として指定しない場合は、現在の作業ディレクトリまたは Vivado ツールを起動した ディレクトリにファイルが作成されます。

既存のファイルにコマンドの結果を追加するには、-file オプションと共に -append オプションを使用します。次の例では、1つのファイル all violations.rpt を作成し、2つのコマンドの結果を保存しています。

```
report_timing -delay_type max -file all_violations.rpt report_timing -delay_type min -file -append all_violations.rpt
```

ファイルが作成されたら、ファイルシステムからファイルを開いて確認したり書き込んだりできます。Tclシェルには、ファイルにアクセスするさまざまなコマンドがあります。詳細は、20ページの「ファイルへのアクセス」を参照してください。

多くの report\_\* コマンドでは、-return\_string オプションもサポートされています。このオプションを使用すると、コマンドの出力が Tcl 変数に代入可能な文字列として返されます。出力を文字列変数に代入すると、Tcl スクリプトでのその後の処理に有益で、主要な情報を抽出してフロー制御や分岐を可能にしたり、スクリプトで使用するほかの変数を設定したりできます。

次に、-return string オプションをサポートするコマンドの一部を示します。

report\_clocks
report\_clock\_interaction
report\_disable\_timing
report\_environment
report\_high\_fanout\_nets
report\_operating\_conditions
report\_power
report\_property
report\_pulse\_width
report\_route\_status
report\_utilization



レポート コマンドから返された文字列を改行文字 \n で分離して、文字列をリストとして行ごとに処理できます。

set timeLines [split [report timing -return string -max paths 10] \n ]

Tcl シェルには、文字列を操作するさまざまなコマンドがあります。詳細は、22 ページの「文字列の操作」を参照してください。

#### ファイルへのアクセス

ファイルシステムにファイルを記述すると、ファイルを処理するさまざまな Tcl コマンドを使用できます。ファイルパス、ファイル名、ファイル拡張子など、ファイルの要素を抽出できます。ファイルに関する情報を調べる次のようなコマンドもあります。

- file exists filename: filename が存在し、その場所の読み取り権限がある場合は 1、それ以外の場合は 0を返します。ファイルが既に存在しているかどうかを調べるのに使用します。
- file type *filename*:ファイルのタイプを示す文字列を返します。可能な値は file、directory、characterSpecial、blockSpecial、fifo、link、socket です。
- file dirname filename: filename の最後のスラッシュまでのディレクトリ構造を最後のスラッシュを含めずに返します。
- file rootname filename: filenameの最後のピリオドまでの文字を最後のピリオドを含めずに返します。
- file tail filename: filenameの最後のスラッシュより後の文字すべてを返します。
- file extension filename: filename の最後のピリオド以降の文字を最後のピリオドを含めて返します。

次に、使用可能な Tcl コマンドのいくつかの例を示します。

```
set filePath {C:/Data/carry_chain.txt}
file dirname $filePath ; # Returns C:/Data
file tail $filePath ; # Returns carry_chain.txt
file extension $filePath ; # Returns .txt
```

report\_\* コマンドまたは write\_\* コマンドでファイルを作成したら、Tcl スクリプトでファイルを開いて、その内容を読み出したり、追加の内容を記述したりできます。ファイルを開いたり閉じたり、ファイルの読み出しまたは書き込みを実行するには、次の Tcl コマンドを使用できます。

- open <filename> [access] [perms]:filename を開き、ファイルにアクセスするのに使用したファイルハンドル(fileID)を返します。必要に応じてファイルハンドルを参照できるようにするため、fileIDを Tcl変数に代入するのが一般的です。新しいファイルの権限は、perms とプロセス umask の組み合わせで設定します。access モードは、開いたファイルの読み取り権および書き込み権を指定します。一般的なアクセスモードは次のとおりです。
  - 。 r: 読み出しモード。ファイルが存在している必要があり、作成はされません。アクセス モードを指定しない場合、これがデフォルトです。
  - 。 w:書き込みモード。ファイルが存在しない場合は、作成されます。データはファイルの冒頭から記述され、 既存のファイルの内容は切り捨てられるか、上書きされます。
  - 。 a: 追加モード。ファイルが存在しない場合は、作成されます。データはファイルの末尾に記述され、既存のファイルの内容に追加されます。
- read [-nonewline] fileId:fileIdから残りすべてのバイトを読み出し、オプションで最後の文字が改行 \n の場合はその最後の文字を破棄します。ファイルを開いた直後にこの形式を使用すると、read コマンドでファイル全体が一度に読み出されます。
- read fileId numBytes: fileIdから指定したバイト数 numBytes を読み出します。この形式は、ファイルの ブロックをファイルの最後まで読み出す場合に使用します。
- eof fileId: fileIdで EOF (End Of File) が発生した場合は 1、それ以外の場合は 0 を返します。



• gets fileId [varName]: fileIdから次の行を読み出します。改行文字は破棄されます。\$varName を指定した場合は行の文字列をその変数に代入し、それ以外の場合は文字列をコマンドシェルに返します。次に、getsコマンドの異なる形式を示します。

```
gets $fileHandle

Append line 4 of file.

gets $fileHandle line

28

puts $line

Append line 5 of file.

set line [gets $fileHandle]

Append line 6 of file.

puts $line

Append line 6 of file.
```

上記の例では、\$fileHandle がファイルを開いたときに返されるファイル ハンドルです。最初の例は gets の単純な形式で、出力を代入する Tcl 変数は指定していません。この場合、出力は stdout に返されます。2番目の例では出力を \$line という変数に代入しており、gets コマンドで読み出された文字数 28 が返されます。

注記: gets コマンドの戻り値を Tcl 変数に代入できますが、コマンドの形式によって、ファイルの内容または gets コマンドで読み出された文字数が代入されます。

- puts [-nonewline] [fileId] string:指定した fileId に文字列を書き込みます。オプションで、改行文字 \n を省くこともできます。puts コマンドのデフォルトの fileId は stdout です。
- close *fileId*:開いているファイル チャネル *fileId*を閉じます。Tcl スクリプトで開いたファイルを閉じる ようにすることが重要です。そうしないと、Vivado アプリケーションでメモリ リークやその他の問題が発生す る可能性があります。

次の例では、ファイルを読み出しアクセス モードで開き、ファイル ハンドルを \$FH に代入して、1 つの操作でファイルの内容を読み出して \$content に代入し、その内容を Tcl リストに分割しています。完了したら、ファイルを閉じます。

```
set FH [open C:/Data/carry_chains.txt r]
set content [read $FH]; # The entire file content is saved to $content
foreach line [split $content \n] {
    # The current line is saved inside $line variable
    puts $line
}
close $FH
```

注記:パフォーマンスおよびメモリの面から、サイズの大きいファイルを1回の操作で読み出すことはお勧めしません。

次の例では、ファイル全体を一度に読み出してから結果を解析するのではなく、ファイルを1行ずつ最後まで読み出し、stdoutに行数と行の内容を出力しています。完了したら、ファイルを閉じます。

```
set FH [open C:/Data/carry_chains.txt r]
set i 1
while {![eof $FH]} {
    # Read a line from the file, and assign it to the $line variable
    set line [gets $FH]
      puts "Line $i:$line"
      incr i
}
close $FH
```



次の例では、デザインのすべての I/O ポートをその方向と共に名前順に並べ替えて、ファイル ports.rpt に書き込んでいます。

```
set FH [open C:/Data/ports.rpt w]
foreach port [lsort [get_ports *]] {
  puts $FH [format "%-18s %-4s" $port [get_property DIRECTION $port]]
}
close $FH
```

上記の例では、ファイルを書き込みモードで開いています。読み出しモードとは異なり、書き込みモードではファイルが存在していない場合は作成され、ファイルが存在している場合は上書きされます。既存のファイルの最後に新しい内容を追加するには、ファイルを追加モードで開く必要があります。

#### 文字列の操作

-return\_string オプションを使用すると、report\_\* コマンドの出力を stdout ではなく Tcl 文字列として返すことができます。文字列は Tcl 変数に代入したり、解析または処理できます。

```
set rpt [report_timing -return_string]
```

文字列を変数に代入すると、文字列を処理するさまざまな Tcl コマンドを使用できます。

- append string [arg1 arg2 ... argN]:指定した args を string の最後に追加します。
- format formatString [arg1 arg2 ... argN]: 文字列を formatString テンプレートで指定した フォーマットの形式にします。テンプレートは、sprintf で使用されるように%変換指示子を使用して指定する必要があります。追加の引数 arg は、フォーマットされた文字列内で置換する値を指定します。
- regexp [switches] exp string: 正規表現 exp が stringに一致する場合は 1、それ以外の場合は 0を返します。-nocase オプションを指定すると、大文字/小文字は区別されません。
- string match pattern string:glob pattern が string に一致する場合は 1、それ以外の場合は 0 を返します。
- scan *string formatString* [*varName1 varName2 ...*] : 指定の string から値を抽出して変数 varName に代入し、sscanf と同じように % 変換指示子を使用して formatString を適用します。 *varName* を指定しない場合は、値のリストが stdout に出力されます。
- string range string first last: stringから、文字インデックス firstから last までを、それらを 含めて返します。
- string compare string1 string2:2つの文字列に対して辞書式比較を実行し、string1が string2より前の場合は-1、2つが同じ場合は0、string1が string2より後の場合は1を返します。
- string last string1 string2: string2で string1が最初に現れた文字インデックスを返します。 string2で string1が見つからなかった場合は-1を返します。
- string length *string*: *string*の文字数を返します。



次の例では、-return\_string を使用して report\_timing コマンドの結果を \$report Tcl 変数に代入し、各パスの開始点、終点、パスグループ、およびパス タイプを抽出して、最後にそのパスのサマリを Tcl コンソールに出力しています。

```
# Capture return string of timing report, and assign variables
set report [report timing -return string -max paths 10]
set startPoint {}
set endPoint {}
set pathGroup {}
set pathType {}
# Write the header for string output
puts [format " %-12s %-12s %-20s -> %-20s" "Path Type" "Path Group" "Start Point" "End Point"]
puts [format " %-12s %-20s -> %-20s" "-----" "------" "-----"
# Split the return string into multiple lines to allow line by line processing
foreach line [split $report \n] {
 if \{[regexp -nocase -- \{^s*Source:\s*([^[:blank:]]+)((\s+\(?)|\$)\} \\ \$line - startPoint]\} \\ \{ fine (section of the property 
} elseif {[regexp -nocase -- ^\s=Destination:\s*([^[:blank:]]+)((\s+\(?)|$)} $line - endPoint]} {
} elseif {[reqexp -nocase -- {^\s*Path Group:\s*([^[:blank:]]+)\s*$} $line - pathGroup]} {
} elseif {[regexp -nocase -- ^{\star} *Path Type:\s*([^[:blank:]]+)((\s+\(?)|$)} $line - pathType]} {
puts [format " %-12s %-12s %-20s -> %-20s" $pathType $pathGroup $startPoint $endPoint]
```

次は、このコードの出力例です。

| Path Type | Path Group | Start Point -> End Point                                                            |
|-----------|------------|-------------------------------------------------------------------------------------|
|           |            |                                                                                     |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[0]</pre>  |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[10]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[11]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[12]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[13]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[14]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[15]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[16]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[17]</pre> |
| Setup     | bftClk     | <pre>ingressLoop[0]/ram/CLKBWRCLK -&gt; transformLoop[0].ct/xOutReg_reg/A[18]</pre> |

# カスタム DRC の作成

Vivado Design Suite では、Tcl でカスタム デザイン ルール チェックを定義し、使用できます。プロセスは次のとおりです。

- 1. デザイン オブジェクトまたはオブジェクトの属性、およびデザイン ルールを定義するチェック関数を取得する Tcl プロシージャを記述します。Tcl チェッカー プロシージャは別の Tcl スクリプトで定義し、report\_drc を実行する前に Vivado Design Suite に読み込む必要があります。
- 2. Tcl チェッカー内で create\_drc\_violation コマンドを使用して、デザインでルールをチェックした際に検出される違反を指定します。このコマンドにより違反オブジェクトがインメモリ デザインに作成され、そのプロパティをレポートしてその後の処理に使用できます。
- 3. create\_drc\_check コマンドを使用して、DRC ルールの名前を -rule\_body で指定した Tcl チェッカー プロシージャに関連付けるユーザー定義 DRC ルール チェックを定義します。report\_drc コマンドを実行する際、このルールを名前で呼び出します。
- 4. create\_drc\_ruledeck コマンドを使用してルール デックを作成し、add\_drc\_checks コマンドを使用して ユーザー定義 DRC ルールをルール デックに追加します。
- 5. ルール デックまたはユーザー定義 DRC ルールを指定して report\_drc を実行し、違反がないかどうかをチェックします。



#### Tcl DRC チェッカーの記述

デザイン ルールを定義する Tcl スクリプトである Tcl チェッカー プロシージャは、まずチェックするデザイン オブジェクトを選択し、必要なテストまたは評価を実行して、エラーに関連するオブジェクトを特定する DRC 違反オブジェクトを返します。

次の Tcl スクリプトは、WRITE\_B バスの幅をチェックする dataWidthCheck プロシージャを定義しています。 report\_drc を実行する前に、この Tcl スクリプト ファイルを Vivado Design Suite に読み込む必要があります。

```
# This is a simplistic check -- report BRAM cells with WRITE_WIDTH_B wider than 36.
proc dataWidthCheck {} {
 # list to hold violations
 set vios {}
 # iterate through the objects to be checked
 foreach bram [get cells -hier -filter {PRIMITIVE SUBGROUP == bram}] {
   set bwidth [get_property WRITE_WIDTH_B $bram]
   if { $bwidth > 36} {
     # define the message to report when violations are found
    set msg "On cell %ELG, WRITE_WIDTH_B is $bwidth"
    set vio [ create_drc_violation -name {RAMW-1} -msg $msg $bram ]
    lappend vios $vio
    }; # End IF
   }; # End FOR
   if {[llength $vios] > 0} {
    return -code error $vios
   } else {
    return {}
 }; # End IF
} ; # End PROC
```

proc 定義からわかるように、dataWidthCheck プロシージャに引数はありません。必要な情報はすべてデザインから取得されます。

空のリスト変数 \$vios を作成し、create drc violation コマンドで返された違反オブジェクトを保存します。

チェックするデザイン ルールに関連するデザイン オブジェクトを選択し (この場合は BRAM)、各セルの WRITE WIDTH Bプロパティを取得して、値が36を超える場合は違反とします。

違反が検出されると、セルのプレースホルダー値 %ELG と バス幅 \$bwidth を含むメッセージ \$msg を作成します。dataWidthCheck プロシージャでは、create\_drc\_violation コマンドで1つのオブジェクト \$bram のみが返され、メッセージ文字列に定義されている %ELG プレースホルダーに代入されます。



**重要**: create\_drc\_violation コマンドで渡される順序とタイプが、create\_drc\_check コマンドの -msg の指定と一致している必要があります。

WRITE\_WIDTH\_Bプロパティの幅が指定の値を超えている BRAM が検出されるたびに、create\_drc\_violation で違反オブジェクトが作成されます。違反オブジェクトは、名前は関連付けられている Vivado Design Suite の DRC ルールの名前と同じになり、dataWidthCheck プロシージャで定義されたメッセージ文字列を含み、ルールに違反 するオブジェクトを特定します。デザイン ルール違反で返されるオブジェクトには、セル、ポート、ピン、ネット、クロック領域、デバイス サイト、パッケージ I/O バンクなどがあります。dataWidthCheck プロシージャに示すように、違反からのメッセージ文字列には、特定のプロパティ値などのほかの情報を含めることもでき、DRC レポートに必要な詳細情報を提供できます。

違反が検出された場合、dataWidthCheck プロシージャから report\_drc コマンドにチェックの結果を通知するエラー コードが返されます。

return -code error \$vios

エラーコードに加え、\$vios変数により、プロシージャで作成された違反オブジェクトのリストが返されます。



#### Vivado Tcl DRC コマンド

Tcl チェッカー プロシージャを定義したら、Vivado Design Suite 内での DRC レポート システムの一部として DRC チェックを定義する必要があります。

まず create\_drc\_check コマンドを使用して、新しいデザインルールを登録します。このコマンドを使用すると、ユーザー定義ルール チェックに固有の名前または略称を指定できます。これらの名前は、Tcl チェッカー プロシージャで作成した違反の名前と一致している必要があります。先ほど定義した dataWidthCheck プロシージャでは、create\_drc\_violation コマンドで RAMW-1 という名前を使用しているので、DRC チェックを作成する際はこの名前を指定する必要があります。

create\_drc\_check -name {RAMW-1} -category {RAMB Checks} \
 -desc {Block RAM Data Width Check} -rule body dataWidthCheck

DRC チェックをカテゴリにグループ化し、レポート用にルールの説明を指定することもできます。

違反が検出されたときに DRC レポートに追加するメッセージを定義できます。デフォルトでは、Tcl プロシージャの create\_drc\_violation で作成されたメッセージが DRC チェック オブジェクトに渡されます。この場合、create\_drc\_violation の -rule\_body で定義されたメッセージが DRC レポートに記述されます。

最後に、-rule\_body オプションを使用して、ルールをチェックするときに Vivado Design Suite で実行する Tcl プロシージャの名前を指定します。必要なチェックが完全に定義されるようにするため、report\_drc コマンドを実行する前にプロシージャを Vivado Design Suite に読み込んでおく必要があります。

プロシージャが読み込まれていれば、report\_drc コマンドを使用して DRC ルール チェックを個別にまたはほかの ルールと共に実行できます。 create\_drc\_ruledeck および add\_drc\_checks コマンドを使用して、一緒に実行 する関連のルールをグループ化したルール デックを作成できます。ルール デックから DRC チェックを削除するに は、remove\_drc\_checks コマンドを使用します。

DRC ルール チェック オブジェクトには is\_enabled プロパティがあり、set\_property コマンドを使用して TRUE または FALSE に設定できます。新しいルール チェックを作成すると、is\_enabled プロパティはデフォルトで TRUE に設定されます。 report\_drc を実行したときにルール チェックが使用されないようにするには、is\_enabled プロパティを FALSE に設定します。これにより、新しい DRC チェックを作成し、add\_drc\_checks を使用してルール デックに追加した場合に、そのチェックをルール デックから削除せずにイネーブルにしたりディスエーブルにしたりできます。

# Tcl スクリプトの読み込みと実行

Vivado Design Suite では、デザイン セッション中に Tcl スクリプトを読み込んで実行するのに複数の方法があります。 ツールを起動したときにスクリプト ファイルが自動的に読み込まれるようにするか、Tcl コマンド ラインで source コマンドを使用して読み込むか、Vivado IDE のメニューに追加します。

Vivado Design Suite で Tcl スクリプトが自動的に読み込まれるようにするには、init.tcl ファイルで定義します。この方法は、新しいコマンドを定義する Tcl プロシージャを記述し、Vivado のすべてのセッションで使用できるようにする場合に有益です。

Vivado ツールを起動すると、次の2箇所でTcl 初期化スクリプトが検索されます。

- 1. ツールのインストール ディレクトリ: <installdir>/Vivado/version/scripts/init.tcl
- 2. ローカルのユーザーディレクトリ:
  - a. Windows 7: %APPDATA%/Roaming/Xilinx/Vivado/init.tcl
  - b. Linux: \$HOME/.Xilinx/Vivado/init.tcl

<installdir>は Vivado Design Suite のインストール ディレクトリです。



init.tcl が両方の場所で見つかった場合、まず Vivado ツールのインストール ディレクトリにあるファイルが読み込まれ、次にホーム ディレクトリにあるファイルが読み込まれます。

インストール ディレクトリにある init.tcl ファイルを使用すると、企業またはデザイン グループのすべてのユーザーに対して共通の初期化スクリプトをサポートできます。 そのインストール ディレクトリから Vivado ツールを起動すると、共通の init.tcl スクリプトが使用されます。

ホーム ディレクトリにある init.tcl ファイルを使用すると、各ユーザーがそれぞれコマンドを追加したり、デザイン要件を満たすためにツールのインストール ディレクトリに含まれるコマンドを変更できます。

この init.tcl スクリプトは標準の Tcl スクリプト ファイルで、Vivado ツールでサポートされるどの Tcl コマンドも 含めることができます。source コマンドを追加して、init.tcl から別の Tcl スクリプト ファイルを読み込むこともできます。

source コマンドを使用すると、Tcl スクリプト ファイルを Vivado ツールに手動で読み込むことができます。

source <filename>

<filename> はファイル名とファイルの相対パスまたは絶対パスを指定します。パスをファイル名の一部として指定しない場合は、現在の作業ディレクトリまたは Vivado Design Suite ツールを起動したディレクトリにファイルが作成されます。

Vivado IDE で Tcl スクリプトを読み込むには、[Tools] → [Run Tcl Script] をクリックします。

デフォルトでは、ファイルの各行が Tcl コンソールに表示されます。表示されないようにするには、-notrace オプションを使用します。これは、Vivado Tcl インタープリターに特有のオプションです。

source <filename> -notrace

また、 $[Tools] \rightarrow [Custom Commands] \rightarrow [Customize Commands]$  を使用して、 $Vivado\ IDE\ のメイン\ メニューおよびツールバーにシステムまたはユーザー定義の Tcl コマンドを追加できます。カスタム コマンドをメニューに追加する方法は、<math>\mathbb{C}$ Vivado Design Suite ユーザーガイド:  $Vivado\ IDE\ の使用』(UG893)\ の「カスタム メニュー コマンドの追加」を参照してください。$ 



# Tcl スクリプト記述のヒント

いくつかの規則に従うことで、Tcl スクリプトの実行時間と効率を向上できます。次に、Vivado Design Suite で Tcl スクリプト機能を使用する際の推奨事項を示します。

#### オブジェクトのキャッシュ

オブジェクトまたはオブジェクトのリストを Tcl 変数にキャッシュし、再利用します。

たとえば、同じネットのリストをスクリプトで複数回使用する場合は、同じクエリを繰り返し実行するのは効率的ではありません。Vivado ツールの Tcl コマンドは効率的に実行されますが、Tcl クエリを実行するたびに Tcl インタープリターとアプリケーションの 下位 C++ コードの間を行き来することになります。この C++/Tcl のアクセスに時間がかかるので、できる限り避けるようにします。

Vivado ツールの異なるフィルター機能をできる限り利用します。効果的な検索パターン、-of\_objects オプション、-filter オプション、および filter コマンドを使用することで実行時間を短縮できます。これらの機能はアプリケーションの下位にインプリメントされており、実行時間およびメモリの点で非常に効率的です。

クエリの結果をキャッシュして、そのオブジェクトのリストに対して filter コマンドを実行してオブジェクトのサブリストを作成できます。また、インメモリデザインにアクセスせずに、標準 Tcl コマンドを使用して Tcl 変数に代入された結果を解析できます。

```
set allCells [get_cells * -hier]
lsort $allCells ; # Returns a sort ordered list of all cells
filter $allCells {IS_PRIMITIVE} ; # Returns only the primitive cells
filter $allCells {!IS_PRIMITIVE} ; # Returns non-primitive cells
```

#### オブジェクト名と NAMF プロパティ

デザイン オブジェクトを必要とする Tcl コマンドと、文字列入力を必要とする Tcl コマンドがあります。Vivado Design Suite では、文字列引数を必要とする Tcl コマンドであっても、デザイン オブジェクトを直接渡すことができます。この場合、デザイン オブジェクトの階層名が文字列として Tcl コマンドに渡されます。オブジェクトの NAME プロパティを取得して Tcl コマンドに渡す必要はありません。

たとえば、次の regexp コマンドでは、2つの if 文は同等で、オブジェクトの名前が渡されます。

```
if {[regexp {.*enable.*}$MyObject]} { ...}
if {[regexp {.*enable.*}[get_property NAME $MyObject]]} { ...}
```

上記の例では、最初のコードの方が読みやすいだけでなく、オブジェクトのプロパティを取得する必要がないので実行時間も短くなります。2番目の例では get\_property コマンドにより、オブジェクト プロパティを取得するため、Tcl インタープリターと下位 C++ アプリケーション コードの間にアクセスが発生します。これを複数のオブジェクトに対してループで実行すると、Tcl スクリプトの実行時間が大幅に増加します。

#### オブジェクトのリストのフォーマット

get\_\* コマンドから返されたリストはフォーマットされておらず、stdout にスペースで区切られて1行で表示されます。この例を次に示します。

```
get_cells
```

A B clk\_IBUF\_inst rst\_IBUF\_inst din0\_IBUF\_inst din1\_IBUF\_inst dout0\_OBUF\_inst dout1 OBUF inst dout2 OBUF inst dout3 OBUF inst clk IBUF BUFG inst



このフォーマットされていないリストでは、Tcl コンソールおよび Vivado IDE で何が返されたかを確認するのが困難です。リストの各アイテムを個別の行に表示するには、次のようにコマンドを join コマンドにネストし、改行文字 \n を追加します。

```
join [get_cells] \n
A
B
clk_IBUF_inst
rst_IBUF_inst
din0_IBUF_inst
din1_IBUF_inst
dout0_OBUF_inst
dout1_OBUF_inst
dout1_OBUF_inst
dout2_OBUF_inst
dout3_OBUF_inst
clk_IBUF_BUFG_inst
```

get\_\*コマンドで返されるリストは、joinコマンドでは変更されません。

### Vivado Tcl コマンドをオプションで検索

次のプロシージャ findCmd を使用すると、Vivado Design Suite のすべての Tcl コマンドの構文が検索され、指定のオプションをサポートするコマンドがリストされます。

```
proc findCmd {option} {
  foreach cmd [lsort [info commands *]] {
    catch {
    if {[regexp "$option" [help -syntax $cmd]]} {
      puts $cmd
    }
    }
  }
}; # End proc
```

たとえば、-return\_string オプションをサポートする Vivado ツール コマンドを検索するには、次のコマンドを使用します。

findCmd return string



# その他のリソース

# ザイリンクス リソース

アンサー、資料、ダウンロード、フォーラムなどのサポート リソースは、次のザイリンクス サポート サイトを参照 してください。

http://japan.xilinx.com/support

ザイリンクス資料で使用される用語集は、次を参照してください。

http://japan.xilinx.com/company/terms.htm

# ソリューション センター

デバイス、ツール、IP のサポートについては、<u>ザイリンクス ソリューション センター</u>を参照してください。トピックには、デザイン アシスタント、アドバイザリ、トラブルシュート ヒントなどが含まれます。

# リファレンス

Vivado Design Suite 2012.3 の資料

 $\underline{http://japan.xilinx.com/support/documentation/dt\_vivado\_vivado2012-3.htm}$ 

Tcl Developer Xchange

www.tcl.tk