mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 00:02:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			200 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Matthew Olsson <mattco@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Format.h>
 | |
| 
 | |
| namespace PDF {
 | |
| 
 | |
| class Object;
 | |
| 
 | |
| class Value {
 | |
| public:
 | |
|     // We store refs as u32, with 18 bits for the index and 14 bits for the
 | |
|     // generation index. The generation index is stored in the higher bits.
 | |
|     // This may need to be rethought later, as the max generation index is
 | |
|     // 2^16 and the max for the object index is probably 2^32 (I don't know
 | |
|     // exactly)
 | |
|     static constexpr auto max_ref_index = (1 << 19) - 1;            // 2 ^ 18 - 1
 | |
|     static constexpr auto max_ref_generation_index = (1 << 15) - 1; // 2 ^ 14 - 1
 | |
| 
 | |
|     Value()
 | |
|         : m_type(Type::Empty)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     struct NullTag {
 | |
|     };
 | |
| 
 | |
|     Value(NullTag)
 | |
|         : m_type(Type::Null)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Value(bool b)
 | |
|         : m_type(Type::Bool)
 | |
|     {
 | |
|         m_as_bool = b;
 | |
|     }
 | |
| 
 | |
|     Value(int i)
 | |
|         : m_type(Type::Int)
 | |
|     {
 | |
|         m_as_int = i;
 | |
|     }
 | |
| 
 | |
|     Value(float f)
 | |
|         : m_type(Type::Float)
 | |
|     {
 | |
|         m_as_float = f;
 | |
|     }
 | |
| 
 | |
|     Value(u32 index, u32 generation_index)
 | |
|         : m_type(Type::Ref)
 | |
|     {
 | |
|         VERIFY(index < max_ref_index);
 | |
|         VERIFY(generation_index < max_ref_generation_index);
 | |
|         m_as_ref = (generation_index << 14) | index;
 | |
|     }
 | |
| 
 | |
|     template<IsObject T>
 | |
|     Value(RefPtr<T> obj)
 | |
|         : m_type(obj ? Type::Object : Type::Empty)
 | |
|     {
 | |
|         if (obj) {
 | |
|             obj->ref();
 | |
|             m_as_object = obj;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     template<IsObject T>
 | |
|     Value(NonnullRefPtr<T> obj)
 | |
|         : m_type(Type::Object)
 | |
|     {
 | |
|         obj->ref();
 | |
|         m_as_object = obj;
 | |
|     }
 | |
| 
 | |
|     Value(Value const& other)
 | |
|     {
 | |
|         *this = other;
 | |
|     }
 | |
| 
 | |
|     ~Value();
 | |
| 
 | |
|     Value& operator=(Value const& other);
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_empty() const { return m_type == Type::Empty; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_null() const { return m_type == Type::Null; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_bool() const { return m_type == Type::Bool; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_int() const { return m_type == Type::Int; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_float() const { return m_type == Type::Float; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_number() const { return is_int() || is_float(); }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_ref() const { return m_type == Type::Ref; }
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_object() const { return m_type == Type::Object; }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE bool as_bool() const
 | |
|     {
 | |
|         VERIFY(is_bool());
 | |
|         return m_as_bool;
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE int as_int() const
 | |
|     {
 | |
|         VERIFY(is_int());
 | |
|         return m_as_int;
 | |
|     }
 | |
| 
 | |
|     template<typename T>
 | |
|     [[nodiscard]] ALWAYS_INLINE bool is_int_type() const
 | |
|     {
 | |
|         if (!is_int())
 | |
|             return false;
 | |
|         auto as_int = static_cast<T>(m_as_int);
 | |
|         return as_int >= NumericLimits<T>::min() && as_int <= NumericLimits<T>::max();
 | |
|     }
 | |
| 
 | |
|     template<typename T>
 | |
|     [[nodiscard]] ALWAYS_INLINE T as_int_type() const
 | |
|     {
 | |
|         VERIFY(is_int_type<T>());
 | |
|         return static_cast<T>(m_as_int);
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE int to_int() const
 | |
|     {
 | |
|         if (is_int())
 | |
|             return as_int();
 | |
|         return static_cast<int>(as_float());
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE float as_float() const
 | |
|     {
 | |
|         VERIFY(is_float());
 | |
|         return m_as_float;
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE float to_float() const
 | |
|     {
 | |
|         if (is_float())
 | |
|             return as_float();
 | |
|         return static_cast<float>(as_int());
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE u32 as_ref_index() const
 | |
|     {
 | |
|         VERIFY(is_ref());
 | |
|         return m_as_ref & 0x3ffff;
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE u32 as_ref_generation_index() const
 | |
|     {
 | |
|         VERIFY(is_ref());
 | |
|         return m_as_ref >> 18;
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE NonnullRefPtr<Object> as_object() const { return *m_as_object; }
 | |
| 
 | |
|     [[nodiscard]] ALWAYS_INLINE explicit operator bool() const { return !is_empty(); }
 | |
| 
 | |
|     [[nodiscard]] String to_string(int indent = 0) const;
 | |
| 
 | |
| private:
 | |
|     enum class Type {
 | |
|         Empty,
 | |
|         Null,
 | |
|         Bool,
 | |
|         Int,
 | |
|         Float,
 | |
|         Ref,
 | |
|         Object,
 | |
|     };
 | |
| 
 | |
|     union {
 | |
|         bool m_as_bool;
 | |
|         int m_as_int;
 | |
|         u32 m_as_ref;
 | |
|         float m_as_float;
 | |
|         Object* m_as_object;
 | |
|     };
 | |
| 
 | |
|     Type m_type;
 | |
| };
 | |
| 
 | |
| }
 | |
| 
 | |
| namespace AK {
 | |
| 
 | |
| template<>
 | |
| struct Formatter<PDF::Value> : Formatter<StringView> {
 | |
|     void format(FormatBuilder& builder, PDF::Value const& value)
 | |
|     {
 | |
|         Formatter<StringView>::format(builder, value.to_string());
 | |
|     }
 | |
| };
 | |
| 
 | |
| }
 | 
