atomic_t                        head;           /* write position    */
        atomic_t                        events;         /* event limit       */
 
-       atomic_t                        wakeup_head;    /* completed head    */
+       atomic_t                        done_head;      /* completed head    */
        atomic_t                        lock;           /* concurrent writes */
 
+       atomic_t                        wakeup;         /* needs a wakeup    */
+
        struct perf_counter_mmap_page   *user_page;
        void                            *data_pages[0];
 };
 
        struct perf_mmap_data   *data;
        unsigned int            offset;
        unsigned int            head;
-       int                     wakeup;
        int                     nmi;
        int                     overflow;
        int                     locked;
        struct perf_mmap_data *data = handle->data;
        int head, cpu;
 
-       if (handle->wakeup)
-               data->wakeup_head = data->head;
+       data->done_head = data->head;
 
        if (!handle->locked)
                goto out;
         * before we publish the new head, matched by a rmb() in userspace when
         * reading this position.
         */
-       while ((head = atomic_xchg(&data->wakeup_head, 0))) {
+       while ((head = atomic_xchg(&data->done_head, 0)))
                data->user_page->data_head = head;
-               handle->wakeup = 1;
-       }
 
        /*
-        * NMI can happen here, which means we can miss a wakeup_head update.
+        * NMI can happen here, which means we can miss a done_head update.
         */
 
        cpu = atomic_xchg(&data->lock, 0);
        /*
         * Therefore we have to validate we did not indeed do so.
         */
-       if (unlikely(atomic_read(&data->wakeup_head))) {
+       if (unlikely(atomic_read(&data->done_head))) {
                /*
                 * Since we had it locked, we can lock it again.
                 */
                goto again;
        }
 
-       if (handle->wakeup)
+       if (atomic_xchg(&data->wakeup, 0))
                perf_output_wakeup(handle);
 out:
        local_irq_restore(handle->flags);
 
        handle->offset  = offset;
        handle->head    = head;
-       handle->wakeup  = (offset >> PAGE_SHIFT) != (head >> PAGE_SHIFT);
+
+       if ((offset >> PAGE_SHIFT) != (head >> PAGE_SHIFT))
+               atomic_set(&data->wakeup, 1);
 
        return 0;
 
                int events = atomic_inc_return(&data->events);
                if (events >= wakeup_events) {
                        atomic_sub(wakeup_events, &data->events);
-                       handle->wakeup = 1;
+                       atomic_set(&data->wakeup, 1);
                }
        }