/*
 * Copyright (C) 2012 The Android Open Source Project
 *
 * Licensed 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
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#define LOG_TAG "common_time"
#include <utils/Log.h>

#include "utils.h"

namespace android {

void Timeout::setTimeout(int msec) {
    if (msec < 0) {
        mSystemEndTime = 0;
        return;
    }

    mSystemEndTime = systemTime() + (static_cast<nsecs_t>(msec) * 1000000);
}

int Timeout::msecTillTimeout(nsecs_t nowTime) {
    if (!mSystemEndTime) {
        return -1;
    }

    if (mSystemEndTime < nowTime) {
        return 0;
    }

    nsecs_t delta = mSystemEndTime - nowTime;
    delta += 999999;
    delta /= 1000000;
    if (delta > 0x7FFFFFFF) {
        return 0x7FFFFFFF;
    }

    return static_cast<int>(delta);
}

LogRing::LogRing(const char* header, size_t entries)
    : mSize(entries)
    , mWr(0)
    , mIsFull(false)
    , mHeader(header) {
    mRingBuffer = new Entry[mSize];
    if (NULL == mRingBuffer)
        ALOGE("Failed to allocate log ring with %zu entries.", mSize);
}

LogRing::~LogRing() {
    if (NULL != mRingBuffer)
        delete[] mRingBuffer;
}

void LogRing::log(int prio, const char* tag, const char* fmt, ...) {
    va_list argp;
    va_start(argp, fmt);
    internalLog(prio, tag, fmt, argp);
    va_end(argp);
}

void LogRing::log(const char* fmt, ...) {
    va_list argp;
    va_start(argp, fmt);
    internalLog(0, NULL, fmt, argp);
    va_end(argp);
}

void LogRing::internalLog(int prio,
                          const char* tag,
                          const char* fmt,
                          va_list argp) {
    if (NULL != mRingBuffer) {
        Mutex::Autolock lock(&mLock);
        String8 s(String8::formatV(fmt, argp));
        Entry* last = NULL;

        if (mIsFull || mWr)
            last = &(mRingBuffer[(mWr + mSize - 1) % mSize]);


        if ((NULL != last) && !last->s.compare(s)) {
            gettimeofday(&(last->last_ts), NULL);
            ++last->count;
        } else {
            gettimeofday(&mRingBuffer[mWr].first_ts, NULL);
            mRingBuffer[mWr].last_ts = mRingBuffer[mWr].first_ts;
            mRingBuffer[mWr].count = 1;
            mRingBuffer[mWr].s.setTo(s);

            mWr = (mWr + 1) % mSize;
            if (!mWr)
                mIsFull = true;
        }
    }

    if (NULL != tag)
        LOG_PRI_VA(prio, tag, fmt, argp);
}

void LogRing::dumpLog(int fd) {
    if (NULL == mRingBuffer)
        return;

    Mutex::Autolock lock(&mLock);

    if (!mWr && !mIsFull)
        return;

    char buf[1024];
    int res;
    size_t start = mIsFull ? mWr : 0;
    size_t count = mIsFull ? mSize : mWr;
    static const char* kTimeFmt = "%a %b %d %Y %H:%M:%S";

    res = snprintf(buf, sizeof(buf), "\n%s\n", mHeader);
    if (res > 0)
        write(fd, buf, res);

    for (size_t i = 0; i < count; ++i) {
        struct tm t;
        char timebuf[64];
        char repbuf[96];
        size_t ndx = (start + i) % mSize;

        if (1 != mRingBuffer[ndx].count) {
            localtime_r(&mRingBuffer[ndx].last_ts.tv_sec, &t);
            strftime(timebuf, sizeof(timebuf), kTimeFmt, &t);
            snprintf(repbuf, sizeof(repbuf),
                    " (repeated %d times, last was %s.%03ld)",
                     mRingBuffer[ndx].count,
                     timebuf,
                     mRingBuffer[ndx].last_ts.tv_usec / 1000);
            repbuf[sizeof(repbuf) - 1] = 0;
        } else {
            repbuf[0] = 0;
        }

        localtime_r(&mRingBuffer[ndx].first_ts.tv_sec, &t);
        strftime(timebuf, sizeof(timebuf), kTimeFmt, &t);
        res = snprintf(buf, sizeof(buf), "[%2zu] %s.%03ld :: %s%s\n",
                       i, timebuf,
                       mRingBuffer[ndx].first_ts.tv_usec / 1000,
                       mRingBuffer[ndx].s.string(),
                       repbuf);

        if (res > 0)
            write(fd, buf, res);
    }
}

}  // namespace android