/*! \file */
/* ************************************************************************
* Copyright (c) 2020-2021 Advanced Micro Devices, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
* ************************************************************************ */

#include "rocsparse.hpp"
#include "rocsparse_bench.hpp"
#include "rocsparse_routine.hpp"
#include "utility.hpp"
#include <iostream>
#include <rocsparse.h>

#include "rocsparse_bench_app.hpp"

//
// REQUIRED ROUTINES:
// - rocsparse_record_timing
// - display_timing_info_stdout_skip_legend
// - display_timing_info_is_stdout_disabled
//
rocsparse_status rocsparse_record_timing(double msec, double gflops, double gbs)
{
    auto* s_bench_app = rocsparse_bench_app::instance();
    if(s_bench_app)
    {
        return s_bench_app->record_timing(msec, gflops, gbs);
    }
    else
    {
        return rocsparse_status_success;
    }
}

bool display_timing_info_stdout_skip_legend()
{
    auto* s_bench_app = rocsparse_bench_app::instance();
    if(s_bench_app)
    {
        return s_bench_app->stdout_skip_legend();
    }
    else
    {
        return false;
    }
}

bool display_timing_info_is_stdout_disabled()
{
    auto* s_bench_app = rocsparse_bench_app::instance();
    if(s_bench_app)
    {
        return s_bench_app->is_stdout_disabled();
    }
    else
    {
        return false;
    }
}

int main(int argc, char* argv[])
{
    if(rocsparse_bench_app::applies(argc, argv))
    {
        try
        {
            auto* s_bench_app = rocsparse_bench_app::instance(argc, argv);
            //
            // RUN CASES.
            //
            rocsparse_status status = s_bench_app->run_cases();
            if(status != rocsparse_status_success)
            {
                return status;
            }

            //
            // EXPORT FILE.
            //
            status = s_bench_app->export_file();
            if(status != rocsparse_status_success)
            {
                return status;
            }

            return status;
        }
        catch(const rocsparse_status& status)
        {
            return status;
        }
    }
    else
    {
        //
        // old style.
        //
        try
        {
            rocsparse_bench bench(argc, argv);

            //
            // Print info devices.
            //
            bench.info_devices(std::cout);

            //
            // Run benchmark.
            //
            rocsparse_status status = bench.run();
            if(status != rocsparse_status_success)
            {
                return status;
            }
            return status;
        }
        catch(const rocsparse_status& status)
        {
            return status;
        }
    }
}
