forked from JackZhao98/T-rex-verilog
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathVGA.v
177 lines (146 loc) · 5.19 KB
/
VGA.v
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
module VGA(
input wire pixel_clock, //pixel clock: 25MHz
input wire rst, //asynchronous reset
output wire Hsync, //horizontal sync out
output wire Vsync, //vertical sync out
output wire FPSClk,
output wire [9:0] X, // X counter coordinate
output wire [8:0] Y // Y counter Coordinate
);
/**********************
* *
* 4 *
* 8 *
* 0 *
* 640 *
* *******************/
// video structure constants
parameter hpixels = 800;// horizontal pixels per line
parameter vlines = 521; // vertical lines per frame
parameter hpulse = 96; // Hsync pulse length
parameter vpulse = 2; // Vsync pulse length
parameter hbp = 144; // end of horizontal back porch
parameter hfp = 784; // beginning of horizontal front porch
parameter vbp = 31; // end of vertical back porch
parameter vfp = 511; // beginning of vertical front porch
// active horizontal video is therefore: 784 - 144 = 640
// active vertical video is therefore: 511 - 31 = 480
// graphic parameter (top-left is (0, 0))
localparam screen_width = hfp - hbp;
localparam screen_height = vfp - vbp;
// registers for storing the horizontal & vertical counters
reg [9:0] hc;
reg [9:0] vc;
// Horizontal & vertical counters --
// this is how we keep track of where we are on the screen.
// ------------------------
// Sequential "always block", which is a block that is
// only triggered on signal transitions or "edges".
// posedge = rising edge & negedge = falling edge
// Assignment statements can only be used on type "reg" and need to be of the "non-blocking" type: <=
always @(posedge pixel_clock or posedge rst)
begin
// reset condition
if (rst == 1)
begin
hc <= 0;
vc <= 0;
end
else
begin
// keep counting until the end of the line
if (hc < hpixels - 1)
hc <= hc + 1;
else
// When we hit the end of the line, reset the horizontal
// counter and increment the vertical counter.
// If vertical counter is at the end of the frame, then
// reset that one too.
begin
hc <= 0;
if (vc < vlines - 1)
vc <= vc + 1;
else
vc <= 0;
end
end
end
assign Hsync = (hc < hpulse) ? 0:1;
assign Vsync = (vc < vpulse) ? 0:1;
assign X = (hc >= hbp)? (hc-hbp):0;
assign Y = (vc >= vbp)? (vc -vbp):0;
assign FPSClk = ((hc == hpixels) & (vc == screen_height - 1));
endmodule
/* Archived
module VGA(
input wire clk,
input wire pixel_clk,
input wire rst,
output wire Hsync,
output wire Vsync,
output wire blanking,
output wire active,
output wire screened,
output wire animate,
output wire [9:0] x, // x and y represent the current
output wire [8:0] y); // pixel within the screen
localparam RES_H = 480;
localparam RES_W = 640;
/**********************
* *
* 4 *
* 8 *
* 0 *
* 640 *
* ********************
localparam H_Front_Porch = 16;
localparam H_Sync_Pulse = 96;
localparam H_Back_Porch = 48;
localparam V_Front_Porch = 10;
localparam V_Sync_Pulse = 2;
localparam V_Back_Porch = 33;
localparam HS_START = H_Front_Porch;
localparam HS_END = H_Front_Porch + H_Sync_Pulse;
localparam HA_START = H_Front_Porch + H_Sync_Pulse + H_Back_Porch;
// |--16--|-------96-------|----48----|ACTIVE AREA|
localparam VS_START = RES_H + V_Front_Porch;
localparam VS_END = RES_H + V_Front_Porch + V_Sync_Pulse;
localparam VA_END = RES_H;
localparam LINE = H_Front_Porch + H_Sync_Pulse + H_Back_Porch + RES_W; // 800
localparam SCREEN = RES_H + V_Front_Porch + V_Sync_Pulse + V_Back_Porch; // 525
reg [9:0] h_count;
reg [9:0] v_count;
// generate sync signals (1 control active)
assign Hsync = ((h_count >= HS_START) & (h_count <= HS_END));
assign Vsync = ((v_count >= VS_START) & (v_count < VS_END));
// Keep x and y in-bound within active region (640*480)
// When counter is in blank area, keep x = 0, y = 479;
assign x = (h_count < HA_START)? 0 : (h_count - HA_START);
assign y = (v_count >= VA_END)? (RES_H - 1) : v_count;
// Blanking signal
assign blanking = ((h_count < HA_START) | (v_count >= VA_END));
// Active signal
assign active = ~blanking;
// Screened: high for one tick at the end of screen (one frame)
assign screened = ((h_count == LINE) & (v_count == SCREEN - 1));
// Active: high for one tick at tha last active pixel
assign animate = ((h_count == LINE) & (v_count == VA_END - 1));
always @(posedge clk) begin
if (rst) begin
h_count <= 0;
v_count <= 0;
end
if (pixel_clk) begin
if (h_count == LINE) begin
h_count <= 0;
v_count <= v_count + 1;
end
else
h_count <= h_count + 1;
if (v_count == SCREEN)
v_count <= 0;
end // if (pixel_clk)
end // always @ (posedge clk)
endmodule // VGA
// Reference: https://timetoexplore.net/blog/arty-fpga-vga-verilog-01
*/