gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
bitunion.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2007-2008 The Regents of The University of Michigan
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #ifndef __BASE_BITUNION_HH__
30 #define __BASE_BITUNION_HH__
31 
32 #include <functional>
33 #include <iostream>
34 #include <type_traits>
35 #include <typeinfo>
36 
37 #include "base/bitfield.hh"
39 
40 // The following implements the BitUnion system of defining bitfields
41 //on top of an underlying class. This is done through the pervasive use of
42 //both named and unnamed unions which all contain the same actual storage.
43 //Since they're unioned with each other, all of these storage locations
44 //overlap. This allows all of the bitfields to manipulate the same data
45 //without having to have access to each other. More details are provided with
46 //the individual components.
47 
48 //This class wraps around another which defines getter/setter functions which
49 //manipulate the underlying data. The type of the underlying data and the type
50 //of the bitfield itself are inferred from the argument types of the setter
51 //function.
52 template<class Base>
53 class BitfieldTypeImpl : public Base
54 {
55  static_assert(std::is_empty<Base>::value,
56  "Bitfield base class must be empty.");
57 
58  private:
59 
60  struct TypeDeducer
61  {
62  template<typename>
63  struct T;
64 
65  template<typename C, typename Type1, typename Type2>
66  struct T<void (C::*)(Type1 &, Type2)>
67  {
68  typedef Type1 Storage;
69  typedef Type2 Type;
70  };
71 
72  struct Wrapper : public Base
73  {
74  using Base::setter;
75  };
76 
77  typedef typename T<decltype(&Wrapper::setter)>::Storage Storage;
78  typedef typename T<decltype(&Wrapper::setter)>::Type Type;
79  };
80 
81  protected:
82  typedef typename TypeDeducer::Storage Storage;
83  typedef typename TypeDeducer::Type Type;
84 
85  Type getter(const Storage &storage) const = delete;
86  void setter(Storage &storage, Type val) = delete;
87 
88  BitfieldTypeImpl() = default;
89  BitfieldTypeImpl(const BitfieldTypeImpl &) = default;
90 
92 
93  operator Type () const
94  {
95  return Base::getter(__storage);
96  }
97 
98  Type
100  {
101  Base::setter(__storage, val);
102  return val;
103  }
104 
105  Type
107  {
108  return *this = (Type)other;
109  }
110 };
111 
112 //A wrapper for the above class which allows setting and getting.
113 template<class Base>
114 class BitfieldType : public BitfieldTypeImpl<Base>
115 {
116  protected:
118  using typename Impl::Type;
119 
120  public:
121  BitfieldType() = default;
122  BitfieldType(const BitfieldType &) = default;
123 
124  operator Type () const { return Impl::operator Type(); }
125 
126  Type operator=(const Type val) { return Impl::operator=(val); }
127  Type
129  {
130  return Impl::operator=(other);
131  }
132 };
133 
134 //A wrapper which only supports getting.
135 template<class Base>
136 class BitfieldROType : public BitfieldTypeImpl<Base>
137 {
138  public:
140  using typename Impl::Type;
141 
142  BitfieldROType() = default;
143  BitfieldROType(const BitfieldROType &) = default;
144 
145  Type operator=(BitfieldROType<Base> const &other) = delete;
146  operator Type () const { return Impl::operator Type(); }
147 };
148 
149 //A wrapper which only supports setting.
150 template <class Base>
151 class BitfieldWOType : public BitfieldTypeImpl<Base>
152 {
153  protected:
155  using typename Impl::Type;
156 
157  public:
158  BitfieldWOType() = default;
159  BitfieldWOType(const BitfieldWOType &) = default;
160 
161  Type operator=(const Type val) { return Impl::operator=(val); }
162  Type
164  {
165  return Impl::operator=(other);
166  }
167 };
168 
169 //This namespace is for classes which implement the backend of the BitUnion
170 //stuff. Don't use any of these directly.
172 {
173  template<class Storage, int first, int last>
174  class Unsigned
175  {
176  static_assert(first >= last,
177  "Bitfield ranges must be specified as <msb, lsb>");
178 
179  protected:
180  uint64_t
181  getter(const Storage &storage) const
182  {
183  return bits(storage, first, last);
184  }
185 
186  void
187  setter(Storage &storage, uint64_t val)
188  {
189  replaceBits(storage, first, last, val);
190  }
191  };
192 
193  template<class Storage, int first, int last>
194  class Signed
195  {
196  static_assert(first >= last,
197  "Bitfield ranges must be specified as <msb, lsb>");
198 
199  protected:
200  int64_t
201  getter(const Storage &storage) const
202  {
203  return sext<first - last + 1>(bits(storage, first, last));
204  }
205 
206  void
207  setter(Storage &storage, int64_t val)
208  {
209  replaceBits(storage, first, last, val);
210  }
211  };
212 
213  //This class contains the basic bitfield types which are automatically
214  //available within a BitUnion. They inherit their Storage type from the
215  //containing BitUnion.
216  template<class Storage>
218  {
219  protected:
220 
221  template<int first, int last=first>
223  template<int first, int last=first>
224  using BitfieldRO =
226  template<int first, int last=first>
227  using BitfieldWO =
229 
230  template<int first, int last=first>
231  using SignedBitfield =
233  template<int first, int last=first>
234  using SignedBitfieldRO =
236  template<int first, int last=first>
237  using SignedBitfieldWO =
239  };
240 
241  //When a BitUnion is set up, an underlying class is created which holds
242  //the actual union. This class then inherits from it, and provids the
243  //implementations for various operators. Setting things up this way
244  //prevents having to redefine these functions in every different BitUnion
245  //type. More operators could be implemented in the future, as the need
246  //arises.
247  template <class Base>
248  class BitUnionOperators : public Base
249  {
250  static_assert(sizeof(Base) == sizeof(typename Base::__StorageType),
251  "BitUnion larger than its storage type.");
252 
253  public:
254  BitUnionOperators(typename Base::__StorageType const &val)
255  {
256  Base::__storage = val;
257  }
258 
259  BitUnionOperators(const BitUnionOperators &) = default;
260 
262 
263  //Conversion operators.
264  operator const typename Base::__StorageType () const
265  {
266  return Base::__storage;
267  }
268 
269  //Basic assignment operators.
271  operator=(typename Base::__StorageType const &val)
272  {
273  Base::__storage = val;
274  return *this;
275  }
276 
279  {
280  return operator=(other.__storage);
281  }
282 
283  //Increment and decrement operators.
286  {
287  Base::__storage++;
288  return *this;
289  }
290 
293  {
294  BitUnionOperators ret = *this;
295  operator++();
296  return ret;
297  }
298 
301  {
302  Base::__storage--;
303  return *this;
304  }
305 
308  {
309  BitUnionOperators ret = *this;
310  operator--();
311  return ret;
312  }
313 
314  //Operation and assignment operators
316  operator+=(typename Base::__StorageType const &val)
317  {
318  Base::__storage += val;
319  return *this;
320  }
321 
323  operator-=(typename Base::__StorageType const &val)
324  {
325  Base::__storage -= val;
326  return *this;
327  }
328 
330  operator*=(typename Base::__StorageType const &val)
331  {
332  Base::__storage *= val;
333  return *this;
334  }
335 
337  operator/=(typename Base::__StorageType const &val)
338  {
339  Base::__storage /= val;
340  return *this;
341  }
342 
344  operator%=(typename Base::__StorageType const &val)
345  {
346  Base::__storage %= val;
347  return *this;
348  }
349 
351  operator&=(typename Base::__StorageType const &val)
352  {
353  Base::__storage &= val;
354  return *this;
355  }
356 
358  operator|=(typename Base::__StorageType const &val)
359  {
360  Base::__storage |= val;
361  return *this;
362  }
363 
365  operator^=(typename Base::__StorageType const &val)
366  {
367  Base::__storage ^= val;
368  return *this;
369  }
370 
372  operator<<=(typename Base::__StorageType const &val)
373  {
374  Base::__storage <<= val;
375  return *this;
376  }
377 
379  operator>>=(typename Base::__StorageType const &val)
380  {
381  Base::__storage >>= val;
382  return *this;
383  }
384  };
385 }
386 
387 //This macro is a backend for other macros that specialize it slightly.
388 //First, it creates/extends a namespace "BitfieldUnderlyingClasses" and
389 //sticks the class which has the actual union in it, which
390 //BitfieldOperators above inherits from. Putting these classes in a special
391 //namespace ensures that there will be no collisions with other names as long
392 //as the BitUnion names themselves are all distinct and nothing else uses
393 //the BitfieldUnderlyingClasses namespace, which is unlikely. The class itself
394 //creates a typedef of the "type" parameter called __StorageType. This allows
395 //the type to propagate outside of the macro itself in a controlled way.
396 //Finally, the base storage is defined which BitfieldOperators will refer to
397 //in the operators it defines. This macro is intended to be followed by
398 //bitfield definitions which will end up inside it's union. As explained
399 //above, these is overlayed the __storage member in its entirety by each of the
400 //bitfields which are defined in the union, creating shared storage with no
401 //overhead.
402 #define __BitUnion(type, name) \
403  class BitfieldUnderlyingClasses##name : \
404  public BitfieldBackend::BitfieldTypes<type> \
405  { \
406  protected: \
407  typedef type __StorageType; \
408  friend BitfieldBackend::BitUnionBaseType< \
409  BitfieldBackend::BitUnionOperators< \
410  BitfieldUnderlyingClasses##name> >; \
411  friend BitfieldBackend::BitUnionBaseType< \
412  BitfieldUnderlyingClasses##name>; \
413  public: \
414  union { \
415  type __storage;
416 
425 #define EndBitUnion(name) \
426  }; \
427  }; \
428  typedef BitfieldBackend::BitUnionOperators< \
429  BitfieldUnderlyingClasses##name> name;
430 
431 //This sets up a bitfield which has other bitfields nested inside of it. The
432 //__storage member functions like the "underlying storage" of the top level
433 //BitUnion. Like everything else, it overlays with the top level storage, so
434 //making it a regular bitfield type makes the entire thing function as a
435 //regular bitfield when referred to by itself.
436 #define __SubBitUnion(name, fieldType, ...) \
437  class \
438  { \
439  public: \
440  union { \
441  fieldType<__VA_ARGS__> __storage;
442 
452 #define EndSubBitUnion(name) \
453  }; \
454  inline operator __StorageType () const \
455  { return __storage; } \
456  \
457  inline __StorageType operator = (const __StorageType & _storage) \
458  { return __storage = _storage;} \
459  } name;
460 
467 #define SubBitUnion(name, first, last) \
468  __SubBitUnion(name, Bitfield, first, last)
469 
476 #define SignedSubBitUnion(name, first, last) \
477  __SubBitUnion(name, SignedBitfield, first, last)
478 
484 #define BitUnion(type, name) __BitUnion(type, name)
485 
491 #define BitUnion64(name) __BitUnion(uint64_t, name)
492 #define BitUnion32(name) __BitUnion(uint32_t, name)
493 #define BitUnion16(name) __BitUnion(uint16_t, name)
494 #define BitUnion8(name) __BitUnion(uint8_t, name)
495 
496 
497 //These templates make it possible to define other templates related to
498 //BitUnions without having to refer to internal typedefs or the BitfieldBackend
499 //namespace.
500 
501 //To build a template specialization which works for all BitUnions, accept a
502 //template argument T, and then use BitUnionType<T> as an argument in the
503 //template. To refer to the basic type the BitUnion wraps, use
504 //BitUnionBaseType<T>.
505 
506 //For example:
507 //template <typename T>
508 //void func(BitUnionType<T> u) { BitUnionBaseType<T> b = u; }
509 
510 //Also, BitUnionBaseType can be used on a BitUnion type directly.
511 
515 template <typename T>
517 
518 namespace BitfieldBackend
519 {
520  template<typename T>
522  {
524  };
525 
526  template<typename T>
528  {
530  };
531 }
532 
536 template <typename T>
538 
539 
540 //An STL style hash structure for hashing BitUnions based on their base type.
541 namespace std
542 {
543  template <typename T>
544  struct hash<BitUnionType<T> > : public hash<BitUnionBaseType<T> >
545  {
546  size_t
547  operator() (const BitUnionType<T> &val) const
548  {
549  return hash<BitUnionBaseType<T> >::operator()(val);
550  }
551  };
552 }
553 
554 
555 namespace BitfieldBackend
556 {
557 
558  template<typename T>
559  static inline std::ostream &
560  bitfieldBackendPrinter(std::ostream &os, const T &t)
561  {
562  os << t;
563  return os;
564  }
565 
566  //Since BitUnions are generally numerical values and not character codes,
567  //these specializations attempt to ensure that they get cast to integers
568  //of the appropriate type before printing.
569  template <>
570  inline std::ostream &
571  bitfieldBackendPrinter(std::ostream &os, const char &t)
572  {
573  os << (int)t;
574  return os;
575  }
576 
577  template <>
578  inline std::ostream &
579  bitfieldBackendPrinter(std::ostream &os, const unsigned char &t)
580  {
581  os << (unsigned int)t;
582  return os;
583  }
584 }
585 
592 template <typename T>
593 std::ostream &
594 operator << (std::ostream &os, const BitUnionType<T> &bu)
595 {
597  os, (BitUnionBaseType<T>)bu);
598 }
599 
600 // Specialization for BitUnion types.
601 template <class T>
603 {
604  static bool
605  parse(const std::string &s, BitUnionType<T> &value)
606  {
607  // Zero initialize storage to avoid leaking an uninitialized value
609  auto res = to_number(s, storage);
610  value = storage;
611  return res;
612  }
613 };
614 
615 template <class T>
617 {
618  static void
619  show(std::ostream &os, const BitUnionType<T> &value)
620  {
622  os, static_cast<const BitUnionBaseType<T> &>(value));
623  }
624 };
625 
626 #endif // __BASE_BITUNION_HH__
BitfieldBackend::BitUnionOperators::BitUnionOperators
BitUnionOperators(typename Base::__StorageType const &val)
Definition: bitunion.hh:254
BitfieldBackend::BitUnionOperators::operator&=
BitUnionOperators & operator&=(typename Base::__StorageType const &val)
Definition: bitunion.hh:351
BitfieldBackend::BitUnionOperators::operator--
BitUnionOperators operator--(int)
Definition: bitunion.hh:307
BitfieldTypeImpl::TypeDeducer
Definition: bitunion.hh:60
BitfieldTypeImpl::TypeDeducer::T
Definition: bitunion.hh:63
X86ISA::os
Bitfield< 17 > os
Definition: misc.hh:803
BitfieldType::BitfieldType
BitfieldType()=default
BitfieldBackend::BitUnionOperators::operator%=
BitUnionOperators & operator%=(typename Base::__StorageType const &val)
Definition: bitunion.hh:344
BitfieldBackend::BitUnionOperators::operator^=
BitUnionOperators & operator^=(typename Base::__StorageType const &val)
Definition: bitunion.hh:365
BitfieldTypeImpl::__storage
Storage __storage
Definition: bitunion.hh:91
BitfieldBackend::BitUnionOperators::operator-=
BitUnionOperators & operator-=(typename Base::__StorageType const &val)
Definition: bitunion.hh:323
BitfieldTypeImpl::TypeDeducer::Storage
T< decltype(&Wrapper::setter)>::Storage Storage
Definition: bitunion.hh:77
BitfieldROType::BitfieldROType
BitfieldROType()=default
BitfieldBackend::BitUnionOperators::operator+=
BitUnionOperators & operator+=(typename Base::__StorageType const &val)
Definition: bitunion.hh:316
BitfieldBackend::Signed::getter
int64_t getter(const Storage &storage) const
Definition: bitunion.hh:201
BitfieldBackend::BitUnionBaseType
Definition: bitunion.hh:521
BitfieldTypeImpl::TypeDeducer::T< void(C::*)(Type1 &, Type2)>::Type
Type2 Type
Definition: bitunion.hh:69
BitfieldTypeImpl::BitfieldTypeImpl
BitfieldTypeImpl()=default
BitfieldTypeImpl::Storage
TypeDeducer::Storage Storage
Definition: bitunion.hh:82
BitfieldTypeImpl::TypeDeducer::Wrapper
Definition: bitunion.hh:72
BitfieldTypeImpl::getter
Type getter(const Storage &storage) const =delete
ShowParam
Definition: serialize_handlers.hh:122
BitfieldTypeImpl::Type
TypeDeducer::Type Type
Definition: bitunion.hh:83
sext
constexpr uint64_t sext(uint64_t val)
Sign-extend an N-bit value to 64 bits.
Definition: bitfield.hh:121
BitfieldWOType::operator=
Type operator=(const Type val)
Definition: bitunion.hh:161
BitfieldBackend::Unsigned::setter
void setter(Storage &storage, uint64_t val)
Definition: bitunion.hh:187
to_number
bool to_number(const std::string &value, VecPredRegContainer< NumBits, Packed > &p)
Helper functions used for serialization/de-serialization.
Definition: vec_pred_reg.hh:379
BitfieldBackend::BitUnionOperators::operator<<=
BitUnionOperators & operator<<=(typename Base::__StorageType const &val)
Definition: bitunion.hh:372
bitfield.hh
BitfieldBackend::Signed::setter
void setter(Storage &storage, int64_t val)
Definition: bitunion.hh:207
BitfieldTypeImpl::setter
void setter(Storage &storage, Type val)=delete
BitfieldTypeImpl::TypeDeducer::Type
T< decltype(&Wrapper::setter)>::Type Type
Definition: bitunion.hh:78
ParseParam
Definition: serialize_handlers.hh:75
BitfieldBackend::BitUnionOperators::operator/=
BitUnionOperators & operator/=(typename Base::__StorageType const &val)
Definition: bitunion.hh:337
BitfieldBackend::BitUnionOperators::BitUnionOperators
BitUnionOperators()
Definition: bitunion.hh:261
BitfieldBackend::BitUnionOperators::operator--
BitUnionOperators & operator--()
Definition: bitunion.hh:300
BitfieldBackend::BitUnionOperators::operator=
BitUnionOperators & operator=(BitUnionOperators const &other)
Definition: bitunion.hh:278
BitfieldBackend::BitUnionOperators::operator++
BitUnionOperators operator++(int)
Definition: bitunion.hh:292
BitfieldBackend::BitUnionOperators::operator>>=
BitUnionOperators & operator>>=(typename Base::__StorageType const &val)
Definition: bitunion.hh:379
BitfieldType
Definition: bitunion.hh:114
BitfieldBackend::BitUnionOperators
Definition: bitunion.hh:248
X86ISA::val
Bitfield< 63 > val
Definition: misc.hh:769
BitfieldBackend::Signed
Definition: bitunion.hh:194
BitfieldTypeImpl::TypeDeducer::T< void(C::*)(Type1 &, Type2)>::Storage
Type1 Storage
Definition: bitunion.hh:68
BitfieldBackend::BitUnionBaseType< BitUnionType< T > >::Type
BitUnionType< T >::__StorageType Type
Definition: bitunion.hh:529
BitfieldBackend::BitUnionOperators::operator|=
BitUnionOperators & operator|=(typename Base::__StorageType const &val)
Definition: bitunion.hh:358
BitfieldBackend::Unsigned::getter
uint64_t getter(const Storage &storage) const
Definition: bitunion.hh:181
BitfieldTypeImpl::operator=
Type operator=(const Type val)
Definition: bitunion.hh:99
BitfieldType::operator=
Type operator=(BitfieldType< Base > const &other)
Definition: bitunion.hh:128
BitfieldTypeImpl
Definition: bitunion.hh:53
BitfieldWOType::BitfieldWOType
BitfieldWOType()=default
BitfieldBackend::Unsigned
Definition: bitunion.hh:174
serialize_handlers.hh
std
Overload hash function for BasicBlockRange type.
Definition: vec_reg.hh:587
ParseParam< BitUnionType< T > >::parse
static bool parse(const std::string &s, BitUnionType< T > &value)
Definition: bitunion.hh:605
ArmISA::t
Bitfield< 5 > t
Definition: miscregs_types.hh:67
BitfieldBackend::bitfieldBackendPrinter
static std::ostream & bitfieldBackendPrinter(std::ostream &os, const T &t)
Definition: bitunion.hh:560
BitfieldBackend::BitUnionOperators::operator*=
BitUnionOperators & operator*=(typename Base::__StorageType const &val)
Definition: bitunion.hh:330
operator<<
std::ostream & operator<<(std::ostream &os, const BitUnionType< T > &bu)
A default << operator which casts a bitunion to its underlying type and passes it to BitfieldBackend:...
Definition: bitunion.hh:594
bits
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition: bitfield.hh:73
BitfieldROType
Definition: bitunion.hh:136
BitfieldType::operator=
Type operator=(const Type val)
Definition: bitunion.hh:126
BitfieldBackend::BitUnionOperators::operator=
BitUnionOperators & operator=(typename Base::__StorageType const &val)
Definition: bitunion.hh:271
ArmISA::s
Bitfield< 4 > s
Definition: miscregs_types.hh:556
BitfieldBackend::BitUnionOperators::operator++
BitUnionOperators & operator++()
Definition: bitunion.hh:285
BitfieldWOType
Definition: bitunion.hh:151
ShowParam::show
static void show(std::ostream &os, const T &value)
Definition: serialize_handlers.hh:124
BitfieldROType::operator=
Type operator=(BitfieldROType< Base > const &other)=delete
BitfieldBackend::BitUnionBaseType::Type
BitUnionType< T >::__StorageType Type
Definition: bitunion.hh:523
BitfieldBackend::BitfieldTypes
Definition: bitunion.hh:217
BitfieldTypeImpl::operator=
Type operator=(BitfieldTypeImpl< Base > const &other)
Definition: bitunion.hh:106
replaceBits
constexpr void replaceBits(T &val, unsigned first, unsigned last, B bit_val)
A convenience function to replace bits first to last of val with bit_val in place.
Definition: bitfield.hh:174
BitfieldWOType::operator=
Type operator=(BitfieldWOType< Base > const &other)
Definition: bitunion.hh:163
BitUnionBaseType
typename BitfieldBackend::BitUnionBaseType< T >::Type BitUnionBaseType
Definition: bitunion.hh:537
ShowParam< BitUnionType< T > >::show
static void show(std::ostream &os, const BitUnionType< T > &value)
Definition: bitunion.hh:619
BitfieldBackend
Definition: bitunion.hh:171

Generated on Tue Jun 22 2021 15:28:25 for gem5 by doxygen 1.8.17