2018-08-13 18:17:36 -04:00
|
|
|
|
2017-12-04 16:40:10 -05:00
|
|
|
`include "hvsync_generator.v"
|
2018-02-06 18:07:40 -06:00
|
|
|
`include "sprite_rotation.v"
|
2017-12-04 16:40:10 -05:00
|
|
|
|
2018-10-01 12:30:47 -04:00
|
|
|
/*
|
|
|
|
Tank game.
|
|
|
|
|
|
|
|
minefield - Displays the minefield.
|
|
|
|
playfield - Displays the playfield maze.
|
|
|
|
tank_game_top - Runs the tank game, using two tank_controller
|
|
|
|
modules.
|
|
|
|
*/
|
|
|
|
|
2017-12-04 16:40:10 -05:00
|
|
|
module minefield(hpos, vpos, mine_gfx);
|
|
|
|
|
|
|
|
input [8:0] hpos;
|
|
|
|
input [8:0] vpos;
|
|
|
|
output mine_gfx;
|
|
|
|
|
|
|
|
// mine X coordinates in ROM
|
|
|
|
reg [3:0] mine_xpos [0:15];
|
|
|
|
|
|
|
|
// 1 bit for each mine
|
|
|
|
reg [15:0] mine_exploded;
|
|
|
|
|
|
|
|
// screen-wide pattern for mines
|
|
|
|
wire mine_pattern = ~(hpos[0] ^ hpos[1]) ^ (vpos[0] ^ vpos[1])
|
|
|
|
&& hpos[2] && vpos[2];
|
|
|
|
|
|
|
|
// limit mine pattern to a rectangular window
|
|
|
|
wire mine_field = (hpos >= 64 && hpos < 160)
|
|
|
|
&& (vpos >= 48 && vpos < 176)
|
|
|
|
&& mine_pattern;
|
|
|
|
|
|
|
|
// index for each of the 16 mines in window
|
|
|
|
wire [3:0] mine_vindex = vpos[6:3]^8;
|
|
|
|
|
|
|
|
// select only 1 mine per horizontal slice
|
|
|
|
wire mine_all = (hpos[6:3]==(mine_xpos[mine_vindex]^8))
|
|
|
|
&& mine_field;
|
|
|
|
|
|
|
|
// only show mines that haven't exploded
|
|
|
|
wire mine_gfx = mine_all && !mine_exploded[mine_vindex];
|
|
|
|
|
|
|
|
initial begin
|
|
|
|
mine_exploded = 0;
|
|
|
|
mine_xpos[0] = 2;
|
|
|
|
mine_xpos[1] = 10;
|
|
|
|
mine_xpos[2] = 6;
|
|
|
|
mine_xpos[3] = 0;
|
|
|
|
mine_xpos[4] = 9;
|
|
|
|
mine_xpos[5] = 3;
|
|
|
|
mine_xpos[6] = 7;
|
|
|
|
mine_xpos[7] = 11;
|
|
|
|
mine_xpos[8] = 4;
|
|
|
|
mine_xpos[9] = 1;
|
|
|
|
mine_xpos[10] = 10;
|
|
|
|
mine_xpos[11] = 5;
|
|
|
|
mine_xpos[12] = 11;
|
|
|
|
mine_xpos[13] = 3;
|
|
|
|
mine_xpos[14] = 8;
|
|
|
|
mine_xpos[15] = 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
endmodule
|
|
|
|
|
|
|
|
module playfield(hpos, vpos, playfield_gfx);
|
|
|
|
|
|
|
|
input [8:0] hpos;
|
|
|
|
input [8:0] vpos;
|
|
|
|
output playfield_gfx;
|
|
|
|
|
|
|
|
reg [31:0] maze [0:27];
|
|
|
|
|
|
|
|
wire [4:0] x = hpos[7:3];
|
|
|
|
wire [4:0] y = vpos[7:3] - 2;
|
|
|
|
|
|
|
|
assign playfield_gfx = maze[y][x];
|
|
|
|
|
|
|
|
initial begin/*{w:32,h:28,bpw:32}*/
|
|
|
|
maze[0] = 32'b11111111111111111111111111111111;
|
|
|
|
maze[1] = 32'b10000000000100000000001000000001;
|
|
|
|
maze[2] = 32'b10000000000100000000001000000001;
|
|
|
|
maze[3] = 32'b10000000000100000000000000000001;
|
|
|
|
maze[4] = 32'b10011110000000000000000000000001;
|
2018-02-06 18:07:40 -06:00
|
|
|
maze[5] = 32'b10000000000000000000000000000001;
|
|
|
|
maze[6] = 32'b10000000001000000000000011110001;
|
2017-12-04 16:40:10 -05:00
|
|
|
maze[7] = 32'b11100010000000000000000000100001;
|
2018-02-06 18:07:40 -06:00
|
|
|
maze[8] = 32'b10000010000000000000000000100001;
|
2017-12-04 16:40:10 -05:00
|
|
|
maze[9] = 32'b10000011100000000000000000000001;
|
|
|
|
maze[10] = 32'b10000000000000000000000000000001;
|
|
|
|
maze[11] = 32'b10000000000000000000000000000001;
|
|
|
|
maze[12] = 32'b11111000001000000000000000000001;
|
|
|
|
maze[13] = 32'b10001000001000000000000111100001;
|
2018-02-06 18:07:40 -06:00
|
|
|
maze[14] = 32'b10001000001000000000000000000001;
|
2017-12-04 16:40:10 -05:00
|
|
|
maze[15] = 32'b10000000001000000000000000000001;
|
|
|
|
maze[16] = 32'b10000000001000000000000000000001;
|
2018-02-06 18:07:40 -06:00
|
|
|
maze[17] = 32'b10000000000000000000000000000001;
|
|
|
|
maze[18] = 32'b10000010000000000000000100011001;
|
|
|
|
maze[19] = 32'b10001110000000000000000100010001;
|
|
|
|
maze[20] = 32'b10000000001000000000000100010001;
|
|
|
|
maze[21] = 32'b10000000001110000000000100000001;
|
|
|
|
maze[22] = 32'b10000000000000000010001100000001;
|
|
|
|
maze[23] = 32'b10000000000000000000000000000001;
|
|
|
|
maze[24] = 32'b10000010000111100000000000010001;
|
|
|
|
maze[25] = 32'b10000010000000100000000000010001;
|
|
|
|
maze[26] = 32'b10000010000000000010000000010001;
|
2017-12-04 16:40:10 -05:00
|
|
|
maze[27] = 32'b11111111111111111111111111111111;
|
|
|
|
end
|
|
|
|
|
|
|
|
endmodule
|
|
|
|
|
2018-02-21 13:32:11 -06:00
|
|
|
module tank_game_top(clk, reset, hsync, vsync, rgb, switches_p1, switches_p2);
|
2017-12-04 16:40:10 -05:00
|
|
|
|
2018-02-06 18:07:40 -06:00
|
|
|
input clk, reset;
|
|
|
|
input [7:0] switches_p1;
|
|
|
|
input [7:0] switches_p2;
|
2017-12-04 16:40:10 -05:00
|
|
|
output hsync, vsync;
|
|
|
|
output [2:0] rgb;
|
2018-02-06 18:07:40 -06:00
|
|
|
|
2017-12-04 16:40:10 -05:00
|
|
|
wire display_on;
|
|
|
|
wire [8:0] hpos;
|
|
|
|
wire [8:0] vpos;
|
2018-02-21 13:32:11 -06:00
|
|
|
|
2018-10-03 18:49:14 -04:00
|
|
|
wire mine_gfx; // minefield video
|
|
|
|
wire playfield_gfx; // playfield video
|
|
|
|
wire tank1_gfx; // player 1 tank video
|
|
|
|
wire tank2_gfx; // player 2 tank video
|
|
|
|
|
|
|
|
// video sync generator
|
2017-12-04 16:40:10 -05:00
|
|
|
hvsync_generator hvsync_gen(
|
|
|
|
.clk(clk),
|
|
|
|
.reset(0),
|
|
|
|
.hsync(hsync),
|
|
|
|
.vsync(vsync),
|
|
|
|
.display_on(display_on),
|
|
|
|
.hpos(hpos),
|
|
|
|
.vpos(vpos)
|
|
|
|
);
|
|
|
|
|
2018-10-03 18:49:14 -04:00
|
|
|
// minefield (video output -> mine_gfx)
|
2017-12-04 16:40:10 -05:00
|
|
|
minefield mine_gen(
|
|
|
|
.hpos(hpos),
|
|
|
|
.vpos(vpos),
|
|
|
|
.mine_gfx(mine_gfx)
|
|
|
|
);
|
|
|
|
|
2018-10-03 18:49:14 -04:00
|
|
|
// playfield (video output -> playfield_gfx)
|
2017-12-04 16:40:10 -05:00
|
|
|
playfield playfield_gen(
|
|
|
|
.hpos(hpos),
|
|
|
|
.vpos(vpos),
|
|
|
|
.playfield_gfx(playfield_gfx)
|
|
|
|
);
|
|
|
|
|
2018-02-06 18:07:40 -06:00
|
|
|
// multiplex player 1 and 2 load times during hsync
|
2018-09-08 09:59:21 -04:00
|
|
|
wire p2sel = hpos > 280;
|
2018-10-03 18:49:14 -04:00
|
|
|
|
2018-02-06 18:07:40 -06:00
|
|
|
// sprite ROM inputs for each player
|
|
|
|
wire [7:0] tank1_sprite_addr;
|
|
|
|
wire [7:0] tank2_sprite_addr;
|
2018-10-03 18:49:14 -04:00
|
|
|
|
2018-02-06 18:07:40 -06:00
|
|
|
// multiplex sprite ROM output
|
|
|
|
wire [7:0] tank_sprite_bits;
|
|
|
|
|
|
|
|
// bitmap ROM is shared between tank 1 and 2
|
|
|
|
tank_bitmap tank_bmp(
|
|
|
|
.addr(p2sel ? tank2_sprite_addr : tank1_sprite_addr),
|
|
|
|
.bits(tank_sprite_bits));
|
2018-10-03 18:49:14 -04:00
|
|
|
|
|
|
|
// player 1 tank controller
|
2018-02-06 18:07:40 -06:00
|
|
|
tank_controller #(16,36,4) tank1(
|
|
|
|
.clk(clk),
|
|
|
|
.reset(reset),
|
|
|
|
.hpos(hpos),
|
|
|
|
.vpos(vpos),
|
|
|
|
.hsync(hsync && !p2sel),
|
|
|
|
.vsync(vsync),
|
|
|
|
.sprite_addr(tank1_sprite_addr),
|
|
|
|
.sprite_bits(tank_sprite_bits),
|
|
|
|
.gfx(tank1_gfx),
|
|
|
|
.playfield(playfield_gfx),
|
|
|
|
.switch_left(switches_p1[0]),
|
|
|
|
.switch_right(switches_p1[1]),
|
|
|
|
.switch_up(switches_p1[2])
|
|
|
|
);
|
|
|
|
|
2018-10-03 18:49:14 -04:00
|
|
|
// player 2 tank controller
|
2018-02-06 18:07:40 -06:00
|
|
|
tank_controller #(220,190,12) tank2(
|
|
|
|
.clk(clk),
|
|
|
|
.reset(reset),
|
|
|
|
.hpos(hpos),
|
|
|
|
.vpos(vpos),
|
|
|
|
.hsync(hsync && p2sel),
|
|
|
|
.vsync(vsync),
|
|
|
|
.sprite_addr(tank2_sprite_addr),
|
|
|
|
.sprite_bits(tank_sprite_bits),
|
|
|
|
.gfx(tank2_gfx),
|
|
|
|
.playfield(playfield_gfx),
|
|
|
|
.switch_left(switches_p2[0]),
|
|
|
|
.switch_right(switches_p2[1]),
|
|
|
|
.switch_up(switches_p2[2])
|
|
|
|
);
|
|
|
|
|
2018-10-03 18:49:14 -04:00
|
|
|
// video signal mixer
|
2018-02-06 18:07:40 -06:00
|
|
|
wire r = display_on && (mine_gfx || tank2_gfx);
|
|
|
|
wire g = display_on && tank1_gfx;
|
|
|
|
wire b = display_on && (playfield_gfx || tank2_gfx);
|
2017-12-04 16:40:10 -05:00
|
|
|
assign rgb = {b,g,r};
|
|
|
|
|
|
|
|
endmodule
|