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