aboutsummaryrefslogtreecommitdiffstats
path: root/fpga/usrp3/top/x300/dbuf_bootram.v
diff options
context:
space:
mode:
authorBen Hilburn <ben.hilburn@ettus.com>2014-02-14 12:05:07 -0800
committerBen Hilburn <ben.hilburn@ettus.com>2014-02-14 12:05:07 -0800
commitff1546f8137f7f92bb250f685561b0c34cc0e053 (patch)
tree7fa6fd05c8828df256a1b20e2935bd3ba9899e2c /fpga/usrp3/top/x300/dbuf_bootram.v
parent4f691d88123784c2b405816925f1a1aef69d18c1 (diff)
downloaduhd-ff1546f8137f7f92bb250f685561b0c34cc0e053.tar.gz
uhd-ff1546f8137f7f92bb250f685561b0c34cc0e053.tar.bz2
uhd-ff1546f8137f7f92bb250f685561b0c34cc0e053.zip
Pushing the bulk of UHD-3.7.0 code.
Diffstat (limited to 'fpga/usrp3/top/x300/dbuf_bootram.v')
-rw-r--r--fpga/usrp3/top/x300/dbuf_bootram.v122
1 files changed, 122 insertions, 0 deletions
diff --git a/fpga/usrp3/top/x300/dbuf_bootram.v b/fpga/usrp3/top/x300/dbuf_bootram.v
new file mode 100644
index 000000000..69fe81dca
--- /dev/null
+++ b/fpga/usrp3/top/x300/dbuf_bootram.v
@@ -0,0 +1,122 @@
+//
+// Copyright 2013 Ettus Research LLC
+//
+
+
+module dbuf_bootram #(
+ parameter ADDR_WIDTH = 16,
+ parameter DATA_WIDTH = 32,
+ parameter MAX_ADDR = 16'h7FFC
+) (
+ input clk,
+ input rst,
+
+ input mem_stb,
+ input mem_wea,
+ input [ADDR_WIDTH-1:0] mem_addra,
+ input [DATA_WIDTH-1:0] mem_dina,
+ output [DATA_WIDTH-1:0] mem_douta,
+ output reg mem_acka,
+
+ input ldr_stb,
+ input ldr_wea,
+ input [ADDR_WIDTH-1:0] ldr_addra,
+ input [DATA_WIDTH-1:0] ldr_dina,
+ output ldr_acka,
+
+ output reg zpu_rst
+);
+ localparam SR_LDR_ADDR_REG = 4'h0;
+ localparam SR_LDR_DATA_REG = 4'h1;
+
+ //---------------------------------------------------------
+ // Mem ack logic
+ always @(posedge clk) begin
+ if (rst) mem_acka <= 0;
+ else mem_acka <= mem_stb & ~mem_acka;
+ end
+
+ //---------------------------------------------------------
+ // Instantiate 2 bootram modules
+ // They will both initialize with the pre-built ZPU firmware
+ //
+ wire ram0_ena, ram1_ena;
+ wire [3:0] ram0_wea, ram1_wea;
+ wire [12:0] ram0_addra, ram1_addra;
+ wire [31:0] ram0_dina, ram0_douta, ram1_dina, ram1_douta;
+
+ bootram sys_ram0(
+ .clka(clk),
+ .ena(ram0_ena),
+ .wea(ram0_wea),
+ .addra(ram0_addra),
+ .dina(ram0_dina),
+ .douta(ram0_douta));
+
+ bootram sys_ram1(
+ .clka(clk),
+ .ena(ram1_ena),
+ .wea(ram1_wea),
+ .addra(ram1_addra),
+ .dina(ram1_dina),
+ .douta(ram1_douta));
+
+ //---------------------------------------------------------
+ // Settings bus interface for bootloader
+ wire ldr_set_stb;
+ wire [3:0] ldr_set_addr;
+ wire [DATA_WIDTH-1:0] ldr_set_data;
+
+ //@TODO: This address truncation seems unclean. Maybe settings_bus can take a addr_width as a param.
+ wire [7:0] ldr_set_addr_w;
+ settings_bus #(.AWIDTH(ADDR_WIDTH), .DWIDTH(DATA_WIDTH)) ldr_settings_bus
+ (
+ .wb_clk(clk), .wb_rst(rst),
+ .wb_adr_i(ldr_addra), .wb_dat_i(ldr_dina),
+ .wb_stb_i(ldr_stb), .wb_we_i(ldr_wea), .wb_ack_o(ldr_acka),
+ .strobe(ldr_set_stb), .addr(ldr_set_addr_w), .data(ldr_set_data)
+ );
+ assign ldr_set_addr = ldr_set_addr_w[3:0];
+
+ //---------------------------------------------------------
+ // Selection logic
+ //
+ reg bootram_ptr;
+ reg [12:0] ldr_curr_wr_addr;
+ wire ldr_we_stb;
+
+ assign {ram0_ena, ram0_wea, ram0_addra, ram0_dina} = bootram_ptr ? {mem_stb, {4{(mem_wea & ~zpu_rst)}}, mem_addra[14:2], mem_dina} :
+ {ldr_we_stb, {4{ldr_we_stb}}, ldr_curr_wr_addr, ldr_set_data};
+ assign {ram1_ena, ram1_wea, ram1_addra, ram1_dina} = bootram_ptr ? {ldr_we_stb, {4{ldr_we_stb}}, ldr_curr_wr_addr, ldr_set_data} :
+ {mem_stb, {4{(mem_wea & ~zpu_rst)}}, mem_addra[14:2], mem_dina};
+ assign mem_douta = bootram_ptr ? ram0_douta : ram1_douta;
+
+ //---------------------------------------------------------
+ // Boot loader
+ //
+ assign ldr_we_stb = ~zpu_rst && ldr_set_stb && (ldr_set_addr == SR_LDR_DATA_REG);
+
+ always @(posedge clk) begin
+ if (rst) begin
+ zpu_rst <= 1;
+ bootram_ptr <= 1;
+ ldr_curr_wr_addr <= 13'h0;
+ end else if (ldr_set_stb & ~zpu_rst) begin
+ case (ldr_set_addr)
+ SR_LDR_ADDR_REG: begin
+ ldr_curr_wr_addr <= ldr_set_data[14:2];
+ end
+ SR_LDR_DATA_REG: begin
+ ldr_curr_wr_addr <= ldr_curr_wr_addr + 1;
+ if ({1'b0, ldr_curr_wr_addr, 2'b00} == MAX_ADDR) begin
+ zpu_rst <= 1;
+ bootram_ptr <= ~bootram_ptr;
+ end
+ end
+ endcase
+ end else begin
+ zpu_rst <= 0;
+ end
+ end
+
+endmodule