diff options
Diffstat (limited to 'drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c')
-rw-r--r-- | drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c | 46 |
1 files changed, 31 insertions, 15 deletions
diff --git a/drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c b/drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c index deb96de54b00..ee2431a7804e 100644 --- a/drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c +++ b/drivers/gpu/drm/nouveau/nvkm/subdev/pci/base.c | |||
@@ -71,6 +71,10 @@ nvkm_pci_intr(int irq, void *arg) | |||
71 | struct nvkm_pci *pci = arg; | 71 | struct nvkm_pci *pci = arg; |
72 | struct nvkm_device *device = pci->subdev.device; | 72 | struct nvkm_device *device = pci->subdev.device; |
73 | bool handled = false; | 73 | bool handled = false; |
74 | |||
75 | if (pci->irq < 0) | ||
76 | return IRQ_HANDLED; | ||
77 | |||
74 | nvkm_mc_intr_unarm(device); | 78 | nvkm_mc_intr_unarm(device); |
75 | if (pci->msi) | 79 | if (pci->msi) |
76 | pci->func->msi_rearm(pci); | 80 | pci->func->msi_rearm(pci); |
@@ -84,11 +88,6 @@ nvkm_pci_fini(struct nvkm_subdev *subdev, bool suspend) | |||
84 | { | 88 | { |
85 | struct nvkm_pci *pci = nvkm_pci(subdev); | 89 | struct nvkm_pci *pci = nvkm_pci(subdev); |
86 | 90 | ||
87 | if (pci->irq >= 0) { | ||
88 | free_irq(pci->irq, pci); | ||
89 | pci->irq = -1; | ||
90 | } | ||
91 | |||
92 | if (pci->agp.bridge) | 91 | if (pci->agp.bridge) |
93 | nvkm_agp_fini(pci); | 92 | nvkm_agp_fini(pci); |
94 | 93 | ||
@@ -108,8 +107,20 @@ static int | |||
108 | nvkm_pci_oneinit(struct nvkm_subdev *subdev) | 107 | nvkm_pci_oneinit(struct nvkm_subdev *subdev) |
109 | { | 108 | { |
110 | struct nvkm_pci *pci = nvkm_pci(subdev); | 109 | struct nvkm_pci *pci = nvkm_pci(subdev); |
111 | if (pci_is_pcie(pci->pdev)) | 110 | struct pci_dev *pdev = pci->pdev; |
112 | return nvkm_pcie_oneinit(pci); | 111 | int ret; |
112 | |||
113 | if (pci_is_pcie(pci->pdev)) { | ||
114 | ret = nvkm_pcie_oneinit(pci); | ||
115 | if (ret) | ||
116 | return ret; | ||
117 | } | ||
118 | |||
119 | ret = request_irq(pdev->irq, nvkm_pci_intr, IRQF_SHARED, "nvkm", pci); | ||
120 | if (ret) | ||
121 | return ret; | ||
122 | |||
123 | pci->irq = pdev->irq; | ||
113 | return 0; | 124 | return 0; |
114 | } | 125 | } |
115 | 126 | ||
@@ -117,7 +128,6 @@ static int | |||
117 | nvkm_pci_init(struct nvkm_subdev *subdev) | 128 | nvkm_pci_init(struct nvkm_subdev *subdev) |
118 | { | 129 | { |
119 | struct nvkm_pci *pci = nvkm_pci(subdev); | 130 | struct nvkm_pci *pci = nvkm_pci(subdev); |
120 | struct pci_dev *pdev = pci->pdev; | ||
121 | int ret; | 131 | int ret; |
122 | 132 | ||
123 | if (pci->agp.bridge) { | 133 | if (pci->agp.bridge) { |
@@ -131,28 +141,34 @@ nvkm_pci_init(struct nvkm_subdev *subdev) | |||
131 | if (pci->func->init) | 141 | if (pci->func->init) |
132 | pci->func->init(pci); | 142 | pci->func->init(pci); |
133 | 143 | ||
134 | ret = request_irq(pdev->irq, nvkm_pci_intr, IRQF_SHARED, "nvkm", pci); | ||
135 | if (ret) | ||
136 | return ret; | ||
137 | |||
138 | pci->irq = pdev->irq; | ||
139 | |||
140 | /* Ensure MSI interrupts are armed, for the case where there are | 144 | /* Ensure MSI interrupts are armed, for the case where there are |
141 | * already interrupts pending (for whatever reason) at load time. | 145 | * already interrupts pending (for whatever reason) at load time. |
142 | */ | 146 | */ |
143 | if (pci->msi) | 147 | if (pci->msi) |
144 | pci->func->msi_rearm(pci); | 148 | pci->func->msi_rearm(pci); |
145 | 149 | ||
146 | return ret; | 150 | return 0; |
147 | } | 151 | } |
148 | 152 | ||
149 | static void * | 153 | static void * |
150 | nvkm_pci_dtor(struct nvkm_subdev *subdev) | 154 | nvkm_pci_dtor(struct nvkm_subdev *subdev) |
151 | { | 155 | { |
152 | struct nvkm_pci *pci = nvkm_pci(subdev); | 156 | struct nvkm_pci *pci = nvkm_pci(subdev); |
157 | |||
153 | nvkm_agp_dtor(pci); | 158 | nvkm_agp_dtor(pci); |
159 | |||
160 | if (pci->irq >= 0) { | ||
161 | /* freq_irq() will call the handler, we use pci->irq == -1 | ||
162 | * to signal that it's been torn down and should be a noop. | ||
163 | */ | ||
164 | int irq = pci->irq; | ||
165 | pci->irq = -1; | ||
166 | free_irq(irq, pci); | ||
167 | } | ||
168 | |||
154 | if (pci->msi) | 169 | if (pci->msi) |
155 | pci_disable_msi(pci->pdev); | 170 | pci_disable_msi(pci->pdev); |
171 | |||
156 | return nvkm_pci(subdev); | 172 | return nvkm_pci(subdev); |
157 | } | 173 | } |
158 | 174 | ||