// Copyright 2014 PDFium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "public/fpdf_dataavail.h" #include #include #include "core/fpdfapi/page/cpdf_docpagedata.h" #include "core/fpdfapi/parser/cpdf_data_avail.h" #include "core/fpdfapi/parser/cpdf_document.h" #include "core/fpdfapi/render/cpdf_docrenderdata.h" #include "core/fxcrt/fx_safe_types.h" #include "core/fxcrt/fx_stream.h" #include "core/fxcrt/retain_ptr.h" #include "fpdfsdk/cpdfsdk_helpers.h" #include "public/fpdf_formfill.h" #include "third_party/base/ptr_util.h" #ifdef PDF_ENABLE_XFA #include "fpdfsdk/fpdfxfa/cpdfxfa_context.h" #endif // PDF_ENABLE_XFA // These checks are here because core/ and public/ cannot depend on each other. static_assert(CPDF_DataAvail::DataError == PDF_DATA_ERROR, "CPDF_DataAvail::DataError value mismatch"); static_assert(CPDF_DataAvail::DataNotAvailable == PDF_DATA_NOTAVAIL, "CPDF_DataAvail::DataNotAvailable value mismatch"); static_assert(CPDF_DataAvail::DataAvailable == PDF_DATA_AVAIL, "CPDF_DataAvail::DataAvailable value mismatch"); static_assert(CPDF_DataAvail::LinearizationUnknown == PDF_LINEARIZATION_UNKNOWN, "CPDF_DataAvail::LinearizationUnknown value mismatch"); static_assert(CPDF_DataAvail::NotLinearized == PDF_NOT_LINEARIZED, "CPDF_DataAvail::NotLinearized value mismatch"); static_assert(CPDF_DataAvail::Linearized == PDF_LINEARIZED, "CPDF_DataAvail::Linearized value mismatch"); static_assert(CPDF_DataAvail::FormError == PDF_FORM_ERROR, "CPDF_DataAvail::FormError value mismatch"); static_assert(CPDF_DataAvail::FormNotAvailable == PDF_FORM_NOTAVAIL, "CPDF_DataAvail::FormNotAvailable value mismatch"); static_assert(CPDF_DataAvail::FormAvailable == PDF_FORM_AVAIL, "CPDF_DataAvail::FormAvailable value mismatch"); static_assert(CPDF_DataAvail::FormNotExist == PDF_FORM_NOTEXIST, "CPDF_DataAvail::FormNotExist value mismatch"); namespace { class FPDF_FileAvailContext final : public CPDF_DataAvail::FileAvail { public: explicit FPDF_FileAvailContext(FX_FILEAVAIL* avail) : avail_(avail) {} ~FPDF_FileAvailContext() override = default; // CPDF_DataAvail::FileAvail: bool IsDataAvail(FX_FILESIZE offset, size_t size) override { return !!avail_->IsDataAvail(avail_, offset, size); } private: FX_FILEAVAIL* const avail_; }; class FPDF_FileAccessContext final : public IFX_SeekableReadStream { public: template friend RetainPtr pdfium::MakeRetain(Args&&... args); // IFX_SeekableReadStream: FX_FILESIZE GetSize() override { return file_->m_FileLen; } bool ReadBlockAtOffset(void* buffer, FX_FILESIZE offset, size_t size) override { if (!buffer || offset < 0 || !size) return false; if (!pdfium::base::IsValueInRangeForNumericType(size)) return false; FX_SAFE_FILESIZE new_pos = size; new_pos += offset; return new_pos.IsValid() && new_pos.ValueOrDie() <= GetSize() && file_->m_GetBlock(file_->m_Param, offset, static_cast(buffer), size); } private: explicit FPDF_FileAccessContext(FPDF_FILEACCESS* file) : file_(file) {} ~FPDF_FileAccessContext() override = default; FPDF_FILEACCESS* const file_; }; class FPDF_DownloadHintsContext final : public CPDF_DataAvail::DownloadHints { public: explicit FPDF_DownloadHintsContext(FX_DOWNLOADHINTS* pDownloadHints) { m_pDownloadHints = pDownloadHints; } ~FPDF_DownloadHintsContext() override {} public: // IFX_DownloadHints void AddSegment(FX_FILESIZE offset, size_t size) override { if (m_pDownloadHints) m_pDownloadHints->AddSegment(m_pDownloadHints, offset, size); } private: FX_DOWNLOADHINTS* m_pDownloadHints; }; class FPDF_AvailContext { public: FPDF_AvailContext(FX_FILEAVAIL* file_avail, FPDF_FILEACCESS* file) : file_avail_(pdfium::MakeUnique(file_avail)), file_read_(pdfium::MakeRetain(file)), data_avail_(pdfium::MakeUnique(file_avail_.get(), file_read_, true)) {} ~FPDF_AvailContext() = default; CPDF_DataAvail* data_avail() { return data_avail_.get(); } private: std::unique_ptr const file_avail_; RetainPtr const file_read_; std::unique_ptr const data_avail_; }; FPDF_AvailContext* FPDFAvailContextFromFPDFAvail(FPDF_AVAIL avail) { return static_cast(avail); } } // namespace FPDF_EXPORT FPDF_AVAIL FPDF_CALLCONV FPDFAvail_Create(FX_FILEAVAIL* file_avail, FPDF_FILEACCESS* file) { auto pAvail = pdfium::MakeUnique(file_avail, file); return pAvail.release(); // Caller takes ownership. } FPDF_EXPORT void FPDF_CALLCONV FPDFAvail_Destroy(FPDF_AVAIL avail) { // Take ownership back from caller and destroy. std::unique_ptr(FPDFAvailContextFromFPDFAvail(avail)); } FPDF_EXPORT int FPDF_CALLCONV FPDFAvail_IsDocAvail(FPDF_AVAIL avail, FX_DOWNLOADHINTS* hints) { auto* avail_context = FPDFAvailContextFromFPDFAvail(avail); if (!avail_context) return PDF_DATA_ERROR; FPDF_DownloadHintsContext hints_context(hints); return avail_context->data_avail()->IsDocAvail(&hints_context); } FPDF_EXPORT FPDF_DOCUMENT FPDF_CALLCONV FPDFAvail_GetDocument(FPDF_AVAIL avail, FPDF_BYTESTRING password) { auto* avail_context = FPDFAvailContextFromFPDFAvail(avail); if (!avail_context) return nullptr; CPDF_Parser::Error error; std::unique_ptr document; std::tie(error, document) = avail_context->data_avail()->ParseDocument( pdfium::MakeUnique(), pdfium::MakeUnique(), password); if (error != CPDF_Parser::SUCCESS) { ProcessParseError(error); return nullptr; } #ifdef PDF_ENABLE_XFA document->SetExtension(pdfium::MakeUnique(document.get())); #endif // PDF_ENABLE_XFA ReportUnsupportedFeatures(document.get()); return FPDFDocumentFromCPDFDocument(document.release()); } FPDF_EXPORT int FPDF_CALLCONV FPDFAvail_GetFirstPageNum(FPDF_DOCUMENT doc) { CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(doc); return pDoc ? pDoc->GetParser()->GetFirstPageNo() : 0; } FPDF_EXPORT int FPDF_CALLCONV FPDFAvail_IsPageAvail(FPDF_AVAIL avail, int page_index, FX_DOWNLOADHINTS* hints) { auto* avail_context = FPDFAvailContextFromFPDFAvail(avail); if (!avail_context) return PDF_DATA_ERROR; if (page_index < 0) return PDF_DATA_NOTAVAIL; FPDF_DownloadHintsContext hints_context(hints); return avail_context->data_avail()->IsPageAvail(page_index, &hints_context); } FPDF_EXPORT int FPDF_CALLCONV FPDFAvail_IsFormAvail(FPDF_AVAIL avail, FX_DOWNLOADHINTS* hints) { auto* avail_context = FPDFAvailContextFromFPDFAvail(avail); if (!avail_context) return PDF_FORM_ERROR; FPDF_DownloadHintsContext hints_context(hints); return avail_context->data_avail()->IsFormAvail(&hints_context); } FPDF_EXPORT int FPDF_CALLCONV FPDFAvail_IsLinearized(FPDF_AVAIL avail) { auto* avail_context = FPDFAvailContextFromFPDFAvail(avail); if (!avail_context) return PDF_LINEARIZATION_UNKNOWN; return avail_context->data_avail()->IsLinearizedPDF(); }