/*
 * Copyright (c) 2011-2013, 2018 ARM Limited
 * 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.
 *
 * 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: Andreas Hansson
 */

/**
 * @file
 * PortProxy Object Declaration.
 *
 * Port proxies are used when non-structural entities need access to
 * the memory system (or structural entities that want to peak into
 * the memory system without making a real memory access).
 *
 * Proxy objects replace the previous FunctionalPort, TranslatingPort
 * and VirtualPort objects, which provided the same functionality as
 * the proxies, but were instances of ports not corresponding to real
 * structural ports of the simulated system. Via the port proxies all
 * the accesses go through an actual port (either the system port,
 * e.g. for processes or initialisation, or a the data port of the
 * CPU, e.g. for threads) and thus are transparent to a potentially
 * distributed memory and automatically adhere to the memory map of
 * the system.
 */

#ifndef __MEM_PORT_PROXY_HH__
#define __MEM_PORT_PROXY_HH__

#include <functional>
#include <limits>

#include "mem/port.hh"
#include "sim/byteswap.hh"

/**
 * This object is a proxy for a port or other object which implements the
 * functional response protocol, to be used for debug accesses.
 *
 * This proxy object is used when non structural entities
 * (e.g. thread contexts, object file loaders) need access to the
 * memory system. It calls the corresponding functions on the underlying
 * protocol, and provides templatized convenience access functions.
 *
 * The addresses are interpreted as physical addresses.
 *
 * @sa SETranslatingProxy
 * @sa FSTranslatingProxy
 */
class PortProxy : FunctionalRequestProtocol
{
  public:
    typedef std::function<void(PacketPtr pkt)> SendFunctionalFunc;

  private:
    SendFunctionalFunc sendFunctional;

    /** Granularity of any transactions issued through this proxy. */
    const unsigned int _cacheLineSize;

    void
    recvFunctionalSnoop(PacketPtr pkt) override
    {
        // Since port proxies aren't anyone else's peer, they should never
        // receive snoops.
        panic("Port proxies should never receive snoops.");
    }

  public:
    PortProxy(SendFunctionalFunc func, unsigned int cacheLineSize) :
        sendFunctional(func), _cacheLineSize(cacheLineSize)
    {}
    PortProxy(const MasterPort &port, unsigned int cacheLineSize) :
        sendFunctional([&port](PacketPtr pkt)->void {
                port.sendFunctional(pkt);
            }), _cacheLineSize(cacheLineSize)
    {}
    virtual ~PortProxy() { }



    /** Fixed functionality for use in base classes. */

    /**
     * Read size bytes memory at physical address and store in p.
     */
    void readBlobPhys(Addr addr, Request::Flags flags,
                      void *p, int size) const;

    /**
     * Write size bytes from p to physical address.
     */
    void writeBlobPhys(Addr addr, Request::Flags flags,
                       const void *p, int size) const;

    /**
     * Fill size bytes starting at physical addr with byte value val.
     */
    void memsetBlobPhys(Addr addr, Request::Flags flags,
                        uint8_t v, int size) const;



    /** Methods to override in base classes */

    /**
     * Read size bytes memory at address and store in p.
     * Returns true on success and false on failure.
     */
    virtual bool
    tryReadBlob(Addr addr, void *p, int size) const
    {
        readBlobPhys(addr, 0, p, size);
        return true;
    }

    /**
     * Write size bytes from p to address.
     * Returns true on success and false on failure.
     */
    virtual bool
    tryWriteBlob(Addr addr, const void *p, int size) const
    {
        writeBlobPhys(addr, 0, p, size);
        return true;
    }

    /**
     * Fill size bytes starting at addr with byte value val.
     * Returns true on success and false on failure.
     */
    virtual bool
    tryMemsetBlob(Addr addr, uint8_t val, int size) const
    {
        memsetBlobPhys(addr, 0, val, size);
        return true;
    }



    /** Higher level interfaces based on the above. */

    /**
     * Same as tryReadBlob, but insists on success.
     */
    void
    readBlob(Addr addr, void *p, int size) const
    {
        if (!tryReadBlob(addr, p, size))
            fatal("readBlob(%#x, ...) failed", addr);
    }

    /**
     * Same as tryWriteBlob, but insists on success.
     */
    void
    writeBlob(Addr addr, const void *p, int size) const
    {
        if (!tryWriteBlob(addr, p, size))
            fatal("writeBlob(%#x, ...) failed", addr);
    }

    /**
     * Same as tryMemsetBlob, but insists on success.
     */
    void
    memsetBlob(Addr addr, uint8_t v, int size) const
    {
        if (!tryMemsetBlob(addr, v, size))
            fatal("memsetBlob(%#x, ...) failed", addr);
    }

    /**
     * Read sizeof(T) bytes from address and return as object T.
     */
    template <typename T>
    T read(Addr address) const;

    /**
     * Write object T to address. Writes sizeof(T) bytes.
     */
    template <typename T>
    void write(Addr address, const T &data) const;

    /**
     * Read sizeof(T) bytes from address and return as object T.
     * Performs endianness conversion from the selected guest to host order.
     */
    template <typename T>
    T read(Addr address, ByteOrder guest_byte_order) const;

    /**
     * Write object T to address. Writes sizeof(T) bytes.
     * Performs endianness conversion from host to the selected guest order.
     */
    template <typename T>
    void write(Addr address, T data, ByteOrder guest_byte_order) const;

    /**
     * Write the string str into guest memory at address addr.
     * Returns true on success and false on failure.
     */
    bool tryWriteString(Addr addr, const char *str) const;

    /**
     * Same as tryWriteString, but insists on success.
     */
    void
    writeString(Addr addr, const char *str) const
    {
        if (!tryWriteString(addr, str))
            fatal("writeString(%#x, ...) failed", addr);
    }

    /**
     * Reads the string at guest address addr into the std::string str.
     * Returns true on success and false on failure.
     */
    bool tryReadString(std::string &str, Addr addr) const;

    /**
     * Same as tryReadString, but insists on success.
     */
    void
    readString(std::string &str, Addr addr) const
    {
        if (!tryReadString(str, addr))
            fatal("readString(%#x, ...) failed", addr);
    }

    /**
     * Reads the string at guest address addr into the char * str, reading up
     * to maxlen characters. The last character read is always a nul
     * terminator. Returns true on success and false on failure.
     */
    bool tryReadString(char *str, Addr addr, size_t maxlen) const;

    /**
     * Same as tryReadString, but insists on success.
     */
    void
    readString(char *str, Addr addr, size_t maxlen) const
    {
        if (!tryReadString(str, addr, maxlen))
            fatal("readString(%#x, ...) failed", addr);
    }
};


template <typename T>
T
PortProxy::read(Addr address) const
{
    T data;
    readBlob(address, &data, sizeof(T));
    return data;
}

template <typename T>
void
PortProxy::write(Addr address, const T &data) const
{
    writeBlob(address, &data, sizeof(T));
}

template <typename T>
T
PortProxy::read(Addr address, ByteOrder byte_order) const
{
    T data;
    readBlob(address, &data, sizeof(T));
    return gtoh(data, byte_order);
}

template <typename T>
void
PortProxy::write(Addr address, T data, ByteOrder byte_order) const
{
    data = htog(data, byte_order);
    writeBlob(address, &data, sizeof(T));
}

#endif // __MEM_PORT_PROXY_HH__
