nios sgdma(Scatter-Gather dma)示例

在 Quartus7.2以后的版本中,除了原有的基于avalon-mm总线的DMA以外,还增长了Scatter-Gather DMA这种基于avalon-ST流总线的DMA IP核,它更适合与大量数据流传输的场合,使用起来比较灵活,增长了与外设流器件配合的能力。因为网上关于SG-DMA介绍的资料比较少,所以这里简单介 绍一下SG-DMA的使用,利用它能够搭配Altera的千兆网MAC核来实现千兆网方面的应用。css

SG-DMA的 数据手册已经介绍得很是详细(见Scatter-Gather DMA Controller Core
),具体的相关寄存器和功能可能查阅相关手册。Altera为了开发的便利,已经为各个IP核设计好了HAL软件层的代 码,SG-DMA也不例外,所以使用的时候咱们没有必要逐个配置相关寄存器,直接调用HAL层代码便可。这也是使用这类IP核简便的地方,只是须要清楚这 类代码如何调用。
ios

  

1. 首先咱们简单看看SG-DMA的应用环境,从数据手册中截下几张图片简单介绍。web

SG-DMA有 三种工做方式,能够工做在Memory-to-Stream即存储接口到流接口,或者Stream-to-Memory即流接口到存储接口,以及 Memory-to-Memory的存储器到存储器工做方式。工做在存储器到存储器的工做方式与普通DMA并没有差异,没有数据流处理的优点。另外SG- DMA增长了Descriptor Processor,能够实现批量工做,从而进一步减轻Nios处理器的工做。只须要将Descriptor命令字写入到相应的Descriptor memory中。咱们简单看看以上的工做方式。缓存

 

图1. Memory-to-Streamasync

图2. Stream-to-Memoryide

图3. Memory-to-Memory函数

2. 而后咱们直接进入主题,看在Altera的SOPC中如何链接使用SG-DMA器件。测试

M-to-M模 式就不作介绍了,这里主要介绍M-to-S和S-to-M这两种方式。咱们添加两个SG-DMA器件,让它们分别工做在这两个工做方式下。链接示意以下所 示。注意到其中的descriptor memory的设置,原则上只要带有avalon-mm接口的存储器均可以用来作descriptor memroy,所以咱们能够将decriptor memory与主存分离,亦能够直接使用主存的一部分做为descriptor memroy。但为了避免影响主存的使用,最好将descriptor memroy分离。另外SG-DMA的有关设置,例如channel和error的位数控制能够参考avalon-st流接口数据手册,依照须要设置接 口。因为在本例中只有一个通道,也不校验错误,因此咱们都设置为零。spa

3. SG-DMA HAL代码调用。.net

要使得SG-DMA正式工做起来,咱们能够直接调用HAL层代码,省去不少开发时间。下面直接使用一段程序,添加部分注释,相信SG-DMA的基本使用便可完成了,并无相信中的这么复杂。 

  1. #include <stdio.h>
  2. #include "altera_avalon_sgdma_descriptor.h"
  3. #include "altera_avalon_sgdma_regs.h"
  4. #include "altera_avalon_sgdma.h"
  5. #include "system.h"
  6. #include "alt_types.h"
  7. //注意包含这几个头文件
  8. alt_sgdma_dev *sgdma_tx_dev; //sgdma_tx设备文件
  9. alt_sgdma_dev *sgdma_rx_dev; //sgdma_rx设备文件
  10. alt_sgdma_descriptor *desc; //descriptor memory指针
  11. char buf[1000]; //SG-DMA传送缓存,暂定1000字节作测试
  12. alt_u32 rx_payload[256]; //SG-DMA接收缓存
  13. void sgdma_rx_isr(void * context, u_long intnum);
  14. //咱们的基本思路就是,先配置好sgdma_rx和sgdma_tx的基本配置,而后设置好sgdma_rx的回调函数。
  15. //即接收数据完成以后调用的函数,最后启动sgdma_tx完成dma发送。在这个过程当中涵盖了sgdma_tx和sgdma_rx的基本使用
  16. int main()
  17. {
  18.     int i;
  19.     int timeout = 0;
  20.     for(i=0; i<1000; i++) //填充缓存数据
  21.         buf[i] = i%256;
  22.      //重定义desc DISCRIPTOR_MEMORY_BASE定义在system.h中,即descriptor_memory的基地址
  23.     desc = (alt_sgdma_descriptor *)DESC_MEM_BASE;
  24.     //打开sgdma_tx和sgdma_rx
  25.     sgdma_tx_dev = alt_avalon_sgdma_open("/dev/tx_sgdma");
  26.     if(!sgdma_tx_dev) 
  27.     {
  28.         printf("[triple_speed_ethernet_init] Error opening TX SGDMA\n");
  29.         return -1;
  30.     }
  31.     sgdma_rx_dev = alt_avalon_sgdma_open("/dev/rx_sgdma");
  32.     if(!sgdma_rx_dev) 
  33.     {
  34.         printf("[triple_speed_ethernet_init] Error opening RX SGDMA\n");
  35.         return -1;
  36.     }
  37.     
  38.     /* Reset RX-side SGDMA */
  39.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(RX_SGDMA_BASE, ALTERA_AVALON_SGDMA_CONTROL_SOFTWARERESET_MSK);
  40.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(RX_SGDMA_BASE, 0x0);
  41.     /* Reset TX-side SGDMA */
  42.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(TX_SGDMA_BASE, 0);
  43.     IOWR_ALTERA_AVALON_SGDMA_STATUS(TX_SGDMA_BASE, 0xFF);
  44.     //注册sgdma_rx回调函数
  45.     alt_avalon_sgdma_register_callback(
  46.                         sgdma_rx_dev,
  47.                         (alt_avalon_sgdma_callback) &sgdma_rx_isr,
  48.                         //ALTERA_AVALON_SGDMA_CONTROL_IE_DESC_COMPLETED_MSK | //每一个描述符数据传输完产生中断
  49.                         ALTERA_AVALON_SGDMA_CONTROL_IE_CHAIN_COMPLETED_MSK | //全部描述符数据传输完时产生中断
  50.                         ALTERA_AVALON_SGDMA_CONTROL_IE_GLOBAL_MSK //开中断
  51.                         ,
  52.                         0);
  53.     //填充接收decriptor memory 并不须要本身填充,调用函数就行了。
  54.     alt_avalon_sgdma_construct_stream_to_mem_desc(
  55.                         &desc[0], //主描述字
  56.                         &desc[1], //次描述字
  57.                         rx_payload, //接收地址
  58.                         0, //length,为0时当收到EOP时结束
  59.                         0); //write_fixed
  60.                         
  61.     //填充发送decriptor memory 
  62.     alt_avalon_sgdma_construct_mem_to_stream_desc(
  63.                         &desc[2], //主描述字
  64.                         &desc[3], //次描述字
  65.                         (unsigned int*)buf, //发送指针
  66.                         (256), //发送字数
  67.                         0, //read_fixed
  68.                         0, //不发送SOP
  69.                         0, //不发送EOP
  70.                         0); //暂不支持
  71.                         
  72.     alt_avalon_sgdma_construct_mem_to_stream_desc(
  73.                         &desc[3], //主描述字
  74.                         &desc[4], //次描述字
  75.                         (unsigned int*)(buf+256), //发送指针
  76.                         (256), //发送字数
  77.                         0, 
  78.                         1, //发送SOP
  79.                         1, //发送EOP
  80.                         0); 
  81.     //启动sgdma_rx和sgdma_tx
  82.     alt_avalon_sgdma_do_async_transfer(sgdma_rx_dev, &desc[0]);
  83.     alt_avalon_sgdma_do_sync_transfer(sgdma_tx_dev, &desc[2]);
  84.     
  85.     for(i=0; i<256; i++)
  86.         printf("%x ",rx_payload[i]);
  87.     printf("\n\n\n\n");
  88. }
  89. //回调函数,负责处理接收后数据,并重置sgdma_rx,本例中并未对数据进行处理
  90. void sgdma_rx_isr(void * context, u_long intnum)
  91. {
  92.     int i;
  93.     alt_sgdma_descriptor *currdescriptor_ptr = &desc[0];
  94.     if(alt_avalon_sgdma_check_descriptor_status(&desc[0])==0)
  95.     {
  96.         printf("RX descriptor reported OK\n");
  97.     } 
  98.     else 
  99.     {
  100.         printf("RX descriptor reported error\n");
  101.     } 
  102.    
  103. }

或者也能够这样

  1. #include <stdio.h>
  2. #include "altera_avalon_sgdma_descriptor.h"
  3. #include "altera_avalon_sgdma_regs.h"
  4. #include "altera_avalon_sgdma.h"
  5. #include "system.h"
  6. #include "alt_types.h"
  7. //注意包含这几个头文件
  8. alt_sgdma_dev *sgdma_tx_dev; //sgdma_tx设备文件
  9. alt_sgdma_dev *sgdma_rx_dev; //sgdma_rx设备文件
  10. alt_sgdma_descriptor *desc; //descriptor memory指针
  11. char buf[1000]; //SG-DMA传送缓存,暂定1000字节作测试
  12. alt_u32 rx_payload[256]; //SG-DMA接收缓存
  13. void sgdma_rx_isr(void * context, u_long intnum);
  14. //咱们的基本思路就是,先配置好sgdma_rx和sgdma_tx的基本配置,而后设置好sgdma_rx的回调函数。
  15. //即接收数据完成以后调用的函数,最后启动sgdma_tx完成dma发送。在这个过程当中涵盖了sgdma_tx和sgdma_rx的基本使用
  16. int main()
  17. {
  18.     int i;
  19.     int timeout = 0;
  20.     for(i=0; i<1000; i++) //填充缓存数据
  21.         buf[i] = i%256;
  22.      //重定义desc DISCRIPTOR_MEMORY_BASE定义在system.h中,即descriptor_memory的基地址
  23.     desc = (alt_sgdma_descriptor *)DESC_MEM_BASE;
  24.     //打开sgdma_tx和sgdma_rx
  25.     sgdma_tx_dev = alt_avalon_sgdma_open("/dev/tx_sgdma");
  26.     if(!sgdma_tx_dev) 
  27.     {
  28.         printf("[triple_speed_ethernet_init] Error opening TX SGDMA\n");
  29.         return -1;
  30.     }
  31.     sgdma_rx_dev = alt_avalon_sgdma_open("/dev/rx_sgdma");
  32.     if(!sgdma_rx_dev) 
  33.     {
  34.         printf("[triple_speed_ethernet_init] Error opening RX SGDMA\n");
  35.         return -1;
  36.     }
  37.     
  38.     /* Reset RX-side SGDMA */
  39.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(RX_SGDMA_BASE, ALTERA_AVALON_SGDMA_CONTROL_SOFTWARERESET_MSK);
  40.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(RX_SGDMA_BASE, 0x0);
  41.     /* Reset TX-side SGDMA */
  42.     IOWR_ALTERA_AVALON_SGDMA_CONTROL(TX_SGDMA_BASE, 0);
  43.     IOWR_ALTERA_AVALON_SGDMA_STATUS(TX_SGDMA_BASE, 0xFF);
  44.     //注册sgdma_rx回调函数
  45.     alt_avalon_sgdma_register_callback(
  46.                         sgdma_rx_dev,
  47.                         (alt_avalon_sgdma_callback) &sgdma_rx_isr,
  48.                         //ALTERA_AVALON_SGDMA_CONTROL_IE_DESC_COMPLETED_MSK | //每一个描述符数据传输完产生中断
  49.                         ALTERA_AVALON_SGDMA_CONTROL_IE_CHAIN_COMPLETED_MSK | //全部描述符数据传输完时产生中断
  50.                         ALTERA_AVALON_SGDMA_CONTROL_IE_GLOBAL_MSK //开中断
  51.                         ,
  52.                         0);
  53.     //填充接收decriptor memory 并不须要本身填充,调用函数就行了。
  54.     alt_avalon_sgdma_construct_stream_to_mem_desc(
  55.                         &desc[0], //主描述字
  56.                         &desc[1], //次描述字
  57.                         rx_payload, //接收地址
  58.                         0, //length,为0时当收到EOP时结束
  59.                         0); //write_fixed
  60.                         
  61.     //填充发送decriptor memory 
  62.     alt_avalon_sgdma_construct_mem_to_stream_desc(
  63.                         &desc[2], //主描述字
  64.                         &desc[3], //次描述字
  65.                         (unsigned int*)buf, //发送指针
  66.                         (256), //发送字数
  67.                         0, //read_fixed
  68.                         0, //不发送SOP
  69.                         1, //发送EOP
  70.                         0); //暂不支持
  71.                         
  72.     alt_avalon_sgdma_construct_mem_to_stream_desc(
  73.                         &desc[3], //主描述字
  74.                         &desc[4], //次描述字
  75.                         (unsigned int*)(buf+256), //发送指针
  76.                         (256), //发送字数
  77.                         0, 
  78.                         1, //发送SOP
  79.                         1, //发送EOP
  80.                         0); 
  81.     //启动sgdma_rx和sgdma_tx
  82.     alt_avalon_sgdma_do_async_transfer(sgdma_rx_dev, &desc[0]);
  83.     alt_avalon_sgdma_do_sync_transfer(sgdma_tx_dev, &desc[2]);
  84.     
  85.     for(i=0; i<256; i++)
  86.         printf("%x ",rx_payload[i]);
  87.     printf("\n\n\n\n");
  88. }
  89. //回调函数,负责处理接收后数据,并重置sgdma_rx,本例中并未对数据进行处理
  90. void sgdma_rx_isr(void * context, u_long intnum)
  91. {
  92.     int i;
  93.     alt_sgdma_descriptor *currdescriptor_ptr = &desc[0];
  94.     if(alt_avalon_sgdma_check_descriptor_status(&desc[0])==0)
  95.     {
  96.         printf("RX descriptor reported OK\n");
  97.     } 
  98.     else 
  99.     {
  100.         printf("RX descriptor reported error\n");
  101.     } 
  102.     
  103.     alt_avalon_sgdma_construct_stream_to_mem_desc(
  104.                         &desc[0], //主描述字
  105.                         &desc[1], //次描述字
  106.                         rx_payload+64, //接收地址
  107.                         0, //length,为0时当收到EOP时结束
  108.                         0); //write_fixed 
  109.     // Re-start SGDMA (always, if we have a single descriptor)
  110.     alt_avalon_sgdma_do_async_transfer(sgdma_rx_dev, &desc[0]);
  111.     
  112. }
相关文章
相关标签/搜索