# 本科实验报告

课程名称: 数字系统实验设计

姓 名: 姚桂涛

学院: 信息与电子工程学院

专业: 信息工程

学 号: 3190105597

指导老师: 屈民军、唐奕

2021年6月21日

## 一、 实验目的

- (1) 掌握音符产生的方法,了解 DDS 技术的应用。
- (2) 了解音频编解码的应用。
- (3) 掌握系统"自顶而下"的数字系统设计方法。

## 二、 实验任务

设计一个音乐播放器,要求以下条件。

- (1) 可以播放四首乐曲,设置 play/pause \_ button、next \_ button、reset 三个按键。按 play/pause \_ button 键,音乐在播放和暂停之间切换,按 next \_ button 键播放下一首乐曲。
- (2) LEDO 指示播放情况 (播放时点亮)、LED2 和 LED 3 指示当前乐曲序号。

# 三、 实验原理

根据实验任务可将系统划分为时钟管理模块 (DCM)、按键处理、主控制器、乐曲读取、音符播放 (note \_ player)、同步化电路、节拍基准产生器和音频编解码接口电路等子模块。

时钟管理模块 (DCM) 产生 100MHz 的系统时钟 sys \_ clk 和 12.5MHz 的音频时钟 audio \_ clk。 主控制器 (mcu) 模块接收按键信息,通知 song \_ reader 模块是否要播放 (play) 及播放哪首乐曲 (song)。

乐曲读取 (song \_ reader) 模块根据 mcu 模块的要求,逐个取出音符信息 note, duration 送给 note \_ player 模块播放, 当一首乐曲播放完毕, 回复 mcu 模块乐曲播放结束信号 (song \_ done)。

音符播放接收到需播放的音符,在音符的持续时间内,以 48kHz 速率送出该音符的正弦波样品给音频编解码接口模块。当一个音符播放结束,向 song \_ reader 模块发送一个 note \_ done 脉冲索取新的音符。

音频编解码接口模块负责将音符的正弦波样品转换为串行输出并发送给音频编解码芯片 ADAU1761。音频编解码芯片 ADAU1761 接收正弦波样品,再进行 AD 转换并放大,最后送至扬声器播放。注意,note \_ player 模块产生的正弦波样品为 16 位二进制,需在低位加 8 个 0 后送入音频编解码接口模块。

由于音频编解码模块与系统使用不同时钟,因此需要同步化电路协调两部分电路。

节拍基准产生器产生 48Hz 的节拍定时基准脉冲信号 (beat), 而 ready 信号频率为 48kHz, 因此, 节拍基准产生器为分频比为 1000 的分频器。而按键处理模块完成输入同步化、防颤动和脉宽变换等功能。

#### 1. 具体设计

1.1 主控制模块 mcu 的设计

主控制模块 mcu 有响应按键信息、控制系统播放两大任务,表 6.11 为其端口含义。

| 表 1: | 主控制模块 | mcu | 的端口含义 |
|------|-------|-----|-------|
|------|-------|-----|-------|

| 引脚名称       | I/O    | 引脚说明                                           |  |  |
|------------|--------|------------------------------------------------|--|--|
| clk        | Input  | 100MHz 时钟信号                                    |  |  |
| reset      | Input  | 复位信号,高电平有效                                     |  |  |
| play_pause | Input  | 来自按键处理模块的"播放/暂停"控制信号,一个时钟周期宽度的脉冲               |  |  |
| next       | Input  | 来自按键处理模块的"下一曲"控制信号,一个时钟周期宽度的脉冲                 |  |  |
| play       | Output | 输出控制信号,高电平表示播放,控制 song _reader 模块是否要播放         |  |  |
| reset_play | Output | 时钟周期宽度的高电平复位脉冲 reset play,用于同时复位模块 song_reader |  |  |
|            |        | 和 note_ player                                 |  |  |
| song_done  | Input  | song_ reader 模块的应答信号,一个时钟周期宽度的高电平脉冲,表示一        |  |  |
|            |        | 曲播放结束                                          |  |  |
| song[1:0]  | Output | 当前播放乐曲的序号                                      |  |  |

根据设计要求,模块 mcu 的原理框图如图所示。图中的 2 位二进制数用来计算乐曲序号 (song)。



图 1: mcu 的结构框图

根据原理框图设计出 mcu 项层代码如下:

```
module mcu (
1
        clk,
                      //100MHz 时钟信号
 2
                      //复位信号, 高电平有效
 3
        reset,
                      //来自按键处理模块的"播放/暂停"控制信号,一个时钟周期宽度的脉冲
 4
        play_pause,
5
        next,
                      //来自按键处理模块的"下一曲"控制信号,一个时钟周期宽度的脉冲
 6
                      //输出控制信号,高电平表示播放,控制 song _ reader 模块是否要播放
        play,
7
        reset_play,
                      //时钟周期宽度的高电平复位脉冲 reset play,用于同时复位模块 song _
  reader 和 note _ player
8
        song_done,
                     //song _ reader 模块的应答信号,一个时钟周期宽度的高电平脉冲,表示一
  曲播放结束
                      //当前播放乐曲的序号
9
        song
10
     );
        input clk, reset, play_pause, next, song_done;
11
        output play, reset_play;
12
        output [1:0] song;
13
```

```
14
           wire NextSong;
15
16
           //mcu 控制器
           mcu_controller m_ctrl1 (
17
18
                .clk(clk),
                .reset(reset),
19
20
                .play_pause(play_pause),
21
                .next(next),
                .song_done(song_done),
22
                .play(play),
23
24
                .reset_play(reset_play),
25
                .NextSong(NextSong)
           );
26
27
           //2 位二进制计数器
28
29
           counter_n #(.n(4), .counter_bits(2)) m_counter(
                .clk(clk),
30
31
                .en(NextSong),
32
                .r(reset),
               .q(song),
33
               .co()
34
35
           );
      endmodule
36
37
```

mcu 顶层代码

控制器的工作流程图如图 6.46 所示,控制器设置初始复位 (RESET)、播放 (PLAY)、暂停 (PAUSE)和下一首 (NEXT) 四种状态。系统复位后,经 RESET 状态初始化后进入 PAUSE 状态,等待各种命令输入; play \_ pause 脉冲信号使系统在 PLAY、PAUSE 两状态之间互转; 在 PLAY 或 PAUSE 状态下,若按下 next \_ button 按钮,则使系统在进入 NEXT 状态,输出 reset \_ play 脉冲复位 song \_ reader 和 note \_ player 两个模块,同时输出脉冲 NextSong 乐曲序号计数器加 1,进入下一曲播放; 另外,在 PLAY 状态时,若乐曲播放结束 (song \_ done 有效)则结束播放,经 RESET 状态复位 song \_ reader 和 note \_ player 两个模块,并进入 PAUSE 状态,再次等待各种命令输入。



图 2: mcu 控制器的算法流程

根据原理框图设计出 mcu 控制器代码如下:

```
module mcu_controller (
1
2
          clk,
 3
          reset,
 4
          play_pause,
 5
          next,
 6
          song_done,
7
          play,
8
          reset_play,
 9
          NextSong
10
      );
          input clk, reset, play_pause, next, song_done;
11
          output reg play, reset_play, NextSong;
12
          parameter RESET = 0, PAUSE = 1, PLAY = 2, NEXT = 3;
13
14
          reg [1:0] state;
                                   //状态信号
15
          reg [1:0] nextstate;
                                   //驱动信号
16
17
          //状态寄存器
18
19
          always @(posedge clk ) begin
               if(reset) state = RESET;
20
21
               else state = nextstate;
22
          end
23
```

```
//下一状态和输出
24
25
           always @(*) begin
26
               play = 0;NextSong = 0;reset_play = 0;
               case(state)
27
                    RESET:
28
29
                        begin
                            play = 0;
30
31
                            NextSong = 0;
                            reset_play = 1;
32
33
                            nextstate = PAUSE;
34
                        end
                    PAUSE:
35
                        begin
36
37
                            play = 0;
                            NextSong = 0;
38
39
                            reset_play = 0;
                            if(play_pause) nextstate = PLAY;
40
                            else
41
42
                                 begin
43
                                     if(next) nextstate = NEXT;
44
                                     else nextstate = PAUSE;
45
                                 end
                        end
46
                    PLAY:
47
48
                        begin
49
                            play = 1;
50
                            NextSong = 0;
51
                            reset_play = 0;
                            if(play_pause) nextstate = PAUSE;
52
                            else
53
54
                                 begin
55
                                     if(next) nextstate = NEXT;
                                     else
56
57
                                          begin
                                              if(song_done) nextstate = RESET;
58
59
                                              else nextstate = PLAY;
                                          end
60
61
                                 end
                        end
62
                    NEXT:
63
                        begin
64
                            play = 0;
65
66
                            NextSong = 1;
                            reset_play = 1;
67
                            nextstate = PLAY;
68
69
                        end
70
                    default: nextstate = RESET;
               endcase
71
72
           end
      endmodule
73
```

74

#### mcu 控制器代码

#### 1.2 乐曲读取模块 song \_ reader 的设计

乐曲读取模块 song \_ reader 的任务有 (1) 根据 mcu 模块的要求,选择播放乐曲。(2) 响应 note \_ player 模块请求,从 song \_ rom 中逐个取出音符 note,duration 送给 note \_ player 模块播放。(3) 判断乐曲是否播放完毕,若播放完毕,则回复 mcu 模块应答信号。根据 song \_ reader 模块的任务要求,song \_ reader 模块需包含表 6.12 所示的输入、输出端口。

引脚名称 I/O 引脚说明 100MHz 时钟信号 clk Input 复位信号,高电平有效 Input reset 来自 mcu 的控制信号,高电平要求播放 Input play 来自 mcu 的控制信号, 当前播放乐曲的序号 song[1:0]Input 即模块 note player 的应答信号,一个时钟周期宽度的脉冲,表示一个音符 note done Input 播放结束并索取新音符 给 mcu 的应答信号, 当乐曲播放结束, 输出-一个时钟周期宽度的脉冲, 表 song done Output 示乐曲播放结束 音符标记 note[5:0]Output duration[5:0] 音符的持续时间 Output 给模块 note playe 的控制信号,一个时钟周期宽度的高电平脉冲,表示新  $new\_note$ Output

表 2: 乐曲读取模块 song reader 的端口含义

 $song\_rom$  是一个只读存储器,用来存放乐曲,容量为  $2'\times12bits$ 。共存放四首乐曲,每首乐曲占用  $25\times12bits$  空间,即每首乐曲最长由 32 个音符组成。因此, $song\_rom$  高 2 位地址决定哪首乐曲,而低 5 位地址决定这首乐曲的哪个音符。 $song\_rom$  每个地址存放一个音符信息,音符信息由 12 位二进制组成,高 6 位表示音符标记 note,低 6 位表示音长 duration。

的音符需播放

song \_ rom 模块已由作者提供,前三首乐曲已填写,第 4 首乐曲空白,由读者自己填写,这里说明一下,若乐曲不足 32 个音符,多余的空间用数字 0 填补。

根据 song \_ reader 模块的功能及 song \_ rom 结构,可画出图 6.47 所示的结构框图,控制器主要负责接收 mcu 模块与 note \_ player 模块的控制信号,并做出响应。算法流程图如图 6.48 所示。



图 3: song reader 的结构框图

根据原理框图设计出 song reader 顶层代码如下:

```
1
      module song_reader (
 2
         song,
                        //来自 mcu 的控制信号, 当前播放乐曲的序号
 3
         clk,
                        //100MHz 时钟信号
 4
         reset,
                        //复位信号,高电平有效
 5
                        //即模块 note player 的应答信号,一个时钟周期宽度的脉冲,表示一个音符
         note_done,
  播放结束并索取新音符
 6
         play,
                        //来自 mcu 的控制信号,高电平要求播放
7
         song_done,
                        //给 mcu 的应答信号, 当乐曲播放结束, 输出-一个时钟周期宽度的脉冲, 表示
  乐曲播放结束
8
         new_note,
                        //给模块 note _ playe 的控制信号,一个时钟周期宽度的高电平脉冲,表示
  新的音符需播放
9
         note.
                        //音符标记
         duration
                        //音符的持续时间
10
11
      );
         input clk, reset, play, note_done;
12
         input [1:0] song;
13
         output song done, new note;
14
         output [5:0] note, duration;
15
         wire [4:0] q; //song _ rom 的低 5 位地址
16
                        //地址计数器进位
17
         wire co;
18
         //地址计数器
19
20
         counter_n #(.n(32), .counter_bits(5)) song_counter(
21
             .clk(clk),
22
             .en(note_done),
23
             .r(reset),
24
             .q(q),
             .co(co)
25
         );
26
27
```

```
//控制器
28
29
           song_reader_controller s_ctrl1(
                .clk(clk),
30
                .reset(reset),
31
                .play(play),
32
                .note_done(note_done),
33
34
                .new_note(new_note)
35
           );
36
           //结東判断
37
           over over1(
38
                .clk(clk),
39
                .duration(duration),
40
41
                .co(co),
                .out(song_done),
42
                .reset(reset)
43
           );
44
45
46
           //song _ rom
47
           song_rom songs(
48
                .clk(clk),
                .dout({note,duration}),
49
                .addr({song,q})
50
           );
51
52
       endmodule
53
```

song reader 顶层代码

系统复位后一直在 RESET 状态等待 mcu 模块控制信号输入,当 mcu 模块发出播放命令 (play 为高电平) 时,进入 NEW \_ NOTE 状态输出 new \_ note 脉冲要求 note \_ player 模块播放音符; 然后进入 WAIT 状态等待,当 note \_ player 模块播放完音符时,会发出 note \_ done 脉冲信号索取下一音符,note \_ done 脉冲信号一方面让地址计数器递增,并从 song \_ rom 取出一个新的音符,另一方面让控制器进入 NEW \_ NOTE 状态,输出 new \_ note 脉冲通知 note \_ player 模块有新的音符需要播放。当 note \_ done 有效,需要两个时钟周期才能从 song \_ rom 中读取下一个音符信息。因此新音符有效标记信号 new \_ note 也应在新音符数据输出后有效,其时序关系如图 6.49 所示。所以在流程图中插入 NEXT \_ NOTE 状态,目的是延迟一个时钟周期输出信号,以配合 song \_ rom 的读取要求。

地址计数器为 5 位二进制计数器,其中 note \_ done 为计数使能输入,当 note \_ done 为高电平时,允许计数。计数器状态 q 为 song \_ rom 的低 5 位地址.song[1:0] 为 song \_ rom 高两位地址。

当地址计数器出现进位或 duration 为 0 时,表示乐曲结束,应输出一个时钟周期宽度的高电平脉冲信号 song  $\_$  done。



图 4: song \_ reader 控制器的算法流程

根据原理框图设计出 song \_ reader 控制器代码如下:

```
module song_reader_controller (
1
 2
           clk,
 3
           reset,
 4
           play,
 5
          note_done,
 6
          new_note
7
      );
           input clk, reset, play, note_done;
8
9
           output reg new_note;
           parameter RESET = 0, NEW_NOTE = 1, WAIT = 2, NEXT_NOTE = 3;
10
11
           reg [1:0] state, nextstate;
12
          //状态寄存器
13
           always @(posedge clk ) begin
14
15
               if(reset) state = RESET;
               else state = nextstate;
16
          end
17
18
           //下一状态和输出
19
20
           always @(*) begin
               new_note = 0;
21
22
               case(state)
                   RESET:
23
                       begin
24
                            if(play) nextstate = NEW_NOTE;
25
```

```
else nextstate = RESET;
26
27
                         end
                    NEW_NOTE:
28
29
                        begin
30
                         new_note = 1;
                         nextstate = WAIT;
31
32
                         end
33
                    WAIT:
                         begin
34
35
                             if(play)
36
                                      if(note_done) nextstate = NEXT_NOTE;
37
                                      else nextstate = WAIT;
38
39
                                  end
                             else nextstate = RESET;
40
                        end
41
                    NEXT_NOTE:
42
43
                         begin
44
                             nextstate = NEW_NOTE;
45
                    default: nextstate = RESET;
46
47
                endcase
           end
48
       endmodule
49
50
```

## 1.3 音符播放模块 note \_ player 的设计

音符播放模块 note \_ player 是本实验的核心模块,它主要任务包括以下几方面。(1) 从 song \_ reader 模块接收需播放的音符 note, duration。(2) 根据 note 值找出 DDS 的相位增量 k。(3) 以 48kHz 速率从 Sine ROM 取出正弦样品送给音频编解码器接口模块。(4) 当一个音符播放完毕,向 song \_ reader 模块索取新的音符。

根据 note \_ player 模块的任务,进一步划分功能单元,如图 6.50 所示,图中 FreqROM 为只读存储器,完成音符标记 note 与 DDS 模块的相位增量 k 查找表关系。表 6.13 所示为 note \_ player 模块的端口含义。



图 5: note \_ player 的结构框图

表 3: note \_ player 模块的端口含义

| X 9. Hote _ player KXHI/III I IX |                                                                       |  |  |  |
|----------------------------------|-----------------------------------------------------------------------|--|--|--|
| I/O                              | 引脚说明                                                                  |  |  |  |
| Input                            | 系统时钟信号,外接 sys_clk                                                     |  |  |  |
| Input                            | 复位信号,高电平有效,外接 mcu 模块的 reset_ play                                     |  |  |  |
| Input                            | 来自 mcu 模块的 play 信号,高电平表示播放                                            |  |  |  |
| Input                            | 来自 song_reader 模块的音符标记 note,表示需播放的音符                                  |  |  |  |
| Input                            | 来自 song _reader 模块的音符持续时间 duration,表示需播放                              |  |  |  |
|                                  | 音符的音长                                                                 |  |  |  |
| Input                            | 来自 song_reader 模块的 new_note 信号, 一个时钟周期宽度的                             |  |  |  |
|                                  | 高电平脉冲,表示新的音符需播放                                                       |  |  |  |
| Output                           | 给 song_reader 模块的应答信号,一个时钟周期宽度的高电平                                    |  |  |  |
|                                  | 脉冲,表示音符播放完毕                                                           |  |  |  |
| Input                            | 来自同步化电路模块的 ready 信号, 频率 48kHz, 一个时钟周期                                 |  |  |  |
|                                  | 宽度的高电平脉冲,表示索取新的正弦样品                                                   |  |  |  |
| Input                            | 定时基准信号,频率为 48Hz 脉冲,一个时钟周期宽度的高电平                                       |  |  |  |
|                                  | 脉冲                                                                    |  |  |  |
| Output                           | 正弦样品输出                                                                |  |  |  |
|                                  | I/O Input |  |  |  |

根据原理框图设计出 note \_ player 控制器代码如下:

module note\_player (

```
2
         clk,
                            //系统时钟信号,外接 sys clk
                            //复位信号, 高电平有效, 外接 mcu 模块的 reset _ play
 3
          reset,
 4
          play_enable,
                            //来自 mcu 模块的 play 信号, 高电平表示播放
 5
                            //来自 song reader 模块的音符标记 note,表示需播放的音符
         note_to_load,
 6
         duration_to_load,
                            //来自 song _ reader 模块的音符持续时间 duration,表示需播放
  音符的音长
                            //来自 song_ reader 模块的 new_note 信号,一个时钟周期宽度的
7
         load_new_note,
  高电平脉冲,表示新的音符需播放
         note_done,
                            //给 song_ reader 模块的应答信号,一个时钟周期宽度的高电平脉冲,
8
  表示音符播放完毕
9
          sampling_pulse,
                            //来自同步化电路模块的 ready 信号,频率 48kHz,一个时钟周期宽度
  的高电平脉冲,表示索取新的正弦样品
         beat,
                            //定时基准信号,频率为 48Hz 脉冲,一个时钟周期宽度的高电平脉冲
10
11
          sample,
                            //正弦样品输出
12
          sample ready
                            //正弦成功输出信号
13
      );
          input clk, reset, play_enable, load_new_note,sampling_pulse,beat;
14
          input [5:0] note_to_load, duration_to_load;
15
16
         output note_done, sample_ready;
         output [15:0] sample;
17
         wire load;
                                       //读取新的音符
18
19
         wire [5:0] q;
                                       //FregROM 地址输入
         wire [19:0] dout;
20
                                       //FreqROM 读取的相位增量
         wire timer_clear,timer_done;
                                       //清零信号与定时结束标志
21
22
         //D 触发器
23
24
         dffre #(.n(6)) note_dffre(
25
             .d(note_to_load),
26
             .en(load),
             .r(~play_enable||reset),
27
             .clk(clk),
28
29
             .q(q)
30
          );
31
         //FregROM
32
33
          frequency_rom FreqROM(
             .clk(clk),
34
35
             .dout(dout),
             .addr(q)
36
37
          );
38
         //DDS
39
          dds note dds(
40
41
             .clk(clk),
             .reset(~play_enable||reset),
42
             .k({2'b00,dout}),
43
44
             .sampling_pulse(sampling_pulse),
45
             .new_sample_ready(sample_ready),
46
             .sample(sample)
47
          );
```

```
48
49
           //控制器
           note_player_controller note_ctrl(
50
                .clk(clk),
51
                .reset(reset),
52
                .play_enable(play_enable),
53
54
                .load_new_note(load_new_note),
55
                .load(load),
                .note_done(note_done),
56
                .timer clear(timer clear),
57
                .timer_done(timer_done)
58
59
           );
60
           //音符节拍定时器
61
           timer note counter(
62
63
                .clk(clk),
                .r(timer_clear),
64
65
                .en(beat),
                .n(duration_to_load),
66
                .done(timer_done)
67
68
           );
69
       endmodule
70
```

note player 控制器代码

note \_ player 控制器负责与 song \_ reader 模块接口,读取音符信息,并根据音符信息从 Frequency ROM 中读取相应相位增量 k 送给 DDS 子模块。另外,note \_ player 控制器还需要控制音符播放时间。note \_ player 控制器的算法流程如图 6.51 所示。在复位或未播放时,控制器处于 RESET 状态或 PLAY 状态,由于此时高电平 reset 或低电平 play \_ enable 都使图 6.35 中的 D 型寄存器清 0,进而使 k 为 0,不会输出正弦样品。当 play \_ enable 为高电平,系统进入音符播放 PLAY 状态,当一个音符播放结束时,控制器进入 DONE 状态,置位 done \_ with \_ note,向 song \_ reader 模块索取新的音符,此时 song \_ reader 模块输出一个 new \_ note 脉冲信号使控制器进入 LOAD 状态,读取新的音符,然后进入 PLAY 状态播放下一个音符。

音符定时器为 6 位二进制计数器,beat、timer \_ clear 分别为使能、清 信号,均为高电平有效。定时时间由音长信号 duration \_ to \_ load 决定,即 duration \_ to \_ load 个 beat 周期,timer \_ done 为定时结束标志。子模块 DDS 的功能就是利用 DDS 技术产生正弦样品,其工作原理已在实验 15 中介绍了,注意:DDS 模块的输入 k 为 22 位二进制,因此需 FreqROM 输出的 20 位相位增量高位加 2 个 0 后接入 DDS。



图 6: note \_ player 控制器的算法流程

根据原理框图设计出 note \_ player 控制器代码如下:

```
module note_player_controller (
 1
 2
          clk,
 3
           reset,
 4
           play_enable,
 5
           load_new_note,
           load,
 6
 7
           note_done,
 8
          timer_clear,
          timer_done
9
10
      );
           input clk, reset, play_enable, timer_done, load_new_note;
11
           output reg load,timer_clear,note_done;
12
13
           parameter RESET = 0, WAIT = 1, DONE = 2, LOAD = 3;
           reg [1:0] state, nextstate;
14
15
           //状态
           always @(posedge clk ) begin
16
               if(reset) state = RESET;
17
               else state = nextstate;
18
19
          end
20
          //下一状态和输出
21
22
           always @(*) begin
               load = 0; timer_clear = 0; note_done = 0;
23
24
               case(state)
25
                   RESET:
26
                       begin
                            timer_clear = 1;
27
```

```
load = 0:
28
                             note_done = 0;
29
                             nextstate = WAIT;
30
                        end
31
32
                    WAIT:
33
                        begin
                             timer_clear = 0;
34
35
                             load = 0;
                             note_done = 0;
36
37
                             if(play_enable)
                                 begin
38
                                      if(timer_done) nextstate = DONE;
39
                                      else
40
41
                                          begin
                                               if(load_new_note) nextstate = LOAD;
42
43
                                               else nextstate = WAIT;
                                          end
44
45
                                 end
46
                             else nextstate = RESET;
                        end
47
                    DONE:
48
49
                        begin
                             timer_clear = 1;
50
                             load = 0;
51
52
                             note_done = 1;
                             nextstate = WAIT;
53
54
                        end
                    LOAD:
55
56
                        begin
                             timer_clear = 1;
57
                             load = 1;
58
59
                             note done = 0;
60
                             nextstate = WAIT;
61
                        end
                    default: nextstate = RESET;
62
63
               endcase
           end
64
65
       endmodule
66
```

note \_player 控制器代码

#### 1.4 同步化电路

由于音频编解码接口模块和其他模块采用不同的时钟,因此两者之间的控制及应答信号须进行同步化处理。本例中音频编解码接口模块的输出信号 NewFrame 的脉冲宽度为一个 audio \_ clk 时钟周期,需通过同步化处理,产生与 sys \_ clk 同步且脉冲宽度为一个 sys \_ clk 时钟周期的信号 ready。电路如图 6.52 所示,由同步器和脉冲宽度变换电路组成。



图 7: 同步化电路

同步化电路代码如下:

```
1 module synchro (
 2
       clk,
3
       in,
       out
 4
5);
       input clk, in;
 6
 7
       output out;
 8
9
       reg q1,q2;
10
       //非阻塞赋值
11
       always @(posedge clk ) begin
12
           q1 <= in;
13
           q2 <= q1;
14
       end
15
16
       assign out = q1 \&\& (\sim q2);
18 endmodule
19
```

同步化电路代码

#### 1.5 时钟管理模块 (DCM)

IP 内核时钟管理模块的输入时钟 clk 频率为  $100 \mathrm{MHz}$ ,产生  $100 \mathrm{MHz}$  的系统时钟和  $12.50 \mathrm{MHz}$  的音频时钟。

另外, 音频编解码接口模块与按键处理模块按实验 18 和实验 11 介绍设计。也可调用作者提供的设计。不过, 作者是以 BlackBox 方式提供, 即音频编解码接口模块提供综合网表文件 AudioInterface.edf 和端口文件 AudioInterface.v; 而按键处理模块提供综合网表文件 button press \_ unit.edf 和端口文件 button \_ press \_ unit.v。

## 四、 主要仪器设备

Modelsim SE、Vivado、Nexys Video Artix-7 FPGA 多媒体音视频智能互联开发系统、有源音响或耳机。

## 五、 实验过程

#### 1. 仿真测试

#### 1.1 主控制器 mcu 模块

## 仿真图:



图 8: mcu 仿真图

## 1.2 乐曲读取 song \_ reader 模块



图 9: song \_ reader 仿真图

## 1.3 音符播放 note \_ player 模块



图 10: note \_ player 仿真图

## 1.4 次顶层 music \_ player