/* SPDX-License-Identifier: GPL-2.0-only */ /* * Microsemi Switchtec PCIe Driver * Copyright (c) 2017, Microsemi Corporation */ #ifndef _SWITCHTEC_H #define _SWITCHTEC_H #include <linux/pci.h> #include <linux/cdev.h> #define SWITCHTEC_MRPC_PAYLOAD_SIZE 1024 #define SWITCHTEC_MAX_PFF_CSR 255 #define SWITCHTEC_EVENT_OCCURRED BIT(0) #define SWITCHTEC_EVENT_CLEAR BIT(0) #define SWITCHTEC_EVENT_EN_LOG BIT(1) #define SWITCHTEC_EVENT_EN_CLI BIT(2) #define SWITCHTEC_EVENT_EN_IRQ BIT(3) #define SWITCHTEC_EVENT_FATAL BIT(4) #define SWITCHTEC_EVENT_NOT_SUPP BIT(31) #define SWITCHTEC_DMA_MRPC_EN BIT(0) #define MRPC_GAS_READ 0x29 #define MRPC_GAS_WRITE 0x87 #define MRPC_CMD_ID(x) ((x) & 0xffff) enum { SWITCHTEC_GAS_MRPC_OFFSET = 0x0000, SWITCHTEC_GAS_TOP_CFG_OFFSET = 0x1000, SWITCHTEC_GAS_SW_EVENT_OFFSET = 0x1800, SWITCHTEC_GAS_SYS_INFO_OFFSET = 0x2000, SWITCHTEC_GAS_FLASH_INFO_OFFSET = 0x2200, SWITCHTEC_GAS_PART_CFG_OFFSET = 0x4000, SWITCHTEC_GAS_NTB_OFFSET = 0x10000, SWITCHTEC_GAS_PFF_CSR_OFFSET = 0x134000, }; enum switchtec_gen { SWITCHTEC_GEN3, SWITCHTEC_GEN4, }; struct mrpc_regs { u8 input_data[SWITCHTEC_MRPC_PAYLOAD_SIZE]; u8 output_data[SWITCHTEC_MRPC_PAYLOAD_SIZE]; u32 cmd; u32 status; u32 ret_value; u32 dma_en; u64 dma_addr; u32 dma_vector; u32 dma_ver; } __packed; enum mrpc_status { SWITCHTEC_MRPC_STATUS_INPROGRESS = 1, SWITCHTEC_MRPC_STATUS_DONE = 2, SWITCHTEC_MRPC_STATUS_ERROR = 0xFF, SWITCHTEC_MRPC_STATUS_INTERRUPTED = 0x100, }; struct sw_event_regs { u64 event_report_ctrl; u64 reserved1; u64 part_event_bitmap; u64 reserved2; u32 global_summary; u32 reserved3[3]; u32 stack_error_event_hdr; u32 stack_error_event_data; u32 reserved4[4]; u32 ppu_error_event_hdr; u32 ppu_error_event_data; u32 reserved5[4]; u32 isp_error_event_hdr; u32 isp_error_event_data; u32 reserved6[4]; u32 sys_reset_event_hdr; u32 reserved7[5]; u32 fw_exception_hdr; u32 reserved8[5]; u32 fw_nmi_hdr; u32 reserved9[5]; u32 fw_non_fatal_hdr; u32 reserved10[5]; u32 fw_fatal_hdr; u32 reserved11[5]; u32 twi_mrpc_comp_hdr; u32 twi_mrpc_comp_data; u32 reserved12[4]; u32 twi_mrpc_comp_async_hdr; u32 twi_mrpc_comp_async_data; u32 reserved13[4]; u32 cli_mrpc_comp_hdr; u32 cli_mrpc_comp_data; u32 reserved14[4]; u32 cli_mrpc_comp_async_hdr; u32 cli_mrpc_comp_async_data; u32 reserved15[4]; u32 gpio_interrupt_hdr; u32 gpio_interrupt_data; u32 reserved16[4]; u32 gfms_event_hdr; u32 gfms_event_data; u32 reserved17[4]; } __packed; enum { SWITCHTEC_GEN3_CFG0_RUNNING = 0x04, SWITCHTEC_GEN3_CFG1_RUNNING = 0x05, SWITCHTEC_GEN3_IMG0_RUNNING = 0x03, SWITCHTEC_GEN3_IMG1_RUNNING = 0x07, }; enum { SWITCHTEC_GEN4_MAP0_RUNNING = 0x00, SWITCHTEC_GEN4_MAP1_RUNNING = 0x01, SWITCHTEC_GEN4_KEY0_RUNNING = 0x02, SWITCHTEC_GEN4_KEY1_RUNNING = 0x03, SWITCHTEC_GEN4_BL2_0_RUNNING = 0x04, SWITCHTEC_GEN4_BL2_1_RUNNING = 0x05, SWITCHTEC_GEN4_CFG0_RUNNING = 0x06, SWITCHTEC_GEN4_CFG1_RUNNING = 0x07, SWITCHTEC_GEN4_IMG0_RUNNING = 0x08, SWITCHTEC_GEN4_IMG1_RUNNING = 0x09, }; enum { SWITCHTEC_GEN4_KEY0_ACTIVE = 0, SWITCHTEC_GEN4_KEY1_ACTIVE = 1, SWITCHTEC_GEN4_BL2_0_ACTIVE = 0, SWITCHTEC_GEN4_BL2_1_ACTIVE = 1, SWITCHTEC_GEN4_CFG0_ACTIVE = 0, SWITCHTEC_GEN4_CFG1_ACTIVE = 1, SWITCHTEC_GEN4_IMG0_ACTIVE = 0, SWITCHTEC_GEN4_IMG1_ACTIVE = 1, }; struct sys_info_regs_gen3 { u32 reserved1; u32 vendor_table_revision; u32 table_format_version; u32 partition_id; u32 cfg_file_fmt_version; u16 cfg_running; u16 img_running; u32 reserved2[57]; char vendor_id[8]; char product_id[16]; char product_revision[4]; char component_vendor[8]; u16 component_id; u8 component_revision; } __packed; struct sys_info_regs_gen4 { u16 gas_layout_ver; u8 evlist_ver; u8 reserved1; u16 mgmt_cmd_set_ver; u16 fabric_cmd_set_ver; u32 reserved2[2]; u8 mrpc_uart_ver; u8 mrpc_twi_ver; u8 mrpc_eth_ver; u8 mrpc_inband_ver; u32 reserved3[7]; u32 fw_update_tmo; u32 xml_version_cfg; u32 xml_version_img; u32 partition_id; u16 bl2_running; u16 cfg_running; u16 img_running; u16 key_running; u32 reserved4[43]; u32 vendor_seeprom_twi; u32 vendor_table_revision; u32 vendor_specific_info[2]; u16 p2p_vendor_id; u16 p2p_device_id; u8 p2p_revision_id; u8 reserved5[3]; u32 p2p_class_id; u16 subsystem_vendor_id; u16 subsystem_id; u32 p2p_serial_number[2]; u8 mac_addr[6]; u8 reserved6[2]; u32 reserved7[3]; char vendor_id[8]; char product_id[24]; char product_revision[2]; u16 reserved8; } __packed; struct sys_info_regs { u32 device_id; u32 device_version; u32 firmware_version; union { struct sys_info_regs_gen3 gen3; struct sys_info_regs_gen4 gen4; }; } __packed; struct partition_info { u32 address; u32 length; }; struct flash_info_regs_gen3 { u32 flash_part_map_upd_idx; struct active_partition_info_gen3 { u32 address; u32 build_version; u32 build_string; } active_img; struct active_partition_info_gen3 active_cfg; struct active_partition_info_gen3 inactive_img; struct active_partition_info_gen3 inactive_cfg; u32 flash_length; struct partition_info cfg0; struct partition_info cfg1; struct partition_info img0; struct partition_info img1; struct partition_info nvlog; struct partition_info vendor[8]; }; struct flash_info_regs_gen4 { u32 flash_address; u32 flash_length; struct active_partition_info_gen4 { unsigned char bl2; unsigned char cfg; unsigned char img; unsigned char key; } active_flag; u32 reserved[3]; struct partition_info map0; struct partition_info map1; struct partition_info key0; struct partition_info key1; struct partition_info bl2_0; struct partition_info bl2_1; struct partition_info cfg0; struct partition_info cfg1; struct partition_info img0; struct partition_info img1; struct partition_info nvlog; struct partition_info vendor[8]; }; struct flash_info_regs { union { struct flash_info_regs_gen3 gen3; struct flash_info_regs_gen4 gen4; }; }; enum { SWITCHTEC_NTB_REG_INFO_OFFSET = 0x0000, SWITCHTEC_NTB_REG_CTRL_OFFSET = 0x4000, SWITCHTEC_NTB_REG_DBMSG_OFFSET = 0x64000, }; struct ntb_info_regs { u8 partition_count; u8 partition_id; u16 reserved1; u64 ep_map; u16 requester_id; u16 reserved2; u32 reserved3[4]; struct nt_partition_info { u32 xlink_enabled; u32 target_part_low; u32 target_part_high; u32 reserved; } ntp_info[48]; } __packed; struct part_cfg_regs { u32 status; u32 state; u32 port_cnt; u32 usp_port_mode; u32 usp_pff_inst_id; u32 vep_pff_inst_id; u32 dsp_pff_inst_id[47]; u32 reserved1[11]; u16 vep_vector_number; u16 usp_vector_number; u32 port_event_bitmap; u32 reserved2[3]; u32 part_event_summary; u32 reserved3[3]; u32 part_reset_hdr; u32 part_reset_data[5]; u32 mrpc_comp_hdr; u32 mrpc_comp_data[5]; u32 mrpc_comp_async_hdr; u32 mrpc_comp_async_data[5]; u32 dyn_binding_hdr; u32 dyn_binding_data[5]; u32 intercomm_notify_hdr; u32 intercomm_notify_data[5]; u32 reserved4[153]; } __packed; enum { NTB_CTRL_PART_OP_LOCK = 0x1, NTB_CTRL_PART_OP_CFG = 0x2, NTB_CTRL_PART_OP_RESET = 0x3, NTB_CTRL_PART_STATUS_NORMAL = 0x1, NTB_CTRL_PART_STATUS_LOCKED = 0x2, NTB_CTRL_PART_STATUS_LOCKING = 0x3, NTB_CTRL_PART_STATUS_CONFIGURING = 0x4, NTB_CTRL_PART_STATUS_RESETTING = 0x5, NTB_CTRL_BAR_VALID = 1 << 0, NTB_CTRL_BAR_DIR_WIN_EN = 1 << 4, NTB_CTRL_BAR_LUT_WIN_EN = 1 << 5, NTB_CTRL_REQ_ID_EN = 1 << 0, NTB_CTRL_LUT_EN = 1 << 0, }; struct ntb_ctrl_regs { u32 partition_status; u32 partition_op; u32 partition_ctrl; u32 bar_setup; u32 bar_error; u16 lut_table_entries; u16 lut_table_offset; u32 lut_error; u16 req_id_table_size; u16 req_id_table_offset; u32 req_id_error; u32 reserved1[7]; struct { u32 ctl; u32 win_size; u64 xlate_addr; } bar_entry[6]; struct { u32 win_size; u32 reserved[3]; } bar_ext_entry[6]; u32 reserved2[192]; u32 req_id_table[512]; u32 reserved3[256]; u64 lut_entry[512]; } __packed; #define NTB_DBMSG_IMSG_STATUS BIT_ULL(32) #define NTB_DBMSG_IMSG_MASK BIT_ULL(40) struct ntb_dbmsg_regs { u32 reserved1[1024]; u64 odb; u64 odb_mask; u64 idb; u64 idb_mask; u8 idb_vec_map[64]; u32 msg_map; u32 reserved2; struct { u32 msg; u32 status; } omsg[4]; struct { u32 msg; u8 status; u8 mask; u8 src; u8 reserved; } imsg[4]; u8 reserved3[3928]; u8 msix_table[1024]; u8 reserved4[3072]; u8 pba[24]; u8 reserved5[4072]; } __packed; enum { SWITCHTEC_PART_CFG_EVENT_RESET = 1 << 0, SWITCHTEC_PART_CFG_EVENT_MRPC_CMP = 1 << 1, SWITCHTEC_PART_CFG_EVENT_MRPC_ASYNC_CMP = 1 << 2, SWITCHTEC_PART_CFG_EVENT_DYN_PART_CMP = 1 << 3, }; struct pff_csr_regs { u16 vendor_id; u16 device_id; u16 pcicmd; u16 pcists; u32 pci_class; u32 pci_opts; union { u32 pci_bar[6]; u64 pci_bar64[3]; }; u32 pci_cardbus; u32 pci_subsystem_id; u32 pci_expansion_rom; u32 pci_cap_ptr; u32 reserved1; u32 pci_irq; u32 pci_cap_region[48]; u32 pcie_cap_region[448]; u32 indirect_gas_window[128]; u32 indirect_gas_window_off; u32 reserved[127]; u32 pff_event_summary; u32 reserved2[3]; u32 aer_in_p2p_hdr; u32 aer_in_p2p_data[5]; u32 aer_in_vep_hdr; u32 aer_in_vep_data[5]; u32 dpc_hdr; u32 dpc_data[5]; u32 cts_hdr; u32 cts_data[5]; u32 uec_hdr; u32 uec_data[5]; u32 hotplug_hdr; u32 hotplug_data[5]; u32 ier_hdr; u32 ier_data[5]; u32 threshold_hdr; u32 threshold_data[5]; u32 power_mgmt_hdr; u32 power_mgmt_data[5]; u32 tlp_throttling_hdr; u32 tlp_throttling_data[5]; u32 force_speed_hdr; u32 force_speed_data[5]; u32 credit_timeout_hdr; u32 credit_timeout_data[5]; u32 link_state_hdr; u32 link_state_data[5]; u32 reserved4[174]; } __packed; struct switchtec_ntb; struct dma_mrpc_output { u32 status; u32 cmd_id; u32 rtn_code; u32 output_size; u8 data[SWITCHTEC_MRPC_PAYLOAD_SIZE]; }; struct switchtec_dev { struct pci_dev *pdev; struct device dev; struct cdev cdev; enum switchtec_gen gen; int partition; int partition_count; int pff_csr_count; char pff_local[SWITCHTEC_MAX_PFF_CSR]; void __iomem *mmio; struct mrpc_regs __iomem *mmio_mrpc; struct sw_event_regs __iomem *mmio_sw_event; struct sys_info_regs __iomem *mmio_sys_info; struct flash_info_regs __iomem *mmio_flash_info; struct ntb_info_regs __iomem *mmio_ntb; struct part_cfg_regs __iomem *mmio_part_cfg; struct part_cfg_regs __iomem *mmio_part_cfg_all; struct pff_csr_regs __iomem *mmio_pff_csr; /* * The mrpc mutex must be held when accessing the other * mrpc_ fields, alive flag and stuser->state field */ struct mutex mrpc_mutex; struct list_head mrpc_queue; int mrpc_busy; struct work_struct mrpc_work; struct delayed_work mrpc_timeout; bool alive; wait_queue_head_t event_wq; atomic_t event_cnt; struct work_struct link_event_work; void (*link_notifier)(struct switchtec_dev *stdev); u8 link_event_count[SWITCHTEC_MAX_PFF_CSR]; struct switchtec_ntb *sndev; struct dma_mrpc_output *dma_mrpc; dma_addr_t dma_mrpc_dma_addr; }; static inline struct switchtec_dev *to_stdev(struct device *dev) { return container_of(dev, struct switchtec_dev, dev); } extern struct class *switchtec_class; #endif