/*
 * Copyright (C) 2011 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.
 */

#include <common_time/local_clock.h>
#include <utils/String8.h>

#include "common_clock_service.h"
#include "common_clock.h"
#include "common_time_server.h"

namespace android {

sp<CommonClockService> CommonClockService::instantiate(
        CommonTimeServer& timeServer) {
    sp<CommonClockService> tcc = new CommonClockService(timeServer);
    if (tcc == NULL)
        return NULL;

    defaultServiceManager()->addService(ICommonClock::kServiceName, tcc);
    return tcc;
}

status_t CommonClockService::dump(int fd, const Vector<String16>& args) {
    Mutex::Autolock lock(mRegistrationLock);
    return mTimeServer.dumpClockInterface(fd, args, mListeners.size());
}

status_t CommonClockService::isCommonTimeValid(bool* valid,
                                               uint32_t* timelineID) {
    return mTimeServer.isCommonTimeValid(valid, timelineID);
}

status_t CommonClockService::commonTimeToLocalTime(int64_t  commonTime,
                                                   int64_t* localTime) {
    return mTimeServer.getCommonClock().commonToLocal(commonTime, localTime);
}

status_t CommonClockService::localTimeToCommonTime(int64_t  localTime,
                                                   int64_t* commonTime) {
    return mTimeServer.getCommonClock().localToCommon(localTime, commonTime);
}

status_t CommonClockService::getCommonTime(int64_t* commonTime) {
    return localTimeToCommonTime(mTimeServer.getLocalClock().getLocalTime(), commonTime);
}

status_t CommonClockService::getCommonFreq(uint64_t* freq) {
    *freq = mTimeServer.getCommonClock().getCommonFreq();
    return OK;
}

status_t CommonClockService::getLocalTime(int64_t* localTime) {
    *localTime = mTimeServer.getLocalClock().getLocalTime();
    return OK;
}

status_t CommonClockService::getLocalFreq(uint64_t* freq) {
    *freq = mTimeServer.getLocalClock().getLocalFreq();
    return OK;
}

status_t CommonClockService::getEstimatedError(int32_t* estimate) {
    *estimate = mTimeServer.getEstimatedError();
    return OK;
}

status_t CommonClockService::getTimelineID(uint64_t* id) {
    *id = mTimeServer.getTimelineID();
    return OK;
}

status_t CommonClockService::getState(State* state) {
    *state = mTimeServer.getState();
    return OK;
}

status_t CommonClockService::getMasterAddr(struct sockaddr_storage* addr) {
    return mTimeServer.getMasterAddr(addr);
}

status_t CommonClockService::registerListener(
        const sp<ICommonClockListener>& listener) {
    Mutex::Autolock lock(mRegistrationLock);

    {   // scoping for autolock pattern
        Mutex::Autolock lock(mCallbackLock);
        // check whether this is a duplicate
        for (size_t i = 0; i < mListeners.size(); i++) {
            if (mListeners[i]->asBinder() == listener->asBinder())
                return ALREADY_EXISTS;
        }
    }

    mListeners.add(listener);
    mTimeServer.reevaluateAutoDisableState(0 != mListeners.size());
    return listener->asBinder()->linkToDeath(this);
}

status_t CommonClockService::unregisterListener(
        const sp<ICommonClockListener>& listener) {
    Mutex::Autolock lock(mRegistrationLock);
    status_t ret_val = NAME_NOT_FOUND;

    {   // scoping for autolock pattern
        Mutex::Autolock lock(mCallbackLock);
        for (size_t i = 0; i < mListeners.size(); i++) {
            if (mListeners[i]->asBinder() == listener->asBinder()) {
                mListeners[i]->asBinder()->unlinkToDeath(this);
                mListeners.removeAt(i);
                ret_val = OK;
                break;
            }
        }
    }

    mTimeServer.reevaluateAutoDisableState(0 != mListeners.size());
    return ret_val;
}

void CommonClockService::binderDied(const wp<IBinder>& who) {
    Mutex::Autolock lock(mRegistrationLock);

    {   // scoping for autolock pattern
        Mutex::Autolock lock(mCallbackLock);
        for (size_t i = 0; i < mListeners.size(); i++) {
            if (mListeners[i]->asBinder() == who) {
                mListeners.removeAt(i);
                break;
            }
        }
    }

    mTimeServer.reevaluateAutoDisableState(0 != mListeners.size());
}

void CommonClockService::notifyOnTimelineChanged(uint64_t timelineID) {
    Mutex::Autolock lock(mCallbackLock);

    for (size_t i = 0; i < mListeners.size(); i++) {
        mListeners[i]->onTimelineChanged(timelineID);
    }
}

}; // namespace android