]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
LoongArch: KVM: Add EIOINTC read and write functions
authorXianglai Li <lixianglai@loongson.cn>
Wed, 13 Nov 2024 08:18:27 +0000 (16:18 +0800)
committerHuacai Chen <chenhuacai@loongson.cn>
Wed, 13 Nov 2024 08:18:27 +0000 (16:18 +0800)
Add implementation of EIOINTC interrupt controller's address space read
and write function simulation.

Signed-off-by: Tianrui Zhao <zhaotianrui@loongson.cn>
Signed-off-by: Xianglai Li <lixianglai@loongson.cn>
Signed-off-by: Huacai Chen <chenhuacai@loongson.cn>
arch/loongarch/include/asm/kvm_eiointc.h
arch/loongarch/include/asm/kvm_host.h
arch/loongarch/kvm/intc/eiointc.c

index ed65de5a8168a3fed5015a8f7c6deb99c73fa7f1..a3a40aba8acf2985896bfff5f548be90b05b0a9d 100644 (file)
 #define EIOINTC_BASE                   0x1400
 #define EIOINTC_SIZE                   0x900
 
+#define EIOINTC_NODETYPE_START         0xa0
+#define EIOINTC_NODETYPE_END           0xbf
+#define EIOINTC_IPMAP_START            0xc0
+#define EIOINTC_IPMAP_END              0xc7
+#define EIOINTC_ENABLE_START           0x200
+#define EIOINTC_ENABLE_END             0x21f
+#define EIOINTC_BOUNCE_START           0x280
+#define EIOINTC_BOUNCE_END             0x29f
+#define EIOINTC_ISR_START              0x300
+#define EIOINTC_ISR_END                        0x31f
+#define EIOINTC_COREISR_START          0x400
+#define EIOINTC_COREISR_END            0x41f
+#define EIOINTC_COREMAP_START          0x800
+#define EIOINTC_COREMAP_END            0x8ff
+
 #define EIOINTC_VIRT_BASE              (0x40000000)
 #define EIOINTC_VIRT_SIZE              (0x1000)
 
+#define EIOINTC_VIRT_FEATURES          (0x0)
+#define EIOINTC_HAS_VIRT_EXTENSION     (0)
+#define EIOINTC_HAS_ENABLE_OPTION      (1)
+#define EIOINTC_HAS_INT_ENCODE         (2)
+#define EIOINTC_HAS_CPU_ENCODE         (3)
+#define EIOINTC_VIRT_HAS_FEATURES      ((1U << EIOINTC_HAS_VIRT_EXTENSION) \
+                                       | (1U << EIOINTC_HAS_ENABLE_OPTION) \
+                                       | (1U << EIOINTC_HAS_INT_ENCODE)    \
+                                       | (1U << EIOINTC_HAS_CPU_ENCODE))
+#define EIOINTC_VIRT_CONFIG            (0x4)
+#define EIOINTC_ENABLE                 (1)
+#define EIOINTC_ENABLE_INT_ENCODE      (2)
+#define EIOINTC_ENABLE_CPU_ENCODE      (3)
+
 #define LOONGSON_IP_NUM                        8
 
 struct loongarch_eiointc {
@@ -89,5 +118,6 @@ struct loongarch_eiointc {
 };
 
 int kvm_loongarch_register_eiointc_device(void);
+void eiointc_set_irq(struct loongarch_eiointc *s, int irq, int level);
 
 #endif /* __ASM_KVM_EIOINTC_H */
index 2d0476f05148af319a7dac88924cb2061f2f4503..a63c2bf6fae0cc1d572148868278da9b88f989bb 100644 (file)
@@ -48,6 +48,8 @@ struct kvm_vm_stat {
        u64 hugepages;
        u64 ipi_read_exits;
        u64 ipi_write_exits;
+       u64 eiointc_read_exits;
+       u64 eiointc_write_exits;
 };
 
 struct kvm_vcpu_stat {
index 10afa6163643219751c5e60b2c837d26e5adca3a..7369c23be036a42056ccbdb111a685470061b067 100644 (file)
 #include <asm/kvm_vcpu.h>
 #include <linux/count_zeros.h>
 
+static void eiointc_set_sw_coreisr(struct loongarch_eiointc *s)
+{
+       int ipnum, cpu, irq_index, irq_mask, irq;
+
+       for (irq = 0; irq < EIOINTC_IRQS; irq++) {
+               ipnum = s->ipmap.reg_u8[irq / 32];
+               if (!(s->status & BIT(EIOINTC_ENABLE_INT_ENCODE))) {
+                       ipnum = count_trailing_zeros(ipnum);
+                       ipnum = (ipnum >= 0 && ipnum < 4) ? ipnum : 0;
+               }
+               irq_index = irq / 32;
+               irq_mask = BIT(irq & 0x1f);
+
+               cpu = s->coremap.reg_u8[irq];
+               if (!!(s->coreisr.reg_u32[cpu][irq_index] & irq_mask))
+                       set_bit(irq, s->sw_coreisr[cpu][ipnum]);
+               else
+                       clear_bit(irq, s->sw_coreisr[cpu][ipnum]);
+       }
+}
+
+static void eiointc_update_irq(struct loongarch_eiointc *s, int irq, int level)
+{
+       int ipnum, cpu, found, irq_index, irq_mask;
+       struct kvm_vcpu *vcpu;
+       struct kvm_interrupt vcpu_irq;
+
+       ipnum = s->ipmap.reg_u8[irq / 32];
+       if (!(s->status & BIT(EIOINTC_ENABLE_INT_ENCODE))) {
+               ipnum = count_trailing_zeros(ipnum);
+               ipnum = (ipnum >= 0 && ipnum < 4) ? ipnum : 0;
+       }
+
+       cpu = s->sw_coremap[irq];
+       vcpu = kvm_get_vcpu(s->kvm, cpu);
+       irq_index = irq / 32;
+       irq_mask = BIT(irq & 0x1f);
+
+       if (level) {
+               /* if not enable return false */
+               if (((s->enable.reg_u32[irq_index]) & irq_mask) == 0)
+                       return;
+               s->coreisr.reg_u32[cpu][irq_index] |= irq_mask;
+               found = find_first_bit(s->sw_coreisr[cpu][ipnum], EIOINTC_IRQS);
+               set_bit(irq, s->sw_coreisr[cpu][ipnum]);
+       } else {
+               s->coreisr.reg_u32[cpu][irq_index] &= ~irq_mask;
+               clear_bit(irq, s->sw_coreisr[cpu][ipnum]);
+               found = find_first_bit(s->sw_coreisr[cpu][ipnum], EIOINTC_IRQS);
+       }
+
+       if (found < EIOINTC_IRQS)
+               return; /* other irq is handling, needn't update parent irq */
+
+       vcpu_irq.irq = level ? (INT_HWI0 + ipnum) : -(INT_HWI0 + ipnum);
+       kvm_vcpu_ioctl_interrupt(vcpu, &vcpu_irq);
+}
+
+static inline void eiointc_update_sw_coremap(struct loongarch_eiointc *s,
+                                       int irq, void *pvalue, u32 len, bool notify)
+{
+       int i, cpu;
+       u64 val = *(u64 *)pvalue;
+
+       for (i = 0; i < len; i++) {
+               cpu = val & 0xff;
+               val = val >> 8;
+
+               if (!(s->status & BIT(EIOINTC_ENABLE_CPU_ENCODE))) {
+                       cpu = ffs(cpu) - 1;
+                       cpu = (cpu >= 4) ? 0 : cpu;
+               }
+
+               if (s->sw_coremap[irq + i] == cpu)
+                       continue;
+
+               if (notify && test_bit(irq + i, (unsigned long *)s->isr.reg_u8)) {
+                       /* lower irq at old cpu and raise irq at new cpu */
+                       eiointc_update_irq(s, irq + i, 0);
+                       s->sw_coremap[irq + i] = cpu;
+                       eiointc_update_irq(s, irq + i, 1);
+               } else {
+                       s->sw_coremap[irq + i] = cpu;
+               }
+       }
+}
+
+void eiointc_set_irq(struct loongarch_eiointc *s, int irq, int level)
+{
+       unsigned long flags;
+       unsigned long *isr = (unsigned long *)s->isr.reg_u8;
+
+       level ? set_bit(irq, isr) : clear_bit(irq, isr);
+       spin_lock_irqsave(&s->lock, flags);
+       eiointc_update_irq(s, irq, level);
+       spin_unlock_irqrestore(&s->lock, flags);
+}
+
+static inline void eiointc_enable_irq(struct kvm_vcpu *vcpu,
+               struct loongarch_eiointc *s, int index, u8 mask, int level)
+{
+       u8 val;
+       int irq;
+
+       val = mask & s->isr.reg_u8[index];
+       irq = ffs(val);
+       while (irq != 0) {
+               /*
+                * enable bit change from 0 to 1,
+                * need to update irq by pending bits
+                */
+               eiointc_update_irq(s, irq - 1 + index * 8, level);
+               val &= ~BIT(irq - 1);
+               irq = ffs(val);
+       }
+}
+
+static int loongarch_eiointc_readb(struct kvm_vcpu *vcpu, struct loongarch_eiointc *s,
+                               gpa_t addr, int len, void *val)
+{
+       int index, ret = 0;
+       u8 data = 0;
+       gpa_t offset;
+
+       offset = addr - EIOINTC_BASE;
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = offset - EIOINTC_NODETYPE_START;
+               data = s->nodetype.reg_u8[index];
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               index = offset - EIOINTC_IPMAP_START;
+               data = s->ipmap.reg_u8[index];
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = offset - EIOINTC_ENABLE_START;
+               data = s->enable.reg_u8[index];
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               index = offset - EIOINTC_BOUNCE_START;
+               data = s->bounce.reg_u8[index];
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = offset - EIOINTC_COREISR_START;
+               data = s->coreisr.reg_u8[vcpu->vcpu_id][index];
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               index = offset - EIOINTC_COREMAP_START;
+               data = s->coremap.reg_u8[index];
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+       *(u8 *)val = data;
+
+       return ret;
+}
+
+static int loongarch_eiointc_readw(struct kvm_vcpu *vcpu, struct loongarch_eiointc *s,
+                               gpa_t addr, int len, void *val)
+{
+       int index, ret = 0;
+       u16 data = 0;
+       gpa_t offset;
+
+       offset = addr - EIOINTC_BASE;
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 1;
+               data = s->nodetype.reg_u16[index];
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               index = (offset - EIOINTC_IPMAP_START) >> 1;
+               data = s->ipmap.reg_u16[index];
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 1;
+               data = s->enable.reg_u16[index];
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               index = (offset - EIOINTC_BOUNCE_START) >> 1;
+               data = s->bounce.reg_u16[index];
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 1;
+               data = s->coreisr.reg_u16[vcpu->vcpu_id][index];
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               index = (offset - EIOINTC_COREMAP_START) >> 1;
+               data = s->coremap.reg_u16[index];
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+       *(u16 *)val = data;
+
+       return ret;
+}
+
+static int loongarch_eiointc_readl(struct kvm_vcpu *vcpu, struct loongarch_eiointc *s,
+                               gpa_t addr, int len, void *val)
+{
+       int index, ret = 0;
+       u32 data = 0;
+       gpa_t offset;
+
+       offset = addr - EIOINTC_BASE;
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 2;
+               data = s->nodetype.reg_u32[index];
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               index = (offset - EIOINTC_IPMAP_START) >> 2;
+               data = s->ipmap.reg_u32[index];
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 2;
+               data = s->enable.reg_u32[index];
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               index = (offset - EIOINTC_BOUNCE_START) >> 2;
+               data = s->bounce.reg_u32[index];
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 2;
+               data = s->coreisr.reg_u32[vcpu->vcpu_id][index];
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               index = (offset - EIOINTC_COREMAP_START) >> 2;
+               data = s->coremap.reg_u32[index];
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+       *(u32 *)val = data;
+
+       return ret;
+}
+
+static int loongarch_eiointc_readq(struct kvm_vcpu *vcpu, struct loongarch_eiointc *s,
+                               gpa_t addr, int len, void *val)
+{
+       int index, ret = 0;
+       u64 data = 0;
+       gpa_t offset;
+
+       offset = addr - EIOINTC_BASE;
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 3;
+               data = s->nodetype.reg_u64[index];
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               index = (offset - EIOINTC_IPMAP_START) >> 3;
+               data = s->ipmap.reg_u64;
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 3;
+               data = s->enable.reg_u64[index];
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               index = (offset - EIOINTC_BOUNCE_START) >> 3;
+               data = s->bounce.reg_u64[index];
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 3;
+               data = s->coreisr.reg_u64[vcpu->vcpu_id][index];
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               index = (offset - EIOINTC_COREMAP_START) >> 3;
+               data = s->coremap.reg_u64[index];
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+       *(u64 *)val = data;
+
+       return ret;
+}
+
 static int kvm_eiointc_read(struct kvm_vcpu *vcpu,
                        struct kvm_io_device *dev,
                        gpa_t addr, int len, void *val)
 {
-       return 0;
+       int ret = -EINVAL;
+       unsigned long flags;
+       struct loongarch_eiointc *eiointc = vcpu->kvm->arch.eiointc;
+
+       if (!eiointc) {
+               kvm_err("%s: eiointc irqchip not valid!\n", __func__);
+               return -EINVAL;
+       }
+
+       vcpu->kvm->stat.eiointc_read_exits++;
+       spin_lock_irqsave(&eiointc->lock, flags);
+       switch (len) {
+       case 1:
+               ret = loongarch_eiointc_readb(vcpu, eiointc, addr, len, val);
+               break;
+       case 2:
+               ret = loongarch_eiointc_readw(vcpu, eiointc, addr, len, val);
+               break;
+       case 4:
+               ret = loongarch_eiointc_readl(vcpu, eiointc, addr, len, val);
+               break;
+       case 8:
+               ret = loongarch_eiointc_readq(vcpu, eiointc, addr, len, val);
+               break;
+       default:
+               WARN_ONCE(1, "%s: Abnormal address access: addr 0x%llx, size %d\n",
+                                               __func__, addr, len);
+       }
+       spin_unlock_irqrestore(&eiointc->lock, flags);
+
+       return ret;
+}
+
+static int loongarch_eiointc_writeb(struct kvm_vcpu *vcpu,
+                               struct loongarch_eiointc *s,
+                               gpa_t addr, int len, const void *val)
+{
+       int index, irq, bits, ret = 0;
+       u8 cpu;
+       u8 data, old_data;
+       u8 coreisr, old_coreisr;
+       gpa_t offset;
+
+       data = *(u8 *)val;
+       offset = addr - EIOINTC_BASE;
+
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START);
+               s->nodetype.reg_u8[index] = data;
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               /*
+                * ipmap cannot be set at runtime, can be set only at the beginning
+                * of irqchip driver, need not update upper irq level
+                */
+               index = (offset - EIOINTC_IPMAP_START);
+               s->ipmap.reg_u8[index] = data;
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START);
+               old_data = s->enable.reg_u8[index];
+               s->enable.reg_u8[index] = data;
+               /*
+                * 1: enable irq.
+                * update irq when isr is set.
+                */
+               data = s->enable.reg_u8[index] & ~old_data & s->isr.reg_u8[index];
+               eiointc_enable_irq(vcpu, s, index, data, 1);
+               /*
+                * 0: disable irq.
+                * update irq when isr is set.
+                */
+               data = ~s->enable.reg_u8[index] & old_data & s->isr.reg_u8[index];
+               eiointc_enable_irq(vcpu, s, index, data, 0);
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               /* do not emulate hw bounced irq routing */
+               index = offset - EIOINTC_BOUNCE_START;
+               s->bounce.reg_u8[index] = data;
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START);
+               /* use attrs to get current cpu index */
+               cpu = vcpu->vcpu_id;
+               coreisr = data;
+               old_coreisr = s->coreisr.reg_u8[cpu][index];
+               /* write 1 to clear interrupt */
+               s->coreisr.reg_u8[cpu][index] = old_coreisr & ~coreisr;
+               coreisr &= old_coreisr;
+               bits = sizeof(data) * 8;
+               irq = find_first_bit((void *)&coreisr, bits);
+               while (irq < bits) {
+                       eiointc_update_irq(s, irq + index * bits, 0);
+                       bitmap_clear((void *)&coreisr, irq, 1);
+                       irq = find_first_bit((void *)&coreisr, bits);
+               }
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               irq = offset - EIOINTC_COREMAP_START;
+               index = irq;
+               s->coremap.reg_u8[index] = data;
+               eiointc_update_sw_coremap(s, irq, (void *)&data, sizeof(data), true);
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+
+       return ret;
+}
+
+static int loongarch_eiointc_writew(struct kvm_vcpu *vcpu,
+                               struct loongarch_eiointc *s,
+                               gpa_t addr, int len, const void *val)
+{
+       int i, index, irq, bits, ret = 0;
+       u8 cpu;
+       u16 data, old_data;
+       u16 coreisr, old_coreisr;
+       gpa_t offset;
+
+       data = *(u16 *)val;
+       offset = addr - EIOINTC_BASE;
+
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 1;
+               s->nodetype.reg_u16[index] = data;
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               /*
+                * ipmap cannot be set at runtime, can be set only at the beginning
+                * of irqchip driver, need not update upper irq level
+                */
+               index = (offset - EIOINTC_IPMAP_START) >> 1;
+               s->ipmap.reg_u16[index] = data;
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 1;
+               old_data = s->enable.reg_u32[index];
+               s->enable.reg_u16[index] = data;
+               /*
+                * 1: enable irq.
+                * update irq when isr is set.
+                */
+               data = s->enable.reg_u16[index] & ~old_data & s->isr.reg_u16[index];
+               index = index << 1;
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index + i, mask, 1);
+               }
+               /*
+                * 0: disable irq.
+                * update irq when isr is set.
+                */
+               data = ~s->enable.reg_u16[index] & old_data & s->isr.reg_u16[index];
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index, mask, 0);
+               }
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               /* do not emulate hw bounced irq routing */
+               index = (offset - EIOINTC_BOUNCE_START) >> 1;
+               s->bounce.reg_u16[index] = data;
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 1;
+               /* use attrs to get current cpu index */
+               cpu = vcpu->vcpu_id;
+               coreisr = data;
+               old_coreisr = s->coreisr.reg_u16[cpu][index];
+               /* write 1 to clear interrupt */
+               s->coreisr.reg_u16[cpu][index] = old_coreisr & ~coreisr;
+               coreisr &= old_coreisr;
+               bits = sizeof(data) * 8;
+               irq = find_first_bit((void *)&coreisr, bits);
+               while (irq < bits) {
+                       eiointc_update_irq(s, irq + index * bits, 0);
+                       bitmap_clear((void *)&coreisr, irq, 1);
+                       irq = find_first_bit((void *)&coreisr, bits);
+               }
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               irq = offset - EIOINTC_COREMAP_START;
+               index = irq >> 1;
+               s->coremap.reg_u16[index] = data;
+               eiointc_update_sw_coremap(s, irq, (void *)&data, sizeof(data), true);
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+
+       return ret;
+}
+
+static int loongarch_eiointc_writel(struct kvm_vcpu *vcpu,
+                               struct loongarch_eiointc *s,
+                               gpa_t addr, int len, const void *val)
+{
+       int i, index, irq, bits, ret = 0;
+       u8 cpu;
+       u32 data, old_data;
+       u32 coreisr, old_coreisr;
+       gpa_t offset;
+
+       data = *(u32 *)val;
+       offset = addr - EIOINTC_BASE;
+
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 2;
+               s->nodetype.reg_u32[index] = data;
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               /*
+                * ipmap cannot be set at runtime, can be set only at the beginning
+                * of irqchip driver, need not update upper irq level
+                */
+               index = (offset - EIOINTC_IPMAP_START) >> 2;
+               s->ipmap.reg_u32[index] = data;
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 2;
+               old_data = s->enable.reg_u32[index];
+               s->enable.reg_u32[index] = data;
+               /*
+                * 1: enable irq.
+                * update irq when isr is set.
+                */
+               data = s->enable.reg_u32[index] & ~old_data & s->isr.reg_u32[index];
+               index = index << 2;
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index + i, mask, 1);
+               }
+               /*
+                * 0: disable irq.
+                * update irq when isr is set.
+                */
+               data = ~s->enable.reg_u32[index] & old_data & s->isr.reg_u32[index];
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index, mask, 0);
+               }
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               /* do not emulate hw bounced irq routing */
+               index = (offset - EIOINTC_BOUNCE_START) >> 2;
+               s->bounce.reg_u32[index] = data;
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 2;
+               /* use attrs to get current cpu index */
+               cpu = vcpu->vcpu_id;
+               coreisr = data;
+               old_coreisr = s->coreisr.reg_u32[cpu][index];
+               /* write 1 to clear interrupt */
+               s->coreisr.reg_u32[cpu][index] = old_coreisr & ~coreisr;
+               coreisr &= old_coreisr;
+               bits = sizeof(data) * 8;
+               irq = find_first_bit((void *)&coreisr, bits);
+               while (irq < bits) {
+                       eiointc_update_irq(s, irq + index * bits, 0);
+                       bitmap_clear((void *)&coreisr, irq, 1);
+                       irq = find_first_bit((void *)&coreisr, bits);
+               }
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               irq = offset - EIOINTC_COREMAP_START;
+               index = irq >> 2;
+               s->coremap.reg_u32[index] = data;
+               eiointc_update_sw_coremap(s, irq, (void *)&data, sizeof(data), true);
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+
+       return ret;
+}
+
+static int loongarch_eiointc_writeq(struct kvm_vcpu *vcpu,
+                               struct loongarch_eiointc *s,
+                               gpa_t addr, int len, const void *val)
+{
+       int i, index, irq, bits, ret = 0;
+       u8 cpu;
+       u64 data, old_data;
+       u64 coreisr, old_coreisr;
+       gpa_t offset;
+
+       data = *(u64 *)val;
+       offset = addr - EIOINTC_BASE;
+
+       switch (offset) {
+       case EIOINTC_NODETYPE_START ... EIOINTC_NODETYPE_END:
+               index = (offset - EIOINTC_NODETYPE_START) >> 3;
+               s->nodetype.reg_u64[index] = data;
+               break;
+       case EIOINTC_IPMAP_START ... EIOINTC_IPMAP_END:
+               /*
+                * ipmap cannot be set at runtime, can be set only at the beginning
+                * of irqchip driver, need not update upper irq level
+                */
+               index = (offset - EIOINTC_IPMAP_START) >> 3;
+               s->ipmap.reg_u64 = data;
+               break;
+       case EIOINTC_ENABLE_START ... EIOINTC_ENABLE_END:
+               index = (offset - EIOINTC_ENABLE_START) >> 3;
+               old_data = s->enable.reg_u64[index];
+               s->enable.reg_u64[index] = data;
+               /*
+                * 1: enable irq.
+                * update irq when isr is set.
+                */
+               data = s->enable.reg_u64[index] & ~old_data & s->isr.reg_u64[index];
+               index = index << 3;
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index + i, mask, 1);
+               }
+               /*
+                * 0: disable irq.
+                * update irq when isr is set.
+                */
+               data = ~s->enable.reg_u64[index] & old_data & s->isr.reg_u64[index];
+               for (i = 0; i < sizeof(data); i++) {
+                       u8 mask = (data >> (i * 8)) & 0xff;
+                       eiointc_enable_irq(vcpu, s, index, mask, 0);
+               }
+               break;
+       case EIOINTC_BOUNCE_START ... EIOINTC_BOUNCE_END:
+               /* do not emulate hw bounced irq routing */
+               index = (offset - EIOINTC_BOUNCE_START) >> 3;
+               s->bounce.reg_u64[index] = data;
+               break;
+       case EIOINTC_COREISR_START ... EIOINTC_COREISR_END:
+               index = (offset - EIOINTC_COREISR_START) >> 3;
+               /* use attrs to get current cpu index */
+               cpu = vcpu->vcpu_id;
+               coreisr = data;
+               old_coreisr = s->coreisr.reg_u64[cpu][index];
+               /* write 1 to clear interrupt */
+               s->coreisr.reg_u64[cpu][index] = old_coreisr & ~coreisr;
+               coreisr &= old_coreisr;
+               bits = sizeof(data) * 8;
+               irq = find_first_bit((void *)&coreisr, bits);
+               while (irq < bits) {
+                       eiointc_update_irq(s, irq + index * bits, 0);
+                       bitmap_clear((void *)&coreisr, irq, 1);
+                       irq = find_first_bit((void *)&coreisr, bits);
+               }
+               break;
+       case EIOINTC_COREMAP_START ... EIOINTC_COREMAP_END:
+               irq = offset - EIOINTC_COREMAP_START;
+               index = irq >> 3;
+               s->coremap.reg_u64[index] = data;
+               eiointc_update_sw_coremap(s, irq, (void *)&data, sizeof(data), true);
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+
+       return ret;
 }
 
 static int kvm_eiointc_write(struct kvm_vcpu *vcpu,
                        struct kvm_io_device *dev,
                        gpa_t addr, int len, const void *val)
 {
-       return 0;
+       int ret = -EINVAL;
+       unsigned long flags;
+       struct loongarch_eiointc *eiointc = vcpu->kvm->arch.eiointc;
+
+       if (!eiointc) {
+               kvm_err("%s: eiointc irqchip not valid!\n", __func__);
+               return -EINVAL;
+       }
+
+       vcpu->kvm->stat.eiointc_write_exits++;
+       spin_lock_irqsave(&eiointc->lock, flags);
+       switch (len) {
+       case 1:
+               ret = loongarch_eiointc_writeb(vcpu, eiointc, addr, len, val);
+               break;
+       case 2:
+               ret = loongarch_eiointc_writew(vcpu, eiointc, addr, len, val);
+               break;
+       case 4:
+               ret = loongarch_eiointc_writel(vcpu, eiointc, addr, len, val);
+               break;
+       case 8:
+               ret = loongarch_eiointc_writeq(vcpu, eiointc, addr, len, val);
+               break;
+       default:
+               WARN_ONCE(1, "%s: Abnormal address access: addr 0x%llx, size %d\n",
+                                               __func__, addr, len);
+       }
+       spin_unlock_irqrestore(&eiointc->lock, flags);
+
+       return ret;
 }
 
 static const struct kvm_io_device_ops kvm_eiointc_ops = {
@@ -30,6 +712,29 @@ static int kvm_eiointc_virt_read(struct kvm_vcpu *vcpu,
                                struct kvm_io_device *dev,
                                gpa_t addr, int len, void *val)
 {
+       unsigned long flags;
+       u32 *data = val;
+       struct loongarch_eiointc *eiointc = vcpu->kvm->arch.eiointc;
+
+       if (!eiointc) {
+               kvm_err("%s: eiointc irqchip not valid!\n", __func__);
+               return -EINVAL;
+       }
+
+       addr -= EIOINTC_VIRT_BASE;
+       spin_lock_irqsave(&eiointc->lock, flags);
+       switch (addr) {
+       case EIOINTC_VIRT_FEATURES:
+               *data = eiointc->features;
+               break;
+       case EIOINTC_VIRT_CONFIG:
+               *data = eiointc->status;
+               break;
+       default:
+               break;
+       }
+       spin_unlock_irqrestore(&eiointc->lock, flags);
+
        return 0;
 }
 
@@ -37,7 +742,38 @@ static int kvm_eiointc_virt_write(struct kvm_vcpu *vcpu,
                                struct kvm_io_device *dev,
                                gpa_t addr, int len, const void *val)
 {
-       return 0;
+       int ret = 0;
+       unsigned long flags;
+       u32 value = *(u32 *)val;
+       struct loongarch_eiointc *eiointc = vcpu->kvm->arch.eiointc;
+
+       if (!eiointc) {
+               kvm_err("%s: eiointc irqchip not valid!\n", __func__);
+               return -EINVAL;
+       }
+
+       addr -= EIOINTC_VIRT_BASE;
+       spin_lock_irqsave(&eiointc->lock, flags);
+       switch (addr) {
+       case EIOINTC_VIRT_FEATURES:
+               ret = -EPERM;
+               break;
+       case EIOINTC_VIRT_CONFIG:
+               /*
+                * eiointc features can only be set at disabled status
+                */
+               if ((eiointc->status & BIT(EIOINTC_ENABLE)) && value) {
+                       ret = -EPERM;
+                       break;
+               }
+               eiointc->status = value & eiointc->features;
+               break;
+       default:
+               break;
+       }
+       spin_unlock_irqrestore(&eiointc->lock, flags);
+
+       return ret;
 }
 
 static const struct kvm_io_device_ops kvm_eiointc_virt_ops = {