2016-02-22 23:03:50 +00:00
|
|
|
//
|
2017-05-18 20:35:43 +00:00
|
|
|
// Copyright (C) 2015-2017 Markus Hiienkari <mhiienka@niksula.hut.fi>
|
2016-02-22 23:03:50 +00:00
|
|
|
//
|
|
|
|
// This file is part of Open Source Scan Converter project.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//
|
|
|
|
|
2017-10-28 09:10:54 +00:00
|
|
|
`include "lat_tester_includes.v"
|
2017-10-22 19:45:29 +00:00
|
|
|
|
2016-02-22 23:03:50 +00:00
|
|
|
module videogen (
|
2016-03-27 19:40:44 +00:00
|
|
|
input clk27,
|
|
|
|
input reset_n,
|
2017-10-22 19:45:29 +00:00
|
|
|
input lt_active,
|
|
|
|
input [1:0] lt_mode,
|
2019-10-02 23:03:43 +00:00
|
|
|
output reg [7:0] R_out,
|
|
|
|
output reg [7:0] G_out,
|
|
|
|
output reg [7:0] B_out,
|
2016-03-27 19:40:44 +00:00
|
|
|
output reg HSYNC_out,
|
|
|
|
output reg VSYNC_out,
|
2019-10-05 21:26:50 +00:00
|
|
|
output reg DE_out,
|
2019-10-02 23:03:43 +00:00
|
|
|
output reg [9:0] xpos,
|
|
|
|
output reg [9:0] ypos
|
2016-02-22 23:03:50 +00:00
|
|
|
);
|
|
|
|
|
2016-03-27 19:40:44 +00:00
|
|
|
//Parameters for 720x480@59.94Hz (858px x 525lines, pclk 27MHz -> 59.94Hz)
|
2018-03-02 20:57:32 +00:00
|
|
|
parameter H_SYNCLEN = 10'd62;
|
|
|
|
parameter H_BACKPORCH = 10'd60;
|
|
|
|
parameter H_ACTIVE = 10'd720;
|
|
|
|
parameter H_FRONTPORCH = 10'd16;
|
|
|
|
parameter H_TOTAL = 10'd858;
|
2016-02-22 23:03:50 +00:00
|
|
|
|
2018-03-02 20:57:32 +00:00
|
|
|
parameter V_SYNCLEN = 10'd6;
|
|
|
|
parameter V_BACKPORCH = 10'd30;
|
|
|
|
parameter V_ACTIVE = 10'd480;
|
|
|
|
parameter V_FRONTPORCH = 10'd9;
|
|
|
|
parameter V_TOTAL = 10'd525;
|
2016-02-22 23:03:50 +00:00
|
|
|
|
2018-03-02 20:57:32 +00:00
|
|
|
parameter H_OVERSCAN = 10'd40; //at both sides
|
|
|
|
parameter V_OVERSCAN = 10'd16; //top and bottom
|
|
|
|
parameter H_AREA = 10'd640;
|
|
|
|
parameter V_AREA = 10'd448;
|
|
|
|
parameter H_GRADIENT = 10'd512;
|
|
|
|
parameter V_GRADIENT = 10'd256;
|
|
|
|
parameter V_GRAYRAMP = 10'd84;
|
|
|
|
parameter H_BORDER = ((H_AREA-H_GRADIENT)>>1);
|
|
|
|
parameter V_BORDER = ((V_AREA-V_GRADIENT)>>1);
|
2016-02-22 23:03:50 +00:00
|
|
|
|
2016-03-27 19:40:44 +00:00
|
|
|
parameter X_START = H_SYNCLEN + H_BACKPORCH;
|
|
|
|
parameter Y_START = V_SYNCLEN + V_BACKPORCH;
|
2016-02-22 23:03:50 +00:00
|
|
|
|
|
|
|
//Counters
|
|
|
|
reg [9:0] h_cnt; //max. 1024
|
|
|
|
reg [9:0] v_cnt; //max. 1024
|
|
|
|
|
|
|
|
//HSYNC gen (negative polarity)
|
|
|
|
always @(posedge clk27 or negedge reset_n)
|
|
|
|
begin
|
2017-05-18 20:35:43 +00:00
|
|
|
if (!reset_n) begin
|
|
|
|
h_cnt <= 0;
|
2019-10-02 23:03:43 +00:00
|
|
|
xpos <= 0;
|
2017-05-18 20:35:43 +00:00
|
|
|
HSYNC_out <= 0;
|
|
|
|
end else begin
|
|
|
|
//Hsync counter
|
2019-10-02 23:03:43 +00:00
|
|
|
if (h_cnt < H_TOTAL-1) begin
|
2017-05-18 20:35:43 +00:00
|
|
|
h_cnt <= h_cnt + 1'b1;
|
2019-10-02 23:03:43 +00:00
|
|
|
if (h_cnt >= X_START)
|
|
|
|
xpos <= xpos + 1'b1;
|
|
|
|
end else begin
|
2016-03-27 19:40:44 +00:00
|
|
|
h_cnt <= 0;
|
2019-10-02 23:03:43 +00:00
|
|
|
xpos <= 0;
|
|
|
|
end
|
2016-02-22 23:03:50 +00:00
|
|
|
|
2017-05-18 20:35:43 +00:00
|
|
|
//Hsync signal
|
|
|
|
HSYNC_out <= (h_cnt < H_SYNCLEN) ? 1'b0 : 1'b1;
|
|
|
|
end
|
2016-02-22 23:03:50 +00:00
|
|
|
end
|
|
|
|
|
2017-05-18 20:35:43 +00:00
|
|
|
//VSYNC gen (negative polarity)
|
2016-02-22 23:03:50 +00:00
|
|
|
always @(posedge clk27 or negedge reset_n)
|
|
|
|
begin
|
2017-05-18 20:35:43 +00:00
|
|
|
if (!reset_n) begin
|
|
|
|
v_cnt <= 0;
|
2019-10-02 23:03:43 +00:00
|
|
|
ypos <= 0;
|
2017-05-18 20:35:43 +00:00
|
|
|
VSYNC_out <= 0;
|
|
|
|
end else begin
|
|
|
|
//Vsync counter
|
|
|
|
if (h_cnt == H_TOTAL-1) begin
|
2019-10-02 23:03:43 +00:00
|
|
|
if (v_cnt < V_TOTAL-1) begin
|
2017-05-18 20:35:43 +00:00
|
|
|
v_cnt <= v_cnt + 1'b1;
|
2019-10-02 23:03:43 +00:00
|
|
|
if (v_cnt >= Y_START)
|
|
|
|
ypos <= ypos + 1'b1;
|
|
|
|
end else begin
|
2017-05-18 20:35:43 +00:00
|
|
|
v_cnt <= 0;
|
2019-10-02 23:03:43 +00:00
|
|
|
ypos <= 0;
|
|
|
|
end
|
2016-02-22 23:03:50 +00:00
|
|
|
end
|
2017-05-18 20:35:43 +00:00
|
|
|
|
|
|
|
//Vsync signal
|
|
|
|
VSYNC_out <= (v_cnt < V_SYNCLEN) ? 1'b0 : 1'b1;
|
|
|
|
end
|
2016-02-22 23:03:50 +00:00
|
|
|
end
|
|
|
|
|
2017-05-18 20:35:43 +00:00
|
|
|
//Data and ENABLE gen
|
2016-02-22 23:03:50 +00:00
|
|
|
always @(posedge clk27 or negedge reset_n)
|
|
|
|
begin
|
2017-05-18 20:35:43 +00:00
|
|
|
if (!reset_n) begin
|
2019-10-02 23:03:43 +00:00
|
|
|
R_out <= 8'h00;
|
|
|
|
G_out <= 8'h00;
|
|
|
|
B_out <= 8'h00;
|
2019-10-05 21:26:50 +00:00
|
|
|
DE_out <= 1'b0;
|
2017-05-18 20:35:43 +00:00
|
|
|
end else begin
|
2019-10-06 22:25:33 +00:00
|
|
|
if (lt_active) begin
|
2017-10-22 19:45:29 +00:00
|
|
|
case (lt_mode)
|
|
|
|
default: begin
|
2019-10-02 23:03:43 +00:00
|
|
|
{R_out, G_out, B_out} <= {3{8'h00}};
|
2017-10-22 19:45:29 +00:00
|
|
|
end
|
|
|
|
`LT_POS_TOPLEFT: begin
|
2019-10-02 23:03:43 +00:00
|
|
|
{R_out, G_out, B_out} <= {3{((xpos < (H_ACTIVE/`LT_WIDTH_DIV)) && (ypos < (V_ACTIVE/`LT_HEIGHT_DIV))) ? 8'hff : 8'h00}};
|
2017-10-22 19:45:29 +00:00
|
|
|
end
|
|
|
|
`LT_POS_CENTER: begin
|
2019-10-02 23:03:43 +00:00
|
|
|
{R_out, G_out, B_out} <= {3{((xpos >= ((H_ACTIVE/2)-(H_ACTIVE/(`LT_WIDTH_DIV*2)))) && (xpos < ((H_ACTIVE/2)+(H_ACTIVE/(`LT_WIDTH_DIV*2)))) && (ypos >= ((V_ACTIVE/2)-(V_ACTIVE/(`LT_HEIGHT_DIV*2)))) && (ypos < ((V_ACTIVE/2)+(V_ACTIVE/(`LT_HEIGHT_DIV*2))))) ? 8'hff : 8'h00}};
|
2017-10-22 19:45:29 +00:00
|
|
|
end
|
|
|
|
`LT_POS_BOTTOMRIGHT: begin
|
2019-10-02 23:03:43 +00:00
|
|
|
{R_out, G_out, B_out} <= {3{((xpos >= (H_ACTIVE-(H_ACTIVE/`LT_WIDTH_DIV))) && (ypos >= (V_ACTIVE-(V_ACTIVE/`LT_HEIGHT_DIV)))) ? 8'hff : 8'h00}};
|
2017-10-22 19:45:29 +00:00
|
|
|
end
|
|
|
|
endcase
|
|
|
|
end else begin
|
2019-10-02 23:03:43 +00:00
|
|
|
if ((xpos < H_OVERSCAN) || (xpos >= H_OVERSCAN+H_AREA) || (ypos < V_OVERSCAN) || (ypos >= V_OVERSCAN+V_AREA))
|
|
|
|
{R_out, G_out, B_out} <= {3{(xpos[0] ^ ypos[0]) ? 8'hff : 8'h00}};
|
|
|
|
else if ((xpos < H_OVERSCAN+H_BORDER) || (xpos >= H_OVERSCAN+H_AREA-H_BORDER) || (ypos < V_OVERSCAN+V_BORDER) || (ypos >= V_OVERSCAN+V_AREA-V_BORDER))
|
|
|
|
{R_out, G_out, B_out} <= {3{8'h50}};
|
|
|
|
else if (ypos >= V_OVERSCAN+V_BORDER+V_GRADIENT-V_GRAYRAMP)
|
|
|
|
{R_out, G_out, B_out} <= {3{8'((((xpos - (H_OVERSCAN+H_BORDER)) >> 4) << 3) + (xpos - (H_OVERSCAN+H_BORDER) >> 6))}};
|
2017-10-22 19:45:29 +00:00
|
|
|
else
|
2019-10-02 23:03:43 +00:00
|
|
|
{R_out, G_out, B_out} <= {3{8'((xpos - (H_OVERSCAN+H_BORDER)) >> 1)}};
|
2017-10-22 19:45:29 +00:00
|
|
|
end
|
2017-05-18 20:35:43 +00:00
|
|
|
|
2019-10-05 21:26:50 +00:00
|
|
|
DE_out <= (h_cnt >= X_START && h_cnt < X_START + H_ACTIVE && v_cnt >= Y_START && v_cnt < Y_START + V_ACTIVE);
|
2017-05-18 20:35:43 +00:00
|
|
|
end
|
2016-02-22 23:03:50 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
endmodule
|