【【VDMA彩条显示实验之四 含C语言代码】】

VDMA彩条显示实验之四 含C语言代码

VTC 手册简介

所有的视频都需要有时序 有时序的地方就需要有 时序控制器

VTC的 主要作用是 产生 视频时序

相对于上一节 在这里 我们会理解的更多

观察 这个 HB 信号 其实这个和上一节的图片差不多

在 行同步信号 前面就是前沿 在 行同步信号的后侧 就是 后沿

VTC 还可以最多支持 16 个 帧同步信号

我们来配置 VTC

首先观察我们需要配置的 各项参数 设置

这是 LCD 时序参数

我们需要做的是将 这个参数 与 VTC的 配置 关联起来



下面我们来配置 帧时序



最终设计的block design

我们观察一下 像素时钟接到了 哪几个端口上

我们会发现 由 PLL 锁相环产生 的 像素时钟 连接到 了 VTC 模块 Video out的 一个 clk 还有 连接到了 lcd_clk 上

其他上用的大多是 100M的时钟

int run_triple_frame_buffer(XAxiVdma* InstancePtr , int DeviceId , int hsize, int vsize, int buf_base_addr, int number_frame_count, int enable_frm_cnt_intr)

第一个参数是 XAxiVdma* InstancePtr ----> VDMA 数据结构的 句柄

第二个参数是 int DeviceId -------> VDMA 器件 的 ID

第三个参数是 int hsize --------> 一帧水平方向上的大小

第四个参数是 int vsize ----------> 竖直方向上的大小

第五个参数是 int buf_base_addr ----------> VDMA 起始帧缓存的地址

第六个参数 是 int number_frame_count ----------> 指定了经过多少帧之后 会迎来中断

第七个参数是 int enable_frm_cnt_intr --------> 告诉我们是否需要使能帧计数器这样的 一个中断

因为在本次实验中 我们并不是很需要中断 所以最后两个参数可以不用去考虑

下面展示整个C语言代码

c 复制代码
#include "stdio.h"
#include"xparameters.h"
#include "xaxivdma.h"
#include "vdma_api.h"
#define VDMA_ID            XPAR_AXIVDMA_0_DEVICE_ID
#define DDR_BASE_ADDR      XPAR_PS7_DDR_0_S_AXI_BASEADDR  // DDR存储空间 的起始地址
//这个是 DDR的基地址 我们是需要写一个彩条图案 但是 彩条图案总不能从基地址
//开始,因为我们程序 也是 从基地址 开始运行的 。 如果彩条图案是从基地址开始
//运行的话 ,就会给我们程序带来冲突  所以我们要重新定义一个帧缓存的地址
// 把宏定义的形式 改写成 变量的形式 这样在 后续可以更方便使用
int frame_buffer_addr  = (DDR_BASE_ADDR + 0x1000000)  ;// VDMA 帧缓存的地址
#define WIDTH   800  // 图像的宽度
#define HEIGHT  480  // 图像的高度
int main()
{
int i , j ;
u8* vdma_buffer_addr ;
vdma_buffer_addr = (u8*) frame_buffer_addr ;


XAxiVdma vdma_inst;


// 配置并启动 VDMA

run_triple_frame_buffer(
&vdma_inst,        // vdma驱动实例
VDMA_ID,           // VDMA 的 ID信息
WIDTH,             //  图像的水平尺寸  宽度
HEIGHT,            //  图像的高度
frame_buffer_addr,   //  VDMA 究竟从DMA的哪个地址开始读取图像// VDMA 帧缓存的起始地址
0,
0);
//  往 VDMA 的帧缓存里面 写入 图案
for(j=0;j<HEIGHT ; j++){
for(i=0 ; i<WIDTH ; i++)
{
// 因为我们用的是 RGB 888 相当于 占据了 3个字节
// 我们使用的是 u8* 类型的 所以就如同下面写的那样 三个才表示一个数据
//
(vdma_buffer_addr + jWIDTH3 + i3+0 ) = 0xff ; //往像素的红色通道写入FF
(vdma_buffer_addr + jWIDTH3 + i3+1 ) = 0x00 ; //往像素的绿色通道写入00
(vdma_buffer_addr + jWIDTH3 + i3+2 ) = 0x00 ; //往像素的蓝色通道写入00
}
}
return 0  ;
}

我们在上电之后发现现象不对 开始修改

一个是红色和蓝色 的不对 还有一个是 只显示了一部分的颜色

为什么会没写好呢 是因为 我们 PS端的DDR控制器 会缓存一部分数据

我们需要把缓存的数据强行冲出来

void Xil_DCacheFlush(void); 添加 函数

下面是修改之后的代码

c 复制代码
#include "stdio.h"
#include"xparameters.h"
#include "xaxivdma.h"
#include "vdma_api.h"
#include "xil_cache.h"
#define VDMA_ID            XPAR_AXIVDMA_0_DEVICE_ID
#define DDR_BASE_ADDR      XPAR_PS7_DDR_0_S_AXI_BASEADDR  // DDR存储空间 的起始地址
//这个是 DDR的基地址 我们是需要写一个彩条图案 但是 彩条图案总不能从基地址
//开始,因为我们程序 也是 从基地址 开始运行的 。 如果彩条图案是从基地址开始
//运行的话 ,就会给我们程序带来冲突  所以我们要重新定义一个帧缓存的地址
// 把宏定义的形式 改写成 变量的形式 这样在 后续可以更方便使用
int frame_buffer_addr  = (DDR_BASE_ADDR + 0x1000000)  ;// VDMA 帧缓存的地址
#define WIDTH   800  // 图像的宽度
#define HEIGHT  480  // 图像的高度
int main()
{
int i , j ;
u8* vdma_buffer_addr ;
vdma_buffer_addr = (u8*) frame_buffer_addr ;


XAxiVdma vdma_inst;


// 配置并启动 VDMA

run_triple_frame_buffer(
&vdma_inst,        // vdma驱动实例
VDMA_ID,           // VDMA 的 ID信息
WIDTH,             //  图像的水平尺寸  宽度
HEIGHT,            //  图像的高度
frame_buffer_addr,   //  VDMA 究竟从DMA的哪个地址开始读取图像// VDMA 帧缓存的起始地址
0,
0);
//  往 VDMA 的帧缓存里面 写入 图案
for(j=0;j<HEIGHT ; j++){
for(i=0 ; i<WIDTH ; i++)
{
// 因为我们用的是 RGB 888 相当于 占据了 3个字节
// 我们使用的是 u8* 类型的 所以就如同下面写的那样 三个才表示一个数据
//
(vdma_buffer_addr + jWIDTH3 + i3+0 ) = 0x00 ; //往像素的蓝色通道写入FF
(vdma_buffer_addr + jWIDTH3 + i3+1 ) = 0x00 ; //往像素的绿色通道写入00
(vdma_buffer_addr + jWIDTH3 + i3+2 ) = 0xff ; //往像素的红色通道写入00
}
}
//将cache缓存的数据冲出来
Xil_DCacheFlush();
return 0  ;
}

纯色模块显示完毕 下面展示彩条代码的书写

c 复制代码
#include "stdio.h"
#include"xparameters.h"
#include "xaxivdma.h"
#include "vdma_api.h"
#include "xil_cache.h"
#define VDMA_ID            XPAR_AXIVDMA_0_DEVICE_ID
#define DDR_BASE_ADDR      XPAR_PS7_DDR_0_S_AXI_BASEADDR  // DDR存储空间 的起始地址
//这个是 DDR的基地址 我们是需要写一个彩条图案 但是 彩条图案总不能从基地址
//开始,因为我们程序 也是 从基地址 开始运行的 。 如果彩条图案是从基地址开始
//运行的话 ,就会给我们程序带来冲突  所以我们要重新定义一个帧缓存的地址
// 把宏定义的形式 改写成 变量的形式 这样在 后续可以更方便使用
int frame_buffer_addr  = (DDR_BASE_ADDR + 0x1000000)  ;// VDMA 帧缓存的地址
#define WIDTH   800  // 图像的宽度
#define HEIGHT  480  // 图像的高度
int main()
{
int i , j ;
u8* vdma_buffer_addr ;
vdma_buffer_addr = (u8*) frame_buffer_addr ;


XAxiVdma vdma_inst;


// 配置并启动 VDMA

run_triple_frame_buffer(
&vdma_inst,        // vdma驱动实例
VDMA_ID,           // VDMA 的 ID信息
WIDTH,             //  图像的水平尺寸  宽度
HEIGHT,            //  图像的高度
frame_buffer_addr,   //  VDMA 究竟从DMA的哪个地址开始读取图像// VDMA 帧缓存的起始地址
0,
0);
//  往 VDMA 的帧缓存里面 写入 图案
for(j=0;j<HEIGHT ; j++){
for(i=0 ; i<WIDTH ; i++)
{
// 因为我们用的是 RGB 888 相当于 占据了 3个字节
// 我们使用的是 u8* 类型的 所以就如同下面写的那样 三个才表示一个数据
//
if(i < WIDTH/3) {
(vdma_buffer_addr + jWIDTH3 + i3+0 ) = 0x00 ; //往像素的蓝色通道写入FF
(vdma_buffer_addr + jWIDTH3 + i3+1 ) = 0x00 ; //往像素的绿色通道写入00
(vdma_buffer_addr + jWIDTH3 + i3+2 ) = 0xff ; //往像素的红色通道写入00
}
else if( i <(2WIDTH)/3   ){(vdma_buffer_addr + jWIDTH3 + i3+0 ) = 0x00 ; //往像素的蓝色通道写入FF(vdma_buffer_addr + jWIDTH3 + i3+1 ) = 0xff ; //往像素的绿色通道写入00(vdma_buffer_addr + jWIDTH3 + i3+2 ) = 0x00 ; //往像素的红色通道写入00}else{(vdma_buffer_addr + jWIDTH3 + i3+0 ) = 0xff ; //往像素的蓝色通道写入FF(vdma_buffer_addr + jWIDTH3 + i3+1 ) = 0x00 ; //往像素的绿色通道写入00(vdma_buffer_addr + jWIDTH3 + i*3+2 ) = 0x00 ; //往像素的红色通道写入00
}
 }

}
//将cache缓存的数据冲出来
Xil_DCacheFlush();
return 0  ;
}
相关推荐
tiger1199 小时前
FPGA独立实现LLM推理方案——FlighLLM
fpga开发·llm·fpga·ai推理
fei_sun10 小时前
Systemverilog和Verilog区别
fpga开发
史蒂芬_丁12 小时前
TI F28P65 使用 ePWM 模块模拟 SPI 时钟的详细方法
单片机·嵌入式硬件·fpga开发
fei_sun15 小时前
HDLBits-Verilog Practice
fpga开发
Aaron158816 小时前
RFSOC+VU13P中在线部分可重构技术的应用分析
人工智能·算法·matlab·fpga开发·重构·信息与通信·信号处理
qxl_79991517 小时前
PCB元件对位:相机采集+YOLO定位完整工程方案(含坐标转换公式)
数码相机·yolo·fpga开发
daxi1501 天前
Verilog入门实战——第5讲:Testbench 仿真编写 + 波形查看与分析
fpga开发
FPGA的花路2 天前
UDP协议
fpga开发·以太网·udp协议
LCMICRO-133108477462 天前
长芯微LPS123完全P2P替代ADP123,高性能、低压差的线性稳压器
单片机·嵌入式硬件·fpga开发·硬件工程·dsp开发·线性稳压器
fei_sun2 天前
面经、笔试(持续更新中)
fpga开发·面试