]> git.baikalelectronics.ru Git - kernel.git/commitdiff
PCI: hv: Reuse existing IRTE allocation in compose_msi_msg()
authorJeffrey Hugo <quic_jhugo@quicinc.com>
Fri, 15 Jul 2022 21:36:10 +0000 (21:36 +0000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 29 Jul 2022 15:14:09 +0000 (17:14 +0200)
commit cc424c224004b54c5e25c5293571de9247b2e56f upstream.

Currently if compose_msi_msg() is called multiple times, it will free any
previous IRTE allocation, and generate a new allocation.  While nothing
prevents this from occurring, it is extraneous when Linux could just reuse
the existing allocation and avoid a bunch of overhead.

However, when future IRTE allocations operate on blocks of MSIs instead of
a single line, freeing the allocation will impact all of the lines.  This
could cause an issue where an allocation of N MSIs occurs, then some of
the lines are retargeted, and finally the allocation is freed/reallocated.
The freeing of the allocation removes all of the configuration for the
entire block, which requires all the lines to be retargeted, which might
not happen since some lines might already be unmasked/active.

Signed-off-by: Jeffrey Hugo <quic_jhugo@quicinc.com>
Reviewed-by: Dexuan Cui <decui@microsoft.com>
Tested-by: Dexuan Cui <decui@microsoft.com>
Tested-by: Michael Kelley <mikelley@microsoft.com>
Link: https://lore.kernel.org/r/1652282582-21595-1-git-send-email-quic_jhugo@quicinc.com
Signed-off-by: Wei Liu <wei.liu@kernel.org>
Signed-off-by: Carl Vanderlip <quic_carlv@quicinc.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/pci/controller/pci-hyperv.c

index 9fe94b28e36e9eff78aefe64f16085c6a45d5487..a8d73bfe279c8475f29375cfd79260cc207b4389 100644 (file)
@@ -1387,6 +1387,15 @@ static void hv_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)
        u32 size;
        int ret;
 
+       /* Reuse the previous allocation */
+       if (data->chip_data) {
+               int_desc = data->chip_data;
+               msg->address_hi = int_desc->address >> 32;
+               msg->address_lo = int_desc->address & 0xffffffff;
+               msg->data = int_desc->data;
+               return;
+       }
+
        pdev = msi_desc_to_pci_dev(irq_data_get_msi_desc(data));
        dest = irq_data_get_effective_affinity_mask(data);
        pbus = pdev->bus;
@@ -1395,13 +1404,6 @@ static void hv_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)
        if (!hpdev)
                goto return_null_message;
 
-       /* Free any previous message that might have already been composed. */
-       if (data->chip_data) {
-               int_desc = data->chip_data;
-               data->chip_data = NULL;
-               hv_int_desc_free(hpdev, int_desc);
-       }
-
        int_desc = kzalloc(sizeof(*int_desc), GFP_ATOMIC);
        if (!int_desc)
                goto drop_reference;