PCI: Fix reference leak in pci_register_host_bridge()
[ Upstream commit 804443c1f27883926de94c849d91f5b7d7d696e9 ]
If device_register() fails, call put_device() to give up the reference to
avoid a memory leak, per the comment at device_register().
Found by code review.
Link: https://lore.kernel.org/r/20250225021440.3130264-1-make24@iscas.ac.cn
Fixes: 37d6a0a6f4 ("PCI: Add pci_register_host_bridge() interface")
Signed-off-by: Ma Ke <make24@iscas.ac.cn>
[bhelgaas: squash Dan Carpenter's double free fix from
https://lore.kernel.org/r/db806a6c-a91b-4e5a-a84b-6b7e01bdac85@stanley.mountain]
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Cc: stable@vger.kernel.org
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
81435b85b2
commit
bd2a352a0d
@@ -885,6 +885,7 @@ static int pci_register_host_bridge(struct pci_host_bridge *bridge)
|
|||||||
resource_size_t offset, next_offset;
|
resource_size_t offset, next_offset;
|
||||||
LIST_HEAD(resources);
|
LIST_HEAD(resources);
|
||||||
struct resource *res, *next_res;
|
struct resource *res, *next_res;
|
||||||
|
bool bus_registered = false;
|
||||||
char addr[64], *fmt;
|
char addr[64], *fmt;
|
||||||
const char *name;
|
const char *name;
|
||||||
int err;
|
int err;
|
||||||
@@ -948,6 +949,7 @@ static int pci_register_host_bridge(struct pci_host_bridge *bridge)
|
|||||||
name = dev_name(&bus->dev);
|
name = dev_name(&bus->dev);
|
||||||
|
|
||||||
err = device_register(&bus->dev);
|
err = device_register(&bus->dev);
|
||||||
|
bus_registered = true;
|
||||||
if (err)
|
if (err)
|
||||||
goto unregister;
|
goto unregister;
|
||||||
|
|
||||||
@@ -1031,12 +1033,15 @@ static int pci_register_host_bridge(struct pci_host_bridge *bridge)
|
|||||||
unregister:
|
unregister:
|
||||||
put_device(&bridge->dev);
|
put_device(&bridge->dev);
|
||||||
device_del(&bridge->dev);
|
device_del(&bridge->dev);
|
||||||
|
|
||||||
free:
|
free:
|
||||||
#ifdef CONFIG_PCI_DOMAINS_GENERIC
|
#ifdef CONFIG_PCI_DOMAINS_GENERIC
|
||||||
pci_bus_release_domain_nr(bus, parent);
|
pci_bus_release_domain_nr(bus, parent);
|
||||||
#endif
|
#endif
|
||||||
kfree(bus);
|
if (bus_registered)
|
||||||
|
put_device(&bus->dev);
|
||||||
|
else
|
||||||
|
kfree(bus);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user