【【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  ;
}
相关推荐
一口一口吃成大V2 小时前
FPGA随记——FPGA时序优化小经验
fpga开发
贾saisai3 小时前
Xilinx系FPGA学习笔记(九)DDR3学习
笔记·学习·fpga开发
redcocal8 小时前
地平线秋招
python·嵌入式硬件·算法·fpga开发·求职招聘
思尔芯S2C1 天前
高密原型验证系统解决方案(下篇)
fpga开发·soc设计·debugging·fpga原型验证·prototyping·深度调试·多fpga 调试
坚持每天写程序1 天前
xilinx vivado PULLMODE 设置思路
fpga开发
redcocal2 天前
地平线内推码 kbrfck
c++·嵌入式硬件·mcu·算法·fpga开发·求职招聘
邹莉斯3 天前
FPGA基本结构和简单原理
fpga开发·硬件工程
悲喜自渡7213 天前
易灵思FPGA开发(一)——软件安装
fpga开发
ZxsLoves3 天前
【【通信协议ARP的verilog实现】】
fpga开发
爱奔跑的虎子3 天前
FPGA与Matlab图像处理之伽马校正
图像处理·matlab·fpga开发·fpga·vivado·xilinx