blob: b53f1a39057470fc3dffa8b5fca651b37d64a6fb [file] [log] [blame]
Naseer Ahmed29a26812012-06-14 00:56:20 -07001/*
Duy Truong73d36df2013-02-09 20:33:23 -08002 * Copyright (c) 2012, The Linux Foundation. All rights reserved.
Naseer Ahmed29a26812012-06-14 00:56:20 -07003
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above
10 * copyright notice, this list of conditions and the following
11 * disclaimer in the documentation and/or other materials provided
12 * with the distribution.
Duy Truong73d36df2013-02-09 20:33:23 -080013 * * Neither the name of The Linux Foundation nor the names of its
Naseer Ahmed29a26812012-06-14 00:56:20 -070014 * contributors may be used to endorse or promote products derived
15 * from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
Naseer Ahmed7c958d42012-07-31 18:57:03 -070030#include "idle_invalidator.h"
Naseer Ahmed29a26812012-06-14 00:56:20 -070031#include <unistd.h>
32
Naseer Ahmed7c958d42012-07-31 18:57:03 -070033#define II_DEBUG 0
Naseer Ahmed29a26812012-06-14 00:56:20 -070034
35static const char *threadName = "Invalidator";
36InvalidatorHandler IdleInvalidator::mHandler = NULL;
37android::sp<IdleInvalidator> IdleInvalidator::sInstance(0);
38
39IdleInvalidator::IdleInvalidator(): Thread(false), mHwcContext(0),
40 mSleepAgain(false), mSleepTime(0) {
Naseer Ahmed7c958d42012-07-31 18:57:03 -070041 ALOGD_IF(II_DEBUG, "%s", __func__);
Naseer Ahmed29a26812012-06-14 00:56:20 -070042 }
43
44int IdleInvalidator::init(InvalidatorHandler reg_handler, void* user_data,
45 unsigned int idleSleepTime) {
Naseer Ahmed7c958d42012-07-31 18:57:03 -070046 ALOGD_IF(II_DEBUG, "%s", __func__);
Naseer Ahmed29a26812012-06-14 00:56:20 -070047
Prabhanjan Kandula04167cb2013-07-09 15:41:08 +053048 Locker::Autolock _l(mLock);
Naseer Ahmed29a26812012-06-14 00:56:20 -070049 /* store registered handler */
50 mHandler = reg_handler;
51 mHwcContext = user_data;
52 mSleepTime = idleSleepTime; //Time in millis
53 return 0;
54}
55
56bool IdleInvalidator::threadLoop() {
Saurabh Shahb2117fe2014-01-23 18:39:01 -080057 struct timeval lastUpdateTime;
Naseer Ahmed7c958d42012-07-31 18:57:03 -070058 ALOGD_IF(II_DEBUG, "%s", __func__);
Saurabh Shahb2117fe2014-01-23 18:39:01 -080059
60 {
61 //If we are here, update(s) happened, i.e mSleepAgain is set
62 Locker::Autolock _l(mLock);
63 mSleepAgain = false;
64 lastUpdateTime = mLastUpdateTime;
65 }
66
67 struct timeval currentTime;
68 gettimeofday(&currentTime, NULL);
69 int timeSinceUpdateUs = (currentTime.tv_sec - lastUpdateTime.tv_sec) *
70 1000000 + (currentTime.tv_usec - lastUpdateTime.tv_usec);
71 int sleepDurationUs = mSleepTime * 1000 - timeSinceUpdateUs;
72
73 //Sleep only if the duration required is > 1ms, otherwise its not worth it.
74 if(sleepDurationUs > 1000) {
75 usleep(sleepDurationUs);
76 ALOGD_IF(II_DEBUG, "Slept for %d ms", sleepDurationUs / 1000);
77 }
Prabhanjan Kandula04167cb2013-07-09 15:41:08 +053078
79 Locker::Autolock _l(mLock);
Saurabh Shahb2117fe2014-01-23 18:39:01 -080080 //If an update happened while we were asleep, sleep again
Naseer Ahmed29a26812012-06-14 00:56:20 -070081 if(mSleepAgain) {
82 //We need to sleep again!
83 mSleepAgain = false;
84 return true;
85 }
86
Saurabh Shahb2117fe2014-01-23 18:39:01 -080087#if II_DEBUG
88 gettimeofday(&currentTime, NULL);
89 timeSinceUpdateUs = (currentTime.tv_sec - lastUpdateTime.tv_sec) *
90 1000000 + (currentTime.tv_usec - lastUpdateTime.tv_usec);
91 ALOGD("Idle refresh after %dms", timeSinceUpdateUs / 1000);
92#endif
93
Naseer Ahmed29a26812012-06-14 00:56:20 -070094 mHandler((void*)mHwcContext);
95 return false;
96}
97
98int IdleInvalidator::readyToRun() {
Naseer Ahmed7c958d42012-07-31 18:57:03 -070099 ALOGD_IF(II_DEBUG, "%s", __func__);
Naseer Ahmed29a26812012-06-14 00:56:20 -0700100 return 0; /*NO_ERROR*/
101}
102
103void IdleInvalidator::onFirstRef() {
Naseer Ahmed7c958d42012-07-31 18:57:03 -0700104 ALOGD_IF(II_DEBUG, "%s", __func__);
Naseer Ahmed29a26812012-06-14 00:56:20 -0700105}
106
Saurabh Shahb2117fe2014-01-23 18:39:01 -0800107void IdleInvalidator::handleUpdateEvent() {
Prabhanjan Kandula04167cb2013-07-09 15:41:08 +0530108 Locker::Autolock _l(mLock);
Saurabh Shahb2117fe2014-01-23 18:39:01 -0800109 gettimeofday(&mLastUpdateTime, NULL);
Naseer Ahmed29a26812012-06-14 00:56:20 -0700110 mSleepAgain = true;
111 //Triggers the threadLoop to run, if not already running.
112 run(threadName, android::PRIORITY_AUDIO);
113}
114
115IdleInvalidator *IdleInvalidator::getInstance() {
Naseer Ahmed7c958d42012-07-31 18:57:03 -0700116 ALOGD_IF(II_DEBUG, "%s", __func__);
Naseer Ahmed29a26812012-06-14 00:56:20 -0700117 if(sInstance.get() == NULL)
118 sInstance = new IdleInvalidator();
119 return sInstance.get();
120}