xref: /aosp_15_r20/external/pdfium/fxjs/cfx_globaldata.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 "fxjs/cfx_globaldata.h"
8*3ac0a46fSAndroid Build Coastguard Worker 
9*3ac0a46fSAndroid Build Coastguard Worker #include <utility>
10*3ac0a46fSAndroid Build Coastguard Worker 
11*3ac0a46fSAndroid Build Coastguard Worker #include "core/fdrm/fx_crypt.h"
12*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/stl_util.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/numerics/safe_conversions.h"
14*3ac0a46fSAndroid Build Coastguard Worker 
15*3ac0a46fSAndroid Build Coastguard Worker namespace {
16*3ac0a46fSAndroid Build Coastguard Worker 
17*3ac0a46fSAndroid Build Coastguard Worker constexpr size_t kMinGlobalDataBytes = 12;
18*3ac0a46fSAndroid Build Coastguard Worker constexpr size_t kMaxGlobalDataBytes = 4 * 1024 - 8;
19*3ac0a46fSAndroid Build Coastguard Worker constexpr uint16_t kMagic = ('X' << 8) | 'F';
20*3ac0a46fSAndroid Build Coastguard Worker constexpr uint16_t kMaxVersion = 2;
21*3ac0a46fSAndroid Build Coastguard Worker 
22*3ac0a46fSAndroid Build Coastguard Worker const uint8_t kRC4KEY[] = {
23*3ac0a46fSAndroid Build Coastguard Worker     0x19, 0xa8, 0xe8, 0x01, 0xf6, 0xa8, 0xb6, 0x4d, 0x82, 0x04, 0x45, 0x6d,
24*3ac0a46fSAndroid Build Coastguard Worker     0xb4, 0xcf, 0xd7, 0x77, 0x67, 0xf9, 0x75, 0x9f, 0xf0, 0xe0, 0x1e, 0x51,
25*3ac0a46fSAndroid Build Coastguard Worker     0xee, 0x46, 0xfd, 0x0b, 0xc9, 0x93, 0x25, 0x55, 0x4a, 0xee, 0xe0, 0x16,
26*3ac0a46fSAndroid Build Coastguard Worker     0xd0, 0xdf, 0x8c, 0xfa, 0x2a, 0xa9, 0x49, 0xfd, 0x97, 0x1c, 0x0e, 0x22,
27*3ac0a46fSAndroid Build Coastguard Worker     0x13, 0x28, 0x7c, 0xaf, 0xc4, 0xfc, 0x9c, 0x12, 0x65, 0x8c, 0x4e, 0x5b,
28*3ac0a46fSAndroid Build Coastguard Worker     0x04, 0x75, 0x89, 0xc9, 0xb1, 0xed, 0x50, 0xca, 0x96, 0x6f, 0x1a, 0x7a,
29*3ac0a46fSAndroid Build Coastguard Worker     0xfe, 0x58, 0x5d, 0xec, 0x19, 0x4a, 0xf6, 0x35, 0x6a, 0x97, 0x14, 0x00,
30*3ac0a46fSAndroid Build Coastguard Worker     0x0e, 0xd0, 0x6b, 0xbb, 0xd5, 0x75, 0x55, 0x8b, 0x6e, 0x6b, 0x19, 0xa0,
31*3ac0a46fSAndroid Build Coastguard Worker     0xf8, 0x77, 0xd5, 0xa3};
32*3ac0a46fSAndroid Build Coastguard Worker 
33*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData* g_pInstance = nullptr;
34*3ac0a46fSAndroid Build Coastguard Worker 
35*3ac0a46fSAndroid Build Coastguard Worker // Returns true if non-empty, setting sPropName
TrimPropName(ByteString * sPropName)36*3ac0a46fSAndroid Build Coastguard Worker bool TrimPropName(ByteString* sPropName) {
37*3ac0a46fSAndroid Build Coastguard Worker   sPropName->Trim();
38*3ac0a46fSAndroid Build Coastguard Worker   return sPropName->GetLength() != 0;
39*3ac0a46fSAndroid Build Coastguard Worker }
40*3ac0a46fSAndroid Build Coastguard Worker 
MakeNameTypeString(const ByteString & name,CFX_Value::DataType eType,BinaryBuffer * result)41*3ac0a46fSAndroid Build Coastguard Worker void MakeNameTypeString(const ByteString& name,
42*3ac0a46fSAndroid Build Coastguard Worker                         CFX_Value::DataType eType,
43*3ac0a46fSAndroid Build Coastguard Worker                         BinaryBuffer* result) {
44*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwNameLen = pdfium::base::checked_cast<uint32_t>(name.GetLength());
45*3ac0a46fSAndroid Build Coastguard Worker   result->AppendUint32(dwNameLen);
46*3ac0a46fSAndroid Build Coastguard Worker   result->AppendString(name);
47*3ac0a46fSAndroid Build Coastguard Worker   result->AppendUint16(static_cast<uint16_t>(eType));
48*3ac0a46fSAndroid Build Coastguard Worker }
49*3ac0a46fSAndroid Build Coastguard Worker 
MakeByteString(const ByteString & name,const CFX_KeyValue & pData,BinaryBuffer * result)50*3ac0a46fSAndroid Build Coastguard Worker bool MakeByteString(const ByteString& name,
51*3ac0a46fSAndroid Build Coastguard Worker                     const CFX_KeyValue& pData,
52*3ac0a46fSAndroid Build Coastguard Worker                     BinaryBuffer* result) {
53*3ac0a46fSAndroid Build Coastguard Worker   switch (pData.nType) {
54*3ac0a46fSAndroid Build Coastguard Worker     case CFX_Value::DataType::kNumber: {
55*3ac0a46fSAndroid Build Coastguard Worker       MakeNameTypeString(name, pData.nType, result);
56*3ac0a46fSAndroid Build Coastguard Worker       result->AppendDouble(pData.dData);
57*3ac0a46fSAndroid Build Coastguard Worker       return true;
58*3ac0a46fSAndroid Build Coastguard Worker     }
59*3ac0a46fSAndroid Build Coastguard Worker     case CFX_Value::DataType::kBoolean: {
60*3ac0a46fSAndroid Build Coastguard Worker       MakeNameTypeString(name, pData.nType, result);
61*3ac0a46fSAndroid Build Coastguard Worker       result->AppendUint16(static_cast<uint16_t>(pData.bData));
62*3ac0a46fSAndroid Build Coastguard Worker       return true;
63*3ac0a46fSAndroid Build Coastguard Worker     }
64*3ac0a46fSAndroid Build Coastguard Worker     case CFX_Value::DataType::kString: {
65*3ac0a46fSAndroid Build Coastguard Worker       MakeNameTypeString(name, pData.nType, result);
66*3ac0a46fSAndroid Build Coastguard Worker       uint32_t dwDataLen =
67*3ac0a46fSAndroid Build Coastguard Worker           pdfium::base::checked_cast<uint32_t>(pData.sData.GetLength());
68*3ac0a46fSAndroid Build Coastguard Worker       result->AppendUint32(dwDataLen);
69*3ac0a46fSAndroid Build Coastguard Worker       result->AppendString(pData.sData);
70*3ac0a46fSAndroid Build Coastguard Worker       return true;
71*3ac0a46fSAndroid Build Coastguard Worker     }
72*3ac0a46fSAndroid Build Coastguard Worker     case CFX_Value::DataType::kNull: {
73*3ac0a46fSAndroid Build Coastguard Worker       MakeNameTypeString(name, pData.nType, result);
74*3ac0a46fSAndroid Build Coastguard Worker       return true;
75*3ac0a46fSAndroid Build Coastguard Worker     }
76*3ac0a46fSAndroid Build Coastguard Worker     // Arrays don't get persisted per JS spec page 484.
77*3ac0a46fSAndroid Build Coastguard Worker     case CFX_Value::DataType::kObject:
78*3ac0a46fSAndroid Build Coastguard Worker       break;
79*3ac0a46fSAndroid Build Coastguard Worker   }
80*3ac0a46fSAndroid Build Coastguard Worker   return false;
81*3ac0a46fSAndroid Build Coastguard Worker }
82*3ac0a46fSAndroid Build Coastguard Worker 
83*3ac0a46fSAndroid Build Coastguard Worker }  // namespace
84*3ac0a46fSAndroid Build Coastguard Worker 
85*3ac0a46fSAndroid Build Coastguard Worker // static
GetRetainedInstance(Delegate * pDelegate)86*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData* CFX_GlobalData::GetRetainedInstance(Delegate* pDelegate) {
87*3ac0a46fSAndroid Build Coastguard Worker   if (!g_pInstance) {
88*3ac0a46fSAndroid Build Coastguard Worker     g_pInstance = new CFX_GlobalData(pDelegate);
89*3ac0a46fSAndroid Build Coastguard Worker   }
90*3ac0a46fSAndroid Build Coastguard Worker   ++g_pInstance->m_RefCount;
91*3ac0a46fSAndroid Build Coastguard Worker   return g_pInstance;
92*3ac0a46fSAndroid Build Coastguard Worker }
93*3ac0a46fSAndroid Build Coastguard Worker 
Release()94*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::Release() {
95*3ac0a46fSAndroid Build Coastguard Worker   if (--m_RefCount)
96*3ac0a46fSAndroid Build Coastguard Worker     return false;
97*3ac0a46fSAndroid Build Coastguard Worker 
98*3ac0a46fSAndroid Build Coastguard Worker   delete g_pInstance;
99*3ac0a46fSAndroid Build Coastguard Worker   g_pInstance = nullptr;
100*3ac0a46fSAndroid Build Coastguard Worker   return true;
101*3ac0a46fSAndroid Build Coastguard Worker }
102*3ac0a46fSAndroid Build Coastguard Worker 
CFX_GlobalData(Delegate * pDelegate)103*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::CFX_GlobalData(Delegate* pDelegate) : m_pDelegate(pDelegate) {
104*3ac0a46fSAndroid Build Coastguard Worker   LoadGlobalPersistentVariables();
105*3ac0a46fSAndroid Build Coastguard Worker }
106*3ac0a46fSAndroid Build Coastguard Worker 
~CFX_GlobalData()107*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::~CFX_GlobalData() {
108*3ac0a46fSAndroid Build Coastguard Worker   SaveGlobalPersisitentVariables();
109*3ac0a46fSAndroid Build Coastguard Worker }
110*3ac0a46fSAndroid Build Coastguard Worker 
FindGlobalVariable(const ByteString & propname)111*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::iterator CFX_GlobalData::FindGlobalVariable(
112*3ac0a46fSAndroid Build Coastguard Worker     const ByteString& propname) {
113*3ac0a46fSAndroid Build Coastguard Worker   for (auto it = m_arrayGlobalData.begin(); it != m_arrayGlobalData.end();
114*3ac0a46fSAndroid Build Coastguard Worker        ++it) {
115*3ac0a46fSAndroid Build Coastguard Worker     if ((*it)->data.sKey == propname)
116*3ac0a46fSAndroid Build Coastguard Worker       return it;
117*3ac0a46fSAndroid Build Coastguard Worker   }
118*3ac0a46fSAndroid Build Coastguard Worker   return m_arrayGlobalData.end();
119*3ac0a46fSAndroid Build Coastguard Worker }
120*3ac0a46fSAndroid Build Coastguard Worker 
GetGlobalVariable(const ByteString & propname)121*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::Element* CFX_GlobalData::GetGlobalVariable(
122*3ac0a46fSAndroid Build Coastguard Worker     const ByteString& propname) {
123*3ac0a46fSAndroid Build Coastguard Worker   auto iter = FindGlobalVariable(propname);
124*3ac0a46fSAndroid Build Coastguard Worker   return iter != m_arrayGlobalData.end() ? iter->get() : nullptr;
125*3ac0a46fSAndroid Build Coastguard Worker }
126*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariableNumber(ByteString sPropName,double dData)127*3ac0a46fSAndroid Build Coastguard Worker void CFX_GlobalData::SetGlobalVariableNumber(ByteString sPropName,
128*3ac0a46fSAndroid Build Coastguard Worker                                              double dData) {
129*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
130*3ac0a46fSAndroid Build Coastguard Worker     return;
131*3ac0a46fSAndroid Build Coastguard Worker 
132*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
133*3ac0a46fSAndroid Build Coastguard Worker   if (pData) {
134*3ac0a46fSAndroid Build Coastguard Worker     pData->data.nType = CFX_Value::DataType::kNumber;
135*3ac0a46fSAndroid Build Coastguard Worker     pData->data.dData = dData;
136*3ac0a46fSAndroid Build Coastguard Worker     return;
137*3ac0a46fSAndroid Build Coastguard Worker   }
138*3ac0a46fSAndroid Build Coastguard Worker   auto pNewData = std::make_unique<CFX_GlobalData::Element>();
139*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sKey = std::move(sPropName);
140*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.nType = CFX_Value::DataType::kNumber;
141*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.dData = dData;
142*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.push_back(std::move(pNewData));
143*3ac0a46fSAndroid Build Coastguard Worker }
144*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariableBoolean(ByteString sPropName,bool bData)145*3ac0a46fSAndroid Build Coastguard Worker void CFX_GlobalData::SetGlobalVariableBoolean(ByteString sPropName,
146*3ac0a46fSAndroid Build Coastguard Worker                                               bool bData) {
147*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
148*3ac0a46fSAndroid Build Coastguard Worker     return;
149*3ac0a46fSAndroid Build Coastguard Worker 
150*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
151*3ac0a46fSAndroid Build Coastguard Worker   if (pData) {
152*3ac0a46fSAndroid Build Coastguard Worker     pData->data.nType = CFX_Value::DataType::kBoolean;
153*3ac0a46fSAndroid Build Coastguard Worker     pData->data.bData = bData;
154*3ac0a46fSAndroid Build Coastguard Worker     return;
155*3ac0a46fSAndroid Build Coastguard Worker   }
156*3ac0a46fSAndroid Build Coastguard Worker   auto pNewData = std::make_unique<CFX_GlobalData::Element>();
157*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sKey = std::move(sPropName);
158*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.nType = CFX_Value::DataType::kBoolean;
159*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.bData = bData;
160*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.push_back(std::move(pNewData));
161*3ac0a46fSAndroid Build Coastguard Worker }
162*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariableString(ByteString sPropName,const ByteString & sData)163*3ac0a46fSAndroid Build Coastguard Worker void CFX_GlobalData::SetGlobalVariableString(ByteString sPropName,
164*3ac0a46fSAndroid Build Coastguard Worker                                              const ByteString& sData) {
165*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
166*3ac0a46fSAndroid Build Coastguard Worker     return;
167*3ac0a46fSAndroid Build Coastguard Worker 
168*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
169*3ac0a46fSAndroid Build Coastguard Worker   if (pData) {
170*3ac0a46fSAndroid Build Coastguard Worker     pData->data.nType = CFX_Value::DataType::kString;
171*3ac0a46fSAndroid Build Coastguard Worker     pData->data.sData = sData;
172*3ac0a46fSAndroid Build Coastguard Worker     return;
173*3ac0a46fSAndroid Build Coastguard Worker   }
174*3ac0a46fSAndroid Build Coastguard Worker   auto pNewData = std::make_unique<CFX_GlobalData::Element>();
175*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sKey = std::move(sPropName);
176*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.nType = CFX_Value::DataType::kString;
177*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sData = sData;
178*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.push_back(std::move(pNewData));
179*3ac0a46fSAndroid Build Coastguard Worker }
180*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariableObject(ByteString sPropName,std::vector<std::unique_ptr<CFX_KeyValue>> array)181*3ac0a46fSAndroid Build Coastguard Worker void CFX_GlobalData::SetGlobalVariableObject(
182*3ac0a46fSAndroid Build Coastguard Worker     ByteString sPropName,
183*3ac0a46fSAndroid Build Coastguard Worker     std::vector<std::unique_ptr<CFX_KeyValue>> array) {
184*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
185*3ac0a46fSAndroid Build Coastguard Worker     return;
186*3ac0a46fSAndroid Build Coastguard Worker 
187*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
188*3ac0a46fSAndroid Build Coastguard Worker   if (pData) {
189*3ac0a46fSAndroid Build Coastguard Worker     pData->data.nType = CFX_Value::DataType::kObject;
190*3ac0a46fSAndroid Build Coastguard Worker     pData->data.objData = std::move(array);
191*3ac0a46fSAndroid Build Coastguard Worker     return;
192*3ac0a46fSAndroid Build Coastguard Worker   }
193*3ac0a46fSAndroid Build Coastguard Worker   auto pNewData = std::make_unique<CFX_GlobalData::Element>();
194*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sKey = std::move(sPropName);
195*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.nType = CFX_Value::DataType::kObject;
196*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.objData = std::move(array);
197*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.push_back(std::move(pNewData));
198*3ac0a46fSAndroid Build Coastguard Worker }
199*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariableNull(ByteString sPropName)200*3ac0a46fSAndroid Build Coastguard Worker void CFX_GlobalData::SetGlobalVariableNull(ByteString sPropName) {
201*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
202*3ac0a46fSAndroid Build Coastguard Worker     return;
203*3ac0a46fSAndroid Build Coastguard Worker 
204*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
205*3ac0a46fSAndroid Build Coastguard Worker   if (pData) {
206*3ac0a46fSAndroid Build Coastguard Worker     pData->data.nType = CFX_Value::DataType::kNull;
207*3ac0a46fSAndroid Build Coastguard Worker     return;
208*3ac0a46fSAndroid Build Coastguard Worker   }
209*3ac0a46fSAndroid Build Coastguard Worker   auto pNewData = std::make_unique<CFX_GlobalData::Element>();
210*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.sKey = std::move(sPropName);
211*3ac0a46fSAndroid Build Coastguard Worker   pNewData->data.nType = CFX_Value::DataType::kNull;
212*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.push_back(std::move(pNewData));
213*3ac0a46fSAndroid Build Coastguard Worker }
214*3ac0a46fSAndroid Build Coastguard Worker 
SetGlobalVariablePersistent(ByteString sPropName,bool bPersistent)215*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::SetGlobalVariablePersistent(ByteString sPropName,
216*3ac0a46fSAndroid Build Coastguard Worker                                                  bool bPersistent) {
217*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
218*3ac0a46fSAndroid Build Coastguard Worker     return false;
219*3ac0a46fSAndroid Build Coastguard Worker 
220*3ac0a46fSAndroid Build Coastguard Worker   CFX_GlobalData::Element* pData = GetGlobalVariable(sPropName);
221*3ac0a46fSAndroid Build Coastguard Worker   if (!pData)
222*3ac0a46fSAndroid Build Coastguard Worker     return false;
223*3ac0a46fSAndroid Build Coastguard Worker 
224*3ac0a46fSAndroid Build Coastguard Worker   pData->bPersistent = bPersistent;
225*3ac0a46fSAndroid Build Coastguard Worker   return true;
226*3ac0a46fSAndroid Build Coastguard Worker }
227*3ac0a46fSAndroid Build Coastguard Worker 
DeleteGlobalVariable(ByteString sPropName)228*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::DeleteGlobalVariable(ByteString sPropName) {
229*3ac0a46fSAndroid Build Coastguard Worker   if (!TrimPropName(&sPropName))
230*3ac0a46fSAndroid Build Coastguard Worker     return false;
231*3ac0a46fSAndroid Build Coastguard Worker 
232*3ac0a46fSAndroid Build Coastguard Worker   auto iter = FindGlobalVariable(sPropName);
233*3ac0a46fSAndroid Build Coastguard Worker   if (iter == m_arrayGlobalData.end())
234*3ac0a46fSAndroid Build Coastguard Worker     return false;
235*3ac0a46fSAndroid Build Coastguard Worker 
236*3ac0a46fSAndroid Build Coastguard Worker   m_arrayGlobalData.erase(iter);
237*3ac0a46fSAndroid Build Coastguard Worker   return true;
238*3ac0a46fSAndroid Build Coastguard Worker }
239*3ac0a46fSAndroid Build Coastguard Worker 
GetSize() const240*3ac0a46fSAndroid Build Coastguard Worker int32_t CFX_GlobalData::GetSize() const {
241*3ac0a46fSAndroid Build Coastguard Worker   return fxcrt::CollectionSize<int32_t>(m_arrayGlobalData);
242*3ac0a46fSAndroid Build Coastguard Worker }
243*3ac0a46fSAndroid Build Coastguard Worker 
GetAt(int index)244*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::Element* CFX_GlobalData::GetAt(int index) {
245*3ac0a46fSAndroid Build Coastguard Worker   if (index < 0 || index >= GetSize())
246*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
247*3ac0a46fSAndroid Build Coastguard Worker   return m_arrayGlobalData[index].get();
248*3ac0a46fSAndroid Build Coastguard Worker }
249*3ac0a46fSAndroid Build Coastguard Worker 
LoadGlobalPersistentVariables()250*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::LoadGlobalPersistentVariables() {
251*3ac0a46fSAndroid Build Coastguard Worker   if (!m_pDelegate)
252*3ac0a46fSAndroid Build Coastguard Worker     return false;
253*3ac0a46fSAndroid Build Coastguard Worker 
254*3ac0a46fSAndroid Build Coastguard Worker   bool ret;
255*3ac0a46fSAndroid Build Coastguard Worker   {
256*3ac0a46fSAndroid Build Coastguard Worker     // Span can't outlive call to BufferDone().
257*3ac0a46fSAndroid Build Coastguard Worker     absl::optional<pdfium::span<uint8_t>> buffer = m_pDelegate->LoadBuffer();
258*3ac0a46fSAndroid Build Coastguard Worker     if (!buffer.has_value() || buffer.value().empty())
259*3ac0a46fSAndroid Build Coastguard Worker       return false;
260*3ac0a46fSAndroid Build Coastguard Worker 
261*3ac0a46fSAndroid Build Coastguard Worker     ret = LoadGlobalPersistentVariablesFromBuffer(buffer.value());
262*3ac0a46fSAndroid Build Coastguard Worker   }
263*3ac0a46fSAndroid Build Coastguard Worker   m_pDelegate->BufferDone();
264*3ac0a46fSAndroid Build Coastguard Worker   return ret;
265*3ac0a46fSAndroid Build Coastguard Worker }
266*3ac0a46fSAndroid Build Coastguard Worker 
LoadGlobalPersistentVariablesFromBuffer(pdfium::span<uint8_t> buffer)267*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::LoadGlobalPersistentVariablesFromBuffer(
268*3ac0a46fSAndroid Build Coastguard Worker     pdfium::span<uint8_t> buffer) {
269*3ac0a46fSAndroid Build Coastguard Worker   if (buffer.size() < kMinGlobalDataBytes)
270*3ac0a46fSAndroid Build Coastguard Worker     return false;
271*3ac0a46fSAndroid Build Coastguard Worker 
272*3ac0a46fSAndroid Build Coastguard Worker   CRYPT_ArcFourCryptBlock(buffer, kRC4KEY);
273*3ac0a46fSAndroid Build Coastguard Worker 
274*3ac0a46fSAndroid Build Coastguard Worker   uint8_t* p = buffer.data();
275*3ac0a46fSAndroid Build Coastguard Worker   uint16_t wType = *((uint16_t*)p);
276*3ac0a46fSAndroid Build Coastguard Worker   p += sizeof(uint16_t);
277*3ac0a46fSAndroid Build Coastguard Worker   if (wType != kMagic)
278*3ac0a46fSAndroid Build Coastguard Worker     return false;
279*3ac0a46fSAndroid Build Coastguard Worker 
280*3ac0a46fSAndroid Build Coastguard Worker   uint16_t wVersion = *((uint16_t*)p);
281*3ac0a46fSAndroid Build Coastguard Worker   p += sizeof(uint16_t);
282*3ac0a46fSAndroid Build Coastguard Worker   if (wVersion > kMaxVersion)
283*3ac0a46fSAndroid Build Coastguard Worker     return false;
284*3ac0a46fSAndroid Build Coastguard Worker 
285*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwCount = *((uint32_t*)p);
286*3ac0a46fSAndroid Build Coastguard Worker   p += sizeof(uint32_t);
287*3ac0a46fSAndroid Build Coastguard Worker 
288*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwSize = *((uint32_t*)p);
289*3ac0a46fSAndroid Build Coastguard Worker   p += sizeof(uint32_t);
290*3ac0a46fSAndroid Build Coastguard Worker 
291*3ac0a46fSAndroid Build Coastguard Worker   if (dwSize != buffer.size() - sizeof(uint16_t) * 2 - sizeof(uint32_t) * 2)
292*3ac0a46fSAndroid Build Coastguard Worker     return false;
293*3ac0a46fSAndroid Build Coastguard Worker 
294*3ac0a46fSAndroid Build Coastguard Worker   for (int32_t i = 0, sz = dwCount; i < sz; i++) {
295*3ac0a46fSAndroid Build Coastguard Worker     if (p + sizeof(uint32_t) >= buffer.end()) {
296*3ac0a46fSAndroid Build Coastguard Worker       break;
297*3ac0a46fSAndroid Build Coastguard Worker     }
298*3ac0a46fSAndroid Build Coastguard Worker 
299*3ac0a46fSAndroid Build Coastguard Worker     uint32_t dwNameLen = 0;
300*3ac0a46fSAndroid Build Coastguard Worker     memcpy(&dwNameLen, p, sizeof(uint32_t));
301*3ac0a46fSAndroid Build Coastguard Worker     p += sizeof(uint32_t);
302*3ac0a46fSAndroid Build Coastguard Worker     if (p + dwNameLen > buffer.end())
303*3ac0a46fSAndroid Build Coastguard Worker       break;
304*3ac0a46fSAndroid Build Coastguard Worker 
305*3ac0a46fSAndroid Build Coastguard Worker     ByteString sEntry = ByteString(p, dwNameLen);
306*3ac0a46fSAndroid Build Coastguard Worker     p += sizeof(char) * dwNameLen;
307*3ac0a46fSAndroid Build Coastguard Worker 
308*3ac0a46fSAndroid Build Coastguard Worker     uint16_t wDataType = 0;
309*3ac0a46fSAndroid Build Coastguard Worker     memcpy(&wDataType, p, sizeof(uint16_t));
310*3ac0a46fSAndroid Build Coastguard Worker     p += sizeof(uint16_t);
311*3ac0a46fSAndroid Build Coastguard Worker 
312*3ac0a46fSAndroid Build Coastguard Worker     CFX_Value::DataType eDataType = static_cast<CFX_Value::DataType>(wDataType);
313*3ac0a46fSAndroid Build Coastguard Worker 
314*3ac0a46fSAndroid Build Coastguard Worker     switch (eDataType) {
315*3ac0a46fSAndroid Build Coastguard Worker       case CFX_Value::DataType::kNumber: {
316*3ac0a46fSAndroid Build Coastguard Worker         double dData = 0;
317*3ac0a46fSAndroid Build Coastguard Worker         switch (wVersion) {
318*3ac0a46fSAndroid Build Coastguard Worker           case 1: {
319*3ac0a46fSAndroid Build Coastguard Worker             uint32_t dwData = 0;
320*3ac0a46fSAndroid Build Coastguard Worker             memcpy(&dwData, p, sizeof(uint32_t));
321*3ac0a46fSAndroid Build Coastguard Worker             p += sizeof(uint32_t);
322*3ac0a46fSAndroid Build Coastguard Worker             dData = dwData;
323*3ac0a46fSAndroid Build Coastguard Worker           } break;
324*3ac0a46fSAndroid Build Coastguard Worker           case 2: {
325*3ac0a46fSAndroid Build Coastguard Worker             dData = 0;
326*3ac0a46fSAndroid Build Coastguard Worker             memcpy(&dData, p, sizeof(double));
327*3ac0a46fSAndroid Build Coastguard Worker             p += sizeof(double);
328*3ac0a46fSAndroid Build Coastguard Worker           } break;
329*3ac0a46fSAndroid Build Coastguard Worker         }
330*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariableNumber(sEntry, dData);
331*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariablePersistent(sEntry, true);
332*3ac0a46fSAndroid Build Coastguard Worker       } break;
333*3ac0a46fSAndroid Build Coastguard Worker       case CFX_Value::DataType::kBoolean: {
334*3ac0a46fSAndroid Build Coastguard Worker         uint16_t wData = 0;
335*3ac0a46fSAndroid Build Coastguard Worker         memcpy(&wData, p, sizeof(uint16_t));
336*3ac0a46fSAndroid Build Coastguard Worker         p += sizeof(uint16_t);
337*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariableBoolean(sEntry, (bool)(wData == 1));
338*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariablePersistent(sEntry, true);
339*3ac0a46fSAndroid Build Coastguard Worker       } break;
340*3ac0a46fSAndroid Build Coastguard Worker       case CFX_Value::DataType::kString: {
341*3ac0a46fSAndroid Build Coastguard Worker         uint32_t dwLength = 0;
342*3ac0a46fSAndroid Build Coastguard Worker         memcpy(&dwLength, p, sizeof(uint32_t));
343*3ac0a46fSAndroid Build Coastguard Worker         p += sizeof(uint32_t);
344*3ac0a46fSAndroid Build Coastguard Worker         if (p + dwLength > buffer.end())
345*3ac0a46fSAndroid Build Coastguard Worker           break;
346*3ac0a46fSAndroid Build Coastguard Worker 
347*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariableString(sEntry, ByteString(p, dwLength));
348*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariablePersistent(sEntry, true);
349*3ac0a46fSAndroid Build Coastguard Worker         p += sizeof(char) * dwLength;
350*3ac0a46fSAndroid Build Coastguard Worker       } break;
351*3ac0a46fSAndroid Build Coastguard Worker       case CFX_Value::DataType::kNull: {
352*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariableNull(sEntry);
353*3ac0a46fSAndroid Build Coastguard Worker         SetGlobalVariablePersistent(sEntry, true);
354*3ac0a46fSAndroid Build Coastguard Worker       } break;
355*3ac0a46fSAndroid Build Coastguard Worker       case CFX_Value::DataType::kObject:
356*3ac0a46fSAndroid Build Coastguard Worker         // Arrays aren't allowed in these buffers, nor are unrecognized tags.
357*3ac0a46fSAndroid Build Coastguard Worker         return false;
358*3ac0a46fSAndroid Build Coastguard Worker     }
359*3ac0a46fSAndroid Build Coastguard Worker   }
360*3ac0a46fSAndroid Build Coastguard Worker   return true;
361*3ac0a46fSAndroid Build Coastguard Worker }
362*3ac0a46fSAndroid Build Coastguard Worker 
SaveGlobalPersisitentVariables()363*3ac0a46fSAndroid Build Coastguard Worker bool CFX_GlobalData::SaveGlobalPersisitentVariables() {
364*3ac0a46fSAndroid Build Coastguard Worker   if (!m_pDelegate)
365*3ac0a46fSAndroid Build Coastguard Worker     return false;
366*3ac0a46fSAndroid Build Coastguard Worker 
367*3ac0a46fSAndroid Build Coastguard Worker   uint32_t nCount = 0;
368*3ac0a46fSAndroid Build Coastguard Worker   BinaryBuffer sData;
369*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& pElement : m_arrayGlobalData) {
370*3ac0a46fSAndroid Build Coastguard Worker     if (!pElement->bPersistent)
371*3ac0a46fSAndroid Build Coastguard Worker       continue;
372*3ac0a46fSAndroid Build Coastguard Worker 
373*3ac0a46fSAndroid Build Coastguard Worker     BinaryBuffer sElement;
374*3ac0a46fSAndroid Build Coastguard Worker     if (!MakeByteString(pElement->data.sKey, pElement->data, &sElement))
375*3ac0a46fSAndroid Build Coastguard Worker       continue;
376*3ac0a46fSAndroid Build Coastguard Worker 
377*3ac0a46fSAndroid Build Coastguard Worker     if (sData.GetSize() + sElement.GetSize() > kMaxGlobalDataBytes)
378*3ac0a46fSAndroid Build Coastguard Worker       break;
379*3ac0a46fSAndroid Build Coastguard Worker 
380*3ac0a46fSAndroid Build Coastguard Worker     sData.AppendSpan(sElement.GetSpan());
381*3ac0a46fSAndroid Build Coastguard Worker     nCount++;
382*3ac0a46fSAndroid Build Coastguard Worker   }
383*3ac0a46fSAndroid Build Coastguard Worker 
384*3ac0a46fSAndroid Build Coastguard Worker   BinaryBuffer sFile;
385*3ac0a46fSAndroid Build Coastguard Worker   sFile.AppendUint16(kMagic);
386*3ac0a46fSAndroid Build Coastguard Worker   sFile.AppendUint16(kMaxVersion);
387*3ac0a46fSAndroid Build Coastguard Worker   sFile.AppendUint32(nCount);
388*3ac0a46fSAndroid Build Coastguard Worker 
389*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwSize = pdfium::base::checked_cast<uint32_t>(sData.GetSize());
390*3ac0a46fSAndroid Build Coastguard Worker   sFile.AppendUint32(dwSize);
391*3ac0a46fSAndroid Build Coastguard Worker   sFile.AppendSpan(sData.GetSpan());
392*3ac0a46fSAndroid Build Coastguard Worker 
393*3ac0a46fSAndroid Build Coastguard Worker   CRYPT_ArcFourCryptBlock(sFile.GetMutableSpan(), kRC4KEY);
394*3ac0a46fSAndroid Build Coastguard Worker   return m_pDelegate->StoreBuffer(sFile.GetSpan());
395*3ac0a46fSAndroid Build Coastguard Worker }
396*3ac0a46fSAndroid Build Coastguard Worker 
397*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::Element::Element() = default;
398*3ac0a46fSAndroid Build Coastguard Worker 
399*3ac0a46fSAndroid Build Coastguard Worker CFX_GlobalData::Element::~Element() = default;
400