// Copyright (c) 2008 Roberto Raggi // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. #include "FullySpecifiedType.h" #include "Type.h" #include "CoreTypes.h" using namespace CPlusPlus; FullySpecifiedType::FullySpecifiedType(Type *type) : _type(type), _flags(0) { if (! type) _type = UndefinedType::instance(); } FullySpecifiedType::~FullySpecifiedType() { } bool FullySpecifiedType::isValid() const { return _type != UndefinedType::instance(); } Type *FullySpecifiedType::type() const { return _type; } void FullySpecifiedType::setType(Type *type) { _type = type; } FullySpecifiedType FullySpecifiedType::qualifiedType() const { FullySpecifiedType ty = *this; ty.setFriend(false); ty.setRegister(false); ty.setStatic(false); ty.setExtern(false); ty.setMutable(false); ty.setTypedef(false); ty.setInline(false); ty.setVirtual(false); ty.setOverride(false); ty.setFinal(false); ty.setExplicit(false); ty.setDeprecated(false); ty.setUnavailable(false); return ty; } bool FullySpecifiedType::isConst() const { return f._isConst; } void FullySpecifiedType::setConst(bool isConst) { f._isConst = isConst; } bool FullySpecifiedType::isVolatile() const { return f._isVolatile; } void FullySpecifiedType::setVolatile(bool isVolatile) { f._isVolatile = isVolatile; } bool FullySpecifiedType::isSigned() const { return f._isSigned; } void FullySpecifiedType::setSigned(bool isSigned) { f._isSigned = isSigned; } bool FullySpecifiedType::isUnsigned() const { return f._isUnsigned; } void FullySpecifiedType::setUnsigned(bool isUnsigned) { f._isUnsigned = isUnsigned; } bool FullySpecifiedType::isFriend() const { return f._isFriend; } void FullySpecifiedType::setFriend(bool isFriend) { f._isFriend = isFriend; } bool FullySpecifiedType::isAuto() const { return f._isAuto; } void FullySpecifiedType::setAuto(bool isAuto) { f._isAuto = isAuto; } bool FullySpecifiedType::isRegister() const { return f._isRegister; } void FullySpecifiedType::setRegister(bool isRegister) { f._isRegister = isRegister; } bool FullySpecifiedType::isStatic() const { return f._isStatic; } void FullySpecifiedType::setStatic(bool isStatic) { f._isStatic = isStatic; } bool FullySpecifiedType::isExtern() const { return f._isExtern; } void FullySpecifiedType::setExtern(bool isExtern) { f._isExtern = isExtern; } bool FullySpecifiedType::isMutable() const { return f._isMutable; } void FullySpecifiedType::setMutable(bool isMutable) { f._isMutable = isMutable; } bool FullySpecifiedType::isTypedef() const { return f._isTypedef; } void FullySpecifiedType::setTypedef(bool isTypedef) { f._isTypedef = isTypedef; } bool FullySpecifiedType::isInline() const { return f._isInline; } void FullySpecifiedType::setInline(bool isInline) { f._isInline = isInline; } bool FullySpecifiedType::isVirtual() const { return f._isVirtual; } void FullySpecifiedType::setVirtual(bool isVirtual) { f._isVirtual = isVirtual; } bool FullySpecifiedType::isOverride() const { return f._isOverride; } void FullySpecifiedType::setOverride(bool isOverride) { f._isOverride = isOverride; } bool FullySpecifiedType::isFinal() const { return f._isFinal; } void FullySpecifiedType::setFinal(bool isFinal) { f._isFinal = isFinal; } bool FullySpecifiedType::isExplicit() const { return f._isExplicit; } void FullySpecifiedType::setExplicit(bool isExplicit) { f._isExplicit = isExplicit; } bool FullySpecifiedType::isDeprecated() const { return f._isDeprecated; } void FullySpecifiedType::setDeprecated(bool isDeprecated) { f._isDeprecated = isDeprecated; } bool FullySpecifiedType::isUnavailable() const { return f._isUnavailable; } void FullySpecifiedType::setUnavailable(bool isUnavailable) { f._isUnavailable = isUnavailable; } Type &FullySpecifiedType::operator*() { return *_type; } FullySpecifiedType::operator bool() const { return _type != UndefinedType::instance(); } const Type &FullySpecifiedType::operator*() const { return *_type; } Type *FullySpecifiedType::operator->() { return _type; } const Type *FullySpecifiedType::operator->() const { return _type; } bool FullySpecifiedType::operator == (const FullySpecifiedType &other) const { return _type == other._type && _flags == other._flags; } bool FullySpecifiedType::operator != (const FullySpecifiedType &other) const { return ! operator ==(other); } bool FullySpecifiedType::operator < (const FullySpecifiedType &other) const { if (_type == other._type) return _flags < other._flags; return _type < other._type; } FullySpecifiedType FullySpecifiedType::simplified() const { if (const ReferenceType *refTy = type()->asReferenceType()) return refTy->elementType().simplified(); return *this; } unsigned FullySpecifiedType::flags() const { return _flags; } void FullySpecifiedType::setFlags(unsigned flags) { _flags = flags; } bool FullySpecifiedType::match(const FullySpecifiedType &otherTy, Matcher *matcher) const { static const unsigned flagsMask = [](){ FullySpecifiedType ty; ty.f._isConst = true; ty.f._isVolatile = true; ty.f._isSigned = true; ty.f._isUnsigned = true; return ty._flags; }(); if ((_flags & flagsMask) != (otherTy._flags & flagsMask)) return false; return type()->match(otherTy.type(), matcher); }