diff options
author | Xianzhong <b07117@freescale.com> | 2013-01-26 02:20:55 +0800 |
---|---|---|
committer | Tapani <tapani@vmail.me> | 2013-03-29 14:36:14 +0800 |
commit | 982c3846a5cce75f3e20af2abe6d87a2b880c0d2 (patch) | |
tree | 6fd74f7239e521ef8ac329d5a920093f4dc95360 | |
parent | 7371548ce86709d06d7e2f3b958b5f73d7a1609b (diff) |
ENGR00241777 fix rare kernel panic by gpu lowmem killer
task_free notification procedure cannot be registered/unregistered
during gpu memory allocation dynamically, when the lowmem killer triggers
a kill signal to one app, and then rescan to check if memory is available,
if other 3d apps free some memory prior to the killed app in unlikely case,
then task_free notification procedure will be unregistered before the notification
of the killed app is received, that causes kernel panic in task_free processing.
the fix is to move task_free procedures into gpu kernel drv_init and drv_exit.
Signed-off-by: Xianzhong <b07117@freescale.com>
Acked-by: Lily Zhang
-rw-r--r-- | drivers/mxc/gpu-viv/hal/kernel/gc_hal_kernel.c | 36 | ||||
-rw-r--r-- | drivers/mxc/gpu-viv/hal/os/linux/kernel/gc_hal_kernel_driver.c | 27 |
2 files changed, 28 insertions, 35 deletions
diff --git a/drivers/mxc/gpu-viv/hal/kernel/gc_hal_kernel.c b/drivers/mxc/gpu-viv/hal/kernel/gc_hal_kernel.c index 0a0253a23e33..8b76091dbc4c 100644 --- a/drivers/mxc/gpu-viv/hal/kernel/gc_hal_kernel.c +++ b/drivers/mxc/gpu-viv/hal/kernel/gc_hal_kernel.c @@ -502,27 +502,9 @@ gckKERNEL_Destroy( #include <linux/sched.h> #include <linux/notifier.h> -static struct task_struct *lowmem_deathpending; +extern struct task_struct *lowmem_deathpending; static unsigned long lowmem_deathpending_timeout; -static int -task_notify_func(struct notifier_block *self, unsigned long val, void *data); - -static struct notifier_block task_nb = { - .notifier_call = task_notify_func, -}; - -static int -task_notify_func(struct notifier_block *self, unsigned long val, void *data) -{ - struct task_struct *task = data; - - if (task == lowmem_deathpending) - lowmem_deathpending = NULL; - - return NOTIFY_OK; -} - static int force_contiguous_lowmem_shrink(IN gckKERNEL Kernel) { struct task_struct *p; @@ -644,9 +626,6 @@ _AllocateMemory( gcuVIDMEM_NODE_PTR node = gcvNULL; gctBOOL tileStatusInVirtual; gctBOOL forceContiguous = gcvFALSE; -#ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT - gctBOOL forceContiguousShrinking = gcvFALSE; -#endif gcmkHEADER_ARG("Kernel=0x%x *Pool=%d Bytes=%lu Alignment=%lu Type=%d", Kernel, *Pool, Bytes, Alignment, Type); @@ -806,12 +785,6 @@ _AllocateMemory_Retry: #ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT if(forceContiguous == gcvTRUE) { - if(forceContiguousShrinking == gcvFALSE) - { - forceContiguousShrinking = gcvTRUE; - task_free_register(&task_nb); - } - if(force_contiguous_lowmem_shrink(Kernel) == 0) { /* Sleep 1 millisecond. */ @@ -824,13 +797,6 @@ _AllocateMemory_Retry: gcmkONERROR(gcvSTATUS_OUT_OF_MEMORY); } -#ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT - if(forceContiguous == gcvTRUE && forceContiguousShrinking == gcvTRUE) - { - task_free_unregister(&task_nb); - } -#endif - /* Return node and pool used for allocation. */ *Node = node; *Pool = pool; diff --git a/drivers/mxc/gpu-viv/hal/os/linux/kernel/gc_hal_kernel_driver.c b/drivers/mxc/gpu-viv/hal/os/linux/kernel/gc_hal_kernel_driver.c index 9683cb7bedcf..334c495f0d01 100644 --- a/drivers/mxc/gpu-viv/hal/os/linux/kernel/gc_hal_kernel_driver.c +++ b/drivers/mxc/gpu-viv/hal/os/linux/kernel/gc_hal_kernel_driver.c @@ -41,6 +41,31 @@ #ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT # include <linux/resmem_account.h> +# include <linux/kernel.h> +# include <linux/mm.h> +# include <linux/oom.h> +# include <linux/sched.h> +# include <linux/notifier.h> + +struct task_struct *lowmem_deathpending; + +static int +task_notify_func(struct notifier_block *self, unsigned long val, void *data); + +static struct notifier_block task_nb = { + .notifier_call = task_notify_func, +}; + +static int +task_notify_func(struct notifier_block *self, unsigned long val, void *data) +{ + struct task_struct *task = data; + + if (task == lowmem_deathpending) + lowmem_deathpending = NULL; + + return NOTIFY_OK; +} #endif @@ -819,6 +844,7 @@ static int drv_init(void) } #ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT + task_free_register(&task_nb); viv_gpu_resmem_handler.data = device->kernels[gcvCORE_MAJOR]; register_reserved_memory_account(&viv_gpu_resmem_handler); #endif @@ -904,6 +930,7 @@ static void drv_exit(void) gcmkHEADER(); #ifdef CONFIG_ANDROID_RESERVED_MEMORY_ACCOUNT + task_free_unregister(&task_nb); unregister_reserved_memory_account(&viv_gpu_resmem_handler); #endif |