aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/pci/pci-driver.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/pci/pci-driver.c')
-rw-r--r--drivers/pci/pci-driver.c52
1 files changed, 32 insertions, 20 deletions
diff --git a/drivers/pci/pci-driver.c b/drivers/pci/pci-driver.c
index 888191a3b0d1..c3f76be832d4 100644
--- a/drivers/pci/pci-driver.c
+++ b/drivers/pci/pci-driver.c
@@ -16,6 +16,7 @@
16#include <linux/string.h> 16#include <linux/string.h>
17#include <linux/slab.h> 17#include <linux/slab.h>
18#include <linux/sched.h> 18#include <linux/sched.h>
19#include <linux/cpu.h>
19#include "pci.h" 20#include "pci.h"
20 21
21/* 22/*
@@ -185,32 +186,43 @@ static const struct pci_device_id *pci_match_device(struct pci_driver *drv,
185 return pci_match_id(drv->id_table, dev); 186 return pci_match_id(drv->id_table, dev);
186} 187}
187 188
189struct drv_dev_and_id {
190 struct pci_driver *drv;
191 struct pci_dev *dev;
192 const struct pci_device_id *id;
193};
194
195static long local_pci_probe(void *_ddi)
196{
197 struct drv_dev_and_id *ddi = _ddi;
198
199 return ddi->drv->probe(ddi->dev, ddi->id);
200}
201
188static int pci_call_probe(struct pci_driver *drv, struct pci_dev *dev, 202static int pci_call_probe(struct pci_driver *drv, struct pci_dev *dev,
189 const struct pci_device_id *id) 203 const struct pci_device_id *id)
190{ 204{
191 int error; 205 int error, node;
192#ifdef CONFIG_NUMA 206 struct drv_dev_and_id ddi = { drv, dev, id };
193 /* Execute driver initialization on node where the
194 device's bus is attached to. This way the driver likely
195 allocates its local memory on the right node without
196 any need to change it. */
197 struct mempolicy *oldpol;
198 cpumask_t oldmask = current->cpus_allowed;
199 int node = dev_to_node(&dev->dev);
200 207
208 /* Execute driver initialization on node where the device's
209 bus is attached to. This way the driver likely allocates
210 its local memory on the right node without any need to
211 change it. */
212 node = dev_to_node(&dev->dev);
201 if (node >= 0) { 213 if (node >= 0) {
214 int cpu;
202 node_to_cpumask_ptr(nodecpumask, node); 215 node_to_cpumask_ptr(nodecpumask, node);
203 set_cpus_allowed_ptr(current, nodecpumask); 216
204 } 217 get_online_cpus();
205 /* And set default memory allocation policy */ 218 cpu = cpumask_any_and(nodecpumask, cpu_online_mask);
206 oldpol = current->mempolicy; 219 if (cpu < nr_cpu_ids)
207 current->mempolicy = NULL; /* fall back to system default policy */ 220 error = work_on_cpu(cpu, local_pci_probe, &ddi);
208#endif 221 else
209 error = drv->probe(dev, id); 222 error = local_pci_probe(&ddi);
210#ifdef CONFIG_NUMA 223 put_online_cpus();
211 set_cpus_allowed_ptr(current, &oldmask); 224 } else
212 current->mempolicy = oldpol; 225 error = local_pci_probe(&ddi);
213#endif
214 return error; 226 return error;
215} 227}
216 228