Commit 8a09bb1b authored by Sakari Ailus's avatar Sakari Ailus Committed by Mauro Carvalho Chehab

media: intel/ipu6: Don't re-allocate memory for firmware

The ipu6 driver allocated vmalloc memory for the firmware if
request_firmware() somehow managed not to use vmalloc to allocate it.

Still how the memory is allocated by request_firmware() is not specified
in its API, so be prepared for kmalloc-allocated firmware, too. Instead of
allocating new vmalloc-backed buffer for the firmware, obtain the pages
from virtual addresses instead.

Link: https://lore.kernel.org/linux-media/20240502154950.549015-1-sakari.ailus@linux.intel.comReported-by: default avatarStephen Rothwell <sfr@canb.auug.org.au>
Closes: https://lore.kernel.org/all/20240501102236.3b2585d1@canb.auug.org.au/
Fixes: 25fedc02 ("media: intel/ipu6: add Intel IPU6 PCI device driver")
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@kernel.org>
parent 1aa1329a
...@@ -552,12 +552,16 @@ int ipu6_buttress_reset_authentication(struct ipu6_device *isp) ...@@ -552,12 +552,16 @@ int ipu6_buttress_reset_authentication(struct ipu6_device *isp)
int ipu6_buttress_map_fw_image(struct ipu6_bus_device *sys, int ipu6_buttress_map_fw_image(struct ipu6_bus_device *sys,
const struct firmware *fw, struct sg_table *sgt) const struct firmware *fw, struct sg_table *sgt)
{ {
bool is_vmalloc = is_vmalloc_addr(fw->data);
struct page **pages; struct page **pages;
const void *addr; const void *addr;
unsigned long n_pages; unsigned long n_pages;
unsigned int i; unsigned int i;
int ret; int ret;
if (!is_vmalloc && !virt_addr_valid(fw->data))
return -EDOM;
n_pages = PHYS_PFN(PAGE_ALIGN(fw->size)); n_pages = PHYS_PFN(PAGE_ALIGN(fw->size));
pages = kmalloc_array(n_pages, sizeof(*pages), GFP_KERNEL); pages = kmalloc_array(n_pages, sizeof(*pages), GFP_KERNEL);
...@@ -566,7 +570,8 @@ int ipu6_buttress_map_fw_image(struct ipu6_bus_device *sys, ...@@ -566,7 +570,8 @@ int ipu6_buttress_map_fw_image(struct ipu6_bus_device *sys,
addr = fw->data; addr = fw->data;
for (i = 0; i < n_pages; i++) { for (i = 0; i < n_pages; i++) {
struct page *p = vmalloc_to_page(addr); struct page *p = is_vmalloc ?
vmalloc_to_page(addr) : virt_to_page(addr);
if (!p) { if (!p) {
ret = -ENOMEM; ret = -ENOMEM;
......
...@@ -503,45 +503,6 @@ static void ipu6_configure_vc_mechanism(struct ipu6_device *isp) ...@@ -503,45 +503,6 @@ static void ipu6_configure_vc_mechanism(struct ipu6_device *isp)
writel(val, isp->base + BUTTRESS_REG_BTRS_CTRL); writel(val, isp->base + BUTTRESS_REG_BTRS_CTRL);
} }
static int request_cpd_fw(const struct firmware **firmware_p, const char *name,
struct device *device)
{
const struct firmware *fw;
struct firmware *dst;
int ret = 0;
ret = request_firmware(&fw, name, device);
if (ret)
return ret;
if (is_vmalloc_addr(fw->data)) {
*firmware_p = fw;
return 0;
}
dst = kzalloc(sizeof(*dst), GFP_KERNEL);
if (!dst) {
ret = -ENOMEM;
goto release_firmware;
}
dst->size = fw->size;
dst->data = vmalloc(fw->size);
if (!dst->data) {
kfree(dst);
ret = -ENOMEM;
goto release_firmware;
}
memcpy((void *)dst->data, fw->data, fw->size);
*firmware_p = dst;
release_firmware:
release_firmware(fw);
return ret;
}
static int ipu6_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) static int ipu6_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id)
{ {
struct ipu6_buttress_ctrl *isys_ctrl = NULL, *psys_ctrl = NULL; struct ipu6_buttress_ctrl *isys_ctrl = NULL, *psys_ctrl = NULL;
...@@ -627,7 +588,7 @@ static int ipu6_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) ...@@ -627,7 +588,7 @@ static int ipu6_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id)
if (ret) if (ret)
return ret; return ret;
ret = request_cpd_fw(&isp->cpd_fw, isp->cpd_fw_name, dev); ret = request_firmware(&isp->cpd_fw, isp->cpd_fw_name, dev);
if (ret) { if (ret) {
dev_err_probe(&isp->pdev->dev, ret, dev_err_probe(&isp->pdev->dev, ret,
"Requesting signed firmware %s failed\n", "Requesting signed firmware %s failed\n",
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment