1 //===- llvm/TextAPI/ArchitectureSet.h - ArchitectureSet ---------*- C++ -*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 // 9 // Defines the architecture set. 10 // 11 //===----------------------------------------------------------------------===// 12 13 #ifndef LLVM_TEXTAPI_MACHO_ARCHITECTURESET_H 14 #define LLVM_TEXTAPI_MACHO_ARCHITECTURESET_H 15 16 #include "llvm/TextAPI/Architecture.h" 17 #include <cstddef> 18 #include <iterator> 19 #include <limits> 20 #include <string> 21 #include <tuple> 22 #include <vector> 23 24 namespace llvm { 25 class raw_ostream; 26 27 namespace MachO { 28 29 class ArchitectureSet { 30 private: 31 using ArchSetType = uint32_t; 32 33 const static ArchSetType EndIndexVal = 34 std::numeric_limits<ArchSetType>::max(); 35 ArchSetType ArchSet{0}; 36 37 public: 38 constexpr ArchitectureSet() = default; 39 constexpr ArchitectureSet(ArchSetType Raw) : ArchSet(Raw) {} 40 ArchitectureSet(Architecture Arch) : ArchitectureSet() { set(Arch); } 41 ArchitectureSet(const std::vector<Architecture> &Archs); 42 43 void set(Architecture Arch) { 44 if (Arch == AK_unknown) 45 return; 46 ArchSet |= 1U << static_cast<int>(Arch); 47 } 48 49 void clear(Architecture Arch) { ArchSet &= ~(1U << static_cast<int>(Arch)); } 50 51 bool has(Architecture Arch) const { 52 return ArchSet & (1U << static_cast<int>(Arch)); 53 } 54 55 bool contains(ArchitectureSet Archs) const { 56 return (ArchSet & Archs.ArchSet) == Archs.ArchSet; 57 } 58 59 size_t count() const; 60 61 bool empty() const { return ArchSet == 0; } 62 63 ArchSetType rawValue() const { return ArchSet; } 64 65 bool hasX86() const { 66 return has(AK_i386) || has(AK_x86_64) || has(AK_x86_64h); 67 } 68 69 template <typename Ty> class arch_iterator { 70 public: 71 using iterator_category = std::forward_iterator_tag; 72 using value_type = Architecture; 73 using difference_type = std::size_t; 74 using pointer = value_type *; 75 using reference = value_type &; 76 77 private: 78 ArchSetType Index; 79 Ty *ArchSet; 80 81 void findNextSetBit() { 82 if (Index == EndIndexVal) 83 return; 84 while (++Index < sizeof(Ty) * 8) { 85 if (*ArchSet & (1UL << Index)) 86 return; 87 } 88 89 Index = EndIndexVal; 90 } 91 92 public: 93 arch_iterator(Ty *ArchSet, ArchSetType Index = 0) 94 : Index(Index), ArchSet(ArchSet) { 95 if (Index != EndIndexVal && !(*ArchSet & (1UL << Index))) 96 findNextSetBit(); 97 } 98 99 Architecture operator*() const { return static_cast<Architecture>(Index); } 100 101 arch_iterator &operator++() { 102 findNextSetBit(); 103 return *this; 104 } 105 106 arch_iterator operator++(int) { 107 auto tmp = *this; 108 findNextSetBit(); 109 return tmp; 110 } 111 112 bool operator==(const arch_iterator &o) const { 113 return std::tie(Index, ArchSet) == std::tie(o.Index, o.ArchSet); 114 } 115 116 bool operator!=(const arch_iterator &o) const { return !(*this == o); } 117 }; 118 119 ArchitectureSet operator&(const ArchitectureSet &o) { 120 return {ArchSet & o.ArchSet}; 121 } 122 123 ArchitectureSet operator|(const ArchitectureSet &o) { 124 return {ArchSet | o.ArchSet}; 125 } 126 127 ArchitectureSet &operator|=(const ArchitectureSet &o) { 128 ArchSet |= o.ArchSet; 129 return *this; 130 } 131 132 ArchitectureSet &operator|=(const Architecture &Arch) { 133 set(Arch); 134 return *this; 135 } 136 137 bool operator==(const ArchitectureSet &o) const { 138 return ArchSet == o.ArchSet; 139 } 140 141 bool operator!=(const ArchitectureSet &o) const { 142 return ArchSet != o.ArchSet; 143 } 144 145 bool operator<(const ArchitectureSet &o) const { return ArchSet < o.ArchSet; } 146 147 using iterator = arch_iterator<ArchSetType>; 148 using const_iterator = arch_iterator<const ArchSetType>; 149 150 iterator begin() { return {&ArchSet}; } 151 iterator end() { return {&ArchSet, EndIndexVal}; } 152 153 const_iterator begin() const { return {&ArchSet}; } 154 const_iterator end() const { return {&ArchSet, EndIndexVal}; } 155 156 operator std::string() const; 157 operator std::vector<Architecture>() const; 158 void print(raw_ostream &OS) const; 159 }; 160 161 inline ArchitectureSet operator|(const Architecture &lhs, 162 const Architecture &rhs) { 163 return ArchitectureSet(lhs) | ArchitectureSet(rhs); 164 } 165 166 raw_ostream &operator<<(raw_ostream &OS, ArchitectureSet Set); 167 168 } // end namespace MachO. 169 } // end namespace llvm. 170 171 #endif // LLVM_TEXTAPI_MACHO_ARCHITECTURESET_H 172