123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- /*
- * Copyright (c) 2021-2022, Matthew Olsson <mattco@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #pragma once
- #include <AK/NonnullRefPtrVector.h>
- #include <AK/SourceLocation.h>
- #include <AK/WeakPtr.h>
- #include <LibPDF/Object.h>
- #include <LibPDF/Operator.h>
- #include <LibPDF/Reader.h>
- #include <LibPDF/XRefTable.h>
- namespace PDF {
- class Document;
- class Parser final : public RefCounted<Parser> {
- public:
- enum class LinearizationResult {
- NotLinearized,
- Linearized,
- };
- static PDFErrorOr<Vector<Operator>> parse_operators(Document*, ReadonlyBytes);
- Parser(Document*, ReadonlyBytes);
- [[nodiscard]] ALWAYS_INLINE RefPtr<DictObject> const& trailer() const { return m_trailer; }
- void set_document(WeakPtr<Document> const&);
- // Parses the header and initializes the xref table and trailer
- PDFErrorOr<void> initialize();
- PDFErrorOr<Value> parse_object_with_index(u32 index);
- // Specialized version of parse_dict which aborts early if the dict being parsed
- // is not a page object
- PDFErrorOr<RefPtr<DictObject>> conditionally_parse_page_tree_node(u32 object_index);
- private:
- struct LinearizationDictionary {
- u32 length_of_file { 0 };
- u32 primary_hint_stream_offset { 0 };
- u32 primary_hint_stream_length { 0 };
- u32 overflow_hint_stream_offset { 0 };
- u32 overflow_hint_stream_length { 0 };
- u32 first_page_object_number { 0 };
- u32 offset_of_first_page_end { 0 };
- u16 number_of_pages { 0 };
- u32 offset_of_main_xref_table { 0 };
- u32 first_page { 0 }; // The page to initially open (I think, the spec isn't all that clear here)
- };
- struct PageOffsetHintTable {
- u32 least_number_of_objects_in_a_page { 0 };
- u32 location_of_first_page_object { 0 };
- u16 bits_required_for_object_number { 0 };
- u32 least_length_of_a_page { 0 };
- u16 bits_required_for_page_length { 0 };
- u32 least_offset_of_any_content_stream { 0 };
- u16 bits_required_for_content_stream_offsets { 0 };
- u32 least_content_stream_length { 0 };
- u16 bits_required_for_content_stream_length { 0 };
- u16 bits_required_for_number_of_shared_obj_refs { 0 };
- u16 bits_required_for_greatest_shared_obj_identifier { 0 };
- u16 bits_required_for_fraction_numerator { 0 };
- u16 shared_object_reference_fraction_denominator { 0 };
- };
- struct PageOffsetHintTableEntry {
- u32 objects_in_page_number { 0 };
- u32 page_length_number { 0 };
- u32 number_of_shared_objects { 0 };
- Vector<u32> shared_object_identifiers {};
- Vector<u32> shared_object_location_numerators {};
- u32 page_content_stream_offset_number { 0 };
- u32 page_content_stream_length_number { 0 };
- };
- friend struct AK::Formatter<LinearizationDictionary>;
- friend struct AK::Formatter<PageOffsetHintTable>;
- friend struct AK::Formatter<PageOffsetHintTableEntry>;
- explicit Parser(ReadonlyBytes);
- PDFErrorOr<void> parse_header();
- PDFErrorOr<LinearizationResult> initialize_linearization_dict();
- PDFErrorOr<void> initialize_linearized_xref_table();
- PDFErrorOr<void> initialize_non_linearized_xref_table();
- PDFErrorOr<void> initialize_hint_tables();
- PDFErrorOr<PageOffsetHintTable> parse_page_offset_hint_table(ReadonlyBytes hint_stream_bytes);
- Vector<PageOffsetHintTableEntry> parse_all_page_offset_hint_table_entries(PageOffsetHintTable const&, ReadonlyBytes hint_stream_bytes);
- PDFErrorOr<NonnullRefPtr<XRefTable>> parse_xref_table();
- PDFErrorOr<NonnullRefPtr<DictObject>> parse_file_trailer();
- bool navigate_to_before_eof_marker();
- bool navigate_to_after_startxref();
- String parse_comment();
- PDFErrorOr<Value> parse_value();
- PDFErrorOr<Value> parse_possible_indirect_value_or_ref();
- PDFErrorOr<NonnullRefPtr<IndirectValue>> parse_indirect_value(u32 index, u32 generation);
- PDFErrorOr<NonnullRefPtr<IndirectValue>> parse_indirect_value();
- PDFErrorOr<Value> parse_number();
- PDFErrorOr<NonnullRefPtr<NameObject>> parse_name();
- NonnullRefPtr<StringObject> parse_string();
- String parse_literal_string();
- String parse_hex_string();
- PDFErrorOr<NonnullRefPtr<ArrayObject>> parse_array();
- PDFErrorOr<NonnullRefPtr<DictObject>> parse_dict();
- PDFErrorOr<NonnullRefPtr<StreamObject>> parse_stream(NonnullRefPtr<DictObject> dict);
- PDFErrorOr<Vector<Operator>> parse_operators();
- void push_reference(Reference const& ref) { m_current_reference_stack.append(ref); }
- void pop_reference() { m_current_reference_stack.take_last(); }
- bool matches_eol() const;
- bool matches_whitespace() const;
- bool matches_number() const;
- bool matches_delimiter() const;
- bool matches_regular_character() const;
- bool consume_eol();
- bool consume_whitespace();
- char consume();
- void consume(int amount);
- bool consume(char);
- Error error(
- String const& message
- #ifdef PDF_DEBUG
- ,
- SourceLocation loc = SourceLocation::current()
- #endif
- ) const;
- Reader m_reader;
- WeakPtr<Document> m_document;
- RefPtr<XRefTable> m_xref_table;
- RefPtr<DictObject> m_trailer;
- Optional<LinearizationDictionary> m_linearization_dictionary;
- Vector<Reference> m_current_reference_stack;
- bool m_disable_encryption { false };
- };
- };
|