0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

FPGA數(shù)字圖像處理仿真方法

HJ18656750788 ? 來源:Cascatrix ? 2023-01-29 14:09 ? 次閱讀

01FPGA圖像仿真方法概述

VHDL和Verilog代碼編寫后通常需要編寫激勵(lì)文件進(jìn)行仿真以驗(yàn)證代碼的可行性,通過仿真可以及時(shí)排查代碼存在的時(shí)序問題,有效提高代碼實(shí)現(xiàn)效率。

圖像數(shù)據(jù)量通常較大,1920×1080@60Hz分辨率的圖像通常需要1920×1080個(gè)8bit寄存器reg,因此在仿真中通常難以直接編寫語句生成相應(yīng)的代碼,因此需要借助相應(yīng)圖像格式文件的讀取操作和文件格式轉(zhuǎn)換工具輔助。

Matlab是圖像算法設(shè)計(jì)驗(yàn)證常用工具,因此主要以Matlab為輔助工具,對(duì)FPGA圖像仿真方法進(jìn)行介紹,通常采用以下3種方法:

1. .txt格式圖像仿真方法:Verilog代碼通常支持通過$readmemh、$readmemb、$fopen等語句讀取.txt文本文件,因此可以借助Matlab將圖像轉(zhuǎn)換為.txt格式圖像文本,編寫.txt文件讀取代碼實(shí)現(xiàn)FPGA圖像仿真;

2. .coe格式圖像仿真方法:EDA工具通常提供ROM IP核支持.coe格式文件存儲(chǔ),因此可以借助Matlab將圖像轉(zhuǎn)換為.coe格式文件,將.coe文件存入ROM,通過操作ROM讀取圖像數(shù)據(jù)實(shí)現(xiàn)FPGA圖像仿真;

3. 圖像存儲(chǔ)模塊仿真方法:通過編寫圖像存儲(chǔ).v或.vhdl模塊以供頂層文件讀取,由于圖像數(shù)據(jù)量較大,可以借助Matlab工具實(shí)現(xiàn)存儲(chǔ)模塊的編寫過程。

02.txt格式圖像仿真方法

通過讀取.txt格式圖像文件實(shí)現(xiàn)基于Verilog仿真步驟:

1. 利用Matlab轉(zhuǎn)換.jpg或者其他圖像為.txt格式數(shù)據(jù);

2. 通過Vivado完成圖像讀取處理并生成.txt格式文件;

3. 利用Matlab轉(zhuǎn)換顯示Vivado處理后的.txt格式圖像。

2.1 Matlab生成.txt文件

通過Matlab將圖像按16進(jìn)制轉(zhuǎn)換為.txt格式,文件夾中應(yīng)包含輸入圖片cascatrix.jpg:

cc8d783e-990c-11ed-bfe3-dac502259ad0.pngccb5c3ac-990c-11ed-bfe3-dac502259ad0.png

Matlab代碼如下:

%**********************************************************************

% -------------------------------------------------------------------

% Company: Cascatrix

% Engineer: Carson

%

% Create Date: 2023/01/19

% Design Name: rgb_hex

% Module Name: rgb_hex

% Tool Versions: v1.0

% Description: Convert RGB888 into .txt

%-------------------------------------------------------------------

%*********************************************************************/

clear;clear all;clc;

% Load image

image_in = imread('cascatrix.jpg');

% Create .txt file

FileName=['image_in','.txt'];

% Get image size

[row,col,n] = size(image_in);

% Write image into image_in.txt

FileImage = fopen(FileName,'w');

for x = 1:row

for y = 1:col

image_r = dec2hex(image_in(x,y,1));

image_g = dec2hex(image_in(x,y,2));

image_b = dec2hex(image_in(x,y,3));

[r_row,r_col]=size(image_r);

[g_row,g_col]=size(image_g);

[b_row,b_col]=size(image_b);

if r_col == 1

image_r =['0',image_r];

elseif r_col == 0

image_r ='00';

end

if g_col == 1

image_g =['0',image_g];

elseif g_col == 0

image_g ='00';

end

if b_col == 1

image_b =['0',image_b];

elseif b_col == 0

image_b ='00';

end

image_hex=[image_r,image_g,image_b,];

fprintf(FileImage,'%s ',image_hex);

end

end

fclose(FileImage);

rgb_hex代碼運(yùn)行后生成.txt格式圖像image_in.txt:

ccd6c408-990c-11ed-bfe3-dac502259ad0.png

cd01ce5a-990c-11ed-bfe3-dac502259ad0.png

2.2 VIVADO讀取與處理

利用Verilog $readmemh語句讀取image_in.txt文件:

cx_image.v模塊(將.txt文件放置在相應(yīng)路徑):

`timescale 1ns / 1ps

//////////////////////////////////////////////////////////////////////////////////

// Company: Cascatrix

// Engineer: Carson

//

// Create Date: 2023/01/19

// Design Name: Image_Base

// Module Name: cx_image

// Tool Versions: v1.0

// Description: Image read and generate display

//parameters

//////////////////////////////////////////////////////////////////////////////////

`define PIXEL_1920_1080

//`define PIXEL_1680_1050

//`define PIXEL_1280_1024

//`define PIXEL_1280_720

//`define PIXEL_1024_768

//`define PIXEL_800_600

//`define PIXEL_640_480

module cx_image(

inputwireclk,

outputreghsyn,

outputregvsyn,

outputwireen,

outputreg [23:0]data

);

//1920x1080 148.5Mhz

`ifdef PIXEL_1920_1080

parameter H_ACTIVE = 1920;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 88; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 44; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 148; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1080;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 4; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 36; // 列消隱后肩時(shí)間

`endif

//1680x1050 119Mhz

`ifdef PIXEL_1680_1050

parameter H_ACTIVE = 1680;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 32; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 80; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1050;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 21; // 列消隱后肩時(shí)間

`endif

//1280x1024 108Mhz

`ifdef PIXEL_1280_1024

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 112; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 248; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1024;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 3; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 38; // 列消隱后肩時(shí)間

`endif

//1280X720 74.25MHZ

`ifdef PIXEL_1280_720

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 110; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 40; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 220; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 720; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 5; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 20; // 列消隱后肩時(shí)間

`endif

//1024x768 65Mhz

`ifdef PIXEL_1024_768

parameter H_ACTIVE = 1024;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 24; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 136; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 160; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 768; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 29; // 列消隱后肩時(shí)間

`endif

//800x600 40Mhz

`ifdef PIXEL_800_600

parameter H_ACTIVE = 800;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 40 ;// 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 128;// 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 88 ;// 行消隱后肩時(shí)間

parameter V_ACTIVE = 600;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1 ;// 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 4 ;// 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 23 ;// 列消隱后肩時(shí)間

`endif

//640x480 25.175Mhz

`ifdef PIXEL_640_480

parameter H_ACTIVE = 640; // 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 16 ; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 96 ; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 48 ; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 480; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 10 ; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 2 ; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 33 ; // 列消隱后肩時(shí)間

`endif

parameter H_TOTAL_TIME = H_ACTIVE + H_FRONT_PORCH + H_SYNC_TIME + H_BACK_PORCH;

parameter V_TOTAL_TIME = V_ACTIVE + V_FRONT_PORCH + V_SYNC_TIME + V_BACK_PORCH;

reg h_act = 1'b0;

reg v_act = 1'b0;

reg [12:0] h_syn_cnt = 13'b0;

reg [12:0] v_syn_cnt = 13'b0;

reg [23:0] image [0 : H_ACTIVE*V_ACTIVE-1];

reg [31:0] image_cnt = 32'b0;

// 讀取.txt文件到image數(shù)組中

initial begin

$readmemh("D:/FPGA_Document/CX_Document/CX_Image/ 01_Image_process_base/image_src/image_in.txt", image);

end

// 行掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME-1)

h_syn_cnt <= 13'b0;

else

h_syn_cnt <= h_syn_cnt + 1;

end

// 列掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME-1)

begin

if(v_syn_cnt == V_TOTAL_TIME-1)

v_syn_cnt <= 13'b0;

else

v_syn_cnt <= v_syn_cnt + 1;

end

end

// 行同步控制

always@(posedge clk)

begin

if(h_syn_cnt < H_SYNC_TIME)

hsyn <= 1'b0;

else

hsyn <= 1'b1;

end

// 場(chǎng)同步控制

always@(posedge clk)

begin

if(v_syn_cnt < V_SYNC_TIME)

vsyn <= 1'b0;

else

vsyn <= 1'b1;

end

// 有效數(shù)據(jù)使能

assign en = h_act & v_act;

// 行有效數(shù)據(jù)控制

always@(posedge clk)

begin

if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH - 1)

h_act = 1'b1;

else if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH + H_ACTIVE - 1)

h_act = 1'b0;

end

// 列有效數(shù)據(jù)控制

always@(posedge clk)

begin

if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH - 1 && h_syn_cnt == 0)

v_act = 1'b1;

else if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH + V_ACTIVE - 1 && h_syn_cnt == 0)

v_act = 1'b0;

end

// 像素計(jì)數(shù)器

always@(posedge clk)

begin

if(h_act & v_act)

image_cnt <= image_cnt + 1;

else if(image_cnt == H_ACTIVE*V_ACTIVE - 1)

image_cnt <= 32'b0;

end

// 圖像數(shù)據(jù)

always@(posedge clk)

begin

if(h_act & v_act)

data <= image[image_cnt][23:0];

else

data <= 24'b0;

end

endmodule

利用Verilog $fopen和$fwrite生成.txt圖像文件:

cx_tb.v仿真模塊(生成圖像文件到相應(yīng)路徑):

`timescale 1ns / 1ps

//////////////////////////////////////////////////////////////////////////////////

// Company: Cascatrix

// Engineer: Carson

//

// Create Date: 2023/01/19

// Design Name: Image_Base

// Module Name: cx_tb

// Tool Versions: v1.0

// Description: Image output simulation

//////////////////////////////////////////////////////////////////////////////////

module cx_tb(

);

reg clk;

reg [31:0] pixel_cnt;

wire de;

wire [23:0] data;

integer image_txt;

parameter PIXEL_TOTAL = 1920*1080;

//parameter PIXEL_TOTAL = 1680*1050;

//parameter PIXEL_TOTAL = 1280*1024;

//parameter PIXEL_TOTAL = 1280*720;

//parameter PIXEL_TOTAL = 1024*768;

//parameter PIXEL_TOTAL = 800*600;

//parameter PIXEL_TOTAL = 640*480;

cx_image inst_cx_image

(

.clk (clk ),

.hsyn ( ),

.vsyn ( ),

.en (de ),

.data (data)

);

always #1 clk = ~clk;

initial

begin

clk = 1‘b0;

end

initial

begin

image_txt = $fopen("D:/FPGA_Document/CX_Document/CX _Image/01_Image_process_base/image_src/image_out.txt");

end

always@(posedge clk or negedge rst_n)

begin

if(!rst_n)

begin

pixel_cnt <= 0;

end

else if(de)

begin

pixel_cnt = pixel_cnt + 1;

$fwrite(image_txt,"%h ",data);

end

end

always@(posedge clk)

begin

if(pixel_cnt == PIXEL_TOTAL)

begin

$display("CX: image_out.txt is output completed successfully! %t", $realtime, "ps");

$fclose(image_txt);

$stop;

end

end

endmodule

仿真波形結(jié)果:

cd1c34a2-990c-11ed-bfe3-dac502259ad0.png

在指定路徑生成處理后的.txt圖像文件image_out.txt:

cd3dd3dc-990c-11ed-bfe3-dac502259ad0.png

2.3 Matlab顯示.txt文件

通過Matlab將處理后的.txt圖像image_out.txt顯示,并轉(zhuǎn)換為jpg或其他格式圖像,編寫hex_rgb.m文件:

cd6e73f2-990c-11ed-bfe3-dac502259ad0.png

Matlab代碼如下:

%**********************************************************************

% -------------------------------------------------------------------

% Company: Cascatrix

% Engineer: Carson

%

% Create Date: 2023/01/19

% Design Name: hex_rgb

% Module Name: hex_rgb

% Tool Versions: v1.0

% Description: Convert .txt into RGB888

%-------------------------------------------------------------------

%*********************************************************************/

clear;clear all;clc;

% Image resolution

row = 1080;

col = 1920;

n = 3;

% Create output image

image_out = uint8(zeros(row,col,n));

% Write data into output image

FileImage = fopen('image_out.txt','r');

for x = 1:row

for y = 1:col

RGB = fscanf(FileImage,'%s',1);

image_out(x,y,1) = uint8(hex2dec(RGB(1:2)));

image_out(x,y,2) = uint8(hex2dec(RGB(3:4)));

image_out(x,y,3) = uint8(hex2dec(RGB(5:6)));

end

end

fclose(FileImage);

% Show the output image

imshow(image_out),title('Image output');

% Create image in .jpg format

imwrite(image_out,'cascatrix_output.jpg');

運(yùn)行結(jié)果:

cd94dfc4-990c-11ed-bfe3-dac502259ad0.png

生成cascatrix_output.jpg

cdb6c6c0-990c-11ed-bfe3-dac502259ad0.pngcdcee3e0-990c-11ed-bfe3-dac502259ad0.png

03.coe格式圖像仿真方法

通過讀取.coe格式圖像文件實(shí)現(xiàn)FPGA仿真步驟:

1. 利用Matlab通過圖像獲取.coe格式數(shù)據(jù);

2. 通過操作ROM讀取圖像并處理生成.txt格式文件;

3. 利用Matlab轉(zhuǎn)換顯示Vivado處理后的.txt格式圖像。

3.1 Matlab獲取.coe文件

利用Matlab獲取圖像數(shù)據(jù)并生成.coe文件:

cdfa3bee-990c-11ed-bfe3-dac502259ad0.png

Matlab代碼如下:

%**********************************************************************

% -------------------------------------------------------------------

% Company: Cascatrix

% Engineer: Carson

%

% Create Date: 2023/01/19

% Design Name: rgb_coe

% Module Name: rgb_coe

% Tool Versions: v1.0

% Description: Convert RGB888 into .coe

%-------------------------------------------------------------------

%*********************************************************************/

clear;clear all;clc;

% Load image

image_in = imread('cascatrix.jpg');

% Create .coe file

FileName=['image_in','.coe'];

% Get image size

[row,col,n] = size(image_in);

% Write image into image_in.coe

FileImage = fopen(FileName,'w');

% Format of image.coe

fprintf(FileImage,'memory_initialization_radix = 16; ');

fprintf(FileImage,'memory_initialization_vector = ');

for x = 1:row

for y = 1:col

image_r = dec2hex(image_in(x,y,1));

image_g = dec2hex(image_in(x,y,2));

image_b = dec2hex(image_in(x,y,3));

[r_row,r_col]=size(image_r);

[g_row,g_col]=size(image_g);

[b_row,b_col]=size(image_b);

if r_col == 1

image_r =['0',image_r];

elseif r_col == 0

image_r ='00';

end

if g_col == 1

image_g =['0',image_g];

elseif g_col == 0

image_g ='00';

end

if b_col == 1

image_b =['0',image_b];

elseif b_col == 0

image_b ='00';

end

if((x == row)&&(y == col))

image_hex=[image_r,image_g,image_b,';'];

else

image_hex=[image_r,image_g,image_b,','];

end

fprintf(FileImage,'%s ',image_hex);

end

end

fclose(FileImage);

rgb_coe代碼運(yùn)行后生成.coe文件image_in.coe:

ce1f39e4-990c-11ed-bfe3-dac502259ad0.pngce5183ae-990c-11ed-bfe3-dac502259ad0.png

3.2 Vivado操作ROM

調(diào)用配置ROM IP核,通過操作ROM讀取圖像數(shù)據(jù):

ce6a76d4-990c-11ed-bfe3-dac502259ad0.png

由于Vivado中ROM IP支持最大深度為1048576,無法完全存儲(chǔ)1920×1080=2073600的圖像數(shù)據(jù),因此只能存儲(chǔ)1280×720=921600的圖像數(shù)據(jù),上述.coe文件也應(yīng)當(dāng)由1280×720分辨率的圖像生成:

ce8d82dc-990c-11ed-bfe3-dac502259ad0.png

導(dǎo)入Matlab生成的image_in.coe文件為初始數(shù)據(jù)的,.coe文件內(nèi)容一定要與上一步配置的接口寬度和深度相匹配,否則將會(huì)報(bào)錯(cuò):

ceb8bcae-990c-11ed-bfe3-dac502259ad0.png

完成所有配置后,cx_image.v調(diào)用ROM并讀取數(shù)據(jù):

`timescale 1ns / 1ps

//////////////////////////////////////////////////////////////////////////////////

// Company: Cascatrix

// Engineer: Carson

//

// Create Date: 2023/01/15

// Design Name: Image_Base

// Module Name: cx_image

// Tool Versions: v1.0

// Description: Image read and generate display parameters

//

//////////////////////////////////////////////////////////////////////////////////

//`define PIXEL_1920_1080

//`define PIXEL_1680_1050

//`define PIXEL_1280_1024

`define PIXEL_1280_720

//`define PIXEL_1024_768

//`define PIXEL_800_600

//`define PIXEL_640_480

module cx_image(

inputwireclk,

outputreghs,

outputregvs,

outputwireen,

outputwire [23:0]data

);

//1920x1080 148.5Mhz

`ifdef PIXEL_1920_1080

parameter H_ACTIVE = 1920;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 88; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 44; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 148; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1080;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 4; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 36; // 列消隱后肩時(shí)間

`endif

//1680x1050 119Mhz

`ifdef PIXEL_1680_1050

parameter H_ACTIVE = 1680;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 32; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 80; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1050;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 21; // 列消隱后肩時(shí)間

`endif

//1280x1024 108Mhz

`ifdef PIXEL_1280_1024

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 112; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 248; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1024;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 3; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 38; // 列消隱后肩時(shí)間

`endif

//1280X720 74.25MHZ

`ifdef PIXEL_1280_720

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 110; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 40; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 220; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 720; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 5; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 20; // 列消隱后肩時(shí)間

`endif

//1024x768 65Mhz

`ifdef PIXEL_1024_768

parameter H_ACTIVE = 1024;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 24; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 136; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 160; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 768; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 29; // 列消隱后肩時(shí)間

`endif

//800x600 40Mhz

`ifdef PIXEL_800_600

parameter H_ACTIVE = 800;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 40 ;// 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 128;// 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 88 ;// 行消隱后肩時(shí)間

parameter V_ACTIVE = 600;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1 ;// 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 4 ;// 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 23 ;// 列消隱后肩時(shí)間

`endif

//640x480 25.175Mhz

`ifdef PIXEL_640_480

parameter H_ACTIVE = 640; // 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 16 ; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 96 ; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 48 ; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 480; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 10 ; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 2 ; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 33 ; // 列消隱后肩時(shí)間

`endif

parameter H_TOTAL_TIME = H_ACTIVE + H_FRONT_PORCH + H_SYNC_TIME + H_BACK_PORCH;

parameter V_TOTAL_TIME = V_ACTIVE + V_FRONT_PORCH + V_SYNC_TIME + V_BACK_PORCH;

reg h_act;

reg v_act;

reg [12:0] h_syn_cnt;

reg [12:0] v_syn_cnt;

reg [19:0] image_cnt;

// 有效數(shù)據(jù)控制

assign en = h_act & v_act;

// 行掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME - 1)

h_syn_cnt <= 13'b0;

else

h_syn_cnt <= h_syn_cnt + 1'b1;

end

// 列掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME - 1)

begin

if(v_syn_cnt == V_TOTAL_TIME-1)

v_syn_cnt <= 13'b0;

else

v_syn_cnt <= v_syn_cnt + 1'b1;

end

end

// 行同步控制

always@(posedge clk)

begin

if(h_syn_cnt < H_SYNC_TIME)

hs <= 1'b0;

else

hs <= 1'b1;

end

// 場(chǎng)同步控制

always@(posedge clk)

begin

if(v_syn_cnt < V_SYNC_TIME)

vs <= 1'b0;

else

vs <= 1'b1;

end

// 行有效控制

always@(posedge clk)

begin

if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH - 1)

h_act = 1'b1;

else if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH + H_ACTIVE - 1)

h_act = 1'b0;

end

// 列有效控制

always@(posedge clk)

begin

if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH - 1 && h_syn_cnt == 0)

v_act = 1'b1;

else if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH + V_ACTIVE - 1 && h_syn_cnt == 0)

v_act = 1'b0;

end

// 像素?cái)?shù)計(jì)數(shù)器

always@(posedge clk)

begin

if(image_cnt == H_ACTIVE*V_ACTIVE - 1)

image_cnt <= 20'b0;

else if(h_act & v_act)

image_cnt <= image_cnt + 1'b1;

end

// 調(diào)用ROM IP

blk_mem inst_blk_mem

(

.clka (clk),

.ena (h_act & v_act),

.addra (image_cnt),

.douta (data)

);

endmodule

仿真文件仍沿用cx_tb.v,但圖像分辨率應(yīng)改為1280×720,生成image_out.txt文件與仿真波形:

cedd73fa-990c-11ed-bfe3-dac502259ad0.png

3.3 Matlab顯示.txt文件

采用2.3中.txt文件轉(zhuǎn)圖像的Matlab代碼hex_rgb.m,同樣可以得到圖像:

cd94dfc4-990c-11ed-bfe3-dac502259ad0.pngcdcee3e0-990c-11ed-bfe3-dac502259ad0.png

04圖像存儲(chǔ)模塊仿真方法

通過讀取.coe格式圖像文件實(shí)現(xiàn)FPGA仿真步驟:

1. 利用Matlab生成圖像存儲(chǔ)模塊.v文件;

2. 通過VIVADO讀取圖像并處理生成.txt格式文件;

3. 利用Matlab轉(zhuǎn)換顯示Vivado處理后的.txt格式圖像。

4.1 Matlab生成.v文件

利用Matlab獲取圖像數(shù)據(jù)并生成.v文件:

cf2db108-990c-11ed-bfe3-dac502259ad0.png

Matlab代碼如下:

%**********************************************************************

% -------------------------------------------------------------------

% Company: Cascatrix

% Engineer: Carson

%

% Create Date: 2023/01/19

% Design Name: rgb_v

% Module Name: rgb_v

% Tool Versions: v1.0

% Description: Convert RGB888 into .v

%-------------------------------------------------------------------

%*********************************************************************/

clear;clear all;clc;

% Load image

image_in = imread('cascatrix.jpg');

% Create .v file

FileName=['image_in','.v'];

% Get image size

[row,col,n] = size(image_in);

% Write image into image_in.v

FileImage = fopen(FileName,'w');

% Format of image.coe

fprintf(FileImage,'module image_rom ');

fprintf(FileImage,'( ');

fprintf(FileImage,'input clk, ');

fprintf(FileImage,'input[31:0]addr, ');

fprintf(FileImage,'inputen, ');

fprintf(FileImage,'output[23:0]dout ');

fprintf(FileImage,'); ');

fprintf(FileImage,'reg [23:0] DATA; ');

fprintf(FileImage,'assigndouta = DATA; ');

fprintf(FileImage,'always@(*) ');

fprintf(FileImage,'begin ');

fprintf(FileImage,'case(addra) ');

s=0;

for x = 1:row

for y = 1:col

image_R = dec2hex(image_in(x,y,1));

image_G = dec2hex(image_in(x,y,2));

image_B = dec2hex(image_in(x,y,3));

[rm,rn]=size(image_R);

[gm,gn]=size(image_G);

[bm,bn]=size(image_B);

if rn == 1

image_R =['0',image_R];

elseif rn == 0

image_R ='00';

end

if gn == 1

image_G =['0',image_G];

elseif gn == 0

image_G ='00';

end

if bn == 1

image_B =['0',image_B];

elseif bn == 0

image_B ='00';

end

image_hex=[': DATA<=24','''','h',image_R,image_G,image_B,';'];

fprintf(FileImage,'%d %s ',(x-1)*col + y -1,image_hex);

end

end

fprintf(FileImage,' default: DATA<= 0; ');

fprintf(FileImage,'endcase ');

fprintf(FileImage,'end ');

fprintf(FileImage,'endmodule ');

fclose(FileImage);

rgb_v代碼運(yùn)行后生成.v文件image_in.v:

cf59d1fc-990c-11ed-bfe3-dac502259ad0.pngcf8d8452-990c-11ed-bfe3-dac502259ad0.png

4.2 Vivado讀取與處理

將生成的圖像存儲(chǔ)image.v模塊在cx_image調(diào)用:

`timescale 1ns / 1ps

//////////////////////////////////////////////////////////////////////////////////

// Company: Cascatrix

// Engineer: Carson

//

// Create Date: 2023/01/15

// Design Name: Image_Base

// Module Name: cx_image

// Tool Versions: v1.0

// Description: Image read and generate display parameters

//

//////////////////////////////////////////////////////////////////////////////////

`define PIXEL_1920_1080

//`define PIXEL_1680_1050

//`define PIXEL_1280_1024

//`define PIXEL_1280_720

//`define PIXEL_1024_768

//`define PIXEL_800_600

//`define PIXEL_640_480

module cx_image(

inputwireclk,

outputreghs,

outputregvs,

outputwireen,

outputwire [23:0]data

);

//1920x1080 148.5Mhz

`ifdef PIXEL_1920_1080

parameter H_ACTIVE = 1920;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 88; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 44; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 148; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1080;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 4; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 36; // 列消隱后肩時(shí)間

`endif

//1680x1050 119Mhz

`ifdef PIXEL_1680_1050

parameter H_ACTIVE = 1680;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 32; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 80; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1050;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 21; // 列消隱后肩時(shí)間

`endif

//1280x1024 108Mhz

`ifdef PIXEL_1280_1024

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 48; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 112; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 248; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 1024;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 3; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 38; // 列消隱后肩時(shí)間

`endif

//1280X720 74.25MHZ

`ifdef PIXEL_1280_720

parameter H_ACTIVE = 1280;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 110; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 40; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 220; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 720; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 5; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 5; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 20; // 列消隱后肩時(shí)間

`endif

//1024x768 65Mhz

`ifdef PIXEL_1024_768

parameter H_ACTIVE = 1024;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 24; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 136; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 160; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 768; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 3; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 6; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 29; // 列消隱后肩時(shí)間

`endif

//800x600 40Mhz

`ifdef PIXEL_800_600

parameter H_ACTIVE = 800;// 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 40 ;// 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 128;// 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 88 ;// 行消隱后肩時(shí)間

parameter V_ACTIVE = 600;// 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 1 ;// 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 4 ;// 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 23 ;// 列消隱后肩時(shí)間

`endif

//640x480 25.175Mhz

`ifdef PIXEL_640_480

parameter H_ACTIVE = 640; // 行數(shù)據(jù)有效時(shí)間

parameter H_FRONT_PORCH = 16 ; // 行消隱前肩時(shí)間

parameter H_SYNC_TIME = 96 ; // 行同步信號(hào)時(shí)間

parameter H_BACK_PORCH = 48 ; // 行消隱后肩時(shí)間

parameter V_ACTIVE = 480; // 列數(shù)據(jù)有效時(shí)間

parameter V_FRONT_PORCH = 10 ; // 列消隱前肩時(shí)間

parameter V_SYNC_TIME = 2 ; // 列同步信號(hào)時(shí)間

parameter V_BACK_PORCH = 33 ; // 列消隱后肩時(shí)間

`endif

parameter H_TOTAL_TIME = H_ACTIVE + H_FRONT_PORCH + H_SYNC_TIME + H_BACK_PORCH;

parameter V_TOTAL_TIME = V_ACTIVE + V_FRONT_PORCH + V_SYNC_TIME + V_BACK_PORCH;

reg h_act;

reg v_act;

reg [12:0] h_syn_cnt;

reg [12:0] v_syn_cnt;

reg [31:0] image_cnt;

// 有效數(shù)據(jù)控制

assign en = h_act & v_act;

// 行掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME - 1)

h_syn_cnt <= 13'b0;

else

h_syn_cnt <= h_syn_cnt + 1'b1;

end

// 列掃描計(jì)數(shù)器

always@(posedge clk)

begin

if(h_syn_cnt == H_TOTAL_TIME - 1)

begin

if(v_syn_cnt == V_TOTAL_TIME-1)

v_syn_cnt <= 13'b0;

else

v_syn_cnt <= v_syn_cnt + 1'b1;

end

end

// 行同步控制

always@(posedge clk)

begin

if(h_syn_cnt < H_SYNC_TIME)

hs <= 1'b0;

else

hs <= 1'b1;

end

// 場(chǎng)同步控制

always@(posedge clk)

begin

if(v_syn_cnt < V_SYNC_TIME)

vs <= 1'b0;

else

vs <= 1'b1;

end

// 行有效控制

always@(posedge clk)

begin

if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH - 1)

h_act = 1'b1;

else if(h_syn_cnt == H_SYNC_TIME + H_BACK_PORCH + H_ACTIVE - 1)

h_act = 1'b0;

end

// 列有效控制

always@(posedge clk)

begin

if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH - 1 && h_syn_cnt == 0)

v_act = 1'b1;

else if(v_syn_cnt == V_SYNC_TIME + V_BACK_PORCH + V_ACTIVE - 1 && h_syn_cnt == 0)

v_act = 1'b0;

end

// 像素?cái)?shù)計(jì)數(shù)器

always@(posedge clk)

begin

if(image_cnt == H_ACTIVE*V_ACTIVE - 1)

image_cnt <= 32’b0;

else if(h_act & v_act)

image_cnt <= image_cnt + 1'b1;

end

// 圖像像素值存儲(chǔ)模塊

image inst_rom_image

(

.clk (clk),

.en (h_act & v_act),

.addr (image_cnt),

.dout (data)

);

endmodule

仿真文件仍采用cx_tb.v,生成image_out.txt文件與仿真波形:

cfb837a6-990c-11ed-bfe3-dac502259ad0.png

4.3 Matlab顯示.txt文件

采用2.3中.txt文件轉(zhuǎn)圖像的Matlab代碼hex_rgb.m,同樣可以得到圖像:

cd94dfc4-990c-11ed-bfe3-dac502259ad0.pngcdcee3e0-990c-11ed-bfe3-dac502259ad0.png

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • FPGA
    +關(guān)注

    關(guān)注

    1630

    文章

    21783

    瀏覽量

    604994
  • matlab
    +關(guān)注

    關(guān)注

    185

    文章

    2980

    瀏覽量

    230774
  • Verilog
    +關(guān)注

    關(guān)注

    28

    文章

    1351

    瀏覽量

    110224
  • 數(shù)字圖像處理
    +關(guān)注

    關(guān)注

    7

    文章

    103

    瀏覽量

    18940
  • 圖像仿真
    +關(guān)注

    關(guān)注

    0

    文章

    2

    瀏覽量

    5237

原文標(biāo)題:FPGA數(shù)字圖像處理仿真方法

文章出處:【微信號(hào):Carlinx FPGA,微信公眾號(hào):Carlinx FPGA】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    FPGA數(shù)字圖像處理的基礎(chǔ)知識(shí)

    數(shù)字圖像** (Digital Image),是計(jì)算機(jī)視覺與圖像處理的基礎(chǔ),區(qū)別于模擬圖像
    發(fā)表于 09-06 17:11 ?1773次閱讀
    <b class='flag-5'>FPGA</b><b class='flag-5'>數(shù)字圖像</b><b class='flag-5'>處理</b>的基礎(chǔ)知識(shí)

    FreeDev FPGA數(shù)字圖像處理

    FreeDev FPGA數(shù)字圖像處理
    發(fā)表于 08-17 22:34

    數(shù)字圖像處理

    1、圖像的變換和壓縮,利用離散余弦變換(DCT)2、實(shí)現(xiàn)圖像的真彩色增強(qiáng)3、實(shí)現(xiàn)圖像的灰度變換,利用直方圖均衡化的方法4、使用常用的濾波器對(duì)數(shù)字圖像
    發(fā)表于 04-11 23:33

    基于FPGA數(shù)字圖像處理中的邊緣檢測(cè)系統(tǒng)

    `基于FPGA數(shù)字圖像處理領(lǐng)域的邊緣檢測(cè)系統(tǒng)。該系統(tǒng)實(shí)現(xiàn)了從24位真彩色圖片的存儲(chǔ)到VGA顯示邊緣信息。`
    發(fā)表于 06-26 13:36

    有誰用FPGA做過數(shù)字圖像直方圖統(tǒng)計(jì)

    剛開始用FPGA數(shù)字圖像處理,看了一些有關(guān)直方圖方面的資料但是感覺還是不太清晰,請(qǐng)問有誰做過求帶順便求推薦FPGA數(shù)字圖像
    發(fā)表于 08-21 09:48

    有誰用FPGA做過數(shù)字圖像直方圖統(tǒng)計(jì)

    剛開始用FPGA數(shù)字圖像處理,看了一些有關(guān)直方圖方面的資料但是感覺還是不太清晰,請(qǐng)問有誰做過求帶順便求推薦FPGA數(shù)字圖像
    發(fā)表于 08-23 11:01

    數(shù)字圖像處理的技術(shù)方法和應(yīng)用

    圖像,常用的圖像處理方法圖像增強(qiáng)、復(fù)原、編碼、壓縮等。下面維視圖像為您簡(jiǎn)單介紹一下
    發(fā)表于 11-18 11:45

    基于FPGA數(shù)字圖像處理

    分享一本用fpga數(shù)字圖像處理的書。
    發(fā)表于 03-05 21:40

    【電子書】基于FPGA數(shù)字圖像處理

    `一個(gè)FPGA工程師的數(shù)字圖像處理筆記。`
    發(fā)表于 03-29 15:01

    數(shù)字圖像處理實(shí)驗(yàn)報(bào)告

    數(shù)字圖像處理實(shí)驗(yàn)報(bào)告數(shù)字圖像處理實(shí)驗(yàn)報(bào)告數(shù)字圖像處理實(shí)驗(yàn)報(bào)告
    發(fā)表于 11-11 15:33 ?0次下載

    數(shù)字圖像處理試題集

    數(shù)字圖像處理試題集數(shù)字圖像處理試題集數(shù)字圖像處理試題集
    發(fā)表于 12-21 15:13 ?3次下載

    數(shù)字圖像邊緣檢測(cè)的FPGA實(shí)現(xiàn)

    數(shù)字圖像邊緣檢測(cè)的FPGA實(shí)現(xiàn)......
    發(fā)表于 01-04 15:31 ?18次下載

    數(shù)字圖像處理平臺(tái)系統(tǒng)方案詳解

    數(shù)字圖像處理自從出現(xiàn)以來,就一直是前沿的研究學(xué)科,經(jīng)久不衰,同時(shí)隨著數(shù)字化時(shí)代的到來,市場(chǎng)對(duì)于數(shù)字圖像處理的需求也越來越大。因此,本項(xiàng)目--
    發(fā)表于 05-30 11:39 ?4731次閱讀
    <b class='flag-5'>數(shù)字圖像</b><b class='flag-5'>處理</b>平臺(tái)系統(tǒng)方案詳解

    探析數(shù)字圖像處理常用的方法

    數(shù)字圖像處理(Digital Image Processing)是通過計(jì)算機(jī)對(duì)圖像進(jìn)行去除噪聲、增強(qiáng)、復(fù)原、分割、提取特征等處理方法和技術(shù)
    的頭像 發(fā)表于 01-29 11:35 ?5271次閱讀

    數(shù)字圖像處理基本的知識(shí)(一)

    數(shù)字圖像處理基本知識(shí)(一) demi 在 周二, 04/02/2019 - 13:38 提交 1、數(shù)字圖像數(shù)字圖像,又稱為數(shù)碼圖像或數(shù)位
    的頭像 發(fā)表于 03-24 17:24 ?6973次閱讀