module n64_cfg ( input clk, input reset, n64_reg_bus.cfg reg_bus, n64_scb.cfg n64_scb, output logic irq ); typedef enum bit [3:0] { REG_STATUS, REG_COMMAND, REG_DATA_0_H, REG_DATA_0_L, REG_DATA_1_H, REG_DATA_1_L, REG_IDENTIFIER_H, REG_IDENTIFIER_L, REG_KEY_H, REG_KEY_L } e_reg; logic cfg_error; always_comb begin reg_bus.rdata = 16'd0; if (reg_bus.address[16] && (reg_bus.address[15:5] == 11'd0)) begin case (reg_bus.address[4:1]) REG_STATUS: reg_bus.rdata = { n64_scb.cfg_pending, cfg_error, irq, 13'd0 }; REG_COMMAND: reg_bus.rdata = {8'd0, n64_scb.cfg_cmd}; REG_DATA_0_H: reg_bus.rdata = n64_scb.cfg_wdata[0][31:16]; REG_DATA_0_L: reg_bus.rdata = n64_scb.cfg_wdata[0][15:0]; REG_DATA_1_H: reg_bus.rdata = n64_scb.cfg_wdata[1][31:16]; REG_DATA_1_L: reg_bus.rdata = n64_scb.cfg_wdata[1][15:0]; REG_IDENTIFIER_H: reg_bus.rdata = n64_scb.cfg_identifier[31:16]; REG_IDENTIFIER_L: reg_bus.rdata = n64_scb.cfg_identifier[15:0]; REG_KEY_H: reg_bus.rdata = 16'd0; REG_KEY_L: reg_bus.rdata = 16'd0; endcase end end logic unlock_flag; logic lock_sequence_counter; always_ff @(posedge clk) begin if (n64_scb.cfg_done) begin n64_scb.cfg_pending <= 1'b0; cfg_error <= n64_scb.cfg_error; end if (n64_scb.cfg_irq) begin irq <= 1'b1; end if (unlock_flag) begin n64_scb.cfg_unlock <= 1'b1; end if (reset || n64_scb.n64_reset || n64_scb.n64_nmi) begin n64_scb.cfg_unlock <= 1'b0; n64_scb.cfg_pending <= 1'b0; n64_scb.cfg_cmd <= 8'h00; irq <= 1'b0; cfg_error <= 1'b0; lock_sequence_counter <= 1'd0; end else if (n64_scb.cfg_unlock) begin if (reg_bus.write && reg_bus.address[16] && (reg_bus.address[15:5] == 11'd0)) begin case (reg_bus.address[4:1]) REG_COMMAND: begin n64_scb.cfg_pending <= 1'b1; n64_scb.cfg_cmd <= reg_bus.wdata[7:0]; cfg_error <= 1'b0; end REG_DATA_0_H: n64_scb.cfg_rdata[0][31:16] <= reg_bus.wdata; REG_DATA_0_L: n64_scb.cfg_rdata[0][15:0] <= reg_bus.wdata; REG_DATA_1_H: n64_scb.cfg_rdata[1][31:16] <= reg_bus.wdata; REG_DATA_1_L: n64_scb.cfg_rdata[1][15:0] <= reg_bus.wdata; REG_IDENTIFIER_H: irq <= 1'b0; REG_KEY_H, REG_KEY_L: begin lock_sequence_counter <= lock_sequence_counter + 1'd1; if (reg_bus.wdata != 16'hFFFF) begin lock_sequence_counter <= 1'd0; end if (lock_sequence_counter == 1'd1) begin n64_scb.cfg_unlock <= (reg_bus.wdata != 16'hFFFF); end end endcase end end end const bit [15:0] UNLOCK_SEQUENCE [4] = { 16'h5F55, 16'h4E4C, 16'h4F43, 16'h4B5F }; logic [1:0] unlock_sequence_counter; always_ff @(posedge clk) begin unlock_flag <= 1'b0; if (reset || n64_scb.n64_reset || n64_scb.n64_nmi) begin unlock_sequence_counter <= 2'd0; end else if (!n64_scb.cfg_unlock) begin if (reg_bus.write && reg_bus.address[16] && (reg_bus.address[15:5] == 11'd0)) begin case (reg_bus.address[4:1]) REG_KEY_H, REG_KEY_L: begin for (int index = 0; index < $size(UNLOCK_SEQUENCE); index++) begin if (index == unlock_sequence_counter) begin if (reg_bus.wdata == UNLOCK_SEQUENCE[index]) begin unlock_sequence_counter <= unlock_sequence_counter + 1'd1; if (index == ($size(UNLOCK_SEQUENCE) - 1'd1)) begin unlock_flag <= 1'b1; unlock_sequence_counter <= 2'd0; end end else begin unlock_sequence_counter <= 2'd0; end end end end endcase end end end endmodule