// Copyright (c) 2005-2024 Jay Berkenbilt // // This file is part of qpdf. // // Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except // in compliance with the License. You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software distributed under the License // is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express // or implied. See the License for the specific language governing permissions and limitations under // the License. // // Versions of qpdf prior to version 7 were released under the terms of version 2.0 of the Artistic // License. At your option, you may continue to consider qpdf to be licensed under those terms. // Please see the manual for additional information. #ifndef QPDFOBJGEN_HH #define QPDFOBJGEN_HH #include #include #include class QPDFObjectHandle; class QPDFObjectHelper; // This class represents an object ID and generation pair. It is suitable to use as a key in a map // or set. class QPDFObjGen { public: // ABI: change to default. QPDF_DLL QPDFObjGen() { } QPDF_DLL explicit QPDFObjGen(int obj, int gen) : obj(obj), gen(gen) { } QPDF_DLL bool operator<(QPDFObjGen const& rhs) const { return (obj < rhs.obj) || ((obj == rhs.obj) && (gen < rhs.gen)); } QPDF_DLL bool operator==(QPDFObjGen const& rhs) const { return (obj == rhs.obj) && (gen == rhs.gen); } QPDF_DLL bool operator!=(QPDFObjGen const& rhs) const { return (obj != rhs.obj) || (gen != rhs.gen); } QPDF_DLL int getObj() const { return obj; } QPDF_DLL int getGen() const { return gen; } QPDF_DLL bool isIndirect() const { return obj != 0; } QPDF_DLL std::string unparse(char separator = ',') const; QPDF_DLL friend std::ostream& operator<<(std::ostream& os, const QPDFObjGen& og); // Convenience class for loop detection when processing objects. // // The class adds 'add' methods to a std::set which allows to test whether an // QPDFObjGen is present in the set and to insert it in a single operation. The 'add' method is // overloaded to take a QPDFObjGen, QPDFObjectHandle or an QPDFObjectHelper as parameter. // // The erase method is modified to ignore requests to erase QPDFObjGen(0, 0). // // Usage example: // // void process_object(QPDFObjectHandle oh, QPDFObjGen::set& seen) // { // if (seen.add(oh)) { // // handle first encounter of oh // } else { // // handle loop / subsequent encounter of oh // } // } class QPDF_DLL_CLASS set: public std::set { public: // Add 'og' to the set. Return false if 'og' is already present in the set. Attempts to // insert QPDFObjGen(0, 0) are ignored. QPDF_DLL bool add(QPDFObjGen og) { if (og.isIndirect()) { if (count(og) > 0) { return false; } emplace(og); } return true; } QPDF_DLL bool add(QPDFObjectHandle const& oh); QPDF_DLL bool add(QPDFObjectHelper const& oh); QPDF_DLL void erase(QPDFObjGen og) { if (og.isIndirect()) { std::set::erase(og); } } QPDF_DLL void erase(QPDFObjectHandle const& oh); QPDF_DLL void erase(QPDFObjectHelper const& oh); }; private: // This class does not use the Members pattern to avoid a memory allocation for every one of // these. A lot of these get created and destroyed. int obj{0}; int gen{0}; }; #endif // QPDFOBJGEN_HH