AirTmpInlines.h   [plain text]


/*
 * Copyright (C) 2015-2017 Apple Inc. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. 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.
 *
 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 APPLE INC. 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. 
 */

#pragma once

#if ENABLE(B3_JIT)

#include "AirArg.h"
#include "AirCode.h"
#include "AirTmp.h"

namespace JSC { namespace B3 { namespace Air {

inline Tmp::Tmp(const Arg& arg)
{
    *this = arg.tmp();
}

// When a Hash structure is too slow or when Sets contains most values, you can
// use direct array addressing with Tmps.
template<Bank bank>
struct AbsoluteTmpMapper;

template<>
struct AbsoluteTmpMapper<GP> {
    static unsigned absoluteIndex(const Tmp& tmp)
    {
        ASSERT(tmp.isGP());
        ASSERT(static_cast<int>(tmp.internalValue()) > 0);
        return tmp.internalValue();
    }

    static unsigned absoluteIndex(unsigned tmpIndex)
    {
        return absoluteIndex(Tmp::gpTmpForIndex(tmpIndex));
    }

    static unsigned lastMachineRegisterIndex()
    {
        return absoluteIndex(Tmp(MacroAssembler::lastRegister()));
    }

    static Tmp tmpFromAbsoluteIndex(unsigned tmpIndex)
    {
        return Tmp::tmpForInternalValue(tmpIndex);
    }
};

template<>
struct AbsoluteTmpMapper<FP> {
    static unsigned absoluteIndex(const Tmp& tmp)
    {
        ASSERT(tmp.isFP());
        ASSERT(static_cast<int>(tmp.internalValue()) < 0);
        return -tmp.internalValue();
    }

    static unsigned absoluteIndex(unsigned tmpIndex)
    {
        return absoluteIndex(Tmp::fpTmpForIndex(tmpIndex));
    }

    static unsigned lastMachineRegisterIndex()
    {
        return absoluteIndex(Tmp(MacroAssembler::lastFPRegister()));
    }

    static Tmp tmpFromAbsoluteIndex(unsigned tmpIndex)
    {
        return Tmp::tmpForInternalValue(-tmpIndex);
    }
};

template<Bank theBank>
class Tmp::Indexed : public Tmp {
public:
    static const char* const dumpPrefix;
    
    Indexed(Tmp tmp)
        : Tmp(tmp)
    {
    }
    
    unsigned index() const
    {
        return tmpIndex(theBank);
    }
};

template<Bank theBank>
class Tmp::AbsolutelyIndexed : public Tmp {
public:
    static const char* const dumpPrefix;

    AbsolutelyIndexed(Tmp tmp)
        : Tmp(tmp)
    {
    }
    
    unsigned index() const
    {
        return AbsoluteTmpMapper<theBank>::absoluteIndex(*this);
    }
};

class Tmp::LinearlyIndexed : public Tmp {
public:
    static const char* const dumpPrefix;
    
    LinearlyIndexed(Tmp tmp, Code& code)
        : Tmp(tmp)
        , m_code(&code)
    {
    }
    
    ALWAYS_INLINE unsigned index() const
    {
        if (isGP())
            return AbsoluteTmpMapper<GP>::absoluteIndex(*this);
        return absoluteIndexEnd(*m_code, GP) + AbsoluteTmpMapper<FP>::absoluteIndex(*this);
    }

private:
    Code* m_code;
};

template<Bank theBank>
inline Tmp::Indexed<theBank> Tmp::indexed() const
{
    return Indexed<theBank>(*this);
}

template<Bank theBank>
inline Tmp::AbsolutelyIndexed<theBank> Tmp::absolutelyIndexed() const
{
    return AbsolutelyIndexed<theBank>(*this);
}

inline Tmp::LinearlyIndexed Tmp::linearlyIndexed(Code& code) const
{
    return LinearlyIndexed(*this, code);
}

ALWAYS_INLINE unsigned Tmp::indexEnd(Code& code, Bank bank)
{
    return code.numTmps(bank);
}

ALWAYS_INLINE unsigned Tmp::absoluteIndexEnd(Code& code, Bank bank)
{
    if (bank == GP)
        return AbsoluteTmpMapper<GP>::absoluteIndex(code.numTmps(GP));
    return AbsoluteTmpMapper<FP>::absoluteIndex(code.numTmps(FP));
}

ALWAYS_INLINE unsigned Tmp::linearIndexEnd(Code& code)
{
    return absoluteIndexEnd(code, GP) + absoluteIndexEnd(code, FP);
}

ALWAYS_INLINE Tmp Tmp::tmpForAbsoluteIndex(Bank bank, unsigned index)
{
    if (bank == GP)
        return AbsoluteTmpMapper<GP>::tmpFromAbsoluteIndex(index);
    ASSERT(bank == FP);
    return AbsoluteTmpMapper<FP>::tmpFromAbsoluteIndex(index);
}

ALWAYS_INLINE Tmp Tmp::tmpForLinearIndex(Code& code, unsigned index)
{
    unsigned gpEnd = absoluteIndexEnd(code, GP);
    if (index < gpEnd)
        return tmpForAbsoluteIndex(GP, index);
    return tmpForAbsoluteIndex(FP, index - gpEnd);
}

} } } // namespace JSC::B3::Air

#endif // ENABLE(B3_JIT)