FPGA实现AD9361数据接口逻辑

在这里插入图片描述


作者:ShownSun
工作室:时沿科技


FPGA实现AD9361数据接口逻辑

1 引言

  本文通过以高速AD9361芯片为例进行数据接口逻辑代码的编写,利用SelectIO IP快速高效完成芯片驱动的生成。关于SelectIO IP的使用,可以参考Xilinx SelectIO IP使用说明(一)

2 AD9361

2.1 芯片简介

  AD9361是一款面向3G和4G基站应用的高性能、高集成度的射频(RF)Agile Transceiver™捷变收发器。该器件的可编程性和宽带能力使其成为多种收发器应用的理想选择。该器件集RF前端与灵活的混合信号基带部分为一体,集成频率合成器,为处理器提供可配置数字接口,从而简化设计导入。AD9361接收器LO工作频率范围为70 MHz至6.0 GHz,发射器LO工作频率范围为47 MHz至6.0 GHz,涵盖大部分特许执照和免执照频段,支持的通道带宽范围为200 kHz以下至56 MHz,整体结构图如图 1 AD9361整体结构图所示。

图 1 AD9361整体结构图

图 1 AD9361整体结构图
•	集成12位DAC和ADC的RF 2 × 2收发器
•	TX频段:47 MHz至6.0 GHz
•	RX频段:70 MHz至6.0 GHz
•	支持TDD和FDD操作
•	可调谐通道带宽:<200 kHz至56 MHz
•	双通道接收器:6路差分或12路单端输入
•	出色的接收器灵敏度,噪声系数为2 dB (800 MHz LO)
•	RX增益控制
  o	实时监控和控制信号用于手动增益
  o	独立的自动增益控制
•	双发射器:4路差分输出
•	高线性度宽带发射器
  o	TX EVM:≤−40 dB
  o	TX噪声:≤−157 dBm/Hz本底噪声
  o	TX监控器:动态范围≥66 dB,精度=1 dB
•	集成式小数N分频频率合成器
•	2.4 Hz最大本振(LO)步长
•	多器件同步
•	CMOS/LVDS数字接口

2.2 参数配置

  用户可以根据自己的需求将数据接口通过SPI配置成LVDS或CMOS接口,也可以还可以选择FDD或TDD工作方式,以及数据速率可以选择SDR或DDR。只需要通过配置软件设置即可,如图 2 AD9361数据接口配置参数所示,详细的配置教程见AD936x Evaluation Software 详细配置

图 2 AD9361数据接口配置参数

图 2 AD9361数据接口配置参数

  在进行数据验证时,也可以使用测试模式,对收发数据进行验证以保证系统的正确性。另外,还可以对输入时钟进行延时调节或者通过SelectIO的delay、delayctrl功能对时钟信号进行微调,以满足时序要求。芯片数据时钟与数据之间的时序可靠性也可以通过芯片内部的延时寄存器0x006、0x007进行条件,以此达到要求,具体的SPI配置寄存器时序如图 3 AD9361 寄存器配置接口SPI时序所示。此方面不是本文重点,不做展开,更多内容参考官方data sheet。

图 3 AD9361 寄存器配置接口SPI时序

图 3 AD9361 寄存器配置接口SPI时序

  本小节使用的数据接口参数:LVDS、FDD、DDR,对应的时序逻辑也是根据该参数进行设计。

2.3 引脚

RX数据时序接口如下:




TX数据时序接口如下:




2.4 接口时序

  以下使用的数据接口参数:LVDS、FDD、DDR,根据不同的通道数可以得到不同的数据时序,用户在解析数据时只要按照对应的结构进行拼接即可。

图 4 AD9361接收数据路径

图 4 AD9361接收数据路径

图 5 AD9361发射接口路径

图 5 AD9361发射接口路径

3 参考代码

3.1 SelectIO配置

  根据以上对AD9361的了解,就可以轻松的配置SelectIO IP的GUI界面了。芯片既包括发射模块TX又包括接收模块RX,所以IO类型选择chip to chip。

图 6 SelectIO配置界面1

图 6 SelectIO配置界面1

  根据上述参数配置部分,自然就选择DDR。数据接口包括时钟CLK、Frame对齐信号与差分数据端Data[05:0],要同时对Frame与Data信号进行时序解析,所以端口宽度设置为7.

图 7 SelectIO配置界面2

图 7 SelectIO配置界面2

  由于芯片内部寄存器0x006、0x007可以确保时钟与数据满足时序要求,所以不需要延时模块,以节约FPGA逻辑资源。

图 8 SelectIO配置界面3

图 8 SelectIO配置界面3

图 9 SelectIO配置界面4

图 9 SelectIO配置界面4

3.2 数据解析

//-------------------------------------------------------------------
// 用于将接收时钟与数据进行单端与差分的变换
//-------------------------------------------------------------------
	selectio_ip u_selectio_ip (
		// From the system into the device
		 .DATA_IN_FROM_PINS_P	(ad_rx_data_in_p), 	
 		//从AD接收端接收到的单端数据与标志
		 .DATA_IN_FROM_PINS_N	(ad_rx_data_in_n), 	
  		//从AD接收端接收到的单端数据与标志
		 .DATA_IN_TO_DEVICE		(ad_rx_data), 	
  		//将AD接收端接收到的数据与标志转换为单端数据
	  	// From the device out to the system
		 .DATA_OUT_FROM_DEVICE	(ad_tx_data), 	
  		//将要发送的DA数据与标志转换为单端数据
		 .DATA_OUT_TO_PINS_P		(ad_tx_data_out_p), 	
  		//发送端的单端DA数据与标志
		 .DATA_OUT_TO_PINS_N		(ad_tx_data_out_n),	
  		//发送端的单端DA数据与标志
		 .CLK_TO_PINS_P			(ad_fb_clk_p), 			
  		//将AD接收端的输入时钟用于发射时钟
		 .CLK_TO_PINS_N			(ad_fb_clk_n), 			
  		//将AD接收端的输入时钟用于发射时钟
		 .CLK_IN_P					(ad_data_clk_p),      
  		//AD接收端的单端输入时钟
		 .CLK_IN_N					(ad_data_clk_n),      
  		//AD接收端的单端输入时钟
		 .CLK_OUT					(ad9361_data_clk),        
  		//将AD接收端的差分输入时钟转变为单端时钟
		 .CLK_RESET					(reset), 				 
  		//用于AD输入时钟的复位,高有效
		 .IO_RESET					(reset)  				
  		//用于单端、差分变换的复位,高有效
	);

//-------------------------------------------------------------------
//发送数据的生成
//-------------------------------------------------------------------
	assign ad_tx0_msb_q=ad_tx0_data[23:18];
	assign ad_tx0_lsb_q=ad_tx0_data[17:12];
	assign ad_tx0_msb_i=ad_tx0_data[11:06];
	assign ad_tx0_lsb_i=ad_tx0_data[05:00];
	assign ad_tx1_msb_q=ad_tx1_data[23:18];
	assign ad_tx1_lsb_q=ad_tx1_data[17:12];
	assign ad_tx1_msb_i=ad_tx1_data[11:06];
	assign ad_tx1_lsb_i=ad_tx1_data[05:00];
	
	reg [13:0] ad_tx_data;

//-------------------------------------------------------------------
//选择要发送的I与Q数据
//-------------------------------------------------------------------
	always @(posedge ad9361_data_clk or posedge reset) begin
		if(reset)
			ad_tx_data<=0;
		else if((ad_tx_frame_reg==0)&&(ad_tx_frame==1))
			ad_tx_data<={ad_tx_frame,ad_tx0_msb_q,ad_tx_frame,ad_tx0_msb_i};
		else if((ad_tx_frame_reg==1)&&(ad_tx_frame==1))
			ad_tx_data<={ad_tx_frame,ad_tx0_lsb_q,ad_tx_frame,ad_tx0_lsb_i};
		else if((ad_tx_frame_reg==1)&&(ad_tx_frame==0))
			ad_tx_data<={ad_tx_frame,ad_tx1_msb_q,ad_tx_frame,ad_tx1_msb_i};
		else if((ad_tx_frame_reg==0)&&(ad_tx_frame==0))
			ad_tx_data<={ad_tx_frame,ad_tx1_lsb_q,ad_tx_frame,ad_tx1_lsb_i};		
	end	

//-------------------------------------------------------------------
//选择接收的I与Q数据
//-------------------------------------------------------------------
	always @(posedge ad9361_data_clk or posedge reset) begin
		if(reset) begin
			ad_rx0_msb_i<=0;
			ad_rx0_msb_q<=0;
			ad_rx0_lsb_i<=0;
			ad_rx0_lsb_q<=0;
			ad_rx1_msb_i<=0;
			ad_rx1_msb_q<=0;
			ad_rx1_lsb_i<=0;
			ad_rx1_lsb_q<=0;			
		end
		else if((ad_rx_frame_reg==0)&&(ad_rx_frame==1)) begin
			ad_rx0_msb_i<=ad_rx_data[05:0];
			ad_rx0_msb_q<=ad_rx_data[12:7];
		end
		else if((ad_rx_frame_reg==1)&&(ad_rx_frame==1)) begin
			ad_rx0_lsb_i<=ad_rx_data[05:0];
			ad_rx0_lsb_q<=ad_rx_data[12:7];
		end
		else if((ad_rx_frame_reg==1)&&(ad_rx_frame==0)) begin
			ad_rx1_msb_i<=ad_rx_data[05:0];
			ad_rx1_msb_q<=ad_rx_data[12:7];
		end
		else if((ad_rx_frame_reg==0)&&(ad_rx_frame==0)) begin
			ad_rx1_lsb_i<=ad_rx_data[05:0];
			ad_rx1_lsb_q<=ad_rx_data[12:7];
		end	
	end
	
	wire [23:0]	 ad_rx0_fifo_data;
	wire [23:0]	 ad_rx1_fifo_data;
	assign ad_rx0_fifo_data={ad_rx0_msb_q,ad_rx0_lsb_q,ad_rx0_msb_i,ad_rx0_lsb_i};
	assign ad_rx1_fifo_data={ad_rx1_msb_q,ad_rx1_lsb_q,ad_rx1_msb_i,ad_rx1_lsb_i};

  代码资源获取地址:https://download.csdn.net/download/u014447324/18394119

评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ShownSun

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值