time.cc revision 10905
12SN/A/* 21762SN/A * Copyright (c) 2003-2005 The Regents of The University of Michigan 32SN/A * All rights reserved. 42SN/A * 52SN/A * Redistribution and use in source and binary forms, with or without 62SN/A * modification, are permitted provided that the following conditions are 72SN/A * met: redistributions of source code must retain the above copyright 82SN/A * notice, this list of conditions and the following disclaimer; 92SN/A * redistributions in binary form must reproduce the above copyright 102SN/A * notice, this list of conditions and the following disclaimer in the 112SN/A * documentation and/or other materials provided with the distribution; 122SN/A * neither the name of the copyright holders nor the names of its 132SN/A * contributors may be used to endorse or promote products derived from 142SN/A * this software without specific prior written permission. 152SN/A * 162SN/A * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 172SN/A * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 182SN/A * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 192SN/A * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 202SN/A * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 212SN/A * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 222SN/A * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 232SN/A * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 242SN/A * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 252SN/A * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 262SN/A * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 272665Ssaidi@eecs.umich.edu * 282665Ssaidi@eecs.umich.edu * Authors: Nathan Binkert 292SN/A */ 302SN/A 318869SAli.Saidi@ARM.com#include <cstdlib> 328869SAli.Saidi@ARM.com#include <ctime> 332SN/A#include <iostream> 347840Snate@binkert.org#include <sstream> 352SN/A 36400SN/A#include "base/time.hh" 377840Snate@binkert.org#include "config/use_posix_clock.hh" 387862Sgblack@eecs.umich.edu#include "sim/core.hh" 397870Sgblack@eecs.umich.edu#include "sim/serialize.hh" 402SN/A 412SN/Ausing namespace std; 422SN/A 437840Snate@binkert.orgvoid 447840Snate@binkert.orgTime::_set(bool monotonic) 452SN/A{ 467840Snate@binkert.org#if USE_POSIX_CLOCK 477840Snate@binkert.org ::clock_gettime(monotonic ? CLOCK_MONOTONIC : CLOCK_REALTIME, &_time); 487840Snate@binkert.org#else 49400SN/A timeval tv; 507840Snate@binkert.org ::gettimeofday(&tv, NULL); 517840Snate@binkert.org operator=(tv); 527840Snate@binkert.org#endif 53400SN/A} 54400SN/A 557862Sgblack@eecs.umich.eduvoid 567862Sgblack@eecs.umich.eduTime::setTick(Tick ticks) 577862Sgblack@eecs.umich.edu{ 587862Sgblack@eecs.umich.edu uint64_t nsecs = ticks / SimClock::Int::ns; 597862Sgblack@eecs.umich.edu set(nsecs / NSEC_PER_SEC, nsecs % NSEC_PER_SEC); 607862Sgblack@eecs.umich.edu} 617862Sgblack@eecs.umich.edu 627862Sgblack@eecs.umich.eduTick 637862Sgblack@eecs.umich.eduTime::getTick() const 647862Sgblack@eecs.umich.edu{ 657862Sgblack@eecs.umich.edu return (nsec() + sec() * NSEC_PER_SEC) * SimClock::Int::ns; 667862Sgblack@eecs.umich.edu} 677862Sgblack@eecs.umich.edu 68400SN/Astring 697840Snate@binkert.orgTime::date(const string &format) const 70400SN/A{ 717840Snate@binkert.org time_t sec = this->sec(); 72400SN/A char buf[256]; 73400SN/A 74400SN/A if (format.empty()) { 753918Ssaidi@eecs.umich.edu#ifdef __SUNPRO_CC 767840Snate@binkert.org ctime_r(&sec, buf, sizeof(buf)); 773918Ssaidi@eecs.umich.edu#else 78400SN/A ctime_r(&sec, buf); 793918Ssaidi@eecs.umich.edu#endif 80400SN/A buf[24] = '\0'; 81400SN/A return buf; 822SN/A } 832SN/A 84400SN/A struct tm *tm = localtime(&sec); 85400SN/A strftime(buf, sizeof(buf), format.c_str(), tm); 86400SN/A return buf; 87400SN/A} 882SN/A 897840Snate@binkert.orgstring 907840Snate@binkert.orgTime::time() const 91400SN/A{ 927840Snate@binkert.org double time = double(*this); 937840Snate@binkert.org double secs = fmod(time, 60.0); 947840Snate@binkert.org double all_mins = floor(time / 60.0); 957840Snate@binkert.org double mins = fmod(all_mins, 60.0); 967840Snate@binkert.org double hours = floor(all_mins / 60.0); 977840Snate@binkert.org 987840Snate@binkert.org stringstream str; 997840Snate@binkert.org 1007840Snate@binkert.org if (hours > 0.0) { 1017840Snate@binkert.org if (hours < 10.0) 1027840Snate@binkert.org str << '0'; 1037840Snate@binkert.org str << hours << ':'; 1047840Snate@binkert.org } 1057840Snate@binkert.org 1067840Snate@binkert.org if (mins > 0.0) { 1077840Snate@binkert.org if (mins < 10.0) 1087840Snate@binkert.org str << '0'; 1097840Snate@binkert.org str << mins << ':'; 1107840Snate@binkert.org } 1117840Snate@binkert.org 1127840Snate@binkert.org if (secs < 10.0 && !str.str().empty()) 1137840Snate@binkert.org str << '0'; 1147840Snate@binkert.org str << secs; 1157840Snate@binkert.org 1167840Snate@binkert.org return str.str(); 117400SN/A} 1182SN/A 1197840Snate@binkert.orgvoid 12010905Sandreas.sandberg@arm.comTime::serialize(const std::string &base, CheckpointOut &cp) const 1217870Sgblack@eecs.umich.edu{ 12210905Sandreas.sandberg@arm.com paramOut(cp, base + ".sec", sec()); 12310905Sandreas.sandberg@arm.com paramOut(cp, base + ".nsec", nsec()); 1247870Sgblack@eecs.umich.edu} 1257870Sgblack@eecs.umich.edu 1267870Sgblack@eecs.umich.eduvoid 12710905Sandreas.sandberg@arm.comTime::unserialize(const std::string &base, CheckpointIn &cp) 1287870Sgblack@eecs.umich.edu{ 1297870Sgblack@eecs.umich.edu time_t secs; 1307870Sgblack@eecs.umich.edu time_t nsecs; 13110905Sandreas.sandberg@arm.com paramIn(cp, base + ".sec", secs); 13210905Sandreas.sandberg@arm.com paramIn(cp, base + ".nsec", nsecs); 1337870Sgblack@eecs.umich.edu sec(secs); 1347870Sgblack@eecs.umich.edu nsec(nsecs); 1357870Sgblack@eecs.umich.edu} 1367870Sgblack@eecs.umich.edu 1377870Sgblack@eecs.umich.eduvoid 1387840Snate@binkert.orgsleep(const Time &time) 139400SN/A{ 1407840Snate@binkert.org timespec ts = time; 1417840Snate@binkert.org 1427840Snate@binkert.org#if USE_POSIX_CLOCK 1437840Snate@binkert.org clock_nanosleep(CLOCK_MONOTONIC, 0, &ts, NULL); 1447840Snate@binkert.org#else 1457840Snate@binkert.org nanosleep(&ts, NULL); 1467840Snate@binkert.org#endif 147400SN/A} 1488869SAli.Saidi@ARM.com 1498869SAli.Saidi@ARM.comtime_t 1508869SAli.Saidi@ARM.commkutctime(struct tm *time) 1518869SAli.Saidi@ARM.com{ 15210531Sandreas.hansson@arm.com // get the current timezone 15310531Sandreas.hansson@arm.com char *tz = getenv("TZ"); 15410531Sandreas.hansson@arm.com 15510531Sandreas.hansson@arm.com // copy the string as the pointer gets invalidated when updating 15610531Sandreas.hansson@arm.com // the environment 15710531Sandreas.hansson@arm.com if (tz) { 15810531Sandreas.hansson@arm.com tz = strdup(tz); 15910531Sandreas.hansson@arm.com if (!tz) { 16010531Sandreas.hansson@arm.com fatal("Failed to reserve memory for UTC time conversion\n"); 16110531Sandreas.hansson@arm.com } 16210531Sandreas.hansson@arm.com } 16310531Sandreas.hansson@arm.com 16410531Sandreas.hansson@arm.com // change to UTC and get the time 16510531Sandreas.hansson@arm.com setenv("TZ", "", 1); 16610531Sandreas.hansson@arm.com tzset(); 16710531Sandreas.hansson@arm.com time_t ret = mktime(time); 16810531Sandreas.hansson@arm.com 16910531Sandreas.hansson@arm.com // restore the timezone again 17010531Sandreas.hansson@arm.com if (tz) { 17110531Sandreas.hansson@arm.com setenv("TZ", tz, 1); 17210531Sandreas.hansson@arm.com free(tz); 17310531Sandreas.hansson@arm.com } else { 17410531Sandreas.hansson@arm.com unsetenv("TZ"); 17510531Sandreas.hansson@arm.com } 17610531Sandreas.hansson@arm.com tzset(); 17710531Sandreas.hansson@arm.com 17810531Sandreas.hansson@arm.com return ret; 1798869SAli.Saidi@ARM.com} 1808869SAli.Saidi@ARM.com 181