diff --git a/lib/rtti.h b/lib/rtti.h index 223118da9d..bccd8a142e 100644 --- a/lib/rtti.h +++ b/lib/rtti.h @@ -296,18 +296,26 @@ struct Base { }; \ DECLARE_TYPEINFO_COMMON(T, ##__VA_ARGS__) -#define DECLARE_TYPEINFO_COMMON(T, ...) \ - public: \ - static constexpr T *rttiEnabledMarker(T *); \ - using TypeInfo = P4::RTTI::TypeInfo; \ - [[nodiscard]] P4::RTTI::TypeId typeId() const noexcept override { return TypeInfo::id(); } \ - [[nodiscard]] bool isA(P4::RTTI::TypeId typeId) const noexcept override { \ - return TypeInfo::isA(typeId); \ - } \ - \ - protected: \ - [[nodiscard]] const void *toImpl(P4::RTTI::TypeId typeId) const noexcept override { \ - return TypeInfo::isA(typeId) ? TypeInfo::dyn_cast(typeId, this) : nullptr; \ +// Common typeinfo boilerplate methods. Note that they are marked pure / const, so consecutive +// calls to is<> / to<> on the same pointer could be eliminated by a compiler. +// - typeId() / isA are const as they do not access any global state, they +// always return the same value (for same input arguments) +// - toImpl() is pure. It only access global state via its arguments (this pointer). So for +// the same `this` the result is also the same. +#define DECLARE_TYPEINFO_COMMON(T, ...) \ + public: \ + static constexpr T *rttiEnabledMarker(T *); \ + using TypeInfo = P4::RTTI::TypeInfo; \ + [[nodiscard, gnu::const]] P4::RTTI::TypeId typeId() const noexcept override { \ + return TypeInfo::id(); \ + } \ + [[nodiscard, gnu::const]] bool isA(P4::RTTI::TypeId typeId) const noexcept override { \ + return TypeInfo::isA(typeId); \ + } \ + \ + protected: \ + [[nodiscard, gnu::pure]] const void *toImpl(P4::RTTI::TypeId typeId) const noexcept override { \ + return TypeInfo::isA(typeId) ? TypeInfo::dyn_cast(typeId, this) : nullptr; \ } #endif /* LIB_RTTI_H_ */