提交 76730334 编写于 作者: N Nishanth Aravamudan 提交者: Benjamin Herrenschmidt

powerpc: Fix kexec with dynamic dma windows

When we kexec we look for a particular property added by the first
kernel, "linux,direct64-ddr-window-info", per-device where we already
have set up dynamic dma windows. The current code, though, wasn't
initializing the size of this property and thus when we kexec'd, we
would find the property but read uninitialized memory resulting in
garbage ddw values for the kexec'd kernel and panics. Fix this by
setting the size at enable_ddw() time and ensuring that the size of the
found property is valid at dupe_ddw_if_kexec() time.
Signed-off-by: NNishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: NBenjamin Herrenschmidt <benh@kernel.crashing.org>
上级 31355403
...@@ -730,6 +730,9 @@ static u64 dupe_ddw_if_kexec(struct pci_dev *dev, struct device_node *pdn) ...@@ -730,6 +730,9 @@ static u64 dupe_ddw_if_kexec(struct pci_dev *dev, struct device_node *pdn)
pcidn = PCI_DN(dn); pcidn = PCI_DN(dn);
direct64 = of_get_property(pdn, DIRECT64_PROPNAME, &len); direct64 = of_get_property(pdn, DIRECT64_PROPNAME, &len);
if (direct64) { if (direct64) {
if (len < sizeof(struct dynamic_dma_window_prop)) {
remove_ddw(pdn);
} else {
window = kzalloc(sizeof(*window), GFP_KERNEL); window = kzalloc(sizeof(*window), GFP_KERNEL);
if (!window) { if (!window) {
remove_ddw(pdn); remove_ddw(pdn);
...@@ -742,6 +745,7 @@ static u64 dupe_ddw_if_kexec(struct pci_dev *dev, struct device_node *pdn) ...@@ -742,6 +745,7 @@ static u64 dupe_ddw_if_kexec(struct pci_dev *dev, struct device_node *pdn)
dma_addr = direct64->dma_base; dma_addr = direct64->dma_base;
} }
} }
}
return dma_addr; return dma_addr;
} }
...@@ -833,7 +837,7 @@ static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn) ...@@ -833,7 +837,7 @@ static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn)
struct device_node *dn; struct device_node *dn;
const u32 *uninitialized_var(ddr_avail); const u32 *uninitialized_var(ddr_avail);
struct direct_window *window; struct direct_window *window;
struct property *uninitialized_var(win64); struct property *win64;
struct dynamic_dma_window_prop *ddwprop; struct dynamic_dma_window_prop *ddwprop;
mutex_lock(&direct_window_init_mutex); mutex_lock(&direct_window_init_mutex);
...@@ -907,6 +911,7 @@ static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn) ...@@ -907,6 +911,7 @@ static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn)
} }
win64->name = kstrdup(DIRECT64_PROPNAME, GFP_KERNEL); win64->name = kstrdup(DIRECT64_PROPNAME, GFP_KERNEL);
win64->value = ddwprop = kmalloc(sizeof(*ddwprop), GFP_KERNEL); win64->value = ddwprop = kmalloc(sizeof(*ddwprop), GFP_KERNEL);
win64->length = sizeof(*ddwprop);
if (!win64->name || !win64->value) { if (!win64->name || !win64->value) {
dev_info(&dev->dev, dev_info(&dev->dev,
"couldn't allocate property name and value\n"); "couldn't allocate property name and value\n");
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册