From bf72b3fcbe053168f75a0c2295b46d3a18e4ac3f Mon Sep 17 00:00:00 2001 From: Daniel Schultz Date: Sun, 5 Jan 2025 11:25:09 +0100 Subject: [PATCH] designs: ihp-sg13g2: Add I2C GPIO Expander Simple I2C GPIO expander with 8 GPIO pins. This design is a full chip layout with IO ring included. It's missing metal fill and therefore not tape-out ready but a new example to demonstrate OpenROAD and/or the IHP PDK due it's short build time. Signed-off-by: Daniel Schultz --- .../i2c-gpio-expander/I2cGpioExpander.v | 957 ++++++++++++++++++ .../i2c-gpio-expander/I2cGpioExpanderTop.v | 202 ++++ .../ihp-sg13g2/i2c-gpio-expander/config.mk | 21 + .../i2c-gpio-expander/constraint.sdc | 56 + .../ihp-sg13g2/i2c-gpio-expander/pad.tcl | 102 ++ .../ihp-sg13g2/i2c-gpio-expander/pdn.tcl | 60 ++ .../i2c-gpio-expander/sealring.gds.gz | Bin 0 -> 2757 bytes 7 files changed, 1398 insertions(+) create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpander.v create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpanderTop.v create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/config.mk create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/constraint.sdc create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/pad.tcl create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/pdn.tcl create mode 100644 flow/designs/ihp-sg13g2/i2c-gpio-expander/sealring.gds.gz diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpander.v b/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpander.v new file mode 100644 index 0000000000..237a43223c --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpander.v @@ -0,0 +1,957 @@ +// Generator : SpinalHDL v1.10.2a git head : a348a60b7e8b6a455c72e1536ec3d74a2ea16935 +// Component : SG13G2Top +// Git hash : dbd10e6c9374bd7bc3ffc29e541415e2be16643a + +`timescale 1ns/1ps + +module I2cGpioExpander ( + input wire [2:0] io_address, + output wire io_i2c_scl_write, + input wire io_i2c_scl_read, + output wire io_i2c_sda_write, + input wire io_i2c_sda_read, + output wire [0:0] io_i2c_interrupts, + input wire [7:0] io_gpio_pins_read, + output wire [7:0] io_gpio_pins_write, + output wire [7:0] io_gpio_pins_writeEnable, + input wire clock, + input wire reset +); + localparam State_IDLE = 3'd0; + localparam State_REG_1 = 3'd1; + localparam State_READ = 3'd2; + localparam State_WRITE = 3'd3; + localparam State_RESPONSE = 3'd4; + + wire [6:0] i2cCtrl_io_config_deviceAddr; + wire [0:0] i2cCtrl_io_interrupts; + reg i2cCtrl_io_cmd_ready; + reg i2cCtrl_io_rsp_valid; + reg [7:0] irq_high_ctrl_io_clears; + reg [7:0] irq_low_ctrl_io_clears; + reg [7:0] irq_rise_ctrl_io_clears; + reg [7:0] irq_fall_ctrl_io_clears; + wire i2cCtrl_io_i2c_scl_write; + wire i2cCtrl_io_i2c_sda_write; + wire [0:0] i2cCtrl_io_i2c_interrupts; + wire i2cCtrl_io_cmd_valid; + wire [7:0] i2cCtrl_io_cmd_payload_data; + wire i2cCtrl_io_cmd_payload_reg; + wire i2cCtrl_io_cmd_payload_read; + wire i2cCtrl_io_rsp_ready; + wire [7:0] gpioCtrl_1_io_gpio_pins_write; + wire [7:0] gpioCtrl_1_io_gpio_pins_writeEnable; + wire [7:0] gpioCtrl_1_io_value; + wire gpioCtrl_1_io_interrupt; + wire [7:0] gpioCtrl_1_io_irqHigh_valid; + wire [7:0] gpioCtrl_1_io_irqLow_valid; + wire [7:0] gpioCtrl_1_io_irqRise_valid; + wire [7:0] gpioCtrl_1_io_irqFall_valid; + wire [7:0] irq_high_ctrl_io_pendings; + wire [7:0] irq_low_ctrl_io_pendings; + wire [7:0] irq_rise_ctrl_io_pendings; + wire [7:0] irq_fall_ctrl_io_pendings; + reg [2:0] i2cConfig_latchedAddress; + reg i2cConfig_latch; + wire when_I2cGpioExpander_l46; + reg [7:0] gpioConfig_write; + reg [7:0] gpioConfig_direction; + reg [7:0] irq_high_masks; + reg [7:0] irq_low_masks; + reg [7:0] irq_rise_masks; + reg [7:0] irq_fall_masks; + reg [7:0] link_regAddr; + reg [2:0] link_state; + reg [7:0] link_data; + reg link_error; + wire [7:0] switch_I2cGpioExpander_l139; + wire [7:0] switch_I2cGpioExpander_l182; + `ifndef SYNTHESIS + reg [63:0] link_state_string; + `endif + + + I2cDeviceCtrl i2cCtrl ( + .io_config_clockDivider (16'h0001 ), //i + .io_config_timeout (16'h1388 ), //i + .io_config_deviceAddr (i2cCtrl_io_config_deviceAddr[6:0]), //i + .io_i2c_scl_write (i2cCtrl_io_i2c_scl_write ), //o + .io_i2c_scl_read (io_i2c_scl_read ), //i + .io_i2c_sda_write (i2cCtrl_io_i2c_sda_write ), //o + .io_i2c_sda_read (io_i2c_sda_read ), //i + .io_i2c_interrupts (i2cCtrl_io_i2c_interrupts ), //o + .io_interrupts (i2cCtrl_io_interrupts ), //i + .io_cmd_valid (i2cCtrl_io_cmd_valid ), //o + .io_cmd_ready (i2cCtrl_io_cmd_ready ), //i + .io_cmd_payload_data (i2cCtrl_io_cmd_payload_data[7:0] ), //o + .io_cmd_payload_reg (i2cCtrl_io_cmd_payload_reg ), //o + .io_cmd_payload_read (i2cCtrl_io_cmd_payload_read ), //o + .io_rsp_valid (i2cCtrl_io_rsp_valid ), //i + .io_rsp_ready (i2cCtrl_io_rsp_ready ), //o + .io_rsp_payload_data (link_data[7:0] ), //i + .io_rsp_payload_error (link_error ), //i + .clock (clock ), //i + .reset (reset ) //i + ); + GpioCtrl gpioCtrl_1 ( + .io_gpio_pins_read (io_gpio_pins_read[7:0] ), //i + .io_gpio_pins_write (gpioCtrl_1_io_gpio_pins_write[7:0] ), //o + .io_gpio_pins_writeEnable (gpioCtrl_1_io_gpio_pins_writeEnable[7:0]), //o + .io_config_write (gpioConfig_write[7:0] ), //i + .io_config_direction (gpioConfig_direction[7:0] ), //i + .io_value (gpioCtrl_1_io_value[7:0] ), //o + .io_interrupt (gpioCtrl_1_io_interrupt ), //o + .io_irqHigh_valid (gpioCtrl_1_io_irqHigh_valid[7:0] ), //o + .io_irqHigh_pending (irq_high_ctrl_io_pendings[7:0] ), //i + .io_irqLow_valid (gpioCtrl_1_io_irqLow_valid[7:0] ), //o + .io_irqLow_pending (irq_low_ctrl_io_pendings[7:0] ), //i + .io_irqRise_valid (gpioCtrl_1_io_irqRise_valid[7:0] ), //o + .io_irqRise_pending (irq_rise_ctrl_io_pendings[7:0] ), //i + .io_irqFall_valid (gpioCtrl_1_io_irqFall_valid[7:0] ), //o + .io_irqFall_pending (irq_fall_ctrl_io_pendings[7:0] ), //i + .clock (clock ), //i + .reset (reset ) //i + ); + InterruptCtrl irq_high_ctrl ( + .io_inputs (gpioCtrl_1_io_irqHigh_valid[7:0]), //i + .io_clears (irq_high_ctrl_io_clears[7:0] ), //i + .io_masks (irq_high_masks[7:0] ), //i + .io_pendings (irq_high_ctrl_io_pendings[7:0] ), //o + .clock (clock ), //i + .reset (reset ) //i + ); + InterruptCtrl irq_low_ctrl ( + .io_inputs (gpioCtrl_1_io_irqLow_valid[7:0]), //i + .io_clears (irq_low_ctrl_io_clears[7:0] ), //i + .io_masks (irq_low_masks[7:0] ), //i + .io_pendings (irq_low_ctrl_io_pendings[7:0] ), //o + .clock (clock ), //i + .reset (reset ) //i + ); + InterruptCtrl irq_rise_ctrl ( + .io_inputs (gpioCtrl_1_io_irqRise_valid[7:0]), //i + .io_clears (irq_rise_ctrl_io_clears[7:0] ), //i + .io_masks (irq_rise_masks[7:0] ), //i + .io_pendings (irq_rise_ctrl_io_pendings[7:0] ), //o + .clock (clock ), //i + .reset (reset ) //i + ); + InterruptCtrl irq_fall_ctrl ( + .io_inputs (gpioCtrl_1_io_irqFall_valid[7:0]), //i + .io_clears (irq_fall_ctrl_io_clears[7:0] ), //i + .io_masks (irq_fall_masks[7:0] ), //i + .io_pendings (irq_fall_ctrl_io_pendings[7:0] ), //o + .clock (clock ), //i + .reset (reset ) //i + ); + `ifndef SYNTHESIS + always @(*) begin + case(link_state) + State_IDLE : link_state_string = "IDLE "; + State_REG_1 : link_state_string = "REG_1 "; + State_READ : link_state_string = "READ "; + State_WRITE : link_state_string = "WRITE "; + State_RESPONSE : link_state_string = "RESPONSE"; + default : link_state_string = "????????"; + endcase + end + `endif + + assign io_i2c_scl_write = i2cCtrl_io_i2c_scl_write; + assign io_i2c_sda_write = i2cCtrl_io_i2c_sda_write; + assign io_i2c_interrupts = i2cCtrl_io_i2c_interrupts; + assign when_I2cGpioExpander_l46 = (! i2cConfig_latch); + assign i2cCtrl_io_config_deviceAddr = {4'b0110,i2cConfig_latchedAddress}; + assign io_gpio_pins_write = gpioCtrl_1_io_gpio_pins_write; + assign io_gpio_pins_writeEnable = gpioCtrl_1_io_gpio_pins_writeEnable; + always @(*) begin + irq_high_ctrl_io_clears = 8'h0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + end + State_READ : begin + end + State_WRITE : begin + case(switch_I2cGpioExpander_l182) + 8'h01 : begin + end + 8'h02 : begin + end + 8'h03 : begin + irq_high_ctrl_io_clears = i2cCtrl_io_cmd_payload_data; + end + 8'h04 : begin + end + 8'h05 : begin + end + 8'h06 : begin + end + 8'h07 : begin + end + 8'h08 : begin + end + 8'h09 : begin + end + 8'h0a : begin + end + default : begin + end + endcase + end + default : begin + end + endcase + end + + always @(*) begin + irq_low_ctrl_io_clears = 8'h0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + end + State_READ : begin + end + State_WRITE : begin + case(switch_I2cGpioExpander_l182) + 8'h01 : begin + end + 8'h02 : begin + end + 8'h03 : begin + end + 8'h04 : begin + irq_low_ctrl_io_clears = i2cCtrl_io_cmd_payload_data; + end + 8'h05 : begin + end + 8'h06 : begin + end + 8'h07 : begin + end + 8'h08 : begin + end + 8'h09 : begin + end + 8'h0a : begin + end + default : begin + end + endcase + end + default : begin + end + endcase + end + + always @(*) begin + irq_rise_ctrl_io_clears = 8'h0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + end + State_READ : begin + end + State_WRITE : begin + case(switch_I2cGpioExpander_l182) + 8'h01 : begin + end + 8'h02 : begin + end + 8'h03 : begin + end + 8'h04 : begin + end + 8'h05 : begin + irq_rise_ctrl_io_clears = i2cCtrl_io_cmd_payload_data; + end + 8'h06 : begin + end + 8'h07 : begin + end + 8'h08 : begin + end + 8'h09 : begin + end + 8'h0a : begin + end + default : begin + end + endcase + end + default : begin + end + endcase + end + + always @(*) begin + irq_fall_ctrl_io_clears = 8'h0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + end + State_READ : begin + end + State_WRITE : begin + case(switch_I2cGpioExpander_l182) + 8'h01 : begin + end + 8'h02 : begin + end + 8'h03 : begin + end + 8'h04 : begin + end + 8'h05 : begin + end + 8'h06 : begin + irq_fall_ctrl_io_clears = i2cCtrl_io_cmd_payload_data; + end + 8'h07 : begin + end + 8'h08 : begin + end + 8'h09 : begin + end + 8'h0a : begin + end + default : begin + end + endcase + end + default : begin + end + endcase + end + + assign i2cCtrl_io_interrupts = (|(((irq_high_ctrl_io_pendings | irq_low_ctrl_io_pendings) | irq_rise_ctrl_io_pendings) | irq_fall_ctrl_io_pendings)); + always @(*) begin + i2cCtrl_io_cmd_ready = 1'b0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + i2cCtrl_io_cmd_ready = 1'b1; + end + State_READ : begin + i2cCtrl_io_cmd_ready = 1'b1; + end + State_WRITE : begin + i2cCtrl_io_cmd_ready = 1'b1; + end + default : begin + end + endcase + end + + always @(*) begin + i2cCtrl_io_rsp_valid = 1'b0; + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + end + State_READ : begin + end + State_WRITE : begin + end + default : begin + i2cCtrl_io_rsp_valid = 1'b1; + end + endcase + end + + assign switch_I2cGpioExpander_l139 = link_regAddr; + assign switch_I2cGpioExpander_l182 = link_regAddr; + always @(posedge clock) begin + if(!reset) begin + i2cConfig_latch <= 1'b0; + gpioConfig_write <= 8'h0; + gpioConfig_direction <= 8'h0; + irq_high_masks <= 8'h0; + irq_low_masks <= 8'h0; + irq_rise_masks <= 8'h0; + irq_fall_masks <= 8'h0; + link_state <= State_IDLE; + link_data <= 8'h0; + link_error <= 1'b0; + end else begin + if(when_I2cGpioExpander_l46) begin + i2cConfig_latch <= 1'b1; + end + case(link_state) + State_IDLE : begin + if(i2cCtrl_io_cmd_valid) begin + link_error <= 1'b0; + if(i2cCtrl_io_cmd_payload_read) begin + link_state <= State_READ; + end else begin + if(i2cCtrl_io_cmd_payload_reg) begin + link_state <= State_REG_1; + end else begin + link_state <= State_WRITE; + end + end + end + end + State_REG_1 : begin + link_state <= State_RESPONSE; + end + State_READ : begin + case(switch_I2cGpioExpander_l139) + 8'h0 : begin + link_data <= gpioCtrl_1_io_value; + end + 8'h01 : begin + link_data <= gpioConfig_write; + end + 8'h02 : begin + link_data <= gpioConfig_direction; + end + 8'h03 : begin + link_data <= irq_high_ctrl_io_pendings; + end + 8'h04 : begin + link_data <= irq_low_ctrl_io_pendings; + end + 8'h05 : begin + link_data <= irq_rise_ctrl_io_pendings; + end + 8'h06 : begin + link_data <= irq_fall_ctrl_io_pendings; + end + 8'h07 : begin + link_data <= irq_high_masks; + end + 8'h08 : begin + link_data <= irq_low_masks; + end + 8'h09 : begin + link_data <= irq_rise_masks; + end + 8'h0a : begin + link_data <= irq_fall_masks; + end + default : begin + link_error <= 1'b1; + end + endcase + link_state <= State_RESPONSE; + end + State_WRITE : begin + case(switch_I2cGpioExpander_l182) + 8'h01 : begin + gpioConfig_write <= i2cCtrl_io_cmd_payload_data; + end + 8'h02 : begin + gpioConfig_direction <= i2cCtrl_io_cmd_payload_data; + end + 8'h03 : begin + end + 8'h04 : begin + end + 8'h05 : begin + end + 8'h06 : begin + end + 8'h07 : begin + irq_high_masks <= i2cCtrl_io_cmd_payload_data; + end + 8'h08 : begin + irq_low_masks <= i2cCtrl_io_cmd_payload_data; + end + 8'h09 : begin + irq_rise_masks <= i2cCtrl_io_cmd_payload_data; + end + 8'h0a : begin + irq_fall_masks <= i2cCtrl_io_cmd_payload_data; + end + default : begin + link_error <= 1'b1; + end + endcase + link_state <= State_RESPONSE; + end + default : begin + if(i2cCtrl_io_rsp_ready) begin + link_state <= State_IDLE; + end + end + endcase + end + end + + always @(posedge clock) begin + if(when_I2cGpioExpander_l46) begin + i2cConfig_latchedAddress <= io_address; + end + case(link_state) + State_IDLE : begin + end + State_REG_1 : begin + link_regAddr <= i2cCtrl_io_cmd_payload_data; + end + State_READ : begin + link_regAddr <= (link_regAddr + 8'h01); + end + State_WRITE : begin + link_regAddr <= (link_regAddr + 8'h01); + end + default : begin + end + endcase + end + + +endmodule + +//InterruptCtrl_3 replaced by InterruptCtrl + +//InterruptCtrl_2 replaced by InterruptCtrl + +//InterruptCtrl_1 replaced by InterruptCtrl + +module InterruptCtrl ( + input wire [7:0] io_inputs, + input wire [7:0] io_clears, + input wire [7:0] io_masks, + output wire [7:0] io_pendings, + input wire clock, + input wire reset +); + + reg [7:0] pendings; + + assign io_pendings = (pendings & io_masks); + always @(posedge clock) begin + if(!reset) begin + pendings <= 8'h0; + end else begin + pendings <= ((pendings & (~ io_clears)) | io_inputs); + end + end + + +endmodule + +module GpioCtrl ( + input wire [7:0] io_gpio_pins_read, + output wire [7:0] io_gpio_pins_write, + output wire [7:0] io_gpio_pins_writeEnable, + input wire [7:0] io_config_write, + input wire [7:0] io_config_direction, + output wire [7:0] io_value, + output wire io_interrupt, + output wire [7:0] io_irqHigh_valid, + input wire [7:0] io_irqHigh_pending, + output wire [7:0] io_irqLow_valid, + input wire [7:0] io_irqLow_pending, + output wire [7:0] io_irqRise_valid, + input wire [7:0] io_irqRise_pending, + output wire [7:0] io_irqFall_valid, + input wire [7:0] io_irqFall_pending, + input wire clock, + input wire reset +); + + wire [7:0] io_gpio_pins_read_buffercc_io_dataOut; + wire [7:0] synchronized; + reg [7:0] last; + + (* keep_hierarchy = "TRUE" *) BufferCC io_gpio_pins_read_buffercc ( + .io_dataIn (io_gpio_pins_read[7:0] ), //i + .io_dataOut (io_gpio_pins_read_buffercc_io_dataOut[7:0]), //o + .clock (clock ), //i + .reset (reset ) //i + ); + assign io_value = io_gpio_pins_read_buffercc_io_dataOut; + assign synchronized = io_value; + assign io_gpio_pins_write = io_config_write; + assign io_gpio_pins_writeEnable = io_config_direction; + assign io_irqHigh_valid = synchronized; + assign io_irqLow_valid = (~ synchronized); + assign io_irqRise_valid = (synchronized & (~ last)); + assign io_irqFall_valid = ((~ synchronized) & last); + assign io_interrupt = (|(((io_irqHigh_pending | io_irqLow_pending) | io_irqRise_pending) | io_irqFall_pending)); + always @(posedge clock) begin + last <= synchronized; + end + + +endmodule + +module I2cDeviceCtrl ( + input wire [15:0] io_config_clockDivider, + input wire [15:0] io_config_timeout, + input wire [6:0] io_config_deviceAddr, + output wire io_i2c_scl_write, + input wire io_i2c_scl_read, + output wire io_i2c_sda_write, + input wire io_i2c_sda_read, + output wire [0:0] io_i2c_interrupts, + input wire [0:0] io_interrupts, + output reg io_cmd_valid, + input wire io_cmd_ready, + output wire [7:0] io_cmd_payload_data, + output wire io_cmd_payload_reg, + output wire io_cmd_payload_read, + input wire io_rsp_valid, + output reg io_rsp_ready, + input wire [7:0] io_rsp_payload_data, + input wire io_rsp_payload_error, + input wire clock, + input wire reset +); + localparam State_1_IDLE = 2'd0; + localparam State_1_REQ = 2'd1; + localparam State_1_RSP = 2'd2; + + wire io_i2c_scl_read_buffercc_io_dataOut; + wire io_i2c_sda_read_buffercc_io_dataOut; + wire [7:0] _zz_ctrl_sdaWrite; + wire [2:0] _zz_ctrl_sdaWrite_1; + reg [15:0] filter_timer_counter; + wire filter_timer_tick; + wire filter_sampler_sclSync; + wire filter_sampler_sdaSync; + wire filter_sampler_sclSamples_0; + wire filter_sampler_sclSamples_1; + wire filter_sampler_sclSamples_2; + wire _zz_filter_sampler_sclSamples_0; + reg _zz_filter_sampler_sclSamples_1; + reg _zz_filter_sampler_sclSamples_2; + wire filter_sampler_sdaSamples_0; + wire filter_sampler_sdaSamples_1; + wire filter_sampler_sdaSamples_2; + wire _zz_filter_sampler_sdaSamples_0; + reg _zz_filter_sampler_sdaSamples_1; + reg _zz_filter_sampler_sdaSamples_2; + reg filter_sda; + reg filter_scl; + wire when_I2cDeviceCtrl_l52; + wire when_I2cDeviceCtrl_l55; + wire sclEdge_rise; + wire sclEdge_fall; + wire sclEdge_toggle; + reg filter_scl_regNext; + wire sdaEdge_rise; + wire sdaEdge_fall; + wire sdaEdge_toggle; + reg filter_sda_regNext; + wire detector_start; + wire detector_stop; + reg [15:0] timeout_value; + reg timeout_transmission; + wire when_I2cDeviceCtrl_l86; + wire when_I2cDeviceCtrl_l89; + reg [1:0] ctrl_state; + reg [10:0] ctrl_shiftRegister; + reg [4:0] ctrl_bitCounter; + reg [3:0] ctrl_frameCounter; + reg ctrl_transmission; + reg ctrl_sdaWrite; + wire [6:0] ctrl_address; + wire [7:0] ctrl_data; + reg ctrl_read; + wire ctrl_write; + reg ctrl_response_error; + reg [7:0] ctrl_response_data; + reg ctrl_cmdLock; + wire when_I2cDeviceCtrl_l123; + wire [4:0] _zz_ctrl_bitCounter; + wire when_I2cDeviceCtrl_l137; + wire when_I2cDeviceCtrl_l138; + wire when_I2cDeviceCtrl_l141; + wire when_I2cDeviceCtrl_l145; + wire when_I2cDeviceCtrl_l149; + wire when_I2cDeviceCtrl_l150; + wire when_I2cDeviceCtrl_l152; + wire when_I2cDeviceCtrl_l157; + wire when_I2cDeviceCtrl_l160; + wire when_I2cDeviceCtrl_l175; + wire io_cmd_fire; + `ifndef SYNTHESIS + reg [31:0] ctrl_state_string; + `endif + + + assign _zz_ctrl_sdaWrite = {ctrl_response_data[0],{ctrl_response_data[1],{ctrl_response_data[2],{ctrl_response_data[3],{ctrl_response_data[4],{ctrl_response_data[5],{ctrl_response_data[6],ctrl_response_data[7]}}}}}}}; + assign _zz_ctrl_sdaWrite_1 = ctrl_bitCounter[2:0]; + (* keep_hierarchy = "TRUE" *) BufferCC_1 io_i2c_scl_read_buffercc ( + .io_dataIn (io_i2c_scl_read ), //i + .io_dataOut (io_i2c_scl_read_buffercc_io_dataOut), //o + .clock (clock ), //i + .reset (reset ) //i + ); + (* keep_hierarchy = "TRUE" *) BufferCC_1 io_i2c_sda_read_buffercc ( + .io_dataIn (io_i2c_sda_read ), //i + .io_dataOut (io_i2c_sda_read_buffercc_io_dataOut), //o + .clock (clock ), //i + .reset (reset ) //i + ); + `ifndef SYNTHESIS + always @(*) begin + case(ctrl_state) + State_1_IDLE : ctrl_state_string = "IDLE"; + State_1_REQ : ctrl_state_string = "REQ "; + State_1_RSP : ctrl_state_string = "RSP "; + default : ctrl_state_string = "????"; + endcase + end + `endif + + assign filter_timer_tick = (filter_timer_counter == 16'h0); + assign filter_sampler_sclSync = io_i2c_scl_read_buffercc_io_dataOut; + assign filter_sampler_sdaSync = io_i2c_sda_read_buffercc_io_dataOut; + assign _zz_filter_sampler_sclSamples_0 = filter_sampler_sclSync; + assign filter_sampler_sclSamples_0 = _zz_filter_sampler_sclSamples_0; + assign filter_sampler_sclSamples_1 = _zz_filter_sampler_sclSamples_1; + assign filter_sampler_sclSamples_2 = _zz_filter_sampler_sclSamples_2; + assign _zz_filter_sampler_sdaSamples_0 = filter_sampler_sdaSync; + assign filter_sampler_sdaSamples_0 = _zz_filter_sampler_sdaSamples_0; + assign filter_sampler_sdaSamples_1 = _zz_filter_sampler_sdaSamples_1; + assign filter_sampler_sdaSamples_2 = _zz_filter_sampler_sdaSamples_2; + assign when_I2cDeviceCtrl_l52 = (&{(filter_sampler_sdaSamples_2 != filter_sda),{(filter_sampler_sdaSamples_1 != filter_sda),(filter_sampler_sdaSamples_0 != filter_sda)}}); + assign when_I2cDeviceCtrl_l55 = (&{(filter_sampler_sclSamples_2 != filter_scl),{(filter_sampler_sclSamples_1 != filter_scl),(filter_sampler_sclSamples_0 != filter_scl)}}); + assign sclEdge_rise = ((! filter_scl_regNext) && filter_scl); + assign sclEdge_fall = (filter_scl_regNext && (! filter_scl)); + assign sclEdge_toggle = (filter_scl_regNext != filter_scl); + assign sdaEdge_rise = ((! filter_sda_regNext) && filter_sda); + assign sdaEdge_fall = (filter_sda_regNext && (! filter_sda)); + assign sdaEdge_toggle = (filter_sda_regNext != filter_sda); + assign detector_start = (filter_scl && sdaEdge_fall); + assign detector_stop = (filter_scl && sdaEdge_rise); + assign when_I2cDeviceCtrl_l86 = ((detector_start || sclEdge_rise) || sclEdge_fall); + assign when_I2cDeviceCtrl_l89 = (filter_timer_tick && timeout_transmission); + assign ctrl_address = ctrl_shiftRegister[7 : 1]; + assign ctrl_data = ctrl_shiftRegister[7 : 0]; + assign ctrl_write = (! ctrl_read); + assign when_I2cDeviceCtrl_l123 = ((detector_start || detector_stop) || (timeout_value == io_config_timeout)); + assign _zz_ctrl_bitCounter[4 : 0] = 5'h1f; + assign when_I2cDeviceCtrl_l137 = (((ctrl_frameCounter == 4'b0000) || (((ctrl_frameCounter == 4'b0001) || ((! (ctrl_frameCounter == 4'b0000)) && (! (ctrl_frameCounter == 4'b0001)))) && ctrl_write)) && (ctrl_bitCounter == 5'h08)); + assign when_I2cDeviceCtrl_l138 = ((ctrl_frameCounter == 4'b0000) && (ctrl_address == io_config_deviceAddr)); + assign when_I2cDeviceCtrl_l141 = (((ctrl_frameCounter == 4'b0001) || ((! (ctrl_frameCounter == 4'b0000)) && (! (ctrl_frameCounter == 4'b0001)))) && (! ctrl_response_error)); + assign when_I2cDeviceCtrl_l145 = ((ctrl_read && (! (ctrl_frameCounter == 4'b0000))) && (((ctrl_frameCounter == 4'b0000) && (ctrl_bitCounter < 5'h08)) || ((! (ctrl_frameCounter == 4'b0000)) && (ctrl_bitCounter < 5'h08)))); + assign when_I2cDeviceCtrl_l149 = (sclEdge_rise && ctrl_transmission); + assign when_I2cDeviceCtrl_l150 = (((ctrl_frameCounter == 4'b0000) && (ctrl_bitCounter < 5'h08)) || ((! (ctrl_frameCounter == 4'b0000)) && (ctrl_bitCounter < 5'h08))); + assign when_I2cDeviceCtrl_l152 = ((ctrl_frameCounter == 4'b0000) && (ctrl_bitCounter == 5'h07)); + assign when_I2cDeviceCtrl_l157 = (sclEdge_fall && ctrl_transmission); + assign when_I2cDeviceCtrl_l160 = (ctrl_bitCounter == 5'h08); + always @(*) begin + io_cmd_valid = 1'b0; + case(ctrl_state) + State_1_IDLE : begin + end + State_1_REQ : begin + io_cmd_valid = 1'b1; + end + default : begin + end + endcase + end + + assign io_cmd_payload_data = ctrl_data; + assign io_cmd_payload_reg = (ctrl_frameCounter == 4'b0001); + assign io_cmd_payload_read = ctrl_read; + always @(*) begin + io_rsp_ready = 1'b0; + case(ctrl_state) + State_1_IDLE : begin + end + State_1_REQ : begin + end + default : begin + if(io_rsp_valid) begin + io_rsp_ready = 1'b1; + end + end + endcase + end + + assign when_I2cDeviceCtrl_l175 = (((! ctrl_cmdLock) && (ctrl_bitCounter == 5'h08)) && (((ctrl_frameCounter == 4'b0001) || (ctrl_read && (ctrl_frameCounter == 4'b0000))) || (ctrl_write && ((! (ctrl_frameCounter == 4'b0000)) && (! (ctrl_frameCounter == 4'b0001)))))); + assign io_cmd_fire = (io_cmd_valid && io_cmd_ready); + assign io_i2c_scl_write = 1'b0; + assign io_i2c_sda_write = ctrl_sdaWrite; + assign io_i2c_interrupts = io_interrupts; + always @(posedge clock) begin + if(!reset) begin + filter_timer_counter <= 16'h0; + _zz_filter_sampler_sclSamples_1 <= 1'b1; + _zz_filter_sampler_sclSamples_2 <= 1'b1; + _zz_filter_sampler_sdaSamples_1 <= 1'b1; + _zz_filter_sampler_sdaSamples_2 <= 1'b1; + filter_sda <= 1'b1; + filter_scl <= 1'b1; + filter_scl_regNext <= 1'b1; + filter_sda_regNext <= 1'b1; + timeout_value <= 16'h0; + timeout_transmission <= 1'b0; + ctrl_state <= State_1_IDLE; + ctrl_transmission <= 1'b0; + ctrl_sdaWrite <= 1'b0; + end else begin + filter_timer_counter <= (filter_timer_counter - 16'h0001); + if(filter_timer_tick) begin + filter_timer_counter <= io_config_clockDivider; + end + if(filter_timer_tick) begin + _zz_filter_sampler_sclSamples_1 <= _zz_filter_sampler_sclSamples_0; + end + if(filter_timer_tick) begin + _zz_filter_sampler_sclSamples_2 <= _zz_filter_sampler_sclSamples_1; + end + if(filter_timer_tick) begin + _zz_filter_sampler_sdaSamples_1 <= _zz_filter_sampler_sdaSamples_0; + end + if(filter_timer_tick) begin + _zz_filter_sampler_sdaSamples_2 <= _zz_filter_sampler_sdaSamples_1; + end + if(filter_timer_tick) begin + if(when_I2cDeviceCtrl_l52) begin + filter_sda <= filter_sampler_sdaSamples_2; + end + if(when_I2cDeviceCtrl_l55) begin + filter_scl <= filter_sampler_sclSamples_2; + end + end + filter_scl_regNext <= filter_scl; + filter_sda_regNext <= filter_sda; + if(detector_start) begin + timeout_transmission <= 1'b1; + end + if(detector_stop) begin + timeout_transmission <= 1'b0; + end + if(when_I2cDeviceCtrl_l86) begin + timeout_value <= 16'h0; + end + if(when_I2cDeviceCtrl_l89) begin + timeout_value <= (timeout_value + 16'h0001); + end + if(when_I2cDeviceCtrl_l123) begin + ctrl_state <= State_1_IDLE; + if(detector_start) begin + ctrl_transmission <= 1'b1; + end else begin + ctrl_transmission <= 1'b0; + end + end + ctrl_sdaWrite <= 1'b0; + if(when_I2cDeviceCtrl_l137) begin + if(when_I2cDeviceCtrl_l138) begin + ctrl_sdaWrite <= 1'b1; + end + if(when_I2cDeviceCtrl_l141) begin + ctrl_sdaWrite <= 1'b1; + end + end + if(when_I2cDeviceCtrl_l145) begin + ctrl_sdaWrite <= (! _zz_ctrl_sdaWrite[_zz_ctrl_sdaWrite_1]); + end + case(ctrl_state) + State_1_IDLE : begin + if(when_I2cDeviceCtrl_l175) begin + ctrl_state <= State_1_REQ; + end + end + State_1_REQ : begin + if(io_cmd_fire) begin + ctrl_state <= State_1_RSP; + end + end + default : begin + if(io_rsp_valid) begin + ctrl_state <= State_1_IDLE; + end + end + endcase + end + end + + always @(posedge clock) begin + if(when_I2cDeviceCtrl_l123) begin + ctrl_bitCounter <= _zz_ctrl_bitCounter; + ctrl_frameCounter <= 4'b0000; + ctrl_read <= 1'b0; + ctrl_response_error <= 1'b1; + end + if(when_I2cDeviceCtrl_l149) begin + if(when_I2cDeviceCtrl_l150) begin + ctrl_shiftRegister <= {ctrl_shiftRegister[9 : 0],filter_sda}; + if(when_I2cDeviceCtrl_l152) begin + ctrl_read <= filter_sda; + end + end + end + if(when_I2cDeviceCtrl_l157) begin + ctrl_bitCounter <= (ctrl_bitCounter + 5'h01); + ctrl_cmdLock <= 1'b0; + if(when_I2cDeviceCtrl_l160) begin + ctrl_bitCounter <= 5'h0; + ctrl_frameCounter <= (ctrl_frameCounter + 4'b0001); + end + end + case(ctrl_state) + State_1_IDLE : begin + end + State_1_REQ : begin + end + default : begin + if(io_rsp_valid) begin + ctrl_cmdLock <= 1'b1; + ctrl_response_data <= io_rsp_payload_data; + ctrl_response_error <= io_rsp_payload_error; + end + end + endcase + end + + +endmodule + +module BufferCC ( + input wire [7:0] io_dataIn, + output wire [7:0] io_dataOut, + input wire clock, + input wire reset +); + + (* async_reg = "true" *) reg [7:0] buffers_0; + + assign io_dataOut = buffers_0; + always @(posedge clock) begin + buffers_0 <= io_dataIn; + end + + +endmodule + +//BufferCC_2 replaced by BufferCC_1 + +module BufferCC_1 ( + input wire io_dataIn, + output wire io_dataOut, + input wire clock, + input wire reset +); + + (* async_reg = "true" *) reg buffers_0; + (* async_reg = "true" *) reg buffers_1; + + assign io_dataOut = buffers_1; + always @(posedge clock) begin + if(!reset) begin + buffers_0 <= 1'b1; + buffers_1 <= 1'b1; + end else begin + buffers_0 <= io_dataIn; + buffers_1 <= buffers_0; + end + end + + +endmodule diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpanderTop.v b/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpanderTop.v new file mode 100644 index 0000000000..86d9adbaaf --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/I2cGpioExpanderTop.v @@ -0,0 +1,202 @@ +// Generator : SpinalHDL v1.10.2a git head : a348a60b7e8b6a455c72e1536ec3d74a2ea16935 +// Component : SG13G2Top +// Git hash : dbd10e6c9374bd7bc3ffc29e541415e2be16643a + +`timescale 1ns/1ps + +module I2CGpioExpanderTop ( + inout wire io_clock_PAD, + inout wire io_reset_PAD, + inout wire io_i2c_scl_PAD, + inout wire io_i2c_sda_PAD, + inout wire io_i2c_interrupt_PAD, + inout wire io_gpio_0_PAD, + inout wire io_gpio_1_PAD, + inout wire io_gpio_2_PAD, + inout wire io_gpio_3_PAD, + inout wire io_gpio_4_PAD, + inout wire io_gpio_5_PAD, + inout wire io_gpio_6_PAD, + inout wire io_gpio_7_PAD, + inout wire io_address_0_PAD, + inout wire io_address_1_PAD, + inout wire io_address_2_PAD +); + + reg [7:0] system_expander_io_gpio_pins_read; + wire sg13g2_IOPad_io_i2c_interrupt_c2p; + wire sg13g2_IOPad_io_gpio_0_c2p; + wire sg13g2_IOPad_io_gpio_0_c2p_en; + wire sg13g2_IOPad_io_gpio_1_c2p; + wire sg13g2_IOPad_io_gpio_1_c2p_en; + wire sg13g2_IOPad_io_gpio_2_c2p; + wire sg13g2_IOPad_io_gpio_2_c2p_en; + wire sg13g2_IOPad_io_gpio_3_c2p; + wire sg13g2_IOPad_io_gpio_3_c2p_en; + wire sg13g2_IOPad_io_gpio_4_c2p; + wire sg13g2_IOPad_io_gpio_4_c2p_en; + wire sg13g2_IOPad_io_gpio_5_c2p; + wire sg13g2_IOPad_io_gpio_5_c2p_en; + wire sg13g2_IOPad_io_gpio_6_c2p; + wire sg13g2_IOPad_io_gpio_6_c2p_en; + wire sg13g2_IOPad_io_gpio_7_c2p; + wire sg13g2_IOPad_io_gpio_7_c2p_en; + wire sg13g2_IOPad_io_clock_p2c; + wire sg13g2_IOPad_io_reset_p2c; + wire sg13g2_IOPad_io_address_0_p2c; + wire sg13g2_IOPad_io_address_1_p2c; + wire sg13g2_IOPad_io_address_2_p2c; + wire system_expander_io_i2c_scl_write; + wire system_expander_io_i2c_sda_write; + wire [0:0] system_expander_io_i2c_interrupts; + wire [7:0] system_expander_io_gpio_pins_write; + wire [7:0] system_expander_io_gpio_pins_writeEnable; + wire sg13g2_IOPad_io_i2c_scl_p2c; + wire sg13g2_IOPad_io_i2c_sda_p2c; + wire sg13g2_IOPad_io_gpio_0_p2c; + wire sg13g2_IOPad_io_gpio_1_p2c; + wire sg13g2_IOPad_io_gpio_2_p2c; + wire sg13g2_IOPad_io_gpio_3_p2c; + wire sg13g2_IOPad_io_gpio_4_p2c; + wire sg13g2_IOPad_io_gpio_5_p2c; + wire sg13g2_IOPad_io_gpio_6_p2c; + wire sg13g2_IOPad_io_gpio_7_p2c; + wire clock; + wire reset; + reg [2:0] address; + + sg13g2_IOPadIn sg13g2_IOPad_io_clock ( + .p2c (sg13g2_IOPad_io_clock_p2c), //o + .pad (io_clock_PAD ) //~ + ); + sg13g2_IOPadIn sg13g2_IOPad_io_reset ( + .p2c (sg13g2_IOPad_io_reset_p2c), //o + .pad (io_reset_PAD ) //~ + ); + sg13g2_IOPadIn sg13g2_IOPad_io_address_0 ( + .p2c (sg13g2_IOPad_io_address_0_p2c), //o + .pad (io_address_0_PAD ) //~ + ); + sg13g2_IOPadIn sg13g2_IOPad_io_address_1 ( + .p2c (sg13g2_IOPad_io_address_1_p2c), //o + .pad (io_address_1_PAD ) //~ + ); + sg13g2_IOPadIn sg13g2_IOPad_io_address_2 ( + .p2c (sg13g2_IOPad_io_address_2_p2c), //o + .pad (io_address_2_PAD ) //~ + ); + I2cGpioExpander system_expander ( + .io_address (address[2:0] ), //i + .io_i2c_scl_write (system_expander_io_i2c_scl_write ), //o + .io_i2c_scl_read (sg13g2_IOPad_io_i2c_scl_p2c ), //i + .io_i2c_sda_write (system_expander_io_i2c_sda_write ), //o + .io_i2c_sda_read (sg13g2_IOPad_io_i2c_sda_p2c ), //i + .io_i2c_interrupts (system_expander_io_i2c_interrupts ), //o + .io_gpio_pins_read (system_expander_io_gpio_pins_read[7:0] ), //i + .io_gpio_pins_write (system_expander_io_gpio_pins_write[7:0] ), //o + .io_gpio_pins_writeEnable (system_expander_io_gpio_pins_writeEnable[7:0]), //o + .clock (clock ), //i + .reset (reset ) //i + ); + sg13g2_IOPadOut4mA sg13g2_IOPad_io_i2c_interrupt ( + .c2p (sg13g2_IOPad_io_i2c_interrupt_c2p), //i + .pad (io_i2c_interrupt_PAD ) //~ + ); + sg13g2_IOPadInOut4mA sg13g2_IOPad_io_i2c_scl ( + .c2p (1'b0 ), //i + .c2p_en (system_expander_io_i2c_scl_write), //i + .p2c (sg13g2_IOPad_io_i2c_scl_p2c ), //o + .pad (io_i2c_scl_PAD ) //~ + ); + sg13g2_IOPadInOut4mA sg13g2_IOPad_io_i2c_sda ( + .c2p (1'b0 ), //i + .c2p_en (system_expander_io_i2c_sda_write), //i + .p2c (sg13g2_IOPad_io_i2c_sda_p2c ), //o + .pad (io_i2c_sda_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_0 ( + .c2p (sg13g2_IOPad_io_gpio_0_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_0_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_0_p2c ), //o + .pad (io_gpio_0_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_1 ( + .c2p (sg13g2_IOPad_io_gpio_1_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_1_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_1_p2c ), //o + .pad (io_gpio_1_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_2 ( + .c2p (sg13g2_IOPad_io_gpio_2_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_2_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_2_p2c ), //o + .pad (io_gpio_2_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_3 ( + .c2p (sg13g2_IOPad_io_gpio_3_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_3_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_3_p2c ), //o + .pad (io_gpio_3_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_4 ( + .c2p (sg13g2_IOPad_io_gpio_4_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_4_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_4_p2c ), //o + .pad (io_gpio_4_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_5 ( + .c2p (sg13g2_IOPad_io_gpio_5_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_5_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_5_p2c ), //o + .pad (io_gpio_5_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_6 ( + .c2p (sg13g2_IOPad_io_gpio_6_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_6_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_6_p2c ), //o + .pad (io_gpio_6_PAD ) //~ + ); + sg13g2_IOPadInOut16mA sg13g2_IOPad_io_gpio_7 ( + .c2p (sg13g2_IOPad_io_gpio_7_c2p ), //i + .c2p_en (sg13g2_IOPad_io_gpio_7_c2p_en), //i + .p2c (sg13g2_IOPad_io_gpio_7_p2c ), //o + .pad (io_gpio_7_PAD ) //~ + ); + assign clock = sg13g2_IOPad_io_clock_p2c; + assign reset = sg13g2_IOPad_io_reset_p2c; + always @(*) begin + address[0] = sg13g2_IOPad_io_address_0_p2c; + address[1] = sg13g2_IOPad_io_address_1_p2c; + address[2] = sg13g2_IOPad_io_address_2_p2c; + end + + assign sg13g2_IOPad_io_i2c_interrupt_c2p = system_expander_io_i2c_interrupts[0]; + assign sg13g2_IOPad_io_gpio_0_c2p = system_expander_io_gpio_pins_write[0]; + assign sg13g2_IOPad_io_gpio_0_c2p_en = system_expander_io_gpio_pins_writeEnable[0]; + always @(*) begin + system_expander_io_gpio_pins_read[0] = sg13g2_IOPad_io_gpio_0_p2c; + system_expander_io_gpio_pins_read[1] = sg13g2_IOPad_io_gpio_1_p2c; + system_expander_io_gpio_pins_read[2] = sg13g2_IOPad_io_gpio_2_p2c; + system_expander_io_gpio_pins_read[3] = sg13g2_IOPad_io_gpio_3_p2c; + system_expander_io_gpio_pins_read[4] = sg13g2_IOPad_io_gpio_4_p2c; + system_expander_io_gpio_pins_read[5] = sg13g2_IOPad_io_gpio_5_p2c; + system_expander_io_gpio_pins_read[6] = sg13g2_IOPad_io_gpio_6_p2c; + system_expander_io_gpio_pins_read[7] = sg13g2_IOPad_io_gpio_7_p2c; + end + + assign sg13g2_IOPad_io_gpio_1_c2p = system_expander_io_gpio_pins_write[1]; + assign sg13g2_IOPad_io_gpio_1_c2p_en = system_expander_io_gpio_pins_writeEnable[1]; + assign sg13g2_IOPad_io_gpio_2_c2p = system_expander_io_gpio_pins_write[2]; + assign sg13g2_IOPad_io_gpio_2_c2p_en = system_expander_io_gpio_pins_writeEnable[2]; + assign sg13g2_IOPad_io_gpio_3_c2p = system_expander_io_gpio_pins_write[3]; + assign sg13g2_IOPad_io_gpio_3_c2p_en = system_expander_io_gpio_pins_writeEnable[3]; + assign sg13g2_IOPad_io_gpio_4_c2p = system_expander_io_gpio_pins_write[4]; + assign sg13g2_IOPad_io_gpio_4_c2p_en = system_expander_io_gpio_pins_writeEnable[4]; + assign sg13g2_IOPad_io_gpio_5_c2p = system_expander_io_gpio_pins_write[5]; + assign sg13g2_IOPad_io_gpio_5_c2p_en = system_expander_io_gpio_pins_writeEnable[5]; + assign sg13g2_IOPad_io_gpio_6_c2p = system_expander_io_gpio_pins_write[6]; + assign sg13g2_IOPad_io_gpio_6_c2p_en = system_expander_io_gpio_pins_writeEnable[6]; + assign sg13g2_IOPad_io_gpio_7_c2p = system_expander_io_gpio_pins_write[7]; + assign sg13g2_IOPad_io_gpio_7_c2p_en = system_expander_io_gpio_pins_writeEnable[7]; + +endmodule diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/config.mk b/flow/designs/ihp-sg13g2/i2c-gpio-expander/config.mk new file mode 100644 index 0000000000..a832f384e8 --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/config.mk @@ -0,0 +1,21 @@ +export DESIGN_NAME = I2cGpioExpander +export DESIGN_NICKNAME = i2c-gpio-expander +export PLATFORM = ihp-sg13g2 + +export VERILOG_FILES = $(DESIGN_HOME)/$(PLATFORM)/$(DESIGN_NICKNAME)/$(DESIGN_NAME)Top.v \ + $(DESIGN_HOME)/$(PLATFORM)/$(DESIGN_NICKNAME)/$(DESIGN_NAME).v +export SDC_FILE = $(DESIGN_HOME)/$(PLATFORM)/$(DESIGN_NICKNAME)/constraint.sdc + +export SEAL_GDS = sealring.gds.gz + +export DIE_AREA = 0.0 0.0 1050.0 1050.0 +export CORE_AREA = 425.28 427.16 631.2 630.24 + +export MAX_ROUTING_LAYER = TopMetal2 +export HAS_IO_RING = 1 + +export TNS_END_PERCENT = 100 +export PLACE_DENSITY = 0.75 + +export FOOTPRINT_TCL = $(DESIGN_HOME)/$(PLATFORM)/$(DESIGN_NICKNAME)/pad.tcl +export PDN_TCL = $(DESIGN_HOME)/$(PLATFORM)/$(DESIGN_NICKNAME)/pdn.tcl diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/constraint.sdc b/flow/designs/ihp-sg13g2/i2c-gpio-expander/constraint.sdc new file mode 100644 index 0000000000..02aff71773 --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/constraint.sdc @@ -0,0 +1,56 @@ +current_design SG13G2Top +set_units -time ns -resistance kOhm -capacitance pF -voltage V -current uA +set_max_fanout 8 [current_design] +set_max_capacitance 0.5 [current_design] +set_max_transition 3 [current_design] +set_max_area 0 + +set_ideal_network [get_pins sg13g2_IOPad_io_clock/p2c] +create_clock [get_pins sg13g2_IOPad_io_clock/p2c] -name clk_core -period 20.0 -waveform {0 10.0} +set_clock_uncertainty 0.15 [get_clocks clk_core] +set_clock_transition 0.25 [get_clocks clk_core] + +set clock_ports [get_ports { + io_clock_PAD +}] +set_driving_cell -lib_cell sg13g2_IOPadIn -pin pad $clock_ports + +set clk_core_inout_16mA_ports [get_ports { + io_gpio_0_PAD + io_gpio_1_PAD + io_gpio_2_PAD + io_gpio_3_PAD + io_gpio_4_PAD + io_gpio_5_PAD + io_gpio_6_PAD + io_gpio_7_PAD +}] +set_driving_cell -lib_cell sg13g2_IOPadInOut16mA -pin pad $clk_core_inout_16mA_ports +set_input_delay 8 -clock clk_core $clk_core_inout_16mA_ports +set_output_delay 8 -clock clk_core $clk_core_inout_16mA_ports + +set clk_core_inout_4mA_ports [get_ports { + io_i2c_scl_PAD + io_i2c_sda_PAD +}] +set_driving_cell -lib_cell sg13g2_IOPadInOut4mA -pin pad $clk_core_inout_4mA_ports +set_input_delay 8 -clock clk_core $clk_core_inout_4mA_ports +set_output_delay 8 -clock clk_core $clk_core_inout_4mA_ports + +set clk_core_input_ports [get_ports { + io_reset_PAD + io_address_0_PAD + io_address_1_PAD + io_address_2_PAD +}] +set_driving_cell -lib_cell sg13g2_IOPadIn -pin pad $clk_core_input_ports +set_input_delay 8 -clock clk_core $clk_core_input_ports + +set clk_core_output_4mA_ports [get_ports { + io_i2c_interrupt_PAD +}] +set_driving_cell -lib_cell sg13g2_IOPadOut4mA -pin pad $clk_core_output_4mA_ports +set_output_delay 8 -clock clk_core $clk_core_output_4mA_ports + +set_load -pin_load 5 [all_inputs] +set_load -pin_load 5 [all_outputs] diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/pad.tcl b/flow/designs/ihp-sg13g2/i2c-gpio-expander/pad.tcl new file mode 100644 index 0000000000..e15d9e3558 --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/pad.tcl @@ -0,0 +1,102 @@ +set IO_LENGTH 180 +set IO_WIDTH 80 +set BONDPAD_SIZE 70 +set SEALRING_OFFSET 70 + +proc calc_horizontal_pad_location {index total} { + global IO_LENGTH + global IO_WIDTH + global BONDPAD_SIZE + global SEALRING_OFFSET + + set DIE_WIDTH [expr {[lindex $::env(DIE_AREA) 2] - [lindex $::env(DIE_AREA) 0]}] + set PAD_OFFSET [expr {$IO_LENGTH + $BONDPAD_SIZE + $SEALRING_OFFSET}] + set PAD_AREA_WIDTH [expr {$DIE_WIDTH - ($PAD_OFFSET * 2)}] + set HORIZONTAL_PAD_DISTANCE [expr {($PAD_AREA_WIDTH / $total) - $IO_WIDTH}] + + return [expr {$PAD_OFFSET + (($IO_WIDTH + $HORIZONTAL_PAD_DISTANCE) * $index) + ($HORIZONTAL_PAD_DISTANCE / 2)}] +} + +proc calc_vertical_pad_location {index total} { + global IO_LENGTH + global IO_WIDTH + global BONDPAD_SIZE + global SEALRING_OFFSET + + set DIE_HEIGHT [expr {[lindex $::env(DIE_AREA) 3] - [lindex $::env(DIE_AREA) 1]}] + set PAD_OFFSET [expr {$IO_LENGTH + $BONDPAD_SIZE + $SEALRING_OFFSET}] + set PAD_AREA_HEIGHT [expr {$DIE_HEIGHT - ($PAD_OFFSET * 2)}] + set VERTICAL_PAD_DISTANCE [expr {($PAD_AREA_HEIGHT / $total) - $IO_WIDTH}] + + return [expr {$PAD_OFFSET + (($IO_WIDTH + $VERTICAL_PAD_DISTANCE) * $index) + ($VERTICAL_PAD_DISTANCE / 2)}] +} + +make_fake_io_site -name IOLibSite -width 1 -height $IO_LENGTH +make_fake_io_site -name IOLibCSite -width $IO_LENGTH -height $IO_LENGTH + +set IO_OFFSET [expr {$BONDPAD_SIZE + $SEALRING_OFFSET}] +# Create IO Rows +make_io_sites \ + -horizontal_site IOLibSite \ + -vertical_site IOLibSite \ + -corner_site IOLibCSite \ + -offset $IO_OFFSET + +# Place Pads\n# IO pin io_clock +place_pad -row IO_SOUTH -location [calc_horizontal_pad_location 0 5] {sg13g2_IOPad_io_clock} -master sg13g2_IOPadIn +# IO pin io_reset +place_pad -row IO_SOUTH -location [calc_horizontal_pad_location 1 5] {sg13g2_IOPad_io_reset} -master sg13g2_IOPadIn +# IO pin io_i2c_scl +place_pad -row IO_SOUTH -location [calc_horizontal_pad_location 2 5] {sg13g2_IOPad_io_i2c_scl} -master sg13g2_IOPadInOut4mA +# IO pin io_i2c_sda +place_pad -row IO_SOUTH -location [calc_horizontal_pad_location 3 5] {sg13g2_IOPad_io_i2c_sda} -master sg13g2_IOPadInOut4mA +# IO pin io_i2c_interrupt +place_pad -row IO_SOUTH -location [calc_horizontal_pad_location 4 5] {sg13g2_IOPad_io_i2c_interrupt} -master sg13g2_IOPadOut4mA +place_pad -row IO_EAST -location [calc_vertical_pad_location 0 5] {sg13g2_IOPadVdd_east_0} -master sg13g2_IOPadVdd +place_pad -row IO_EAST -location [calc_vertical_pad_location 1 5] {sg13g2_IOPadVss_east_1} -master sg13g2_IOPadVss +# IO pin io_address_0 +place_pad -row IO_EAST -location [calc_vertical_pad_location 2 5] {sg13g2_IOPad_io_address_0} -master sg13g2_IOPadIn +# IO pin io_address_1 +place_pad -row IO_EAST -location [calc_vertical_pad_location 3 5] {sg13g2_IOPad_io_address_1} -master sg13g2_IOPadIn +# IO pin io_address_2 +place_pad -row IO_EAST -location [calc_vertical_pad_location 4 5] {sg13g2_IOPad_io_address_2} -master sg13g2_IOPadIn +# IO pin io_gpio_0 +place_pad -row IO_NORTH -location [calc_horizontal_pad_location 0 5] {sg13g2_IOPad_io_gpio_0} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_1 +place_pad -row IO_NORTH -location [calc_horizontal_pad_location 1 5] {sg13g2_IOPad_io_gpio_1} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_2 +place_pad -row IO_NORTH -location [calc_horizontal_pad_location 2 5] {sg13g2_IOPad_io_gpio_2} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_3 +place_pad -row IO_NORTH -location [calc_horizontal_pad_location 3 5] {sg13g2_IOPad_io_gpio_3} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_4 +place_pad -row IO_NORTH -location [calc_horizontal_pad_location 4 5] {sg13g2_IOPad_io_gpio_4} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_5 +place_pad -row IO_WEST -location [calc_vertical_pad_location 0 5] {sg13g2_IOPad_io_gpio_5} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_6 +place_pad -row IO_WEST -location [calc_vertical_pad_location 1 5] {sg13g2_IOPad_io_gpio_6} -master sg13g2_IOPadInOut16mA +# IO pin io_gpio_7 +place_pad -row IO_WEST -location [calc_vertical_pad_location 2 5] {sg13g2_IOPad_io_gpio_7} -master sg13g2_IOPadInOut16mA +place_pad -row IO_WEST -location [calc_vertical_pad_location 3 5] {sg13g2_IOPadVss_west_3} -master sg13g2_IOPadVss +place_pad -row IO_WEST -location [calc_vertical_pad_location 4 5] {sg13g2_IOPadVdd_west_4} -master sg13g2_IOPadVdd +# Place Corner Cells and Filler +place_corners sg13g2_Corner + +set iofill { + sg13g2_Filler10000 + sg13g2_Filler4000 + sg13g2_Filler2000 + sg13g2_Filler1000 + sg13g2_Filler400 + sg13g2_Filler200 +} + +place_io_fill -row IO_NORTH {*}$iofill +place_io_fill -row IO_SOUTH {*}$iofill +place_io_fill -row IO_WEST {*}$iofill +place_io_fill -row IO_EAST {*}$iofill + +connect_by_abutment + +place_bondpad -bond bondpad_70x70 sg13g2_IOPad* -offset {5.0 -70.0} + +remove_io_rows diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/pdn.tcl b/flow/designs/ihp-sg13g2/i2c-gpio-expander/pdn.tcl new file mode 100644 index 0000000000..ee9da68466 --- /dev/null +++ b/flow/designs/ihp-sg13g2/i2c-gpio-expander/pdn.tcl @@ -0,0 +1,60 @@ + +# stdcell power pins +add_global_connection -net {VDD} -pin_pattern {^VDD$} -power +add_global_connection -net {VDD} -pin_pattern {^VDDPE$} +add_global_connection -net {VDD} -pin_pattern {^VDDCE$} + +add_global_connection -net {VSS} -pin_pattern {^VSS$} -ground +add_global_connection -net {VSS} -pin_pattern {^VSSE$} + +# padframe core power pins +add_global_connection -net {VDD} -pin_pattern {^vdd$} -power +add_global_connection -net {VSS} -pin_pattern {^vss$} -ground + +global_connect + +# core voltage domain +set_voltage_domain -name {CORE} -power {VDD} -ground {VSS} + +# stdcell grid +define_pdn_grid -name {grid} -voltage_domains {CORE} +add_pdn_ring \ + -grid {grid} \ + -layers {Metal5 TopMetal1} \ + -widths {30.0} \ + -spacings {5.0} \ + -core_offsets {4.5} \ + -connect_to_pads +add_pdn_stripe \ + -grid {grid} \ + -layer {Metal1} \ + -width {0.44} \ + -pitch {7.56} \ + -offset {0} \ + -followpins \ + -extend_to_core_ring +add_pdn_stripe \ + -grid {grid} \ + -layer {Metal5} \ + -width {2.200} \ + -pitch {75.6} \ + -offset {13.600} \ + -extend_to_core_ring +add_pdn_stripe \ + -grid {grid} \ + -layer {TopMetal1} \ + -width {2.200} \ + -pitch {75.6} \ + -offset {13.600} \ + -extend_to_core_ring +add_pdn_stripe \ + -grid {grid} \ + -layer {TopMetal2} \ + -width {2.200} \ + -pitch {75.6} \ + -offset {13.600} \ + -extend_to_core_ring +add_pdn_connect -grid {grid} -layers {Metal1 Metal5} +add_pdn_connect -grid {grid} -layers {Metal5 TopMetal1} +add_pdn_connect -grid {grid} -layers {Metal5 TopMetal2} +add_pdn_connect -grid {grid} -layers {TopMetal1 TopMetal2} diff --git a/flow/designs/ihp-sg13g2/i2c-gpio-expander/sealring.gds.gz b/flow/designs/ihp-sg13g2/i2c-gpio-expander/sealring.gds.gz new file mode 100644 index 0000000000000000000000000000000000000000..31bc865e10659e38a9ada3027e2858b170bd34b6 GIT binary patch literal 2757 zcmb7^eK?fq8o(>7A=M}c6=s~v_C;8$@|jno4IwA1QeRQf}v7)<0`p;&;HeA#%ai6&{%{g8~n3mM?$2 zziY5lmvP5IXY5kQjdfn`!KI|{E&JNN0#}okecZOers%R3ZmHAb-!7sTGX+1|nE zOCh+8H*p{uKq@8=)PQcb8^!AcF@xTfwNv(&k3lwA299;D%R@9BPIg$Blf_Vz!`v__~gBQ9iQ>Tqd*O*S=~p8aGpy+?R+ zB)EM~#K~6TiH@wDL&BkgqP1m@w{m}*GXAu4qf1j+$q(JNe3{8)@v8B0TvwT4cisDA z(_7X=;ZoNx1!O6$+)tCu#v{58tK8xZr2BI z^YBRKs%~wkLRVG9Kmu#(lWr#RTR^F`rCpE3(d>7D+qtLL9=6U;oU-xA;^Wu7tM09| zztcW-JfosuF?@=s&so4$y*=rm9=kyd*Aba_=)<^vru!C-`@o+T`Fy#@8Gd7*uh(^hU`w}Z@vR6;6@~#EnJ1KT&emap8Zwg)Dr$pHb#?Hlh3=lsI zMs#ip6CDyXI~Ce)k--idM{=U%$If_^`jZv9Ea%{}ngmn+-6-*O13F^n*;c;N1TSjF zbtJCvXg8+CpWbduRwYIon9!q5Uz_Q7`we4nR(jBIHw5*K#il9IxNTH}TDa2RJl@5< zt>8ZFNW9m#OgJ>S#Ivp76u4sLSnRo?f?v6DL|2KYC6C7@$?q31j{4WYw<-B#`gep5 zdkvaWB6}SKt~2B%L@;pH({ITSgwFcq`IvY51fcKjMYZ4D2>JFYIhRsxUrqBkZR{=M z3!UO!juaF@Z=mPF`0f(9H;-sSZ|n`aMr$h!gpO2nJf9zBbG4(2QWW$f?Iw{UPQtjl zXrTcDtyRD~B_NLnoEU&Y0<*xM%bkE)AU;r_(Wt*PMGCxJ+rY#4YV6j;iwUTVSPllK z2av)#ayhWT4*l0Mn?yoDb`oZ*i;|Zk>~9sYw**|}fz=G)L;4?>t6F3EOC7GMS_ds$ zhCmhyc)J9o^T0|5uqT08Fjuw4@>QLTaTlS5h6rS(fcHqiJs#M{0F@f`44AXp>Z{sV z(-0k$tdFqG74TLGxX1(A44_8>vtX`jjpeVZ_aTB&T5;lYHBhjTZHy#s@C*+uLP9yzySuR zC4pHmXZ7EDtxUpXx@eC9!q&{SzXTNUz!nCek-#jNv-)pyeJe3W#`cFcN$J`UVb z8abKljG+xU z!g`w0(ZtQd%|&0lFqMLAS&x5@hOT(#k>Noxya192ESopRU$=E>x9fqPBkZJ_I3AW*mv6~W%w}?)+isM z#tPPE5cA*3#=^gR$#S)wk(9pDmYWqDT8;C0Y9siyM5r|rK60b4sqG#g#jfiVE-N%k zrD)Z&wE6`L!qj$I68g#}M>xF!30vdpnGQ9QuS%)R>H|Fh;^cB={OPJ~7j6+_QwM-Zf|s2O`ZxLk46?f&-JVA}IykCAEj! zHOWVsJ7tn{-qUvY&^JmRlm*|U^rc(6M<(y+zAz#zFumJYmD9oMdjXjf35~5lT!QY_X zQc;zDj#kUBBL3L4y6rM#(|Ke{t8hswcWFIqe!pOTRAho37Uc6e<(C&!-yy1X zD{8+w@JM!-ybEMu&y}0%BrlSp+8zmZ~3MF)vF!n{LfKYG3Oa7X@Aow`9o}R z+{c(&=G#2m&CV^2+n5h(-#)2Or}cj(Ue?pYA|vHZpm;PJi1%b=ti~Xt=JTb$BLoYl LkOuXkdGr1QIXFL6 literal 0 HcmV?d00001