| // 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 <memory> |
| #include <utility> |
| |
| #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" |
| |
| #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: |
| CONSTRUCT_VIA_MAKE_RETAIN; |
| |
| // 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<FX_FILESIZE>(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<uint8_t*>(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_(std::make_unique<FPDF_FileAvailContext>(file_avail)), |
| file_read_(pdfium::MakeRetain<FPDF_FileAccessContext>(file)), |
| data_avail_(std::make_unique<CPDF_DataAvail>(file_avail_.get(), |
| file_read_, |
| true)) {} |
| ~FPDF_AvailContext() = default; |
| |
| CPDF_DataAvail* data_avail() { return data_avail_.get(); } |
| |
| private: |
| std::unique_ptr<FPDF_FileAvailContext> const file_avail_; |
| RetainPtr<FPDF_FileAccessContext> const file_read_; |
| std::unique_ptr<CPDF_DataAvail> const data_avail_; |
| }; |
| |
| FPDF_AvailContext* FPDFAvailContextFromFPDFAvail(FPDF_AVAIL avail) { |
| return static_cast<FPDF_AvailContext*>(avail); |
| } |
| |
| } // namespace |
| |
| FPDF_EXPORT FPDF_AVAIL FPDF_CALLCONV FPDFAvail_Create(FX_FILEAVAIL* file_avail, |
| FPDF_FILEACCESS* file) { |
| auto pAvail = std::make_unique<FPDF_AvailContext>(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<FPDF_AvailContext>(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<CPDF_Document> document; |
| std::tie(error, document) = avail_context->data_avail()->ParseDocument( |
| std::make_unique<CPDF_DocRenderData>(), |
| std::make_unique<CPDF_DocPageData>(), password); |
| if (error != CPDF_Parser::SUCCESS) { |
| ProcessParseError(error); |
| return nullptr; |
| } |
| |
| #ifdef PDF_ENABLE_XFA |
| document->SetExtension(std::make_unique<CPDFXFA_Context>(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(); |
| } |