/*
 * Copyright 2018 Google, Inc.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met: redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer;
 * redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution;
 * neither the name of the copyright holders nor the names of its
 * contributors may be used to endorse or promote products derived from
 * this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 * Authors: Gabe Black
 */

#include "base/logging.hh"
#include "systemc/ext/core/sc_process_handle.hh"

namespace sc_core
{

const char *
sc_unwind_exception::what() const throw()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return "";
}

bool
sc_unwind_exception::is_reset() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

sc_unwind_exception::sc_unwind_exception()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

sc_unwind_exception::sc_unwind_exception(const sc_unwind_exception &)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

sc_unwind_exception::~sc_unwind_exception() throw()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}


const char *
sc_process_b::name()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return "";
}

const char *
sc_process_b::kind()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return "";
}


sc_process_b *
sc_get_curr_process_handle()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return nullptr;
}


sc_process_handle::sc_process_handle()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

sc_process_handle::sc_process_handle(const sc_process_handle &)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

sc_process_handle::sc_process_handle(sc_object *)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

sc_process_handle::~sc_process_handle()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}


bool
sc_process_handle::valid() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}


sc_process_handle &
sc_process_handle::operator = (const sc_process_handle &)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return *this;
}

bool
sc_process_handle::operator == (const sc_process_handle &) const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return true;
}

bool
sc_process_handle::operator != (const sc_process_handle &) const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

bool
sc_process_handle::operator < (const sc_process_handle &) const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

bool
sc_process_handle::swap(sc_process_handle &)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}


const char *
sc_process_handle::name() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return "";
}

sc_curr_proc_kind
sc_process_handle::proc_kind() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return SC_NO_PROC_;
}

const std::vector<sc_object *> &
sc_process_handle::get_child_objects() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return *(const std::vector<sc_object *> *)nullptr;
}

const std::vector<sc_event *> &
sc_process_handle::get_child_events() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return *(const std::vector<sc_event *> *)nullptr;
}

sc_object *
sc_process_handle::get_parent_object() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return (sc_object *)nullptr;
}

sc_object *
sc_process_handle::get_process_object() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return (sc_object *)nullptr;
}

bool
sc_process_handle::dynamic() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

bool
sc_process_handle::terminated() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

const sc_event &
sc_process_handle::terminated_event() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return *(sc_event *)nullptr;
}


void
sc_process_handle::suspend(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::resume(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::disable(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::enable(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::kill(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::reset(sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

bool
sc_process_handle::is_unwinding()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

const sc_event &
sc_process_handle::reset_event() const
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return *(sc_event *)nullptr;
}


void
sc_process_handle::sync_reset_on(
        sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::sync_reset_off(
        sc_descendent_inclusion_info include_descendants)
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
}

void
sc_process_handle::warn_unimpl(const char *func)
{
    warn("%s not implemented.\n", func);
}


sc_process_handle
sc_get_current_process_handle()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return sc_process_handle();
}

bool
sc_is_unwinding()
{
    warn("%s not implemented.\n", __PRETTY_FUNCTION__);
    return false;
}

bool sc_allow_process_control_corners;

} // namespace sc_core
