ZYNQ实验---IQ调制实现SSB PART2

news2024/11/29 18:38:08

一、前言

  本文实验在ZYNQ实验—IQ调制实现SSB PART1的基础上进行优化完善。

下图为IQ调制实现SSB PART1中设想实现设计框图
在这里插入图片描述
在这里插入图片描述
该图设计存在的几个问题:

  • PC-PS的UDP传输存在丢包
  • 中断控制发包实际不适合流数据的传输
  • 采用的BRAM模块可以存储的空间较小,PC到PL的时间相对较长,很容易出现RAM读空。

针对以上几个问题提出的解决方案

  • 设计对速率和实时性要求不高,UDP在低速情况下丢包率低。
  • PC数据如果不缓存直接送到PL中时间过长。ZYNQ带4Gbit的DDR选择将数据先完整的存在DDR中再进行PS到PL的传输。
  • 输出的数据是一种流数据形式,使用半空读写方式持续读写FIFO即可保持数据的连续性。

新的IQ调制实现SSB设计框图
在这里插入图片描述

二、实验准备

实验平台与工具:ZYNQ7020,HackRF One,AN108 ADDA板,Matlab 2021b,Vivado 2018

2.1 IQ数据生成

使用matlab生成IQ基带数据,不同形式的基带数据决定了我们的输出调制方式,本实验先用正弦信号做测试。

fs_au = 32E3;         % 采样频率
f = 2E3;              % 正弦信号测试频率
ts=128;				  % 采样时间,秒为单位
t = 0:1/fs_au:ts-1/fs_au;  % 生成ts秒的时间序列,步进为1/fs_au

% 定义文件路径和文件名
filename = 'nengcd.mp3'; %测试音频
% 读取MP3文件
[y, Fs] = audioread(filename);
% 转换采样率
y_resampled = resample(y, fs_au, Fs);
% 截取ts秒数据
start_time = 1; % 从缓存结束开始
end_time = ts*fs_au; % 截取ts秒
y_s = y_resampled(start_time : end_time);

% % 生成正弦信号
% x = sin(2*pi*f*t);
% y_hilbert=fi(hilbert(x), 1, 16); %定点化数据

% 进行希尔伯特变换生成IQ数据
y_hilbert=fi(hilbert(y_s), 1, 16); %定点化数据

y_real=real(y_hilbert);
y_imag=imag(y_hilbert);

% 绘制采样信号
plot(t, y_real);
xlabel('时间(秒)');
ylabel('幅度');
ylim([-2 2]);
title('原始信号');
grid on;

% IQ数据保存 保存为16进制格式,4个字符表示一个数据
% 这样的数据格式虽然增加了数据量但是便于输出观察,实际2个字节表示16bit即可
filename = 'music_real.txt';
% 打开文件进行写入
fileID = fopen(filename, 'w');
% 将定点数以格式化的方式写入文件
fprintf(fileID, '%s\n', y_real.hex);
% 关闭文件
fclose(fileID);

% 文件名
filename = 'music_imag.txt';
% 打开文件进行写入
fileID = fopen(filename, 'w');
% 将定点数以格式化的方式写入文件
fprintf(fileID, '%s\n', y_imag.hex);
% 关闭文件
fclose(fileID);

2.2 Simulink 仿真及Verliog代码生成

参考文章:
Simulink HDL–如何生成Verliog代码
IQ调制实现SSB PART1

2.3 C#设计PC端软件

在以往的工程中使用C#设计过简单的软件,实验中也设计了一个调试软件。

  1. 红色区域为串口区域,可以接收PS端的串口信息
  2. 黄色区域为网络区域,读取matlab生成的数据文件通过UDP发送至DDR
    UDP 通信协议帧结构
    数据报文结构 0xAA 1byte类型 2bytes数据长度 2bytes分片数 Lbytes数据 1byte和校验
    指令报文结构 0xAA 1byte类型 2bytes指令长度 2bytes指令 1byte和校验

在这里插入图片描述

2.3 Vivado – PL与PS设计

ZYNQ的设计如图所示
连接关系:PS->AXI Stream FIFO->AXI Stream Data FIFO->上变频和调制模块->DAC输出
在这里插入图片描述
说明:

  1. AXI Stream FIFO设置半空读写,AXI Stream Data FIFO做缓冲作用。ZYNQ实验 FIFO读写实验(如何平衡跨时钟域的读写)
    在这里插入图片描述
    PS端对AXI Stream FIFO的半空读写示例程序
//初始化FIFO
DDR_rd(0,4096,IQchioce);
Status = FIFOTxSend(&FifoInstance, FIFOSourceBuffer);//FIFOTxSend是FIFO示例中的发送函数
while(1)
{
	//UDP是按512个数据发送到DDR中的,因此读取2048是从DDR中读4组数据
	for(i=0;i<cmd_picenum;i+=4)
	{
		 // 半空状态读取
		Status = XLlFifo_Status(&FifoInstance);
		Halfempty= Status & 0x00200000;
		if(Halfempty)
		{
			//DDR读取数据,幅值到FIFOSourceBuffer
			  DDR_rd(0,2048,IQchioce);
			//发送数据至FIFO。循环写入相同的2048个数据
			  XLlFifo_IntClear(&FifoInstance,0xffffffff);
			  Status = FIFOTxSend2(&FifoInstance, FIFOSourceBuffer);//FIFOTxSend2只发送2048个数据的
			  if (Status != XST_SUCCESS){
			  xil_printf("Transmisson of Data failed\n\r");
			  return XST_FAILURE;}
			  memset(FIFOSourceBuffer2,0,2048);
		}
		else
		{
			  i=i-4;
		}
	}
}
  1. Data_interception 模块
    该模块将AXI Stream类型的数据转化为上变频模块可以使用的数据格式。实验中DDR的32bit数据的高16bit存I数据,低16bit存Q数据,因此将DDR中读取到的数据分为两路16bit数据送入上变频模块。
    在这里插入图片描述
  2. PS端代码问题
    IQ调制实现SSB PART1实验中是不涉及PS端设计的,数据存在RAM中进行循环输出。在本实验中PS成为了很关键的沟通PC和FPGA的部分,PS端功能主要涉及串口收发,UDP收发,DDR读写,AXI Stream FIFO读写,某些功能在我的ZYNQ学习专栏中可以找到,网上也有很多实现教程。

三、 实验结果

3.1 输出2kHz正弦信号

ILA 观察信号

在这里插入图片描述
在这里插入图片描述

示波器观察信号

2kHz在1MHz上经过SSB调制,输出1.002MHz信号
在这里插入图片描述

3.2 输出音频信号

HackRF One接收SSB信号,解调出音频信号,最终的效果一般。结果不好与收发端设备有关,整个系统比较简易。
在这里插入图片描述

总结

  整个工程设计虽然简单,但也算我设计的第一个完整的系统。实验数据从PC端一直到DAC输出的过程在框图中看着简单,但在实际调试中遇到了很多的bug和曲折,matlab生成的代码也是存在一定的问题对FPGA设计不熟练的话很容易遇到很多难以发现和解决的问题。这个东西也是抽空做一点慢慢搭起来的,虽然在某些问题上花费了很大的精力但是我也学到了很多东西,各种开发调试的经验,工具平台的了解使用,包括现在对FPGA和嵌入式也有了新的认识等等。

main函数代码

int main()
{
	int Status;
	int Halffull;//半满标记

	XUartPs_Config *Config;
	u16 revnum=0;

	u16 i,j=0;
	u8 state = UART_RXCHECK ;

	ReceivedBufferPtr = ReceivedBuffer ;

	ReceivedFlag = 0 ;
	ReceivedByteNum = 0 ;

	/* Uart init*/
	Config = XUartPs_LookupConfig(UART_DEVICE_ID);
	if (NULL == Config) {
		return XST_FAILURE;
	}
	Status = XUartPs_CfgInitialize(&Uart_PS, Config, Config->BaseAddress);
	if (Status != XST_SUCCESS) {
		return XST_FAILURE;
	}
	/* Use Normal mode. */
	XUartPs_SetOperMode(&Uart_PS, XUARTPS_OPER_MODE_NORMAL);
	/* Set uart mode Baud Rate 115200, 8bits, no parity, 1 stop bit */
	XUartPs_SetDataFormat(&Uart_PS, &UartFormat) ;
	/*Set receiver FIFO interrupt trigger level, here set to 1*/
	XUartPs_SetFifoThreshold(&Uart_PS,1) ;
	/* Enable the receive FIFO trigger level interrupt and empty interrupt for the device */
	XUartPs_SetInterruptMask(&Uart_PS,XUARTPS_IXR_RXOVR|XUARTPS_IXR_RXEMPTY);

	SetupInterruptSystem(&IntcInstPtr, &Uart_PS, UART_INT_IRQ_ID);

	/* UDP init*/
	struct netif *netif, server_netif;
	ip_addr_t ipaddr, netmask, gw;

	/*  开发板MAC地址  */
	unsigned char mac_ethernet_address [] ={0x00, 0x0a, 0x35, 0x00, 0x01, 0x02};
	/*  开启中断系统  */
	Init_Intr_System(&IntcInstPtr);
	Setup_Intr_Exception(&IntcInstPtr);
	netif = &server_netif;
	IP4_ADDR(&ipaddr,  192, 168,  1, 30);
	IP4_ADDR(&netmask, 255, 255, 255, 0);
	IP4_ADDR(&gw,      192, 168,  1,  1);

	lwip_init();   //初始化lwIP库

	/* 添加网络接口并将其设置为默认接口 */
	if (!xemac_add(netif, &ipaddr, &netmask, &gw, mac_ethernet_address, XPAR_XEMACPS_0_BASEADDR)) {
		xil_printf("Error adding N/W interface\r\n");
		return -1;
	}

	netif_set_default(netif);
	netif_set_up(netif);        //启动网络
	user_udp_init();            //初始化UDP

	/* FIFO init*/
	Status = XLlFifoPollingExample(&FifoInstance, FIFO_DEV_ID);//初始化FIFO

	while (1)
	{
		/*  将MAC队列中的包传输的LwIP/IP栈中   */
		xemacif_input(netif);
		//revnum=UartRevdata(); //串口通信
		revnum=udp_ReadData(UDPrecvBuffer);
		memcpy(PC2PScmd,UDPrecvBuffer,revnum);
		if(revnum>=6)
		{
			switch(PC2PScmd[1])
			{
			  case 0x00: //DDR数据输出至FIFO
			  {
			  	if(revnum==7 && RX_CheckSum(PC2PScmd,7)==0)
			  	{
			  		cmd_picenum=(u16)PC2PScmd[4]+((u16)PC2PScmd[5]<<8);
			  		xil_printf("cmd_picenum=%x\n\r",cmd_picenum);
			  		DDR_rdIQ(0,4096,IQchioce);
			  		Status = FIFOTxSend(&FifoInstance, FIFOSourceBuffer); //填满FIFO
			  		while(1) //循环输出数据
			  		{
			  			for(i=0;i<cmd_picenum;i+=4)
			  			{
			  				// FIFO_sent
			  				Status = XLlFifo_Status(&FifoInstance);
			  				//xil_printf("Status=%x\n\r",Status);
			  			    Halffull= Status & 0x00200000;
			  				if(Halffull)
			  				{
			  					DDR_rdIQ(i*512,2048,IQchioce);
			  					/* Transmit the Data Stream */
			  					XLlFifo_IntClear(&FifoInstance,0xffffffff);
			  					Status = FIFOTxSend2(&FifoInstance, FIFOSourceBuffer2);//半满FIFO
			  					if (Status != XST_SUCCESS){
			  					xil_printf("Transmisson of Data failed\n\r");
			  					return XST_FAILURE;}
			  				}
			  				else
			  				{
			  					i=i-4;
			  				}
			  			}
			  		}
					memcpy(UDPsentBuffer,PC2PScmd,7);
					memset(PC2PScmd,0,7);
					udp_SentData(7,UDPsentBuffer);
					countnum=0;
			  	}
			  	break;
			  }
			  case 0x01: //写入DDR
			  {
			  	cmd_datalen=PC2PScmd[2]+(PC2PScmd[3]<<8);
			  	cmd_picenum=PC2PScmd[4]+(PC2PScmd[5]<<8);
			  	if(revnum==1031 && RX_CheckSum(PC2PScmd,1031)==0) //1031 为整个数据帧的长度,数据一次传1024字节
			  	{
			  		DDR_wr(cmd_picenum*512,cmd_datalen,IQchioce); //512*16bits=1024*8bits,16bit才是一个完整的数据
			  		memcpy(UDPsentBuffer,PC2PScmd,3);
			  		UDPsentBuffer[3]=0x01;
			  		udp_SentData(4,UDPsentBuffer);
			  		printf("picenum=%d\n",cmd_picenum);
			  		memset(PC2PScmd,0,1031);
			  		countnum=0;
			  	}
			  		break;
			  }
			  case 0x02: //写入IQ选择,分别存在DDR的不同区域
			  {
				if(revnum==7 && RX_CheckSum(PC2PScmd,7)==0)
				{
					IQchioce=PC2PScmd[4];
					memset(PC2PScmd,0,7);
				}
				countnum=0;
			  	break;
			  }
			  default:
			  {
			  	countnum=0;
			  	break;
			  }
		}
	  }
	}
	return 0;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1159770.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++ 赋值运算重载,const成员,取地址及const取地址操作符重载

C 赋值运算重载&#xff0c;const成员&#xff0c;取地址及const取地址操作符重载 1. 赋值运算符重载1.1 运算符重载1.2 赋值运算符重载1.3 前置/--和后置/--重载 2. const成员3. 取地址及const取地址操作符重载 所属专栏&#xff1a;C“嘎嘎" 系统学习❤️ &#x1f680;…

智慧灌溉平台

1.知识百科 智慧灌溉是运用物联网、云计算、大数据等新一代信息技术&#xff0c;结合农业生产的实际需求&#xff0c;通过传感器采集土壤温湿度、光照强度等信息&#xff0c;利用无线传感网络传输到中央控制系统进行智能控制。智慧灌溉系统由传感器&#xff08;水位传感器&…

解决关于“由于找不到vcruntime140.dll无法继续执行代码”的问题

今天&#xff0c;我就来谈谈关于“由于找不到vcruntime140.dll无法继续执行代码”的问题&#xff0c;为大家提供4个解决方案。希望我的经验和见解能对大家有所帮助。 首先&#xff0c;我们要明确什么是vcruntime140.dll。简单来说&#xff0c;它是一个动态链接库文件&#xff…

进口跨境商城源码:高效、安全、可扩展的电商平台解决方案

电子商务的兴起为跨境贸易提供了前所未有的机会和挑战。在这个全球化的时代&#xff0c;跨境电商平台成为许多企业进军国际市场的首选。然而&#xff0c;搭建一个高效、安全、可扩展的进口跨境商城并非易事。 1. 解决方案概述 我们推出的 "进口跨境商城源码" 提供了一…

3-性能分析-android-基于Choreographer渲染机制详解

3-性能分析-android-基于Choreographer渲染机制详解 一:主线程运行机制的本质1> 引入 Vsync 之前2> 引入 Choreographer二: Choreographer 简介1> 从 Systrace 的角度来看 Choreogrepher 的工作流程2> Choreographer 的工作流程三:Choreographer 处理一帧的逻辑…

【AI视野·今日Sound 声学论文速览 第三十二期】Tue, 24 Oct 2023

AI视野今日CS.Sound 声学论文速览 Tue, 24 Oct 2023 Totally 20 papers &#x1f449;上期速览✈更多精彩请移步主页 Interesting: &#x1f4da;nvas3d, 基于任意录音和室内3D信息合成重建不同听角&#xff08;位置&#xff09;处的新的声音。(from apple cmu) website: htt…

CMake基础【学习笔记(八)】

声明此博客为转载 CMake基础 文章目录 CMake基础一、准备知识1.1 C的编译过程1.2 静态链接库和动态链接库1.3 为什么需要CMake1.3.1 g 命令行编译1.3.2 CMake简介 二、CMake基础知识2.1 安装2.2 第一个CMake例子2.3 语法基础2.3.1 指定版本2.3.2 设置项目2.3.3 添加可执行文件…

Python画图之皮卡丘

Python-turtle画出皮卡丘&#xff08;有趣小游戏&#xff09; 一、效果图二、Python代码 一、效果图 二、Python代码 import turtledef getPosition(x, y):turtle.setx(x)turtle.sety(y)print(x, y)class Pikachu:def __init__(self):self.t turtle.Turtle()t self.tt.pensi…

小样本分割的新视角,Learning What Not to Segment【CVPR 2022】

论文地址&#xff1a;Excellent-Paper-For-Daily-Reading/image-segmentation at main 类别&#xff1a;图像分割 时间&#xff1a;2023/11/01 摘要 目前背景&#xff1a;少样本分割 &#xff08;FSS&#xff09; 得到了广泛的发展。以前的大多数工作都在努力通过分类任务衍…

Linux C语言进阶-D5~D6指针及指针的运算

指针好处&#xff1a; 使程序更加间接、紧凑、高效 有效地表示复杂的数据结构 动态分配内存 得到多于一个的函数返回值 在C语言中&#xff0c;内存单元的地址称为指针&#xff0c;专门用来存放地址的变量&#xff0c;称为指针变量 在不影响理解的情况下&#xff0c;对地址、指…

gRPC之grpcui界面工具

1、grpcui界面工具 简单的说&#xff0c;就是gRPC中的postman&#xff0c;grpcui官方地址&#xff1a;https://github.com/fullstorydev/grpcui。 1.1 安装 go get -u github.com/fullstorydev/grpcui go install github.com/fullstorydev/grpcui/cmd/grpcuiv1.2.0[rootzsx …

VueX介绍和工作原理

一、VueX的作用 VueX就是在Vue中专门集中地管理数据的一个Vue插件。 在VueX中的数据不属于任何一个组件&#xff0c;所有的组价都可以访问和修改这个数据。 因此&#xff0c;当我们的多个组件依赖同一个状态&#xff08;如用户信息&#xff09;时&#xff0c;就可以使用VueX…

【Tomcat Servlet】如何在idea上部署一个maven项目?

目录 1.创建项目 2.引入依赖 3.创建目录 4.编写代码 5.打包程序 6.部署项目 7.验证程序 什么是Tomcat和Servlet? 以idea2019为例&#xff1a; 1.创建项目 1.1 首先创建maven项目 1.2 项目名称 2.引入依赖 2.1 网址输入mvnrepository.com进入maven中央仓库->地址…

Docker 学习路线 9:运行容器

要启动一个新的容器&#xff0c;我们使用 docker run 命令&#xff0c;后跟镜像名称。基本语法如下&#xff1a; docker run [选项] 镜像 [COMMAND] [ARG...] 例如&#xff0c;要运行官方的 Nginx 镜像&#xff0c;我们可以使用&#xff1a; docker run -d -p 8080:80 nginx…

Qt5 安装 phonon

Qt5 安装 phonon Qt5 安装 phonon问题描述安装组件 Qt5 安装 phonon 开发环境&#xff1a;Qt Creator 4.6.2 Based on Qt 5.9.6 问题描述 在运行 Qt5 项目时&#xff0c;显示错误&#xff1a; error: Unknown module(s) in QT: phonon这是缺少组件的原因&#xff0c;QT: pho…

MTK联发科天玑9000旗舰5G移动平台处理器_MT6983芯片定制开发

MT6983天玑9000采用台积电4纳米工艺制程&#xff0c;CPU采用“134”三丛集Armv9架构&#xff0c;APU性能提升&#xff0c;ISP处理速度提升&#xff0c;最高支持3.2亿像素摄像头&#xff0c;采用Mali-G710十核GPU&#xff0c;搭载R16 5G调制解调器。 MT6983天玑9000芯片基本概…

吴恩达《机器学习》4-1->4-5:多变量线性回归

一、引入多维特征 在多维特征中&#xff0c;我们考虑的不再是单一的特征&#xff0c;而是一组特征&#xff0c;例如房价模型中可能包括房间数、楼层等多个特征。这些特征将组成一个向量&#xff0c;表示为(&#x1d465;₁, &#x1d465;₂, . . . , &#x1d465;ₙ)&#x…

英语教育目标转变:更加注重实际应用能力培养

今年九月份,北京市教委发布了《关于深入推进高中阶段学校考试招生改革的实施意见》。按照该意见,北京市2024年初三年级学生的初中学业水平考试英语科目听力口语考试与笔试将分离,首次计算机考试将于2023年12月17日进行。 根据《意见》规定,听力口语计算机考试共有两次考试机会…

c++ 实现二叉搜索树

二叉搜索树的概念 二叉搜索树 (BST&#xff0c;Binary Search Tree)&#xff0c;也称二叉排序树或二叉查找树。它要么是一颗空树&#xff0c;要么是满足以下性质的二叉树&#xff1a; 若它的左子树不为空&#xff0c;则左子树上所有节点的值都小于根节点的值。若它的右子树不为…

css图片保持比例and图片占满整个div

一、非背景图 ①保持宽度固定 img { width: 200px; height: auto; } ②保持高度固定 img { height: 300px; width: auto; } ③保持比例 /* 比例不变 */ img { max-width: 100%; height: auto; } /* 垂直居中 */ img { max-width: 100%; height: auto; display: block; margin:…