1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2025-01-11 23:30:04 +00:00
8bitworkshop/presets/verilog/cpu_platform.v

223 lines
4.7 KiB
Coq
Raw Normal View History

`include "hvsync_generator.v"
`include "font_cp437_8x8.v"
`include "ram.v"
`include "tile_renderer.v"
`include "sprite_scanline_renderer.v"
`include "lfsr.v"
`include "sound_generator.v"
`include "cpu16.v"
2018-10-01 12:30:47 -04:00
/*
A full video game console, with the following components:
64 kilobytes (32,678 16-bit words) of RAM
16-bit CPU running at 4.857 MHz
32x30 tile graphics with 256 x 8 tile ROM
32 16x16 sprites per frame with sprite ROM
16 colors (two per tile, one per sprite)
Two game controllers (four direction switches, two buttons)
One paddle/analog stick controller
*/
2018-07-11 19:53:05 -07:00
module cpu_platform(clk, reset, hsync, vsync,
hpaddle, vpaddle,
switches_p1, switches_p2,
rgb);
input clk, reset;
2018-05-27 11:13:06 -07:00
input hpaddle, vpaddle;
2018-07-11 19:53:05 -07:00
input [7:0] switches_p1;
input [7:0] switches_p2;
output hsync, vsync;
output [3:0] rgb;
wire display_on;
wire [8:0] hpos;
wire [8:0] vpos;
// video RAM bus
2018-02-27 21:35:42 -06:00
wire [15:0] ram_read;
reg [15:0] ram_write;
reg ram_writeenable;
// multiplex sprite and tile RAM
reg [15:0] tile_ram_addr;
2018-02-27 21:35:42 -06:00
reg [5:0] sprite_ram_addr;
wire tile_reading;
wire sprite_reading;
wire [14:0] mux_ram_addr; // 15-bit RAM access
2018-07-11 19:53:05 -07:00
// multiplexor for sprite/tile/CPU RAM
2018-02-27 21:35:42 -06:00
always @(*)
if (cpu_busy) begin
if (sprite_reading)
mux_ram_addr = {9'b111111100, sprite_ram_addr};
2018-02-27 21:35:42 -06:00
else
mux_ram_addr = tile_ram_addr[14:0];
end else
mux_ram_addr = cpu_ram_addr[14:0];
// tile and sprite ROM
wire [10:0] tile_rom_addr;
wire [7:0] tile_rom_data;
wire [15:0] sprite_rom_addr;
wire [15:0] sprite_rom_data;
// gfx outputs
wire [3:0] tile_rgb;
wire [3:0] sprite_rgb;
2018-07-11 19:53:05 -07:00
// video sync generator
hvsync_generator hvsync_gen(
.clk(clk),
.reset(reset),
.hsync(hsync),
.vsync(vsync),
.display_on(display_on),
.hpos(hpos),
.vpos(vpos)
);
2018-02-27 21:35:42 -06:00
// RAM (32k x 16 bits)
RAM_sync #(15,16) ram(
.clk(clk),
2018-02-27 21:35:42 -06:00
.dout(ram_read),
.din(ram_write),
.addr(mux_ram_addr),
.we(ram_writeenable)
);
2018-07-11 19:53:05 -07:00
// tile graphics
tile_renderer tile_gen(
.clk(clk),
.reset(reset),
.hpos(hpos),
.vpos(vpos),
.ram_addr(tile_ram_addr),
2018-02-27 21:35:42 -06:00
.ram_read(ram_read),
.ram_busy(tile_reading),
.rom_addr(tile_rom_addr),
.rom_data(tile_rom_data),
.rgb(tile_rgb)
);
2018-07-11 19:53:05 -07:00
// sprite scanline renderer
sprite_scanline_renderer ssr(
.clk(clk),
.reset(reset),
.hpos(hpos),
.vpos(vpos),
.ram_addr(sprite_ram_addr),
2018-02-27 21:35:42 -06:00
.ram_data(ram_read),
.ram_busy(sprite_reading),
.rom_addr(sprite_rom_addr),
.rom_data(sprite_rom_data),
.rgb(sprite_rgb)
);
2018-07-11 19:53:05 -07:00
// tile ROM
font_cp437_8x8 tile_rom(
.addr(tile_rom_addr),
.data(tile_rom_data)
);
2018-07-11 19:53:05 -07:00
// sprite ROM
example_bitmap_rom bitmap_rom(
.addr(sprite_rom_addr),
.data(sprite_rom_data)
);
2018-02-27 21:35:42 -06:00
// sprites overlay tiles
assign rgb = display_on
? (sprite_rgb>0 ? sprite_rgb : tile_rgb)
: 0;
2018-02-27 21:35:42 -06:00
// CPU
reg cpu_hold = 0;
wire cpu_busy;
wire [15:0] cpu_ram_addr;
wire busy;
2018-02-27 21:35:42 -06:00
wire [15:0] cpu_bus;
2018-05-27 11:13:06 -07:00
wire [15:0] flags = {11'b0, vsync, hsync, vpaddle, hpaddle, display_on};
2018-07-11 19:53:05 -07:00
wire [15:0] switches = {switches_p2, switches_p1};
2018-02-27 21:35:42 -06:00
2018-07-11 19:53:05 -07:00
// select ROM, RAM, switches ($FFFE) or flags ($FFFF)
always @(*)
casez (cpu_ram_addr)
16'hfffe: cpu_bus = switches;
16'hffff: cpu_bus = flags;
16'b0???????????????: cpu_bus = ram_read;
16'b1???????????????: cpu_bus = program_rom[cpu_ram_addr[14:0]];
endcase
// 16-bit CPU
CPU16 cpu(
.clk(clk),
.reset(reset),
2018-07-11 19:53:05 -07:00
.hold(tile_reading | sprite_reading), // hold input
.busy(cpu_busy), // busy output
2018-02-27 21:35:42 -06:00
.address(cpu_ram_addr),
.data_in(cpu_bus),
.data_out(ram_write),
.write(ram_writeenable));
2018-07-11 19:53:05 -07:00
// program ROM ($8000-$FFFE)
reg [15:0] program_rom[0:32767];
2018-07-11 19:53:05 -07:00
// example ROM program code
2018-02-27 21:35:42 -06:00
`ifdef EXT_INLINE_ASM
initial begin
2018-03-01 23:15:33 -06:00
program_rom = '{
2018-02-27 21:35:42 -06:00
__asm
.arch femto16
.org 0x8000
2018-07-11 19:53:05 -07:00
.len 32768
2018-02-27 21:35:42 -06:00
mov sp,@$6fff
mov dx,@InitPageTable
2018-02-27 21:35:42 -06:00
jsr dx
mov ax,@$4ffe
mov dx,@ClearTiles
jsr dx
mov dx,@ClearSprites
2018-02-27 21:35:42 -06:00
jsr dx
reset
InitPageTable:
2018-02-27 21:35:42 -06:00
mov ax,@$6000 ; screen buffer
mov bx,@$7e00 ; page table start
mov cx,#32 ; 32 rows
InitPTLoop:
2018-02-27 21:35:42 -06:00
mov [bx],ax
add ax,#32
inc bx
dec cx
bnz InitPTLoop
2018-02-27 21:35:42 -06:00
rts
ClearTiles:
mov bx,@$6000
2018-03-01 23:15:33 -06:00
mov cx,@$3c0
2018-02-27 21:35:42 -06:00
ClearLoop:
mov [bx],ax
inc bx
dec cx
bnz ClearLoop
rts
ClearSprites:
mov bx,@$7f00
mov ax,#0
2018-03-01 20:17:37 -06:00
mov cx,#$40
ClearSLoop:
mov ax,[bx]
add ax,@$101
mov [bx],ax
inc bx
dec cx
bnz ClearSLoop
2018-07-11 19:53:05 -07:00
rts
2018-02-27 21:35:42 -06:00
__endasm
};
end
`endif
endmodule