list_del(&submit->node);
        spin_unlock_irqrestore(&ring->submit_lock, flags);
 
+       /* Update devfreq on transition from active->idle: */
+       mutex_lock(&gpu->active_lock);
+       gpu->active_submits--;
+       WARN_ON(gpu->active_submits < 0);
+       if (!gpu->active_submits)
+               msm_devfreq_idle(gpu);
+       mutex_unlock(&gpu->active_lock);
+
        msm_gem_submit_put(submit);
 }
 
        list_add_tail(&submit->node, &ring->submits);
        spin_unlock_irqrestore(&ring->submit_lock, flags);
 
+       /* Update devfreq on transition from idle->active: */
+       mutex_lock(&gpu->active_lock);
+       if (!gpu->active_submits)
+               msm_devfreq_active(gpu);
+       gpu->active_submits++;
+       mutex_unlock(&gpu->active_lock);
+
        gpu->funcs->submit(gpu, submit);
        priv->lastctx = submit->queue->ctx;
 
        sched_set_fifo_low(gpu->worker->task);
 
        INIT_LIST_HEAD(&gpu->active_list);
+       mutex_init(&gpu->active_lock);
        kthread_init_work(&gpu->retire_work, retire_worker);
        kthread_init_work(&gpu->recover_work, recover_worker);
        kthread_init_work(&gpu->fault_work, fault_worker);
 
 
        /** time: Time of last sampling period. */
        ktime_t time;
+
+       /** idle_time: Time of last transition to idle: */
+       ktime_t idle_time;
+
+       /**
+        * idle_freq:
+        *
+        * Shadow frequency used while the GPU is idle.  From the PoV of
+        * the devfreq governor, we are continuing to sample busyness and
+        * adjust frequency while the GPU is idle, but we use this shadow
+        * value as the GPU is actually clamped to minimum frequency while
+        * it is inactive.
+        */
+       unsigned long idle_freq;
 };
 
 struct msm_gpu {
         */
        struct list_head active_list;
 
+       /**
+        * active_submits:
+        *
+        * The number of submitted but not yet retired submits, used to
+        * determine transitions between active and idle.
+        *
+        * Protected by lock
+        */
+       int active_submits;
+
+       /** lock: protects active_submits and idle/active transitions */
+       struct mutex active_lock;
+
        /* does gpu need hw_init? */
        bool needs_hw_init;
 
 void msm_devfreq_cleanup(struct msm_gpu *gpu);
 void msm_devfreq_resume(struct msm_gpu *gpu);
 void msm_devfreq_suspend(struct msm_gpu *gpu);
+void msm_devfreq_active(struct msm_gpu *gpu);
+void msm_devfreq_idle(struct msm_gpu *gpu);
 
 int msm_gpu_hw_init(struct msm_gpu *gpu);
 
 
 
        opp = devfreq_recommended_opp(dev, freq, flags);
 
+       /*
+        * If the GPU is idle, devfreq is not aware, so just ignore
+        * it's requests
+        */
+       if (gpu->devfreq.idle_freq) {
+               gpu->devfreq.idle_freq = *freq;
+               return 0;
+       }
+
        if (IS_ERR(opp))
                return PTR_ERR(opp);
 
 
 static unsigned long get_freq(struct msm_gpu *gpu)
 {
+       if (gpu->devfreq.idle_freq)
+               return gpu->devfreq.idle_freq;
+
        if (gpu->funcs->gpu_get_freq)
                return gpu->funcs->gpu_get_freq(gpu);
 
 }
 
 static struct devfreq_dev_profile msm_devfreq_profile = {
-       .polling_ms = 10,
+       .timer = DEVFREQ_TIMER_DELAYED,
+       .polling_ms = 50,
        .target = msm_devfreq_target,
        .get_dev_status = msm_devfreq_get_dev_status,
        .get_cur_freq = msm_devfreq_get_cur_freq,
 {
        devfreq_suspend_device(gpu->devfreq.devfreq);
 }
+
+void msm_devfreq_active(struct msm_gpu *gpu)
+{
+       struct msm_gpu_devfreq *df = &gpu->devfreq;
+       struct devfreq_dev_status status;
+       unsigned int idle_time;
+       unsigned long target_freq = df->idle_freq;
+
+       /*
+        * Hold devfreq lock to synchronize with get_dev_status()/
+        * target() callbacks
+        */
+       mutex_lock(&df->devfreq->lock);
+
+       idle_time = ktime_to_ms(ktime_sub(ktime_get(), df->idle_time));
+
+       /*
+        * If we've been idle for a significant fraction of a polling
+        * interval, then we won't meet the threshold of busyness for
+        * the governor to ramp up the freq.. so give some boost
+        */
+       if (idle_time > msm_devfreq_profile.polling_ms/2) {
+               target_freq *= 2;
+       }
+
+       df->idle_freq = 0;
+
+       msm_devfreq_target(&gpu->pdev->dev, &target_freq, 0);
+
+       /*
+        * Reset the polling interval so we aren't inconsistent
+        * about freq vs busy/total cycles
+        */
+       msm_devfreq_get_dev_status(&gpu->pdev->dev, &status);
+
+       mutex_unlock(&df->devfreq->lock);
+}
+
+void msm_devfreq_idle(struct msm_gpu *gpu)
+{
+       struct msm_gpu_devfreq *df = &gpu->devfreq;
+       unsigned long idle_freq, target_freq = 0;
+
+       /*
+        * Hold devfreq lock to synchronize with get_dev_status()/
+        * target() callbacks
+        */
+       mutex_lock(&df->devfreq->lock);
+
+       idle_freq = get_freq(gpu);
+
+       msm_devfreq_target(&gpu->pdev->dev, &target_freq, 0);
+
+       df->idle_time = ktime_get();
+       df->idle_freq = idle_freq;
+
+       mutex_unlock(&df->devfreq->lock);
+}