创建完成工程之后,需要创建顶层实体。创建完顶层设计文件之后,使用SOPC Builder创建NIOS II 嵌入式处理器,添加、配置系统的外设IP,组成Nios II系统模块。 Nios II 系统模块设计完成之后要加入到该顶层实体中,然后进行其他片上逻辑的开发。( p' Y1 g* ~. F& V: g+ f) ]
2.3.1 创建顶层实体. ^' G- K2 {+ S& ^9 W8 S% B
(1)新建的工程窗口中,选择FileNew;
& N# I+ _9 D0 V7 L3 j/ L(2)在Device Design File页中,选择Block Diagram/Schematic File,即原理图文件,也可以选择硬件描述语言的文件形式。单击OK。 0 h( I3 K' O: t) P
(3)出现一个模块编辑窗口。+ \8 R, h3 @7 y- b$ n6 b
(4) 选择FileSave As,出现Save As对话框,显示的目录为之前设置的工程目录,文件名为之前设置的顶层实体名(由于这是工程的第一个文件,系统会默认为顶层设计实体的名字)。确定Add to Current Project选项被选中,点击Save文件被保存并被加入到工程中。 & H6 Z2 z0 a8 ?& p! g
# D! }- y+ U* b& {0 ~2.3.2 创建Nios II系统模块
, N& S+ ~$ g4 Q' X6 N 创建NIOS II系统模块需要使用SOPC Builder,它是Quartus II中的一个工具,使用SOPC Builder可以创建一个Nios II 系统模块,或者创建多主设备SOPC模块。一个完整Nios II的系统模块包括Nios II处理器和相关的系统外设。所以创建系统模块的流程是先创建一个系统,然后添加Nios II CPU和外设IP,然后进行相应的设置,最后生成实例,然后将其加入到工程的顶层实体中去。 5 T f+ j* m d% f; n% Q3 u& x! K- X
一、创建系统 : M6 \4 p4 S6 i" X$ O0 W6 T: N
启动SOPC Builder,选择ToolsSOPC Builder,出现图2-10的Create New System对话框。键入系统的名字,选择硬件描述语言Verilog或者是VHDL。 ; r! |7 O }% n' r& l$ f
8 o% Z0 G' u0 o( i2 }+ z, P4 b$ I
5 {$ f4 S% ~0 g" y* o6 d& B1 V& T8 b% z+ ~
二 、设置系统主频和指定目标FPGA - Z, J+ z7 d5 I
在Target栏中的Board部分指定我们这本书使用的核心开发板——Cyclone II (EP2C35)。也可以在Board部分选择Unspecified,然后在Device Family选择Cyclone II。 + x2 I; `1 h) I: H
用户需要设置系统的时钟频率,该频率用于计算硬件和软件开发中的定时,比如时钟分频或波特率。这里我们设成85MHz,还可以选择是否选用流水线。
F3 g) W5 ^, @4 g/ X9 }3 P三、加入Nios II CPU和 IP模块
J" O7 ] d1 m 首先加入Nios II软核,Nios II 是软核CPU,共有三种类型的CPU可供选择:Nios II/e(经济型)、Nios II/s(标准型)和Nios II/f(快速型)。用户可以根据实际的情况进行选择。Nios II是一个用户可以自行进行定制的CPU,用户可以增加新的外设、新的指令等。
. ^; Y, a- a' v: p$ h" b- r添加Nios II CPU的步骤如下:
6 P' y4 j/ p W5 }; s1. Avalon Components 下面选择Nios II Processor; 8 S) r$ a! a# @2 `$ f# U# Y, A$ B
2. 点击 Add,出现Nios II CPU的配置向导,共有三种类型的CPU可供选择; & X1 I$ v- R; v$ `& ^
根据需要选择相应的一种Nios II核,我们选择标准型的Nios II核,Hardware Multiply选择none,不选择Hardware Divide点击Next,进入Caches & Tightly Coupled Memories设置窗口; / R0 N6 |3 k% B- {
设置Nios II的Cache和与CPU直接相连的存储器端口(不通过Avalon总线),选择Instruction Cache为4Kbytes,不选中Include tightly coupled instruction master port(s) ,点击Next进入Advanced Features设置页面。
2 h/ d3 K* k+ B Advanced Features的设置页面是Quartus II 6.1 版本才有的设置,这里不选择Include cpu_resetrequest and cpu_resettaken signals点击Next进入JTAG Debug Module设置。 共有4个调试级别可供选择,这里选择Level1即可,该级别支持软件的断点调试。JTAG调试模块要占用较多的逻辑资源,如果整个系统调试完毕了可以选用No Debugger以减少系统占用资源点击Next进入自定义指令的设置。
- M6 v, \% V6 d6 {+ ] 因为本例不用到任何的自定义指令,这里不作任何的设置,点击Finish完成CPU模块的添加。 ( |: e3 H& Y2 A+ [3 G' w
/ a% u: |) o, z# r
除了Nios II CPU,电子钟设计需要添加的IP模块包括:
+ n) W* _$ t3 Y! H8 qTimer
1 @: v* d k& qButton PIO
8 v; G: x3 |: J/ J) BLCD控制器 |3 y# v4 A! Z8 j* K& R
External RAM Bus
7 X/ I# Z0 D) S9 TExternal Flash Bus & }; L. Y' q7 h1 Q" v7 w# b
External RAM interface
/ x5 w/ N2 v5 gExternal Flash Interface $ @) t: x9 W6 J' I: }0 p
1. 添加定时器 ) \( b5 i7 X2 t9 K. X) a& ^, f
定时器和Nios II CPU一样,用户可以对其进 行定制。添加定时器的步骤如下:
! Y1 X1 Q q# j& @4 u! u(1)在Avalon components下的other库中选 择Interval Timer并单击Add,Avalon_Timer-timer_0的向导窗口出现; 1 Z7 c' w" b8 n/ Q$ p
(2)配置定时器,点击Finish,完成定时器的添加。
) f# p% i l9 @5 F, m3 t0 K/ l5 ?(3)可以对Timer进行重命名,我们这里取缺省的名字。 8 \6 K/ F, S9 ?! O+ N/ m: ~
2. 添加Button PIO ' o+ g f" Y D; G/ d7 i( j
(1)在 Avalon components下的other下选择PIO,单击Add,出现Avalon PIO-PIO_0的向导;
: X+ S9 v+ D% f3 q; S& l+ a; G(2)在Basic Settings页中,设置width=4bits,direction为Input Ports Only; $ e9 R4 j6 c3 I7 s1 ?5 V, ?6 t
(3)在Input Options页中,在Edge Capture Register下选中Synchronously Capture,然后选择Either Edge;
8 v/ j+ v1 b4 A (4)在Interrupt下选中Generate IRQ,然后选择Edge;
% i8 s. |; o, K) v4 M% _+ G. u(5)点击Finish,返回到Altera SOPC Builder NIOS2的窗口; 8 P- m7 H% G- B; D
(6)右键单击Module Name下的pio_0,从菜单选择rename,重命名为button_pio.
! `' m- W ?) i/ j' K! ]5 q; z3. 添加LCD控制器
5 e8 {9 r& Z# `(1)在图2-11的窗口中的左侧的Avalon components下的display库中选择Character LCD(162, Optrex 16027),单击Add; ! M9 Z$ I" ^- j/ T8 x
(2)Module Name下出现,lcd_16027_0; - A$ r8 C' k7 g
(3)将其重命名lcd_display。 p& E; e6 O3 I) @. _! H
4.添加外部RAM接口 # x; z' {6 j. _4 e
(1)Avalon components下的Memory下面选择Cypress CY7C1380C SSRAM,单击Add,出现SSRAM (Cypress CY7C1380C)-ext_ssram的向导;
+ V2 t4 M% T$ |) I" G* ^(2)在Timing Parameters下面设置Read Latency 为2 clocks,在SSRAM下面设置Memory Size为2Mbytes ) H9 w/ }" Q& Q1 z% ]# H9 K
(3)单击Finish,返回到Altera SOPC Builder NIOS2的窗口;
6 R$ S: B* d+ U(4)右键单击Module Name下的ssram_0,从菜单选择rename,重命名为ext-ssram。
4 C7 a. D. P; X! C. R 5. 添加外部闪存接口 : a9 J6 \7 k! E4 Z, T3 K* g8 T' h# x
(1)在Avalon components下的Memory下面选择Flash Memory(Common Flash Interface),单击Add,出现外部闪存接口向导; 7 O. } D0 z1 |3 d4 i; Y! ?
(2)在Attributes页中,可以在Presets列表中选择相应的闪存的接口,这些闪存的接口都是经过测试的,如果列表中没有,用户可以自己定义闪存的Size和Data Width; 6 C( a+ D# n" A2 i% F/ J( |
(3)在timing页中,可以设置闪存的读写时序要求,可以设置Setup, Waite, hold时间等参数,通常保留缺省的设置; 5 m4 `) @6 ~4 T& k# [6 g( e
(4)如应用选择的闪存接口不在列表中,需要自己定义Size和Data Width,和时序要求; # R* q( x' p. m
(5)点击Finish,返回到Altera SOPC Builder NIOS2的窗口,Module Name出现cfi_flash_0,对其重命名为ext_flash。 ' J& d: v3 G4 ?2 e
r' u5 P. ~4 @; z; ]0 n7 r5 s+ ^$ f
6. 添加外部RAM总线和外部flash总线(Avalon三态总线桥)
: W3 L% f* J/ Y; q为了使Nios II 系统能与开发板上的外部存储器通信,必须在Avalon总线和外部存储器之间加入Avalon三态桥。
7 }* L' d+ ~8 T$ U/ _步骤如下: % k9 B c7 F; U2 S' A# }5 d! o
(1) 在Bridge下,选择Avalon Tri-State Bridge,点击Add,出现Avalon Tri-State Bridge – tri_state_bridge_0向导; U9 Z$ m6 W0 [" I. o9 v9 z
(2) Registered选项默认为选中; 2 w2 w# x' L l: ]$ m
(3)单击Finish,返回到Altera SOPC Builder NIOS2的窗口;
- a% k D1 F2 w8 Z7 v/ g0 ~' E' K5 f (4) 将其重命名为ext_ssram_bus;
$ ?5 R2 l6 L3 u; N- I(5) 重复前三个步骤,在添加一个Avalon三态总线桥,并重命名为ext_flash_bus.
& Y- b6 Q& S. D7.添加JTAG UART
# _- o. z' s7 ?. S(1)在Avalon components下的Communication下选择 JTAG UART,点击ADD,出现添加向导。
0 w: d1 a' x+ X8 ?8 ](2)按照缺省设置,点击Finish。 3 L; Q! r; u0 Z1 W3 ]1 j( h+ E: N
(3) 将其重命名为JTAG_UART.
+ l: p% L/ ~# W7 @& Z8. Nios II系统的连接 / T+ ^2 s# M: F" W
主要是外部RAM接口要和RAM的三态桥连接,外部的Flash接口要和flash的三态桥连接。所有添加的IP连接都是系统自动完成的,除了上面提到的三态桥和外部存储器的接口的连接之外,其它的连接用户不用修改,但对于三态桥和外部存储器接口的连接,系统的自动连接可能和用户的开发板不匹配,用户需要进行手动的更改。针对我们所用的开发板的情况进行的设置,因为本书用到的开发板外部的SRAM和外部的Flash没有使用共用的数据线和地址线,所以必须为它们分别添加一个三态桥。
/ L2 j3 j r2 m7 s$ W/ ]8 ^ ! _3 j! c* g; J j7 v$ Z$ P
2.3.3分配IP模块的地址和中断号 在以上添加IP的过程中,SOPC Builder为各个IP模块分配了一个默认的基地址,用户可以改变这些默认的分配。如果用户自己分配的地址出现冲突,SOPC Builder会给出警告,用户可以按照下面的步骤来进行分配和解决地址冲突问题。 8 X3 B" k) k2 l ~( K3 K
下面我们给出一种自定义的地址分配实例,将闪存的基地址设定为0x00000000,步骤如下: % v5 x( ?0 R1 u
(1)单击Flash外设的Base栏,将地址改为0x0,然后回车,这时SOPC Builder的消息框中出现错误的提示信息,这是因为闪存的地址和其它的外设地址发生了冲突。
" S1 E/ p) L5 J: n(2)选择Module菜单中的Lock Base Address操作,一个挂锁的图标出现在闪存基地址的旁边。
- W1 l0 G5 C/ K: d C6 y(3)选择System菜单中的Auto Assign Base Address操作,来解决外设的地址分配冲突问题,SOPC Builder调整其它外设的地址来避开与闪存地址的冲突,错误提示的消息就会消失了。
m1 S% n/ l2 i9 v: G(4)用户可以手动修改各个外设的中断号,中断号越低,中断优先级越高,用户也可以采用自动分配中断号,选择System菜单中的Auto Assign IRQs,但SOPC Builder不处理软件操作,采用自动分配中断的策略不一定是最优的,用户最好是根据自己的应用来确定外设的中断优先级,采用手动的分配。
( W5 R- w9 b& k' j, `3 P# ?) j8 E2.3.4 配置NIOS II 系统 系统的IP模块添加完成之后,还需要对系统进行配置。点击Nios II More “cpu_0” Settings 页,用户可以进行如下的设置:
2 ?- L# }: g$ @. s Reset Address:可以选择存放Boot Loader的存储器和设置Boot Loader在存储器中的偏移,我们这里选择ext_flash,偏移选择默认; 3 x6 C, S% G$ } I4 N9 O; `- ?
Exception Address:可以选择存放异常向量表的存放的存储器和设置异常向量表在存储器中的偏移,我们选择ext_ssram,偏移选择默认; ! C) j6 c5 O5 r7 _4 }; A/ R5 ~
Break Location:如果NIOS II 包含有一个JTAG Debug模块,SOPC Builder会显示这项内容。其Memory Module总是JTAG Debug Module,Offset 固定为0x20, 地址由JTAG debug module基地址决定。用户不能修改Break Location的各个域的内容。
9 s; q% Y- n% N3 G& ]* m2 |) p2.3.5 生成NIOS II并加入到工程中 NIOS II系统是工程的一部分,我们首先是生成它,然后加入到工程中去,然后将整个工程下载到FPGA芯片中去。单击System Generation,在Option下进行如下的设置:
6 r7 S! Z0 q- t1 Y t选中 HDL,会生成系统模块的硬件语言文件。 ' Q2 D$ y* ^! x! g
选中 Simulation,如果安装了ModelSim软件,会生成用于仿真的相应的文件。
- Z2 C( z3 [+ y) K9 ^; @) o单击 Generate,SOPC Builder会提示生成系统的进程,系统生成完成时会提示”SUCCESS: SYSTEM GENERATION COMPLETED”。单击exit 退出SOPC Builder。 , M6 g7 ` R% b8 U5 w4 m
系统生成完成之后,SOPC Builder为这个定制的NIOS II系统模块创建了一个符号,要把Nios II系统加入到工程中去,遵循如下的步骤:
9 a! X. v* F i& O1 ~+ R% w9 U(1)在 Quartus II软件中,打开顶层实体(BDF格式),在BDF窗口中任意处双击,出现 Symbol对话框; 1 {! r' c' |& I! o; v
(2)在Symbol对话框中单击Project来展开工程目录,其下出现NIOS2(本例采用的系统名),选中它,右侧出现了系统的符号表示; + i( j% {4 N' G' P5 b
(3)单击OK,Symbol对话框关闭,NIOS2的符号轮廓被附着在鼠标的指针上。
" f* J1 S5 C! ^" c(4)BDF窗口中任意空白处单击一下,NIOS2的符号出现在BDF窗口中,这样我们创建的系统已经被加入到工程中了。 . H/ M4 s% j9 r. v% L9 |
- r8 [& _- y) S: A" ^$ Q 2.3.6 加入引脚和嵌入式锁相环 除了NIOS II系统之外,可能还要有其它的硬件逻辑,我们这里加入一个嵌入式锁相环,嵌入式锁相环有两个时钟输出,一个输出SSRAM提供时钟,另一个时钟的输出为NIOS II CPU提供时钟,然后添加输入、输出、双向引脚,以实现和FPGA外部的外设进行通信。 3 a9 p. G+ J9 t
加入嵌入式锁相环的步骤如下:
; P$ i+ _' I" ] N(1) 点击ToolsMegaWizard Plug-In Manager,出现MegaWizard Plug-In Manager向导Page1窗口,点击next 在MegaWizard Plug-In Manager Page2 窗口中的IO下面选择ALTPLL,器件选择Cyclone II,输出文件类型选择VHDL,文件名为ssram_pll,选中Return to this page for another create operation,然后点击Next,出现MegaWizard Plug-In Manager-ALTPLL [Page 3of 9];
# V) q* F% f/ K* W4 i 在MegaWizard Plug-In Manager-ALTPLL [Page 4 of 10]窗口中,不做任何选择,点击next; , h: o4 v1 ^0 A
在MegaWizard Plug-In Manager-ALTPLL [Page 5 of 10]窗口中,单击next;
4 o, A: L# t* V, s# F 在MegaWizard Plug-In Manager-ALTPLL [Page 6 of 10]窗口中,设置c0输出时钟,首先选中Use this clock选项,设置时钟频率为85MHz,占空比为50%。点击Next会进入c1输出时钟的设置;
. D4 V. b. ~5 |# r 嵌入式PLL可提供3个输出时钟,我们使用其中的两个,MegaWizard Plug-In Manager-ALTPLL [Page 7 of 10]窗口中,在c1时钟的设置页面上,选中Use this clock选项,设置时钟频率为85MHz,点击Next; 5 E: i- b3 M9 A; S* ]! I4 c9 D4 `
在MegaWizard Plug-In Manager-ALTPLL [Page 8 of 10]窗口中,不选中Use this clock选项,即我们不使用c2时钟,点击Next;
% u" R2 H$ @, ~+ K/ |在MegaWizard Plug-In Manager-ALTPLL [Page 9 of 10]窗口中,给出了用于仿真必须要产生的文件,点击Next;
P; J5 v6 U$ p! i在MegaWizard Plug-In Manager-ALTPLL [Page 10 of 10]窗口中,给出了用户选择要产生的文件,不做改变,采用默认配置,点击Finish完成PLL的生成。
* [* @8 n! {1 s在顶层实体的bdf窗口中双击鼠标,出现Symbol添加窗口,在project下面选择刚才建立的ssram_pll,点击OK。ssram_pll的轮廓会附着在鼠标上,点击bdf窗口的空白处,将嵌入式锁相环加入到了工程中。
0 b4 q* a# @2 S3 o+ {引脚添加的步骤如下: 9 Z. M4 F- Z0 Y/ Y8 f5 n t0 D
(1)在顶层实体的bdf窗口的空白处双击鼠标,出现Symbol添加窗口,在altera/quartus/libraries下面选择primitives,再在其下选择pin,在其中有三种类型的引脚,有bidir,input,output,分别为双向、输入和输出引脚,选择相应类型的引脚,点击OK。
! Y% {. J. F' I0 W* x4 p(2)点击bdf窗口的空白处,即将引脚加入到了工程中。
/ l) |1 f2 z/ l5 x7 E(3)重复上面的步骤添加为各个端口添加相应类型的引脚。 , {; q! _$ e+ q! a$ E7 R* @# C9 A& _
连接引脚和命名引脚 . ~/ K+ d, W& u
第一个加入的引脚的名称缺省为pin_name,之后加入的引脚名称依次为pin_name1,pin_name2向上递增,为了便于理解和记忆,需要对引脚重新命名,使其和其传输的信号联系起来。
1 [0 B2 P4 J+ t- a# b3 ^$ V命名引脚的方法如下: (1) 双击引脚的“pin_namen”部分,pin_namen的文字变成高亮,可以对其编辑。 $ Y; ]# T2 i7 F& d( t- q
(2) 对其他的引脚重复以上的操作,修改成具有意义的名字。
/ b9 e6 B. c! I(3) 对于总线型的引脚,引脚名称之后要标识出总线的位数,如ddr_a[12..0],在引脚名称之后加上方括号,然后写上最高位和最低位,用两个..隔开。
- ]; n# U" p2 i0 m, R& y(4)将嵌入式锁相环和系统模块等连接起来,并将引脚连接到相应的端口上。
+ S7 E5 J$ T. j
; N- `, {% B. a2 J& E' ^ ; j' c/ u t% l
! W+ I9 R# H4 U& N+ A9 [
[ 本帖最后由 kenson 于 2009-1-23 14:17 编辑 ] |