Skip to content

Commit

Permalink
loongarch: fix HT_RX_INT_TRANS register
Browse files Browse the repository at this point in the history
On machines with legacy firmware with BPI01000 version, the
HT_RX_INT_TRANS register on the node 5, i.e. the node connected with the
second LS7A bridge chip, is not initialized correctly, causing the
failure of the delivery of interrupts from PCIe devices connected to the
second LS7A bridge chip.

This patch fixes this by correctly pointing the HT_RX_INT_TRANS register
to the EXT_IOI_SEND_OFF register on the corresponding node.
  • Loading branch information
shankerwangmiao authored and MingcongBai committed Aug 14, 2024
1 parent cec549a commit df06a5a
Showing 1 changed file with 24 additions and 0 deletions.
24 changes: 24 additions & 0 deletions arch/loongarch/kernel/legacy_boot.c
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,16 @@ enum bpi_mem_type {
#define LS7A_DMA_NODE_ID_OFFSET_SHF 8
#define LS7A_DMA_NODE_ID_OFFSET_MASK GENMASK(12, 8)

/* Section 14.4.1, Page 100 of Loongson-3A5000 User Manual v1.03 */
#define HT_CTRL_CFG_OFF 0xfdfb000000ull
/* Section 14.5.34, Page 141 of Loongson-3A5000 User Manual v1.03 */
#define HT_CTRL_HT_RX_INT_TRANS_LO_OFF 0x270
#define HT_CTRL_HT_RX_INT_TRANS_HI_OFF 0x274
#define HT_CTRL_HT_RX_INT_TRANS_INT_TRANS_ALLOW BIT(30)

/* Section 11.2.3, Page 61 of Loongson-3A5000 User Manual v1.03 */
#define EXT_IOI_SEND_OFF 0x1140

struct loongarch_bpi_mem_map {
struct loongarch_bpi_ext_hdr header; /*{"M", "E", "M"}*/
u8 map_count;
Expand Down Expand Up @@ -615,6 +625,20 @@ static void __init init_acpi_arch_os_table_override (struct acpi_table_header *e
writel(dma_cfg, dma_node_id_addr);
}
}

// Override HT_RX_INT_TRANS
for(int i = 0; i < io_apic_count; i++){
unsigned int node = eio_pics[i].node;
void __iomem *ht_rx_int_trans_hi = (void __iomem *) TO_UNCACHE(nid_to_addrbase(node) + HT1LO_OFFSET + HT_CTRL_CFG_OFF + HT_CTRL_HT_RX_INT_TRANS_HI_OFF);
void __iomem *ht_rx_int_trans_lo = (void __iomem *) TO_UNCACHE(nid_to_addrbase(node) + HT1LO_OFFSET + HT_CTRL_CFG_OFF + HT_CTRL_HT_RX_INT_TRANS_LO_OFF);
u64 ext_ioi_addr = nid_to_addrbase(node) + LOONGSON_REG_BASE + EXT_IOI_SEND_OFF;
u64 ht_rx_int_trans_cfg_orig = ((u64)readl(ht_rx_int_trans_hi) << 32) | readl(ht_rx_int_trans_lo);
u32 ht_rx_int_trans_cfg_hi = HT_CTRL_HT_RX_INT_TRANS_INT_TRANS_ALLOW | (ext_ioi_addr >> 32);
u32 ht_rx_int_trans_cfg_lo = ext_ioi_addr & GENMASK(31, 0);
pr_info("BPI: HT controller on node %u RX_INT_TRANS is 0x%llx, will set to 0x%llx\n", node, ht_rx_int_trans_cfg_orig, ((u64)ht_rx_int_trans_cfg_hi << 32) | ht_rx_int_trans_cfg_lo);
writel(ht_rx_int_trans_cfg_hi, ht_rx_int_trans_hi);
writel(ht_rx_int_trans_cfg_lo, ht_rx_int_trans_lo);
}
}

void acpi_arch_os_table_override (struct acpi_table_header *existing_table, struct acpi_table_header **new_table){
Expand Down

0 comments on commit df06a5a

Please sign in to comment.