| // Copyright lowRISC contributors. |
| // Licensed under the Apache License, Version 2.0, see LICENSE for details. |
| // SPDX-License-Identifier: Apache-2.0 |
| // |
| // Description: sysrst_ctrl combo Module |
| // |
| module sysrst_ctrl_combo |
| import sysrst_ctrl_pkg::*; |
| import sysrst_ctrl_reg_pkg::*; |
| ( |
| input clk_i, |
| input rst_ni, |
| // (Optionally) inverted input signals on AON clock |
| input pwrb_int_i, |
| input key0_int_i, |
| input key1_int_i, |
| input key2_int_i, |
| input ac_present_int_i, |
| input ec_rst_l_int_i, |
| // CSRs synced to AON clock |
| input sysrst_ctrl_reg2hw_ec_rst_ctl_reg_t ec_rst_ctl_i, |
| input sysrst_ctrl_reg2hw_key_intr_debounce_ctl_reg_t key_intr_debounce_ctl_i, |
| input sysrst_ctrl_reg2hw_com_pre_sel_ctl_mreg_t [NumCombo-1:0] com_pre_sel_ctl_i, |
| input sysrst_ctrl_reg2hw_com_pre_det_ctl_mreg_t [NumCombo-1:0] com_pre_det_ctl_i, |
| input sysrst_ctrl_reg2hw_com_sel_ctl_mreg_t [NumCombo-1:0] com_sel_ctl_i, |
| input sysrst_ctrl_reg2hw_com_det_ctl_mreg_t [NumCombo-1:0] com_det_ctl_i, |
| input sysrst_ctrl_reg2hw_com_out_ctl_mreg_t [NumCombo-1:0] com_out_ctl_i, |
| output sysrst_ctrl_hw2reg_combo_intr_status_reg_t combo_intr_status_o, |
| // Output signals on AON clock |
| output sysrst_ctrl_combo_intr_o, |
| output bat_disable_hw_o, |
| output rst_req_o, |
| output ec_rst_l_hw_o |
| ); |
| |
| // There are four possible combos |
| // Each key combo can select different triggers |
| logic [NumCombo-1:0] combo_bat_disable; |
| logic [NumCombo-1:0] combo_ec_rst_l; |
| logic [NumCombo-1:0] combo_rst_req; |
| logic [NumCombo-1:0] combo_intr_pulse; |
| |
| localparam int unsigned NumInputs = 5; |
| logic [NumInputs-1:0] in; |
| assign in = { |
| pwrb_int_i, |
| key0_int_i, |
| key1_int_i, |
| key2_int_i, |
| ac_present_int_i |
| }; |
| |
| for (genvar k = 0; k < NumCombo; k++) begin : gen_combo_trigger |
| // Generate the pre-condition |
| logic [NumInputs-1:0] cfg_in_pre; |
| assign cfg_in_pre = { |
| com_pre_sel_ctl_i[k].pwrb_in_sel.q, |
| com_pre_sel_ctl_i[k].key0_in_sel.q, |
| com_pre_sel_ctl_i[k].key1_in_sel.q, |
| com_pre_sel_ctl_i[k].key2_in_sel.q, |
| com_pre_sel_ctl_i[k].ac_present_sel.q |
| }; |
| |
| // Combo pre-condition is enabled if any of the keys is selected for this |
| // pre-condition. Note that if no key is selected for this pre-condition, |
| // the precondition is automatically valid - see cfg_combo_en below. |
| logic cfg_combo_pre_en; |
| assign cfg_combo_pre_en = |cfg_in_pre; |
| |
| // Config trigger is asserted if all configured keys are pressed (== 0) |
| logic precond; |
| assign precond = (in & cfg_in_pre) == '0; |
| |
| logic precond_valid; |
| sysrst_ctrl_detect #( |
| .DebounceTimerWidth(TimerWidth), |
| .DetectTimerWidth(DetTimerWidth), |
| // This detects a high level. |
| .EventType(HighLevel), |
| .Sticky(0) |
| ) u_sysrst_ctrl_detect_pre ( |
| .clk_i, |
| .rst_ni, |
| .trigger_i (precond), |
| .cfg_debounce_timer_i (key_intr_debounce_ctl_i.q), |
| .cfg_detect_timer_i (com_pre_det_ctl_i[k].q), |
| .cfg_enable_i (cfg_combo_pre_en), |
| .event_detected_o (precond_valid), |
| .event_detected_pulse_o() |
| ); |
| |
| // Generate the trigger for each combo |
| logic [NumInputs-1:0] cfg_in_sel; |
| assign cfg_in_sel = { |
| com_sel_ctl_i[k].pwrb_in_sel.q, |
| com_sel_ctl_i[k].key0_in_sel.q, |
| com_sel_ctl_i[k].key1_in_sel.q, |
| com_sel_ctl_i[k].key2_in_sel.q, |
| com_sel_ctl_i[k].ac_present_sel.q |
| }; |
| |
| // Combo detection is enabled if |
| // - the pre-condition is valid and at least one of the keys is selected |
| // for this combo. |
| // - the pre-condition is disabled and at least one of the keys is |
| // selected for this combo. |
| logic cfg_combo_en; |
| assign cfg_combo_en = (|cfg_in_sel) && (precond_valid && cfg_combo_pre_en || |
| !cfg_combo_pre_en); |
| |
| // Config trigger is asserted if all configured keys are pressed (== 0) |
| logic trigger; |
| assign trigger = (in & cfg_in_sel) == '0; |
| |
| logic combo_det_pulse; |
| sysrst_ctrl_detect #( |
| .DebounceTimerWidth(TimerWidth), |
| .DetectTimerWidth(DetTimerWidth), |
| // This detects a positive edge |
| .EventType(EdgeToHigh), |
| .Sticky(0) |
| ) u_sysrst_ctrl_detect ( |
| .clk_i, |
| .rst_ni, |
| .trigger_i (trigger), |
| .cfg_debounce_timer_i (key_intr_debounce_ctl_i.q), |
| .cfg_detect_timer_i (com_det_ctl_i[k].q), |
| .cfg_enable_i (cfg_combo_en), |
| .event_detected_o (), |
| .event_detected_pulse_o(combo_det_pulse) |
| ); |
| |
| //Instantiate the combo action module |
| sysrst_ctrl_comboact u_combo_act ( |
| .clk_i, |
| .rst_ni, |
| .cfg_intr_en_i(com_out_ctl_i[k].interrupt.q), |
| .cfg_bat_disable_en_i(com_out_ctl_i[k].bat_disable.q), |
| .cfg_ec_rst_en_i(com_out_ctl_i[k].ec_rst.q), |
| .cfg_rst_req_en_i(com_out_ctl_i[k].rst_req.q), |
| .combo_det_pulse_i(combo_det_pulse), |
| .ec_rst_l_i(ec_rst_l_int_i), |
| .ec_rst_ctl_i(ec_rst_ctl_i), |
| .combo_intr_pulse_o(combo_intr_pulse[k]), |
| .bat_disable_o(combo_bat_disable[k]), |
| .rst_req_o(combo_rst_req[k]), |
| .ec_rst_l_o(combo_ec_rst_l[k]) |
| ); |
| end |
| |
| // bat_disable |
| // If any combo triggers bat_disable, assert the signal |
| assign bat_disable_hw_o = |(combo_bat_disable); |
| |
| // If any combo triggers OT or EC RST(active low), assert the signal |
| assign rst_req_o = |(combo_rst_req); |
| assign ec_rst_l_hw_o = &(combo_ec_rst_l); |
| |
| // Write interrupt status registers using the synced IRQ pulses. |
| assign {combo_intr_status_o.combo3_h2l.de, |
| combo_intr_status_o.combo2_h2l.de, |
| combo_intr_status_o.combo1_h2l.de, |
| combo_intr_status_o.combo0_h2l.de} = combo_intr_pulse; |
| |
| assign sysrst_ctrl_combo_intr_o = |combo_intr_pulse; |
| |
| assign combo_intr_status_o.combo0_h2l.d = 1'b1; |
| assign combo_intr_status_o.combo1_h2l.d = 1'b1; |
| assign combo_intr_status_o.combo2_h2l.d = 1'b1; |
| assign combo_intr_status_o.combo3_h2l.d = 1'b1; |
| |
| endmodule |