/*
 * Copyright (c) 2011-2012,2015,2018,2020 ARM Limited
 * Copyright (c) 2013 Advanced Micro Devices, Inc.
 * All rights reserved
 *
 * The license below extends only to copyright in the software and shall
 * not be construed as granting a license to any other intellectual
 * property including but not limited to intellectual property relating
 * to a hardware implementation of the functionality of the software
 * licensed hereunder.  You may use the software subject to the license
 * terms below provided that you ensure that this notice is replicated
 * unmodified and in its entirety in all distributions of the software,
 * modified or unmodified, in source code or in binary form.
 *
 * Copyright (c) 2002-2005 The Regents of The University of Michigan
 * All rights reserved.
 *
 * 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.
 */

#ifndef __CPU_SIMPLE_BASE_HH__
#define __CPU_SIMPLE_BASE_HH__

#include <memory>

#include "arch/generic/pcstate.hh"
#include "base/statistics.hh"
#include "cpu/base.hh"
#include "cpu/checker/cpu.hh"
#include "cpu/exec_context.hh"
#include "cpu/pc_event.hh"
#include "cpu/simple_thread.hh"
#include "cpu/static_inst.hh"
#include "mem/packet.hh"
#include "mem/port.hh"
#include "mem/request.hh"
#include "sim/eventq.hh"
#include "sim/full_system.hh"
#include "sim/system.hh"

namespace gem5
{

// forward declarations
class Checkpoint;
class Process;
class Processor;
class ThreadContext;

namespace Trace
{
    class InstRecord;
}

struct BaseSimpleCPUParams;
namespace branch_prediction
{
    class BPredUnit;
} // namespace branch_prediction
class SimpleExecContext;

class BaseSimpleCPU : public BaseCPU
{
  protected:
    ThreadID curThread;
    branch_prediction::BPredUnit *branchPred;

    const RegIndex zeroReg;

    void checkPcEventQueue();
    void swapActiveThread();

  public:
    BaseSimpleCPU(const BaseSimpleCPUParams &params);
    virtual ~BaseSimpleCPU();
    void wakeup(ThreadID tid) override;
  public:
    Trace::InstRecord *traceData;
    CheckerCPU *checker;

    std::vector<SimpleExecContext*> threadInfo;
    std::list<ThreadID> activeThreads;

    /** Current instruction */
    StaticInstPtr curStaticInst;
    StaticInstPtr curMacroStaticInst;

  protected:
    enum Status
    {
        Idle,
        Running,
        Faulting,
        ITBWaitResponse,
        IcacheRetry,
        IcacheWaitResponse,
        IcacheWaitSwitch,
        DTBWaitResponse,
        DcacheRetry,
        DcacheWaitResponse,
        DcacheWaitSwitch,
    };

    Status _status;

    /**
     * Handler used when encountering a fault; its purpose is to
     * tear down the InstRecord. If a fault is meant to be traced,
     * the handler won't delete the record and it will annotate
     * the record as coming from a faulting instruction.
     */
    void traceFault();

    std::unique_ptr<PCStateBase> preExecuteTempPC;

  public:
    void checkForInterrupts();
    void setupFetchRequest(const RequestPtr &req);
    void serviceInstCountEvents();
    void preExecute();
    void postExecute();
    void advancePC(const Fault &fault);

    void haltContext(ThreadID thread_num) override;

    // statistics
    void resetStats() override;

    virtual Fault
    readMem(Addr addr, uint8_t* data, unsigned size, Request::Flags flags,
            const std::vector<bool>& byte_enable=std::vector<bool>())
    {
        panic("readMem() is not implemented");
    }

    virtual Fault
    initiateMemRead(Addr addr, unsigned size, Request::Flags flags,
            const std::vector<bool>& byte_enable=std::vector<bool>())
    {
        panic("initiateMemRead() is not implemented\n");
    }

    virtual Fault
    writeMem(uint8_t* data, unsigned size, Addr addr, Request::Flags flags,
            uint64_t* res,
            const std::vector<bool>& byte_enable=std::vector<bool>())
    {
        panic("writeMem() is not implemented\n");
    }

    virtual Fault
    amoMem(Addr addr, uint8_t* data, unsigned size, Request::Flags flags,
            AtomicOpFunctorPtr amo_op)
    {
        panic("amoMem() is not implemented\n");
    }

    virtual Fault
    initiateMemAMO(Addr addr, unsigned size, Request::Flags flags,
            AtomicOpFunctorPtr amo_op)
    {
        panic("initiateMemAMO() is not implemented\n");
    }

    void countInst();
    Counter totalInsts() const override;
    Counter totalOps() const override;

    void serializeThread(CheckpointOut &cp, ThreadID tid) const override;
    void unserializeThread(CheckpointIn &cp, ThreadID tid) override;

    /** Hardware transactional memory commands (HtmCmds), e.g. start a
     * transaction and commit a transaction, are memory operations but are
     * neither really (true) loads nor stores. For this reason the interface
     * is extended and initiateHtmCmd() is used to instigate the command. */
    virtual Fault initiateHtmCmd(Request::Flags flags) = 0;
};

} // namespace gem5

#endif // __CPU_SIMPLE_BASE_HH__
