staging: ion: always initialize the free list parameters
authorMitchel Humpherys <mitchelh@codeaurora.org>
Fri, 9 Jan 2015 01:24:27 +0000 (17:24 -0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sun, 18 Jan 2015 00:16:00 +0000 (16:16 -0800)
Currently we initialize the heap free_lock and free list size in
ion_heap_init_deferred_free, which is only called when the
ION_HEAP_FLAG_DEFER_FREE heap flag is given.  However, the lock and size
are used in the shrinker path as well as the deferred free path, and we
can register a shrinker *without* enabling deferred freeing.  So, if a
heap provides a shrinker but *doesn't* set the DEFER_FREE flag we will
use these parameters uninitialized (resulting in a spinlock bug and
broken shrinker accounting).

Fix these problems by initializing the free list parameters directly in
ion_device_add_heap, which is always called no matter which heap
features are being used.

Signed-off-by: Mitchel Humpherys <mitchelh@codeaurora.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/staging/android/ion/ion.c
drivers/staging/android/ion/ion_heap.c

index 296d347660fc090515ed4c5e3e95ad2f37b5dcb4..b8f1c491553e2619bfbbbbb9ff52318e2f8017cc 100644 (file)
@@ -1508,6 +1508,9 @@ void ion_device_add_heap(struct ion_device *dev, struct ion_heap *heap)
                pr_err("%s: can not add heap with invalid ops struct.\n",
                       __func__);
 
+       spin_lock_init(&heap->free_lock);
+       heap->free_list_size = 0;
+
        if (heap->flags & ION_HEAP_FLAG_DEFER_FREE)
                ion_heap_init_deferred_free(heap);
 
index 4605e04712aaeaa951363875fe4817bedfde7ea4..fd13d05b538a665344893a750c673d52a608d0c4 100644 (file)
@@ -253,8 +253,6 @@ int ion_heap_init_deferred_free(struct ion_heap *heap)
        struct sched_param param = { .sched_priority = 0 };
 
        INIT_LIST_HEAD(&heap->free_list);
-       heap->free_list_size = 0;
-       spin_lock_init(&heap->free_lock);
        init_waitqueue_head(&heap->waitqueue);
        heap->task = kthread_run(ion_heap_deferred_free, heap,
                                 "%s", heap->name);