xref: /aosp_15_r20/external/pdfium/fpdfsdk/fpdf_save.cpp (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
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