/** * @file lllivefile.cpp * * $LicenseInfo:firstyear=2006&license=viewerlgpl$ * Second Life Viewer Source Code * Copyright (C) 2010, Linden Research, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License only. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA * $/LicenseInfo$ */ #include "linden_common.h" #include "lllivefile.h" #include "llframetimer.h" #include "lleventtimer.h" const F32 DEFAULT_CONFIG_FILE_REFRESH = 5.0f; class LLLiveFile::Impl { public: Impl(const std::string& filename, const F32 refresh_period); ~Impl(); bool check(); void changed(); bool mForceCheck; F32 mRefreshPeriod; LLFrameTimer mRefreshTimer; std::string mFilename; time_t mLastModTime; time_t mLastStatTime; bool mLastExists; LLEventTimer* mEventTimer; private: LOG_CLASS(LLLiveFile); }; LLLiveFile::Impl::Impl(const std::string& filename, const F32 refresh_period) : mForceCheck(true), mRefreshPeriod(refresh_period), mFilename(filename), mLastModTime(0), mLastStatTime(0), mLastExists(false), mEventTimer(NULL) { } LLLiveFile::Impl::~Impl() { delete mEventTimer; } LLLiveFile::LLLiveFile(const std::string& filename, const F32 refresh_period) : impl(* new Impl(filename, refresh_period)) { } LLLiveFile::~LLLiveFile() { delete &impl; } bool LLLiveFile::Impl::check() { bool detected_change = false; // Skip the check if not enough time has elapsed and we're not // forcing a check of the file if (mForceCheck || mRefreshTimer.getElapsedTimeF32() >= mRefreshPeriod) { mForceCheck = false; // force only forces one check mRefreshTimer.reset(); // don't check again until mRefreshPeriod has passed // Stat the file to see if it exists and when it was last modified. llstat stat_data; if (LLFile::stat(mFilename, &stat_data)) { // Couldn't stat the file, that means it doesn't exist or is // broken somehow. if (mLastExists) { mLastExists = false; detected_change = true; // no longer existing is a change! LL_DEBUGS() << "detected deleted file '" << mFilename << "'" << LL_ENDL; } } else { // The file exists if ( ! mLastExists ) { // last check, it did not exist - that counts as a change LL_DEBUGS() << "detected created file '" << mFilename << "'" << LL_ENDL; detected_change = true; } else if ( stat_data.st_mtime > mLastModTime ) { // file modification time is newer than last check LL_DEBUGS() << "detected updated file '" << mFilename << "'" << LL_ENDL; detected_change = true; } mLastExists = true; mLastStatTime = stat_data.st_mtime; } } if (detected_change) { LL_INFOS() << "detected file change '" << mFilename << "'" << LL_ENDL; } return detected_change; } void LLLiveFile::Impl::changed() { // we wanted to read this file, and we were successful. mLastModTime = mLastStatTime; } bool LLLiveFile::checkAndReload() { bool changed = impl.check(); if (changed) { if(loadFile()) { impl.changed(); this->changed(); } else { changed = false; } } return changed; } std::string LLLiveFile::filename() const { return impl.mFilename; } namespace { class LiveFileEventTimer : public LLEventTimer { public: LiveFileEventTimer(LLLiveFile& f, F32 refresh) : LLEventTimer(refresh), mLiveFile(f) { } bool tick() override { mLiveFile.checkAndReload(); return FALSE; } private: LLLiveFile& mLiveFile; }; } void LLLiveFile::addToEventTimer() { impl.mEventTimer = new LiveFileEventTimer(*this, impl.mRefreshPeriod); } void LLLiveFile::setRefreshPeriod(F32 seconds) { if (seconds < 0.f) { seconds = -seconds; } impl.mRefreshPeriod = seconds; }