#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <chrono>
#include <omp.h>
#include <cstdlib>
#include <malloc.h>
#include <time.h>

#include "sgx_tseal.h"
#include "sgx_urts.h"
#include "Enclave_u.h"

/* OCall functions */
void ocall_print_string(const char *str)
{
    /* Proxy/Bridge will check the length and null-terminate 
    * the input string to prevent buffer overflow. 
    */
    printf("%s", str);
}

thread_local std::chrono::time_point<std::chrono::high_resolution_clock> start;
double sgx_exe_time = 0.0;

void ocall_start_clock()
{
    start = std::chrono::high_resolution_clock::now();
}

void ocall_end_clock(const char * str)
{
    auto finish = std::chrono::high_resolution_clock::now();
    std::chrono::duration<double, std::milli> elapsed = finish - start;
    printf(str, elapsed.count());
}

void ocall_log_clock()
{
    auto finish = std::chrono::high_resolution_clock::now();
    std::chrono::duration<double, std::milli> elapsed = finish - start;
    sgx_exe_time += elapsed.count();
}

double ocall_get_time()
{
    auto now = std::chrono::high_resolution_clock::now();
    return std::chrono::time_point_cast<std::chrono::microseconds>(now).time_since_epoch().count();
}