Skip to content

Commit

Permalink
i3c_controller: Add I2C_MOD parameter
Browse files Browse the repository at this point in the history
Signed-off-by: Jorge Marques <jorge.marques@analog.com>
  • Loading branch information
gastmaier committed Jan 18, 2025
1 parent 646f1b9 commit 95d598a
Show file tree
Hide file tree
Showing 3 changed files with 40 additions and 26 deletions.
3 changes: 2 additions & 1 deletion testbenches/ip/i3c_controller/system_bd.tcl
Original file line number Diff line number Diff line change
Expand Up @@ -53,10 +53,11 @@ source $ad_hdl_dir/library/i3c_controller/scripts/i3c_controller_bd.tcl

set async_clk 0
set clk_mod $ad_project_params(CLK_MOD)
set i2c_mod 1
set offload 1
set max_devs 16

i3c_controller_create i3c $async_clk $clk_mod $offload $max_devs
i3c_controller_create i3c $async_clk $clk_mod $i2c_mod $offload $max_devs

ad_connect i3c/m_i3c i3c
ad_connect i3c/offload_sdi offload_sdi
Expand Down
61 changes: 37 additions & 24 deletions testbenches/ip/i3c_controller/tests/test_program.sv
Original file line number Diff line number Diff line change
Expand Up @@ -125,12 +125,14 @@ localparam I3C_CMD_3 = {3'b010, 12'd5, DEVICE_DA2[6:0], 1'b1};
localparam I3C_CMD_4 = {3'b010, 12'd2, START_DA[6:0], 1'b0};
// Read 1 byte, no bcast address
localparam I3C_CMD_5 = {3'b000, 12'd1, DEVICE_DA2[6:0], 1'b1};
// Write 2 bytes
localparam I2C_CMD_1 = {3'b010, 12'd2, DEVICE_SA1[6:0], 1'b0};
// Write 1 bytes with SR
localparam I2C_CMD_1 = {3'b001, 12'd1, DEVICE_SA1[6:0], 1'b0};
// Read 5 bytes
localparam I2C_CMD_2 = {3'b010, 12'd5, DEVICE_SA2[6:0], 1'b1};
localparam I2C_CMD_2 = {3'b000, 12'd5, DEVICE_SA2[6:0], 1'b1};
// Write 6 bytes
localparam I2C_CMD_3 = {3'b010, 12'd6, DEVICE_SA1[6:0], 1'b0};
localparam I2C_CMD_3 = {3'b000, 12'd6, DEVICE_SA1[6:0], 1'b0};
// Write 1 byte
localparam I2C_CMD_4 = {3'b000, 12'd1, DEVICE_SA1[6:0], 1'b0};

program test_program (
input i3c_irq,
Expand Down Expand Up @@ -180,7 +182,7 @@ endtask
reg i3c_dev_sda = 1'bZ;
assign i3c_sda = i3c_dev_sda;
// Logic to drive the offload trigger
logic offload_trigger_l = 1'b1;
logic offload_trigger_l = 1'b0;
assign offload_trigger = offload_trigger_l;
// Always ready to receive offload data
assign offload_sdi_ready = 1'b1;
Expand Down Expand Up @@ -581,35 +583,29 @@ task priv_i2c_test();
// Unmask CMDR interrupt
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_IRQ_MASK, 32'h20);

// Test #1, controller does private write transfer that is ACK
`INFO(("PRIV I2C Test #1"), ADI_VERBOSITY_LOW);
// Test #1, controller does private write transfer that is ACK and SR
// Test #2, controller does private read transfer and ACKs all receiving
// Mocks a Write address then read transfer.
`INFO(("PRIV I2C Test #1, #2"), ADI_VERBOSITY_LOW);
auto_ack <= 1'b1;

// Write SDO payload
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_SDO_FIFO, 32'hDEAD_BEEF);
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_SDO_FIFO, 32'h0000_00EF);

// Write CMD instruction
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_CMD_FIFO, I2C_CMD_1);
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_CMD_FIFO, I2C_CMD_2);
wait (`DUT_I3C_BIT_MOD.nop == 0);
// Assert is in I²C mode
if (`DUT_I3C_BIT_MOD.i2c_mode !== 1)
`FATAL(("Not in I2C mode!"));
wait (`DUT_I3C_BIT_MOD.nop == 1);

// Test #2, controller does private read transfer and ACKs all receiving
// bytes.
`INFO(("PRIV I2C Test #2"), ADI_VERBOSITY_LOW);

// Write CMD instruction
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_CMD_FIFO, I2C_CMD_2);

`ERROR(("Not in I2C mode!"));
// Dummy LOW peripheral write + ACK continue
wait (`DUT_I3C_WORD.st == `CMDW_I2C_RX);
auto_ack <= 1'b0;
i3c_dev_sda <= 1'b0;
// Count 5 ACK-bit asserted low by the controller (sampling before
// Count n ACK-bit asserted low by the controller (sampling before
// tri-state)
repeat (5) @(negedge `DUT_I3C_BIT_MOD.sdo);
repeat (I2C_CMD_2[15:8]) @(negedge `DUT_I3C_BIT_MOD.sdo);
i3c_dev_sda <= 1'bZ;

wait (`DUT_I3C_BIT_MOD.nop == 0);
Expand All @@ -634,6 +630,18 @@ task priv_i2c_test();
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_SDO_FIFO, 32'h0000_00DE);
wait (`DUT_I3C_BIT_MOD.nop == 1);

// Test #4, controller does private write transfer that is NACK
`INFO(("PRIV I2C Test #4"), ADI_VERBOSITY_LOW);
auto_ack <= 1'b0;

// Write SDO payload
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_SDO_FIFO, 32'h0000_00EF);
// Write CMD instruction
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_CMD_FIFO, I2C_CMD_4);

wait (`DUT_I3C_BIT_MOD.nop == 0);
wait (`DUT_I3C_BIT_MOD.nop == 1);

// Read Results

if (~`DUT_I3C_REGMAP.up_irq_pending[`I3C_REGMAP_IRQ_CMDR_PENDING])
Expand All @@ -655,6 +663,13 @@ task priv_i2c_test();
if (cmdr_fifo_data[19:8] != 12'd6) // Wrote 6 bytes
`FATAL(("#3: CMD -> CMDR write length test FAILED"));

axi_read (`I3C_CONTROLLER_BA, `I3C_REGMAP_CMDR_FIFO, cmdr_fifo_data);
print_cmdr (cmdr_fifo_data);
if (cmdr_fifo_data[19:8] != 12'd0) // Wrote 0 bytes
`ERROR(("#4: CMD -> CMDR write length test FAILED"));
if (cmdr_fifo_data[23:20] != 4'd6)
`ERROR(("#4: CMD -> CMDR NACK_RESP check FAILED"));

// Mask CMDR interrupt
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_IRQ_MASK, 32'h00);
// Clear all pending IRQs
Expand Down Expand Up @@ -939,21 +954,19 @@ task offload_i3c_test();

`INFO(("Offload I3C Test Started"), ADI_VERBOSITY_LOW);

offload_trigger_l = 1'b1;
#10ns offload_trigger_l = 1'b0;
offload_trigger_l = 1'b0;

// Write SDO payload
axi_write (`I3C_CONTROLLER_BA, {`I3C_REGMAP_OFFLOAD_SDO_, 4'h0}, 32'hDEAD_BEEF);

// Write CMD instruction
axi_write (`I3C_CONTROLLER_BA, {`I3C_REGMAP_OFFLOAD_CMD_, 4'h0}, I3C_CMD_1);

// Write CMD instruction
axi_write (`I3C_CONTROLLER_BA, {`I3C_REGMAP_OFFLOAD_CMD_, 4'h1}, I3C_CMD_3);

// Set offload length and enter offload mode
axi_write (`I3C_CONTROLLER_BA, `I3C_REGMAP_OPS, {2'b11, 4'd2, 1'b1});

auto_ack <= 1'b1;
offload_trigger_l = 1'b1;
#10ns offload_trigger_l = 1'b0;

Expand Down
2 changes: 1 addition & 1 deletion testbenches/ip/i3c_controller/waves/cfg1.wcfg
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@
</wvobject>
<wvobject type="array" fp_name="/system_tb/test_harness/i3c/core/inst/i_i3c_controller_framing/sm">
<obj_property name="ElementShortName">sm[2:0]</obj_property>
<obj_property name="ObjectShortName">sm[2:0]</obj_property>
<obj_property name="ObjectShortName">framing sm[2:0]</obj_property>
<obj_property name="CustomSignalColor">#FFFF00</obj_property>
<obj_property name="UseCustomSignalColor">true</obj_property>
</wvobject>
Expand Down

0 comments on commit 95d598a

Please sign in to comment.