1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2014 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker
5*3ac0a46fSAndroid Build Coastguard Worker // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6*3ac0a46fSAndroid Build Coastguard Worker
7*3ac0a46fSAndroid Build Coastguard Worker #include "public/fpdf_save.h"
8*3ac0a46fSAndroid Build Coastguard Worker
9*3ac0a46fSAndroid Build Coastguard Worker #include <utility>
10*3ac0a46fSAndroid Build Coastguard Worker #include <vector>
11*3ac0a46fSAndroid Build Coastguard Worker
12*3ac0a46fSAndroid Build Coastguard Worker #include "build/build_config.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/edit/cpdf_creator.h"
14*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_array.h"
15*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_dictionary.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_document.h"
17*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_reference.h"
18*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_stream_acc.h"
19*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_string.h"
20*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_extension.h"
21*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/stl_util.h"
22*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_filewriteadapter.h"
23*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_helpers.h"
24*3ac0a46fSAndroid Build Coastguard Worker #include "public/fpdf_edit.h"
25*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/abseil-cpp/absl/types/optional.h"
26*3ac0a46fSAndroid Build Coastguard Worker
27*3ac0a46fSAndroid Build Coastguard Worker #ifdef PDF_ENABLE_XFA
28*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_stream.h"
29*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/cfx_memorystream.h"
30*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/fpdfxfa/cpdfxfa_context.h"
31*3ac0a46fSAndroid Build Coastguard Worker #include "public/fpdf_formfill.h"
32*3ac0a46fSAndroid Build Coastguard Worker #endif
33*3ac0a46fSAndroid Build Coastguard Worker
34*3ac0a46fSAndroid Build Coastguard Worker namespace {
35*3ac0a46fSAndroid Build Coastguard Worker
36*3ac0a46fSAndroid Build Coastguard Worker #ifdef PDF_ENABLE_XFA
SaveXFADocumentData(CPDFXFA_Context * pContext,std::vector<RetainPtr<IFX_SeekableStream>> * fileList)37*3ac0a46fSAndroid Build Coastguard Worker bool SaveXFADocumentData(CPDFXFA_Context* pContext,
38*3ac0a46fSAndroid Build Coastguard Worker std::vector<RetainPtr<IFX_SeekableStream>>* fileList) {
39*3ac0a46fSAndroid Build Coastguard Worker if (!pContext)
40*3ac0a46fSAndroid Build Coastguard Worker return false;
41*3ac0a46fSAndroid Build Coastguard Worker
42*3ac0a46fSAndroid Build Coastguard Worker if (!pContext->ContainsExtensionForm())
43*3ac0a46fSAndroid Build Coastguard Worker return true;
44*3ac0a46fSAndroid Build Coastguard Worker
45*3ac0a46fSAndroid Build Coastguard Worker CPDF_Document* pPDFDocument = pContext->GetPDFDoc();
46*3ac0a46fSAndroid Build Coastguard Worker if (!pPDFDocument)
47*3ac0a46fSAndroid Build Coastguard Worker return false;
48*3ac0a46fSAndroid Build Coastguard Worker
49*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Dictionary> pRoot = pPDFDocument->GetMutableRoot();
50*3ac0a46fSAndroid Build Coastguard Worker if (!pRoot)
51*3ac0a46fSAndroid Build Coastguard Worker return false;
52*3ac0a46fSAndroid Build Coastguard Worker
53*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Dictionary> pAcroForm = pRoot->GetMutableDictFor("AcroForm");
54*3ac0a46fSAndroid Build Coastguard Worker if (!pAcroForm)
55*3ac0a46fSAndroid Build Coastguard Worker return false;
56*3ac0a46fSAndroid Build Coastguard Worker
57*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Object> pXFA = pAcroForm->GetMutableObjectFor("XFA");
58*3ac0a46fSAndroid Build Coastguard Worker if (!pXFA)
59*3ac0a46fSAndroid Build Coastguard Worker return true;
60*3ac0a46fSAndroid Build Coastguard Worker
61*3ac0a46fSAndroid Build Coastguard Worker CPDF_Array* pArray = pXFA->AsMutableArray();
62*3ac0a46fSAndroid Build Coastguard Worker if (!pArray)
63*3ac0a46fSAndroid Build Coastguard Worker return false;
64*3ac0a46fSAndroid Build Coastguard Worker
65*3ac0a46fSAndroid Build Coastguard Worker int size = fxcrt::CollectionSize<int>(*pArray);
66*3ac0a46fSAndroid Build Coastguard Worker int iFormIndex = -1;
67*3ac0a46fSAndroid Build Coastguard Worker int iDataSetsIndex = -1;
68*3ac0a46fSAndroid Build Coastguard Worker for (int i = 0; i < size - 1; i++) {
69*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<const CPDF_Object> pPDFObj = pArray->GetObjectAt(i);
70*3ac0a46fSAndroid Build Coastguard Worker if (!pPDFObj->IsString())
71*3ac0a46fSAndroid Build Coastguard Worker continue;
72*3ac0a46fSAndroid Build Coastguard Worker if (pPDFObj->GetString() == "form")
73*3ac0a46fSAndroid Build Coastguard Worker iFormIndex = i + 1;
74*3ac0a46fSAndroid Build Coastguard Worker else if (pPDFObj->GetString() == "datasets")
75*3ac0a46fSAndroid Build Coastguard Worker iDataSetsIndex = i + 1;
76*3ac0a46fSAndroid Build Coastguard Worker }
77*3ac0a46fSAndroid Build Coastguard Worker
78*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Stream> pFormStream;
79*3ac0a46fSAndroid Build Coastguard Worker if (iFormIndex != -1) {
80*3ac0a46fSAndroid Build Coastguard Worker // Get form CPDF_Stream
81*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Object> pFormPDFObj = pArray->GetMutableObjectAt(iFormIndex);
82*3ac0a46fSAndroid Build Coastguard Worker if (pFormPDFObj->IsReference()) {
83*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Object> pFormDirectObj = pFormPDFObj->GetMutableDirect();
84*3ac0a46fSAndroid Build Coastguard Worker if (pFormDirectObj && pFormDirectObj->IsStream()) {
85*3ac0a46fSAndroid Build Coastguard Worker pFormStream.Reset(pFormDirectObj->AsMutableStream());
86*3ac0a46fSAndroid Build Coastguard Worker }
87*3ac0a46fSAndroid Build Coastguard Worker } else if (pFormPDFObj->IsStream()) {
88*3ac0a46fSAndroid Build Coastguard Worker pFormStream.Reset(pFormPDFObj->AsMutableStream());
89*3ac0a46fSAndroid Build Coastguard Worker }
90*3ac0a46fSAndroid Build Coastguard Worker }
91*3ac0a46fSAndroid Build Coastguard Worker
92*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Stream> pDataSetsStream;
93*3ac0a46fSAndroid Build Coastguard Worker if (iDataSetsIndex != -1) {
94*3ac0a46fSAndroid Build Coastguard Worker // Get datasets CPDF_Stream
95*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Object> pDataSetsPDFObj =
96*3ac0a46fSAndroid Build Coastguard Worker pArray->GetMutableObjectAt(iDataSetsIndex);
97*3ac0a46fSAndroid Build Coastguard Worker if (pDataSetsPDFObj->IsReference()) {
98*3ac0a46fSAndroid Build Coastguard Worker CPDF_Reference* pDataSetsRefObj = pDataSetsPDFObj->AsMutableReference();
99*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Object> pDataSetsDirectObj =
100*3ac0a46fSAndroid Build Coastguard Worker pDataSetsRefObj->GetMutableDirect();
101*3ac0a46fSAndroid Build Coastguard Worker if (pDataSetsDirectObj && pDataSetsDirectObj->IsStream()) {
102*3ac0a46fSAndroid Build Coastguard Worker pDataSetsStream.Reset(pDataSetsDirectObj->AsMutableStream());
103*3ac0a46fSAndroid Build Coastguard Worker }
104*3ac0a46fSAndroid Build Coastguard Worker } else if (pDataSetsPDFObj->IsStream()) {
105*3ac0a46fSAndroid Build Coastguard Worker pDataSetsStream.Reset(pDataSetsPDFObj->AsMutableStream());
106*3ac0a46fSAndroid Build Coastguard Worker }
107*3ac0a46fSAndroid Build Coastguard Worker }
108*3ac0a46fSAndroid Build Coastguard Worker // L"datasets"
109*3ac0a46fSAndroid Build Coastguard Worker {
110*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<IFX_SeekableStream> pFileWrite =
111*3ac0a46fSAndroid Build Coastguard Worker pdfium::MakeRetain<CFX_MemoryStream>();
112*3ac0a46fSAndroid Build Coastguard Worker if (pContext->SaveDatasetsPackage(pFileWrite) &&
113*3ac0a46fSAndroid Build Coastguard Worker pFileWrite->GetSize() > 0) {
114*3ac0a46fSAndroid Build Coastguard Worker auto pDataDict = pPDFDocument->New<CPDF_Dictionary>();
115*3ac0a46fSAndroid Build Coastguard Worker if (iDataSetsIndex != -1) {
116*3ac0a46fSAndroid Build Coastguard Worker if (pDataSetsStream) {
117*3ac0a46fSAndroid Build Coastguard Worker pDataSetsStream->InitStreamFromFile(pFileWrite, std::move(pDataDict));
118*3ac0a46fSAndroid Build Coastguard Worker }
119*3ac0a46fSAndroid Build Coastguard Worker } else {
120*3ac0a46fSAndroid Build Coastguard Worker auto pData = pPDFDocument->NewIndirect<CPDF_Stream>();
121*3ac0a46fSAndroid Build Coastguard Worker pData->InitStreamFromFile(pFileWrite, std::move(pDataDict));
122*3ac0a46fSAndroid Build Coastguard Worker int iLast = fxcrt::CollectionSize<int>(*pArray) - 2;
123*3ac0a46fSAndroid Build Coastguard Worker pArray->InsertNewAt<CPDF_String>(iLast, "datasets", false);
124*3ac0a46fSAndroid Build Coastguard Worker pArray->InsertNewAt<CPDF_Reference>(iLast + 1, pPDFDocument,
125*3ac0a46fSAndroid Build Coastguard Worker pData->GetObjNum());
126*3ac0a46fSAndroid Build Coastguard Worker }
127*3ac0a46fSAndroid Build Coastguard Worker fileList->push_back(std::move(pFileWrite));
128*3ac0a46fSAndroid Build Coastguard Worker }
129*3ac0a46fSAndroid Build Coastguard Worker }
130*3ac0a46fSAndroid Build Coastguard Worker // L"form"
131*3ac0a46fSAndroid Build Coastguard Worker {
132*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<IFX_SeekableStream> pFileWrite =
133*3ac0a46fSAndroid Build Coastguard Worker pdfium::MakeRetain<CFX_MemoryStream>();
134*3ac0a46fSAndroid Build Coastguard Worker if (pContext->SaveFormPackage(pFileWrite) && pFileWrite->GetSize() > 0) {
135*3ac0a46fSAndroid Build Coastguard Worker auto pDataDict = pPDFDocument->New<CPDF_Dictionary>();
136*3ac0a46fSAndroid Build Coastguard Worker if (iFormIndex != -1) {
137*3ac0a46fSAndroid Build Coastguard Worker if (pFormStream)
138*3ac0a46fSAndroid Build Coastguard Worker pFormStream->InitStreamFromFile(pFileWrite, std::move(pDataDict));
139*3ac0a46fSAndroid Build Coastguard Worker } else {
140*3ac0a46fSAndroid Build Coastguard Worker auto pData = pPDFDocument->NewIndirect<CPDF_Stream>();
141*3ac0a46fSAndroid Build Coastguard Worker pData->InitStreamFromFile(pFileWrite, std::move(pDataDict));
142*3ac0a46fSAndroid Build Coastguard Worker int iLast = fxcrt::CollectionSize<int>(*pArray) - 2;
143*3ac0a46fSAndroid Build Coastguard Worker pArray->InsertNewAt<CPDF_String>(iLast, "form", false);
144*3ac0a46fSAndroid Build Coastguard Worker pArray->InsertNewAt<CPDF_Reference>(iLast + 1, pPDFDocument,
145*3ac0a46fSAndroid Build Coastguard Worker pData->GetObjNum());
146*3ac0a46fSAndroid Build Coastguard Worker }
147*3ac0a46fSAndroid Build Coastguard Worker fileList->push_back(std::move(pFileWrite));
148*3ac0a46fSAndroid Build Coastguard Worker }
149*3ac0a46fSAndroid Build Coastguard Worker }
150*3ac0a46fSAndroid Build Coastguard Worker return true;
151*3ac0a46fSAndroid Build Coastguard Worker }
152*3ac0a46fSAndroid Build Coastguard Worker #endif // PDF_ENABLE_XFA
153*3ac0a46fSAndroid Build Coastguard Worker
DoDocSave(FPDF_DOCUMENT document,FPDF_FILEWRITE * pFileWrite,FPDF_DWORD flags,absl::optional<int> version)154*3ac0a46fSAndroid Build Coastguard Worker bool DoDocSave(FPDF_DOCUMENT document,
155*3ac0a46fSAndroid Build Coastguard Worker FPDF_FILEWRITE* pFileWrite,
156*3ac0a46fSAndroid Build Coastguard Worker FPDF_DWORD flags,
157*3ac0a46fSAndroid Build Coastguard Worker absl::optional<int> version) {
158*3ac0a46fSAndroid Build Coastguard Worker CPDF_Document* pPDFDoc = CPDFDocumentFromFPDFDocument(document);
159*3ac0a46fSAndroid Build Coastguard Worker if (!pPDFDoc)
160*3ac0a46fSAndroid Build Coastguard Worker return false;
161*3ac0a46fSAndroid Build Coastguard Worker
162*3ac0a46fSAndroid Build Coastguard Worker #ifdef PDF_ENABLE_XFA
163*3ac0a46fSAndroid Build Coastguard Worker auto* pContext = static_cast<CPDFXFA_Context*>(pPDFDoc->GetExtension());
164*3ac0a46fSAndroid Build Coastguard Worker if (pContext) {
165*3ac0a46fSAndroid Build Coastguard Worker std::vector<RetainPtr<IFX_SeekableStream>> fileList;
166*3ac0a46fSAndroid Build Coastguard Worker pContext->SendPreSaveToXFADoc(&fileList);
167*3ac0a46fSAndroid Build Coastguard Worker SaveXFADocumentData(pContext, &fileList);
168*3ac0a46fSAndroid Build Coastguard Worker }
169*3ac0a46fSAndroid Build Coastguard Worker #endif // PDF_ENABLE_XFA
170*3ac0a46fSAndroid Build Coastguard Worker
171*3ac0a46fSAndroid Build Coastguard Worker if (flags < FPDF_INCREMENTAL || flags > FPDF_REMOVE_SECURITY)
172*3ac0a46fSAndroid Build Coastguard Worker flags = 0;
173*3ac0a46fSAndroid Build Coastguard Worker
174*3ac0a46fSAndroid Build Coastguard Worker CPDF_Creator fileMaker(
175*3ac0a46fSAndroid Build Coastguard Worker pPDFDoc, pdfium::MakeRetain<CPDFSDK_FileWriteAdapter>(pFileWrite));
176*3ac0a46fSAndroid Build Coastguard Worker if (version.has_value())
177*3ac0a46fSAndroid Build Coastguard Worker fileMaker.SetFileVersion(version.value());
178*3ac0a46fSAndroid Build Coastguard Worker if (flags == FPDF_REMOVE_SECURITY) {
179*3ac0a46fSAndroid Build Coastguard Worker flags = 0;
180*3ac0a46fSAndroid Build Coastguard Worker fileMaker.RemoveSecurity();
181*3ac0a46fSAndroid Build Coastguard Worker }
182*3ac0a46fSAndroid Build Coastguard Worker
183*3ac0a46fSAndroid Build Coastguard Worker bool bRet = fileMaker.Create(static_cast<uint32_t>(flags));
184*3ac0a46fSAndroid Build Coastguard Worker
185*3ac0a46fSAndroid Build Coastguard Worker #ifdef PDF_ENABLE_XFA
186*3ac0a46fSAndroid Build Coastguard Worker if (pContext)
187*3ac0a46fSAndroid Build Coastguard Worker pContext->SendPostSaveToXFADoc();
188*3ac0a46fSAndroid Build Coastguard Worker #endif // PDF_ENABLE_XFA
189*3ac0a46fSAndroid Build Coastguard Worker
190*3ac0a46fSAndroid Build Coastguard Worker return bRet;
191*3ac0a46fSAndroid Build Coastguard Worker }
192*3ac0a46fSAndroid Build Coastguard Worker
193*3ac0a46fSAndroid Build Coastguard Worker } // namespace
194*3ac0a46fSAndroid Build Coastguard Worker
FPDF_SaveAsCopy(FPDF_DOCUMENT document,FPDF_FILEWRITE * pFileWrite,FPDF_DWORD flags)195*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDF_SaveAsCopy(FPDF_DOCUMENT document,
196*3ac0a46fSAndroid Build Coastguard Worker FPDF_FILEWRITE* pFileWrite,
197*3ac0a46fSAndroid Build Coastguard Worker FPDF_DWORD flags) {
198*3ac0a46fSAndroid Build Coastguard Worker return DoDocSave(document, pFileWrite, flags, {});
199*3ac0a46fSAndroid Build Coastguard Worker }
200*3ac0a46fSAndroid Build Coastguard Worker
201*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDF_SaveWithVersion(FPDF_DOCUMENT document,FPDF_FILEWRITE * pFileWrite,FPDF_DWORD flags,int fileVersion)202*3ac0a46fSAndroid Build Coastguard Worker FPDF_SaveWithVersion(FPDF_DOCUMENT document,
203*3ac0a46fSAndroid Build Coastguard Worker FPDF_FILEWRITE* pFileWrite,
204*3ac0a46fSAndroid Build Coastguard Worker FPDF_DWORD flags,
205*3ac0a46fSAndroid Build Coastguard Worker int fileVersion) {
206*3ac0a46fSAndroid Build Coastguard Worker return DoDocSave(document, pFileWrite, flags, fileVersion);
207*3ac0a46fSAndroid Build Coastguard Worker }
208