Added basic VGA graphics support

Generated some basic FastAdders
Generalized FastAdder to include all fast math modules
Other minor changes
This commit is contained in:
Gabriel Tofvesson 2018-10-12 16:59:08 +02:00
parent 571a280d3e
commit 8b9bd6e106
8 changed files with 204 additions and 150 deletions

1
.gitignore vendored
View File

@ -24,3 +24,4 @@ incremental_db/*
*.qws
*.qsf
*.qpf
.sopc_builder/*

60
ALU.v
View File

@ -25,15 +25,19 @@ H: N/A
reg [15:0] i_z;
reg [7:0] i_flg;
wire [8:0] add_out;
assign z = i_z[7:0];
assign o_flags = i_flg;
FastAdder8 fa8(.cin(), .a(a), .b(b), .out(add_out[7:0]), .cout(add_out[8]));
always @* begin
case(op & 8'b00011111) // 5-bit instructions: 3 flag bits
// ADD
0: begin
i_z <= a+b;
i_flg <= z < a ? 1 : 0; // Set overflow flag if necessary
i_z <= add_out;
i_flg <= add_out[8] ? 8'b1 : 8'b0; // Set overflow flag if necessary
end
// SUB
@ -45,17 +49,17 @@ always @* begin
// MUL
2: begin
i_z <= a*b;
i_flg <= i_z[15:8] != 8'b00000000 ? 1 : 0;
i_flg <= i_z[15:8] != 8'b0 ? 8'b1 : 8'b0;
end
// DIV
3: begin
if(b != 8'b00000000) begin
if(b != 8'b0) begin
i_z <= a/b;
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end else begin
i_z <= 8'b00000000;
i_flg <= 8'b00010000;
i_z <= 16'b0;
i_flg <= 8'b10000;
end
end
@ -74,65 +78,65 @@ always @* begin
111 -> No output
*/
i_z <= (op[7:5] == 3'b000) || (op[7:5] == 3'b011) || (op[7:5] == 3'b111) ? 0 : (op[5] && a > b) || (op[6] && a < b ) || (op[7] && a == b) ? 1 : 0;
i_z <= (op[7:5] == 3'b000) || (op[7:5] == 3'b011) || (op[7:5] == 3'b111) ? 16'b0 : (op[5] && a > b) || (op[6] && a < b ) || (op[7] && a == b) ? 16'b1 : 16'b0;
i_flg <=
(a > b ? 4 : 0) | // a > b
(a == b ? 8 : 0); // a == b
(a > b ? 8'b100 : 8'b0) | // a > b
(a == b ? 8'b1000 : 8'b0); // a == b
end
// AND
5: begin
i_z <= a & b;
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// OR
6: begin
i_z <= a | b;
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// XOR
7: begin
i_z <= a ^ b;
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// NOT
8: begin
i_z <= ~a;
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// NAND
9: begin
i_z <= ~(a & b);
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// NOR
10: begin
i_z <= ~(a | b);
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// XNOR
11: begin
i_z <= ~(a ^ b);
i_flg <= 8'b00000000;
i_flg <= 8'b0;
end
// CL_MUL
12: begin
i_z <=
(a[7] ? b << 7 : 0) ^
(a[6] ? b << 6 : 0) ^
(a[5] ? b << 5 : 0) ^
(a[4] ? b << 4 : 0) ^
(a[3] ? b << 3 : 0) ^
(a[2] ? b << 2 : 0) ^
(a[1] ? b << 1 : 0) ^
(a[0] ? b : 0);
(a[7] ? b << 7 : 16'b0) ^
(a[6] ? b << 6 : 16'b0) ^
(a[5] ? b << 5 : 16'b0) ^
(a[4] ? b << 4 : 16'b0) ^
(a[3] ? b << 3 : 16'b0) ^
(a[2] ? b << 2 : 16'b0) ^
(a[1] ? b << 1 : 16'b0) ^
(a[0] ? b : 16'b0);
i_flg <=
(a[7] && (b[1] || b[2] || b[3] || b[4] || b[5] || b[6] || b[7])) ||
@ -142,11 +146,11 @@ always @* begin
(a[3] && (b[5] || b[6] || b[7])) ||
(a[2] && (b[6] || b[7])) ||
(a[1] && b[7])
? 1 : 0;
? 8'b1 : 8'b0;
end
default: begin
i_z <= 0;
i_flg <= 32; // Unknown opcode
i_z <= 16'b0;
i_flg <= 8'b100000; // Unknown opcode
end
endcase
end

View File

@ -16,11 +16,11 @@ assign divided = div_int; // Assign internal result state to external output
// Division
always @ (posedge clk) begin
if(div == divideby) begin
div_int <= pulsemode ? 1 : ~div_int;
div_int <= pulsemode ? 1'b1 : ~div_int;
div <= 0;
end else begin
if(pulsemode) div_int <= 0;
div <= div + 1;
div <= div + 1'b1;
end
end

View File

@ -1,56 +0,0 @@
module FastAdder4(
input wire c_in,
input wire [3:0] a,
input wire [3:0] b,
output wire [3:0] out,
output wire c_out
);
wire [3:0] gen = a & b; // Generator
wire [3:0] prp = a ^ b; // Propogator
assign out = a ^ b ^ {
gen[2] | (prp[2] & gen[1]) | (prp[2] & prp[1] & gen[0]) | (prp[2] & prp[1] & prp[0] & c_in), // Carry 2
gen[1] | (prp[1] & gen[0]) | (prp[1] & prp[0] & c_in), // Carry 1
gen[0] | (prp[0] & c_in), // Carry 0
c_in // Carry -1 (in)
};
assign c_out = gen[3] | (prp[3] & gen[2]) | (prp[3] & prp[2] & gen[1]) | (prp[3] & prp[2] & prp[1] & gen[0]) | (prp[3] & prp[2] & prp[1] & prp[0] & c_in);
endmodule
module FastAdder8(
input wire c_in,
input wire [7:0] a,
input wire [7:0] b,
output wire [7:0] out,
output wire c_out
);
wire [7:0] gen = a & b; // Generator
wire [7:0] prp = a ^ b; // Propogator
assign out = a ^ b ^ {
gen[2] | (prp[2] & gen[1]) | (prp[2] & prp[1] & gen[0]) | (prp[2] & prp[1] & prp[0] & c_in), // Carry 2
gen[1] | (prp[1] & gen[0]) | (prp[1] & prp[0] & c_in), // Carry 1
gen[0] | (prp[0] & c_in), // Carry 0
c_in // Carry -1 (in)
};
assign c_out = gen[3] | (prp[3] & gen[2]) | (prp[3] & prp[2] & gen[1]) | (prp[3] & prp[2] & prp[1] & gen[0]) | (prp[3] & prp[2] & prp[1] & prp[0] & c_in);
endmodule
module FastAdder2(
input wire cin,
input wire [1:0] a,
input wire [1:0] b,
output wire [1:0] out,
output wire cout
);
wire [1:0] g = a & b;
wire [1:0] p = a ^ b;
assign out = a ^ b ^ {
(g[0]) | (p[0] & cin),
(cin)
};
assign cout = (g[1]) | (p[1] & g[0]) | (p[1] & p[0] & cin);
endmodule

58
FastMath.v Normal file
View File

@ -0,0 +1,58 @@
// 2-bit fast adder
module FastAdder2(
input wire cin,
input wire [1:0] a,
input wire [1:0] b,
output wire [1:0] out,
output wire cout
);
wire [1:0] g = a & b;
wire [1:0] p = a ^ b;
assign out = a ^ b ^ {
(g[0]) | (p[0] & cin),
(cin)
};
assign cout = (g[1]) | (p[1] & g[0]) | (p[1] & p[0] & cin);
endmodule
// 4-bit fast adder
module FastAdder4(
input wire cin,
input wire [3:0] a,
input wire [3:0] b,
output wire [3:0] out,
output wire cout
);
wire [3:0] g = a & b;
wire [3:0] p = a ^ b;
assign out = a ^ b ^ {
(g[2]) | (p[2] & g[1]) | (p[2] & p[1] & g[0]) | (p[2] & p[1] & p[0] & cin),
(g[1]) | (p[1] & g[0]) | (p[1] & p[0] & cin),
(g[0]) | (p[0] & cin),
(cin)
};
assign cout = (g[3]) | (p[3] & g[2]) | (p[3] & p[2] & g[1]) | (p[3] & p[2] & p[1] & g[0]) | (p[3] & p[2] & p[1] & p[0] & cin);
endmodule
// 8-bit fast adder
module FastAdder8(
input wire cin,
input wire [7:0] a,
input wire [7:0] b,
output wire [7:0] out,
output wire cout
);
wire [7:0] g = a & b;
wire [7:0] p = a ^ b;
assign out = a ^ b ^ {
(g[6]) | (p[6] & g[5]) | (p[6] & p[5] & g[4]) | (p[6] & p[5] & p[4] & g[3]) | (p[6] & p[5] & p[4] & p[3] & g[2]) | (p[6] & p[5] & p[4] & p[3] & p[2] & g[1]) | (p[6] & p[5] & p[4] & p[3] & p[2] & p[1] & g[0]) | (p[6] & p[5] & p[4] & p[3] & p[2] & p[1] & p[0] & cin),
(g[5]) | (p[5] & g[4]) | (p[5] & p[4] & g[3]) | (p[5] & p[4] & p[3] & g[2]) | (p[5] & p[4] & p[3] & p[2] & g[1]) | (p[5] & p[4] & p[3] & p[2] & p[1] & g[0]) | (p[5] & p[4] & p[3] & p[2] & p[1] & p[0] & cin),
(g[4]) | (p[4] & g[3]) | (p[4] & p[3] & g[2]) | (p[4] & p[3] & p[2] & g[1]) | (p[4] & p[3] & p[2] & p[1] & g[0]) | (p[4] & p[3] & p[2] & p[1] & p[0] & cin),
(g[3]) | (p[3] & g[2]) | (p[3] & p[2] & g[1]) | (p[3] & p[2] & p[1] & g[0]) | (p[3] & p[2] & p[1] & p[0] & cin),
(g[2]) | (p[2] & g[1]) | (p[2] & p[1] & g[0]) | (p[2] & p[1] & p[0] & cin),
(g[1]) | (p[1] & g[0]) | (p[1] & p[0] & cin),
(g[0]) | (p[0] & cin),
(cin)
};
assign cout = (g[7]) | (p[7] & g[6]) | (p[7] & p[6] & g[5]) | (p[7] & p[6] & p[5] & g[4]) | (p[7] & p[6] & p[5] & p[4] & g[3]) | (p[7] & p[6] & p[5] & p[4] & p[3] & g[2]) | (p[7] & p[6] & p[5] & p[4] & p[3] & p[2] & g[1]) | (p[7] & p[6] & p[5] & p[4] & p[3] & p[2] & p[1] & g[0]) | (p[7] & p[6] & p[5] & p[4] & p[3] & p[2] & p[1] & p[0] & cin);
endmodule

View File

@ -20,7 +20,7 @@ Divider #(.divideby(25000), .divide_reg_size(17)) divider_audio(clk, clk_graphic
// Change the active segment data
always @(posedge clk_graphics) begin
segment_select <=(segment_select << 1) | segment_select[3];
seg_sel_track <= seg_sel_track + 1;
seg_sel_track <= seg_sel_track + 1'b1;
end
// Assign the active segment data to the segment bus

View File

@ -1,4 +1,4 @@
module SevenSegment (
module SevenSegment(
input wire latch, // S4
input wire next, // S2
input wire value, // S3
@ -9,7 +9,7 @@ module SevenSegment (
output [7:0] seg_write, // a-g + dp [0-7] + 8
output reg beep, // Buzzer
output wire [10:0] RAM_addr, // RAM address buffer
output wire RAM_A10,
output wire RAM_A10, // RAM A10 precharge/address
output wire [1:0] RAM_bank_sel, // RAM bank selection
inout wire [15:0] RAM_data, // RAM data bus
output wire RAM_clk, // RAM clock signal
@ -17,11 +17,14 @@ module SevenSegment (
output wire RAM_enable, // RAM chip enable
output wire RAM_strobe_row, // RAM row strobe
output wire RAM_strobe_col, // RAM column strobe
output wire RAM_write_enable // RAM data bus write enable
output wire RAM_write_enable, // RAM data bus write enable
output wire VGA_vsync, // VGA display vsync trigger
output wire VGA_hsync, // VGA display hsync trigger
output wire [2:0] VGA_rgb // VGA color channels [0]: RED, [1]: GREEN, [2]: BLUE
);
// ---- SETTINGS ---- //
localparam PLL_SELECT = 4; // 0: 100MHz, 1: 200MHz, 2: 300MHz, 3: 400MHz, 4: 50MHz
localparam PLL_SELECT = 3; // 0: 100MHz, 1: 200MHz, 2: 300MHz, 3: 400MHz, 4: 50MHz
// ---- REGISTERS ---- //
reg debounce; // Input debouncer
@ -31,23 +34,22 @@ reg [1:0] stage; // Computational stage
reg [7:0] alu_a; // ALU (core0) input a
reg [7:0] alu_b; // ALU (core0) input b
reg [7:0] alu_op; // ALU (core0) opcode
reg [2:0] gfx_rgb; // VGA color channels
// ---- WIRES ---- //
wire [7:0] seg_buf[0:3]; // Encoded segment buffer (8-bit expanded 4-bit number buffer)
wire [7:0] alu_out; // ALU (core0) output
wire [7:0] alu_flags; // ALU (core0) output flags
wire [4:0] pll; // Phase-locked-loop connections (+ source clock)
wire vga_clk; // VGA data clock
// ---- WIRE ASSIGNS ---- //
assign pll[4] = clk;
// ---- INITIAL VALUES ---- //
initial select_out = 4'b1111;
initial seg_buf_numbers[0] = 4'b0000;
initial seg_buf_numbers[1] = 4'b0000;
initial seg_buf_numbers[2] = 4'b0000;
initial seg_buf_numbers[3] = 4'b0000;
initial debounce = 0;
initial db_trap = 1;
initial debounce = 1'b0;
initial db_trap = 1'b1;
// Hex encoders for each 4-bit input set. Generates an 8-bit hex output
SegmentHexEncoder enc0(.number (seg_buf_numbers[0]), .encoded (seg_buf[0]));
@ -66,37 +68,41 @@ SegmentManager seg_display(
.segments (seg_write)
);
VGA screen(clk, gfx_rgb, vga_clk, VGA_rgb, VGA_hsync, VGA_vsync);
ALU core0(.a(alu_a), .b(alu_b), .op(alu_op), .z(alu_out), .o_flags(alu_flags));
altpll0 pll_gen(clk, pll[0], pll[1], pll[2], pll[3]);
always @(posedge vga_clk)
gfx_rgb <= alu_a[2:0];
always @(posedge pll[PLL_SELECT]) begin
if(!latch && write && next) begin
debounce <= 1;
debounce <= 1'b1;
end
if(write && next && db_trap) begin
debounce <= 0;
db_trap <= 0;
debounce <= 1'b0;
db_trap <= 1'b0;
end
if(!write && debounce && !db_trap) begin
db_trap <= 1;
if(stage == 0) begin
alu_a <= alu_a + 1;
end else if(stage == 1) begin
alu_b <= alu_b + 1;
end else if(stage == 2) begin
alu_op <= alu_op + 1;
db_trap <= 1'b1;
if(stage == 2'b0) begin
alu_a <= alu_a + 8'b1;
end else if(stage == 2'b1) begin
alu_b <= alu_b + 8'b1;
end else if(stage == 2'b10) begin
alu_op <= alu_op + 8'b1;
end
end else if (!next && debounce && !db_trap) begin
db_trap <= 1;
stage <= stage + 1;
db_trap <= 1'b1;
stage <= stage + 2'b1;
if(stage == 2'b01) begin
seg_buf_numbers[0] <= 0;
seg_buf_numbers[1] <= 0;
seg_buf_numbers[2] <= 0;
seg_buf_numbers[3] <= 0;
seg_buf_numbers[0] <= 4'b0;
seg_buf_numbers[1] <= 4'b0;
seg_buf_numbers[2] <= 4'b0;
seg_buf_numbers[3] <= 4'b0;
end
else if(stage == 2'b10) begin
seg_buf_numbers[0] <= alu_out[7:4];
@ -105,13 +111,13 @@ always @(posedge pll[PLL_SELECT]) begin
seg_buf_numbers[3] <= alu_flags[3:0];
end
else if(stage == 2'b11) begin
seg_buf_numbers[0] <= alu_a[7:4];
seg_buf_numbers[1] <= alu_a[3:0];
seg_buf_numbers[2] <= alu_b[7:4];
seg_buf_numbers[3] <= alu_b[3:0];
alu_a <= 0;
alu_b <= 0;
alu_op <= 0;
seg_buf_numbers[0] <= 4'b0;
seg_buf_numbers[1] <= 4'b0;
seg_buf_numbers[2] <= 4'b0;
seg_buf_numbers[3] <= 4'b0;
alu_a <= 8'b0;
alu_b <= 8'b0;
alu_op <= 8'b0;
end
end
@ -123,8 +129,8 @@ always @(posedge pll[PLL_SELECT]) begin
end else if(stage == 2'b10) begin
seg_buf_numbers[0] <= alu_op[7:4];
seg_buf_numbers[1] <= alu_op[3:0];
seg_buf_numbers[2] <= 0;
seg_buf_numbers[3] <= 0;
seg_buf_numbers[2] <= 4'b0;
seg_buf_numbers[3] <= 4'b0;
end else if(stage == 2'b11) begin
seg_buf_numbers[0] <= alu_out[7:4];
seg_buf_numbers[1] <= alu_out[3:0];

41
VGA.v Normal file
View File

@ -0,0 +1,41 @@
module VGA(
input wire clk,
input wire [2:0] rgb_data,
output reg graphics_clk,
output wire [2:0] VGA_rgb,
output wire VGA_hsync,
output wire VGA_vsync
);
parameter
hsync_end = 10'd95,
hdat_begin = 10'd143,
hdat_end = 10'd783,
hpixel_end = 10'd799,
vsync_end = 10'd1,
vdat_begin = 10'd34,
vdat_end = 10'd514,
vline_end = 10'd524;
reg [9:0] hcount;
reg [9:0] vcount;
wire hcount_ov = (hcount == hpixel_end);
wire vcount_ov = (vcount == vline_end);
wire dat_act = ((hcount >= hdat_begin) && (hcount < hdat_end)) && ((vcount >= vdat_begin) && (vcount < vdat_end));
assign VGA_hsync = (hcount > hsync_end);
assign VGA_vsync = (vcount > vsync_end);
assign VGA_rgb = (dat_act) ? rgb_data : 3'b0;
// Clock divider
always @(posedge clk) graphics_clk = ~graphics_clk;
// Graphics boundary calculation
always @(posedge graphics_clk) begin
if(hcount_ov && vcount_ov) vcount <= 10'b0;
else if(hcount_ov) vcount <= vcount + 10'b1;
if (hcount_ov) hcount <= 10'd0;
else hcount <= hcount + 10'd1;
end
endmodule