/**
 * Copyright (c) 2020, 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 "carwatchdogd"

#include "ServiceManager.h"

#include <binder/IPCThreadState.h>
#include <binder/ProcessState.h>
#include <log/log.h>
#include <utils/Looper.h>

using ::android::IPCThreadState;
using ::android::Looper;
using ::android::ProcessState;
using ::android::sp;
using ::android::automotive::watchdog::ServiceManager;

const size_t kMaxBinderThreadCount = 16;

int main(int /*argc*/, char** /*argv*/) {
    // Set up the binder
    sp<ProcessState> ps(ProcessState::self());
    ps->setThreadPoolMaxThreadCount(kMaxBinderThreadCount);
    ps->startThreadPool();
    ps->giveThreadPoolName();
    IPCThreadState::self()->disableBackgroundScheduling(true);

    sp<Looper> mainLooper(Looper::prepare(/*opts=*/0));

    auto result = ServiceManager::getInstance()->startServices(mainLooper);
    if (!result.ok()) {
        ALOGE("Failed to start services: %s", result.error().message().c_str());
        ServiceManager::terminate();
        exit(result.error().code());
    }

    // Loop forever -- the health check runs on this thread in a handler, and the binder calls
    // remain responsive in their pool of threads.
    while (true) {
        mainLooper->pollAll(/*timeoutMillis=*/-1);
    }
    ALOGW("Car watchdog server escaped from its loop.");
    ServiceManager::terminate();

    return 0;
}
