Files
libreoffice/vcl/source/app/scheduler.cxx
Jan-Marek Glogowski 17bb382624 Convert bTimer => bIdle
All other places already refer to being Idle, so change the
Scheduler::ProcessTaskScheduling argument to bIdle and adapt
all other scheduler-related functions.

Change-Id: If5a605abbc3e620092127b65ada29f11215a0343
2017-01-17 16:08:47 +01:00

380 lines
11 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <svdata.hxx>
#include <tools/time.hxx>
#include <vcl/scheduler.hxx>
#include <saltimer.hxx>
#include <svdata.hxx>
#include <salinst.hxx>
namespace {
const sal_uInt64 MaximumTimeoutMs = 1000 * 60; // 1 minute
void InitSystemTimer(ImplSVData* pSVData);
}
void ImplSchedulerData::Invoke()
{
DBG_TESTSOLARMUTEX();
assert(!mbInScheduler);
if (mbDelete || mbInScheduler )
return;
// prepare Scheduler Object for deletion after handling
mpScheduler->SetDeletionFlags();
// tdf#92036 Reset the period to avoid re-firing immediately.
mpScheduler->mpSchedulerData->mnUpdateTime = tools::Time::GetSystemTicks();
// invoke it
mbInScheduler = true;
mpScheduler->Invoke();
mbInScheduler = false;
}
ImplSchedulerData *ImplSchedulerData::GetMostImportantTask( bool bIdle, sal_uInt64 nTimeNow )
{
ImplSVData* pSVData = ImplGetSVData();
ImplSchedulerData *pMostUrgent = nullptr;
for ( ImplSchedulerData *pSchedulerData = pSVData->mpFirstSchedulerData; pSchedulerData; pSchedulerData = pSchedulerData->mpNext )
{
if ( !pSchedulerData->mpScheduler || pSchedulerData->mbDelete || pSchedulerData->mbInScheduler ||
!pSchedulerData->mpScheduler->ReadyForSchedule( bIdle, nTimeNow ) ||
!pSchedulerData->mpScheduler->IsActive())
continue;
if (!pMostUrgent)
pMostUrgent = pSchedulerData;
else
{
// Find the highest priority.
// If the priority of the current task is higher (numerical value is lower) than
// the priority of the most urgent, the current task gets the new most urgent.
if ( pSchedulerData->mpScheduler->GetPriority() < pMostUrgent->mpScheduler->GetPriority() )
pMostUrgent = pSchedulerData;
}
}
return pMostUrgent;
}
void Scheduler::SetDeletionFlags()
{
mpSchedulerData->mbDelete = true;
mbActive = false;
}
void Scheduler::ImplDeInitScheduler()
{
ImplSVData* pSVData = ImplGetSVData();
ImplSchedulerData* pSchedulerData = pSVData->mpFirstSchedulerData;
if (pSVData->mpSalTimer)
{
pSVData->mpSalTimer->Stop();
}
if ( pSchedulerData )
{
do
{
ImplSchedulerData* pTempSchedulerData = pSchedulerData;
if ( pSchedulerData->mpScheduler )
{
pSchedulerData->mpScheduler->mbActive = false;
pSchedulerData->mpScheduler->mpSchedulerData = nullptr;
}
pSchedulerData = pSchedulerData->mpNext;
delete pTempSchedulerData;
}
while ( pSchedulerData );
pSVData->mpFirstSchedulerData = nullptr;
pSVData->mnTimerPeriod = 0;
}
delete pSVData->mpSalTimer;
pSVData->mpSalTimer = nullptr;
}
/**
* Start a new timer if we need to for nMS duration.
*
* if this is longer than the existing duration we're
* waiting for, do nothing - unless bForce - which means
* to reset the minimum period; used by the scheduled itself.
*/
void Scheduler::ImplStartTimer(sal_uInt64 nMS, bool bForce)
{
ImplSVData* pSVData = ImplGetSVData();
if (pSVData->mbDeInit)
{
// do not start new timers during shutdown - if that happens after
// ImplSalStopTimer() on WNT the timer queue is restarted and never ends
return;
}
DBG_TESTSOLARMUTEX();
InitSystemTimer(pSVData);
if ( !nMS )
nMS = 1;
// Only if smaller timeout, to avoid skipping.
if (bForce || nMS < pSVData->mnTimerPeriod)
{
pSVData->mnTimerPeriod = nMS;
pSVData->mpSalTimer->Start(nMS);
}
}
namespace {
/**
* Initialize the platform specific timer on which all the
* platform independent timers are built
*/
void InitSystemTimer(ImplSVData* pSVData)
{
assert(pSVData != nullptr);
if (!pSVData->mpSalTimer)
{
pSVData->mnTimerPeriod = MaximumTimeoutMs;
pSVData->mpSalTimer = pSVData->mpDefInst->CreateSalTimer();
pSVData->mpSalTimer->SetCallback(Scheduler::CallbackTaskScheduling);
}
}
}
void Scheduler::CallbackTaskScheduling( bool bIdle )
{
// this function is for the saltimer callback
Scheduler::ProcessTaskScheduling( bIdle );
}
bool Scheduler::ProcessTaskScheduling( bool bIdle )
{
ImplSchedulerData* pSchedulerData;
sal_uInt64 nTime = tools::Time::GetSystemTicks();
DBG_TESTSOLARMUTEX();
if ((pSchedulerData = ImplSchedulerData::GetMostImportantTask(bIdle, nTime)))
{
SAL_INFO("vcl.schedule", "Invoke task " << pSchedulerData->GetDebugName());
pSchedulerData->mnUpdateTime = nTime;
pSchedulerData->Invoke();
return true;
}
else
return false;
}
static bool g_bDeterministicMode = false;
void Scheduler::SetDeterministicMode(bool bDeterministic)
{
g_bDeterministicMode = bDeterministic;
}
bool Scheduler::GetDeterministicMode()
{
return g_bDeterministicMode;
}
sal_uInt64 Scheduler::CalculateMinimumTimeout( bool &bHasActiveIdles )
{
ImplSchedulerData* pSchedulerData = nullptr;
ImplSchedulerData* pPrevSchedulerData = nullptr;
ImplSVData* pSVData = ImplGetSVData();
sal_uInt64 nTime = tools::Time::GetSystemTicks();
sal_uInt64 nMinPeriod = MaximumTimeoutMs;
DBG_TESTSOLARMUTEX();
SAL_INFO("vcl.schedule", "Calculating minimum timeout:");
pSchedulerData = pSVData->mpFirstSchedulerData;
while ( pSchedulerData )
{
ImplSchedulerData *pNext = pSchedulerData->mpNext;
// Should Task be released from scheduling?
if ( !pSchedulerData->mbInScheduler &&
pSchedulerData->mbDelete )
{
if ( pPrevSchedulerData )
pPrevSchedulerData->mpNext = pSchedulerData->mpNext;
else
pSVData->mpFirstSchedulerData = pSchedulerData->mpNext;
if ( pSchedulerData->mpScheduler )
pSchedulerData->mpScheduler->mpSchedulerData = nullptr;
pNext = pSchedulerData->mpNext;
delete pSchedulerData;
}
else
{
if (!pSchedulerData->mbInScheduler)
{
if ( !pSchedulerData->mpScheduler->IsIdle() )
{
sal_uInt64 nOldMinPeriod = nMinPeriod;
nMinPeriod = pSchedulerData->mpScheduler->UpdateMinPeriod(
nOldMinPeriod, nTime );
SAL_INFO("vcl.schedule", "Have active timer '" <<
pSchedulerData->GetDebugName() <<
"' update min period from " << nOldMinPeriod <<
" to " << nMinPeriod);
assert( nMinPeriod <= nOldMinPeriod );
if ( nMinPeriod > nOldMinPeriod )
{
nMinPeriod = nOldMinPeriod;
SAL_WARN("vcl.schedule",
"New update min period > old period - using old");
}
}
else
{
SAL_INFO("vcl.schedule", "Have active idle '" <<
pSchedulerData->GetDebugName() << "'");
bHasActiveIdles = true;
}
}
pPrevSchedulerData = pSchedulerData;
}
pSchedulerData = pNext;
}
// delete clock if no more timers available,
if ( !pSVData->mpFirstSchedulerData )
{
if ( pSVData->mpSalTimer )
pSVData->mpSalTimer->Stop();
nMinPeriod = MaximumTimeoutMs;
pSVData->mnTimerPeriod = nMinPeriod;
SAL_INFO("vcl.schedule", "Unusual - no more timers available - stop timer");
}
else
{
Scheduler::ImplStartTimer(nMinPeriod, true);
SAL_INFO("vcl.schedule", "Calculated minimum timeout as " << nMinPeriod << " and " <<
(bHasActiveIdles ? "has active idles" : "no idles"));
}
return nMinPeriod;
}
void Scheduler::Start()
{
ImplSVData *const pSVData = ImplGetSVData();
if (pSVData->mbDeInit)
{
return;
}
DBG_TESTSOLARMUTEX();
// Mark timer active
mbActive = true;
if ( !mpSchedulerData )
{
// insert Scheduler
mpSchedulerData = new ImplSchedulerData;
mpSchedulerData->mpScheduler = this;
mpSchedulerData->mbInScheduler = false;
// insert last due to SFX!
ImplSchedulerData* pPrev = nullptr;
ImplSchedulerData* pData = pSVData->mpFirstSchedulerData;
while ( pData )
{
pPrev = pData;
pData = pData->mpNext;
}
mpSchedulerData->mpNext = nullptr;
if ( pPrev )
pPrev->mpNext = mpSchedulerData;
else
pSVData->mpFirstSchedulerData = mpSchedulerData;
}
mpSchedulerData->mbDelete = false;
mpSchedulerData->mnUpdateTime = tools::Time::GetSystemTicks();
}
void Scheduler::Stop()
{
mbActive = false;
if ( mpSchedulerData )
mpSchedulerData->mbDelete = true;
}
Scheduler& Scheduler::operator=( const Scheduler& rScheduler )
{
if ( IsActive() )
Stop();
mbActive = false;
mePriority = rScheduler.mePriority;
if ( rScheduler.IsActive() )
Start();
return *this;
}
Scheduler::Scheduler(const sal_Char *pDebugName):
mpSchedulerData(nullptr),
mpDebugName(pDebugName),
mePriority(SchedulerPriority::HIGH),
mbActive(false)
{
}
Scheduler::Scheduler( const Scheduler& rScheduler ):
mpSchedulerData(nullptr),
mpDebugName(rScheduler.mpDebugName),
mePriority(rScheduler.mePriority),
mbActive(false)
{
if ( rScheduler.IsActive() )
Start();
}
Scheduler::~Scheduler()
{
if ( mpSchedulerData )
{
mpSchedulerData->mbDelete = true;
mpSchedulerData->mpScheduler = nullptr;
}
}
const char *ImplSchedulerData::GetDebugName() const
{
return mpScheduler && mpScheduler->GetDebugName() ?
mpScheduler->GetDebugName() : "unknown";
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */