# 基于FPGA的cy7c68013a双向通信实验



置项 春哥笔记 于 2018-05-11 20:47:35 发布 22909 收藏 105

分类专栏: FPGA 嵌入式 USB CY7C68013

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

本文链接: https://blog.csdn.net/chengfengwenalan/article/details/80282946

版权



FPGA 同时被 3 个专栏收录

33 篇文章 8 订阅订阅专栏



嵌入式

2 篇文章 0 订阅 订阅专栏



USB

2 篇文章 0 订阅 订阅专栏

## 基于FPGA的cy7c68013a双向通信实验

本实验是基于FPGA的cy7c68013a的USB双向通信实验,以前折腾过一段时间cy7c68013a,没有入门时感觉好难,入门了就会感觉很简单。本教程主要内容:

1.cy7c68013a的固件编写,以及生成iic固件和下载固件。

2.cy7c68013a的slave模式,以及他的读写时序

3.cy7c68013a的FPGA的上板测试,包括发送和接受两部分

#### 一、驱动

在进行试验前要先安装好cypress提供的usb驱动,插上usb后,电脑就会检测到未识别的设备,这时打开设备管理器,右键未识别的usb,然后手动选择驱动,驱动会在本教程最后的链接中给出





## 二、固件

cy7c68013a想要正常工作,就得给他编写好相应的固件,然后再固化到其内部,当然PC也是需要安装相应的驱动的,固件的编写主要是确定IN和OUT端点,以及一些标志信号。

```
📙 periph. c🛚
      void TD_Init(void)
                                      // Called once at startup
 54
    ₽{
 55
             CPUCS = ((CPUCS & ~bmCLKSPD) | bmCLKSPD1);
      //
              CPUCS = 0x10; // CLKSPD[1:0]=10, for 48MHz operation, output CLKOUT
 57
              IFCONFIG = 0x43; // external clock source,
                                                            IFCLK Tri-state, synchronous slave FIFO mode
 58
 59
              PINFLAGSAB = 0x08; // FLAGA - EP2 Emtpy flag
 60
              SYNCDELAY:
                                                                        这个是标志信号,很重要
 61
              PINFLAGSCD = 0xE0; // FLAGD - EP6 FULL flag
                                                             低有效
 62
 63
              SYNCDELAY;
 64
              PORTACEG = 0x80;
 65
              EP4CFG = 0x02; //clear the valid bits on ep4 and ep8
 66
 67
              SYNCDELAY;
              EP8CFG = 0x02:
 68
 69
               SYNCDELAY,
 70
              EP2CFG = 0xA0; // OUT, 512-bytes,
                                                4x, bulk
 71
              SYNCDELAY;
                                                                这个是设置IN和OUT端点
 72
              EP6CFG = 0xE0; // IN, 512-bytes, 4x, bulk
 73
 74
 75
              SYNCDELAY;
 76
              FIFORESET = 0x80; // activate NAK-ALL to avoid race conditions
 77
              SYNCDELAY; // see TRM section 15.14
 78
              FIFORESET = 0x02; // reset, FIFO 2
 79
              SYNCDELAY; //
 80
              FIFORESET = 0x04; // reset, FIFO 4
 81
              SYNCDELAY; //
 82
              FIFORESET = 0x06; // reset, FIFO 6
 83
              SYNCDELAY; //
              FIFORESET = 0x08; // reset, FIFO 8
 84
 85
              SYNCDELAY; //
 86
              FIFORESET = 0x00; // deactivate NAK-ALL
 87
 88
 89
              // handle the case where we were already in AUTO mode...
 90
              // ...for example: back to back firmware downloads...
 91
              SYNCDELAY; //
 92
              EP2FIFOCFG = 0x00; // AUTOOUT=0, WORDWIDE=0
 93
              // core needs to see AUTOOUT=0 to AUTOOUT=1 switch to arm endp's
 94
              SYNCDELAY; //
 95
              EP2FIFOCFG = 0x11; // AUTOOUT=1, WORDWIDE=1
 96
              SYNCDELAY; //
 97
              EP6FIFOCFG = 0x09; // AUTOIN=1, ZEROLENIN=0, WORDWIDE=1
              SYNCDELAY;
 99
```

固件只需要改这些参数,这里我都写好了,大家就不需要再改了,很容易看出我设置的时钟是48MHz,,然后设置

EP2为OUT端点,512字节,4缓冲,bulk (注意OUT,IN都是相对PC来说的,OUT表示PC--->cy7c68013a,IN则相反)

EP6为IN端点,512字节,4缓冲,bulk

flag a 为EP2的EF,也就是空标志信号,为低时表示空,也就是没有数据过来,为高则表示有数据来了

flag\_d 为EP6的FF,也就是满标志信号,为低时表示写满了,这时再去写就是无效写了,为高则表示没有写满,可以继续写。

Figure 9-1. Slave FIFOs' Role in the EZ-USB System



https://blog.csdn.net/chengfengwenalan

## 我写好的固件所在文件夹: 固件源码什么的都在Firmware文件中



https://blog.csdn.net/chengfengwenalan

## 三、slave的写时序

Figure 9-12. Timing Example: Synchronous FIFO Writes, Waveform 1



https://blog.csdn.net/chengfengwenalan

有图很容易看出,再写之前要先把FIFOADR确认好,这个决定了你写的对象是谁

Table 9-2. FIFO Selection via FIFOADR[1:0]

| FIFOADR[1:0]              | Selected FIFO            |     |
|---------------------------|--------------------------|-----|
| 00                        | EP2                      |     |
| 01                        | EP4                      |     |
| 10                        | EP6                      |     |
| tps://b <b>1</b> bg.csdn. | net/ch <b>EP8</b> fengwe | na1 |

然后在该fifo非满时(相应的FF标志位高),才可以进行写操作,这个时序很简单,就是拉低slwr信号就可以了,注意FD要与slwr对齐。

注意:写操作时,slwr与FD的数据都是FPGA来控制的,为了让cy7c68013a更好的采样,ifclk与clk反向之后再发送给cy7c68013a.

assligns:ifclkg.csdn.net/chengfclkenalan

四、slave读操作

Figure 9-17. Timing: Synchronous FIFO Reads, Waveform 1



读时序跟写也是类似的,再读之前先确定FIFOADR,然后拉低sloe,这时FD总线就会出现第一个数据,然后检测 到slrd为低时, FD就会显示下一个数据。

## 五、FPGA与cy7c68013a通信

前面主要是一些准备工作,现在开始进入正式的通信过程,项目工程如下:



```
//state_n
always@(*)begin
    case(state_c)

IDLE:begin

if(flag_a == 1'b1)begin //ep2非空,说明有数据来了,进入读状态

state_n = READ;
end
            else if(flag_d == 1'b1)begin //flag_d是EP6的FF, 低有效, 为高时表示该fifo现在空闲, 往ep6写数据
                 state_n = WRITE;
             else begin
                 state_n = state_c;
            end
        end
READ: begin
            if(flag_a == 1'b0)begin //flag_a位低,说明数据已经读完了,进入空闲状态
state_n = IDLE;
end
            else begin
                 state_n = state_c;
       end
end
WRITE: begin
if (flag_d == 1'b0) begin //flag_d为低,说明写满了,进入空闲状态,这时cypress自己就会把这些数据打包好,然后发送给pc,
state_n = IDLE;
end
else begin
            end
       end
default:begin
state_n = IDLE:
    endcase
end
```

具体代码都已经有了详细注释了,自己去看看代码就知道了,这里就不详细说了 本教程所用的调试工具是官方的工具



it tps.//blog. csun. net/chenglengwenatar

## 下面给出写的signal tap 的调试截图

写是一次写512个字节数据,0-255,注意usb的fifo是一次发送16位的,也就是2个字节。先发送低字节,然后再发送高字节,这我直接把低字节给赋值为0了

| trigger: 2018/05/10 22:52:05 #0 |       | Lock mode:                         | 🕋 Allow all changes               |              |                 |
|---------------------------------|-------|------------------------------------|-----------------------------------|--------------|-----------------|
| Node                            |       | Data Enable                        | Trigger Enable Trigger Conditions |              |                 |
| Туре                            | Alias | Name                               | 61                                | 61           | 1 ☑ Basic AND ▼ |
| •*                              |       | flag_a                             | $\checkmark$                      | $\checkmark$ | <b>3</b>        |
| •                               |       | flag_d                             | $\checkmark$                      | $\checkmark$ | ■               |
| *                               |       | sloe                               | ~                                 | $\checkmark$ | 2               |
| *                               |       | slrd                               | $\checkmark$                      | $\checkmark$ | 2               |
| *                               |       | slwr                               | ✓                                 | ~            | <u>0</u>        |
| <b>E</b>                        |       | ⊞ fifo_addr[10]                    | $\checkmark$                      | $\checkmark$ | Xh              |
| <b>\$</b>                       |       | ⊞usb_data[150]                     | $\checkmark$                      | $\checkmark$ | XXXXh           |
| *                               |       | usb:usb_inst cmd_flag              | $\checkmark$                      | $\checkmark$ | <b>2</b>        |
| <b>\$</b>                       |       | ⊞ usb:usb_inst cmd_data[150]       | $\checkmark$                      | $\checkmark$ | XXXXh           |
| Range                           |       | ⊞ usb:usb_inst cnt[70]             | $\checkmark$                      | abla         | XXh             |
| R.                              |       | ⊞usb:usb_inst state_c[20]          | $\checkmark$                      | $\checkmark$ | Xh              |
| R.                              |       | ⊞::usb:usb_inst cnt0[9,:0];ps://b] | og. <b>Z</b> sdn.                 | net/⊠heng    | XXXXXXXXXXX     |



https://blog.csdn.net/chengfengwenalar

## 前面局部放大图



后面局部放大图,注意只有在flag\_d为高时,slwr为低才是有效写,否则就是无效写,因为当flag\_d为低时,表示写满了,这时fifo就会弃之后的写数据了(因为已经写满了,也装不下了是吧,总不能硬塞是吧,哈哈~)



https://blog.csdn.net/chengfengwenalan

PC端接受到的数据要2个字节一起读,因为usb是16位发送的,可以看出接受到的数据的确是0000-00FF注意: pc接受数据按我标的编码顺序执行





## 下面轮到"读"出场了



pc发送数据按1-->2-->3的步骤,可以看出我们发送了12 34 56 78 这4个字节





注意这里我是设置了cmd\_flag标志信号的,只有cmd\_flag为高时的cmd\_data的数据才是有效的,也就是pc发送过来的数据。

这里顺带插一嘴:

Figure 7. Writing 16-bit Words and Reading 8-bit Words



In this example, the read port is operating at twice the frequency of the write port. Writing two 16-bit words to the FIFO buffer increases the wrusedw flag to two and the rusedw flag to four. Four 8-bit read operations empty the FIFO buffer. The read begins with the least-significant 8 bits from the 16-bit word written followed by the most-significant 8 bits.

Figure 8. Writing 8-Bit Words and Reading 16-Bit Words



alter 的fifo ip 是可以读写位宽不一致的,具体看下面的图



由上图可以看出这个和usb是一样的格式,都是先发低字节,然后再发高字节。或者说先接受低字节,然后再接受高字节

至此本教程就全部介绍完了,讲的虽然有点简陋,但是基本的流程都已经讲到了,比起网上其他的教程还是要好不少的哈(自我陶醉一下),具体的看我的源代码。可以看出,使用cypress的usb还是比较简单的,因为usb协议什么的,他们都已经做好了,我们不需要考虑这些,要不然那一堆协议就看着头疼,具体的看我给的下载链接,我会把本教程所用到的工具,代码全部分享出来,也欢迎大家评论提问,不足之处还望指正~

## 六、福利

为了能及时回复大家,现在获取源码方式如下:

微信扫描下面的二维码关注【春哥笔记】公众号,回复"usb2"即可Get源码的获取方式:

