drm/lima: fix shared irq handling on driver remove
authorErico Nunes <nunes.erico@gmail.com>
Mon, 1 Apr 2024 22:43:28 +0000 (00:43 +0200)
committerQiang Yu <yuq825@gmail.com>
Mon, 15 Apr 2024 01:06:18 +0000 (09:06 +0800)
lima uses a shared interrupt, so the interrupt handlers must be prepared
to be called at any time. At driver removal time, the clocks are
disabled early and the interrupts stay registered until the very end of
the remove process due to the devm usage.
This is potentially a bug as the interrupts access device registers
which assumes clocks are enabled. A crash can be triggered by removing
the driver in a kernel with CONFIG_DEBUG_SHIRQ enabled.
This patch frees the interrupts at each lima device finishing callback
so that the handlers are already unregistered by the time we fully
disable clocks.

Signed-off-by: Erico Nunes <nunes.erico@gmail.com>
Signed-off-by: Qiang Yu <yuq825@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20240401224329.1228468-2-nunes.erico@gmail.com
drivers/gpu/drm/lima/lima_gp.c
drivers/gpu/drm/lima/lima_mmu.c
drivers/gpu/drm/lima/lima_pp.c

index e15295071533b5387e450dd4d9ff96d34d71750b..3282997a0358dfd2c8fab4223fa7885fd1eb232e 100644 (file)
@@ -345,7 +345,9 @@ int lima_gp_init(struct lima_ip *ip)
 
 void lima_gp_fini(struct lima_ip *ip)
 {
+       struct lima_device *dev = ip->dev;
 
+       devm_free_irq(dev->dev, ip->irq, ip);
 }
 
 int lima_gp_pipe_init(struct lima_device *dev)
index e18317c5ca8c19eafc32f8b4e4950f5a7d414a3d..6611e2836bf0dd7cf7283c5798b238854194f857 100644 (file)
@@ -118,7 +118,12 @@ int lima_mmu_init(struct lima_ip *ip)
 
 void lima_mmu_fini(struct lima_ip *ip)
 {
+       struct lima_device *dev = ip->dev;
+
+       if (ip->id == lima_ip_ppmmu_bcast)
+               return;
 
+       devm_free_irq(dev->dev, ip->irq, ip);
 }
 
 void lima_mmu_flush_tlb(struct lima_ip *ip)
index a4a2ffe6527c2cc59ea0246392340a8ec4a920c6..eaab4788dff494f2cbc4f2ebca70d444050920f9 100644 (file)
@@ -286,7 +286,9 @@ int lima_pp_init(struct lima_ip *ip)
 
 void lima_pp_fini(struct lima_ip *ip)
 {
+       struct lima_device *dev = ip->dev;
 
+       devm_free_irq(dev->dev, ip->irq, ip);
 }
 
 int lima_pp_bcast_resume(struct lima_ip *ip)
@@ -319,7 +321,9 @@ int lima_pp_bcast_init(struct lima_ip *ip)
 
 void lima_pp_bcast_fini(struct lima_ip *ip)
 {
+       struct lima_device *dev = ip->dev;
 
+       devm_free_irq(dev->dev, ip->irq, ip);
 }
 
 static int lima_pp_task_validate(struct lima_sched_pipe *pipe,