1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
7 #include "../../include/javascript/JavaScript.h"
8 #include "../../include/javascript/IJavaScript.h"
9 #include "../../include/javascript/JS_GlobalData.h"
11 #define JS_MAXGLOBALDATA (1024 * 4 - 8)
13 /* --------------------- CJS_GlobalVariableArray --------------------- */
15 CJS_GlobalVariableArray::CJS_GlobalVariableArray()
19 CJS_GlobalVariableArray::~CJS_GlobalVariableArray()
24 void CJS_GlobalVariableArray::Copy(const CJS_GlobalVariableArray& array)
27 for (int i=0,sz=array.Count(); i<sz; i++)
29 CJS_KeyValue* pOldObjData = array.GetAt(i);
30 ASSERT(pOldObjData != NULL);
32 switch (pOldObjData->nType)
34 case JS_GLOBALDATA_TYPE_NUMBER:
36 CJS_KeyValue* pNewObjData = new CJS_KeyValue;
37 pNewObjData->sKey = pOldObjData->sKey;
38 pNewObjData->nType = pOldObjData->nType;
39 pNewObjData->dData = pOldObjData->dData;
43 case JS_GLOBALDATA_TYPE_BOOLEAN:
45 CJS_KeyValue* pNewObjData = new CJS_KeyValue;
46 pNewObjData->sKey = pOldObjData->sKey;
47 pNewObjData->nType = pOldObjData->nType;
48 pNewObjData->bData = pOldObjData->bData;
52 case JS_GLOBALDATA_TYPE_STRING:
54 CJS_KeyValue* pNewObjData = new CJS_KeyValue;
55 pNewObjData->sKey = pOldObjData->sKey;
56 pNewObjData->nType = pOldObjData->nType;
57 pNewObjData->sData = pOldObjData->sData;
61 case JS_GLOBALDATA_TYPE_OBJECT:
63 CJS_KeyValue* pNewObjData = new CJS_KeyValue;
64 pNewObjData->sKey = pOldObjData->sKey;
65 pNewObjData->nType = pOldObjData->nType;
66 pNewObjData->objData.Copy(pOldObjData->objData);
69 case JS_GLOBALDATA_TYPE_NULL:
71 CJS_KeyValue* pNewObjData = new CJS_KeyValue;
72 pNewObjData->sKey = pOldObjData->sKey;
73 pNewObjData->nType = pOldObjData->nType;
80 void CJS_GlobalVariableArray::Add(CJS_KeyValue* p)
85 int CJS_GlobalVariableArray::Count() const
87 return array.GetSize();
90 CJS_KeyValue* CJS_GlobalVariableArray::GetAt(int index) const
92 return array.GetAt(index);
95 void CJS_GlobalVariableArray::Empty()
97 for (int i=0,sz=array.GetSize(); i<sz; i++)
98 delete array.GetAt(i);
102 /* -------------------------- CJS_GlobalData -------------------------- */
104 #define READER_JS_GLOBALDATA_FILENAME L"Reader_JsGlobal.Data"
105 #define PHANTOM_JS_GLOBALDATA_FILENAME L"Phantom_JsGlobal.Data"
106 #define SDK_JS_GLOBALDATA_FILENAME L"SDK_JsGlobal.Data"
108 static const FX_BYTE JS_RC4KEY[] = {0x19,0xa8,0xe8,0x01,0xf6,0xa8,0xb6,0x4d,0x82,0x04,
109 0x45,0x6d,0xb4,0xcf,0xd7,0x77,0x67,0xf9,0x75,0x9f,
110 0xf0,0xe0,0x1e,0x51,0xee,0x46,0xfd,0x0b,0xc9,0x93,
111 0x25,0x55,0x4a,0xee,0xe0,0x16,0xd0,0xdf,0x8c,0xfa,
112 0x2a,0xa9,0x49,0xfd,0x97,0x1c,0x0e,0x22,0x13,0x28,
113 0x7c,0xaf,0xc4,0xfc,0x9c,0x12,0x65,0x8c,0x4e,0x5b,
114 0x04,0x75,0x89,0xc9,0xb1,0xed,0x50,0xca,0x96,0x6f,
115 0x1a,0x7a,0xfe,0x58,0x5d,0xec,0x19,0x4a,0xf6,0x35,
116 0x6a,0x97,0x14,0x00,0x0e,0xd0,0x6b,0xbb,0xd5,0x75,
117 0x55,0x8b,0x6e,0x6b,0x19,0xa0,0xf8,0x77,0xd5,0xa3
120 CJS_GlobalData::CJS_GlobalData(CPDFDoc_Environment* pApp)
122 // IBaseAnnot* pBaseAnnot = IBaseAnnot::GetBaseAnnot(m_pApp);
123 // ASSERT(pBaseAnnot != NULL);
125 // m_sFilePath = pBaseAnnot->GetUserPath();
126 m_sFilePath += SDK_JS_GLOBALDATA_FILENAME;
128 LoadGlobalPersistentVariables();
131 CJS_GlobalData::~CJS_GlobalData()
133 SaveGlobalPersisitentVariables();
135 for (int i=0,sz=m_arrayGlobalData.GetSize(); i<sz; i++)
136 delete m_arrayGlobalData.GetAt(i);
138 m_arrayGlobalData.RemoveAll();
141 int CJS_GlobalData::FindGlobalVariable(FX_LPCSTR propname)
143 ASSERT(propname != NULL);
147 for (int i=0,sz=m_arrayGlobalData.GetSize(); i<sz; i++)
149 CJS_GlobalData_Element* pTemp = m_arrayGlobalData.GetAt(i);
150 if (pTemp->data.sKey[0] == *propname && pTemp->data.sKey == propname)
160 CJS_GlobalData_Element* CJS_GlobalData::GetGlobalVariable(FX_LPCSTR propname)
162 ASSERT(propname != NULL);
164 int nFind = FindGlobalVariable(propname);
167 return m_arrayGlobalData.GetAt(nFind);
172 void CJS_GlobalData::SetGlobalVariableNumber(FX_LPCSTR propname, double dData)
174 ASSERT(propname != NULL);
175 CFX_ByteString sPropName = propname;
177 sPropName.TrimLeft();
178 sPropName.TrimRight();
180 if (sPropName.GetLength() == 0) return;
182 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
184 pData->data.nType = JS_GLOBALDATA_TYPE_NUMBER;
185 pData->data.dData = dData;
189 CJS_GlobalData_Element* pNewData = new CJS_GlobalData_Element;
190 pNewData->data.sKey = sPropName;
191 pNewData->data.nType = JS_GLOBALDATA_TYPE_NUMBER;
192 pNewData->data.dData = dData;
194 m_arrayGlobalData.Add(pNewData);
198 void CJS_GlobalData::SetGlobalVariableBoolean(FX_LPCSTR propname, bool bData)
200 ASSERT(propname != NULL);
201 CFX_ByteString sPropName = propname;
203 sPropName.TrimLeft();
204 sPropName.TrimRight();
206 if (sPropName.GetLength() == 0) return;
208 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
210 pData->data.nType = JS_GLOBALDATA_TYPE_BOOLEAN;
211 pData->data.bData = bData;
215 CJS_GlobalData_Element* pNewData = new CJS_GlobalData_Element;
216 pNewData->data.sKey = sPropName;
217 pNewData->data.nType = JS_GLOBALDATA_TYPE_BOOLEAN;
218 pNewData->data.bData = bData;
220 m_arrayGlobalData.Add(pNewData);
224 void CJS_GlobalData::SetGlobalVariableString(FX_LPCSTR propname, const CFX_ByteString& sData)
226 ASSERT(propname != NULL);
227 CFX_ByteString sPropName = propname;
229 sPropName.TrimLeft();
230 sPropName.TrimRight();
232 if (sPropName.GetLength() == 0) return;
234 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
236 pData->data.nType = JS_GLOBALDATA_TYPE_STRING;
237 pData->data.sData = sData;
241 CJS_GlobalData_Element* pNewData = new CJS_GlobalData_Element;
242 pNewData->data.sKey = sPropName;
243 pNewData->data.nType = JS_GLOBALDATA_TYPE_STRING;
244 pNewData->data.sData = sData;
246 m_arrayGlobalData.Add(pNewData);
250 void CJS_GlobalData::SetGlobalVariableObject(FX_LPCSTR propname, const CJS_GlobalVariableArray& array)
252 ASSERT(propname != NULL);
253 CFX_ByteString sPropName = propname;
255 sPropName.TrimLeft();
256 sPropName.TrimRight();
258 if (sPropName.GetLength() == 0) return;
260 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
262 pData->data.nType = JS_GLOBALDATA_TYPE_OBJECT;
263 pData->data.objData.Copy(array);
267 CJS_GlobalData_Element* pNewData = new CJS_GlobalData_Element;
268 pNewData->data.sKey = sPropName;
269 pNewData->data.nType = JS_GLOBALDATA_TYPE_OBJECT;
270 pNewData->data.objData.Copy(array);
272 m_arrayGlobalData.Add(pNewData);
276 void CJS_GlobalData::SetGlobalVariableNull(FX_LPCSTR propname)
278 ASSERT(propname != NULL);
279 CFX_ByteString sPropName = propname;
281 sPropName.TrimLeft();
282 sPropName.TrimRight();
284 if (sPropName.GetLength() == 0) return;
286 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
288 pData->data.nType = JS_GLOBALDATA_TYPE_NULL;
292 CJS_GlobalData_Element* pNewData = new CJS_GlobalData_Element;
293 pNewData->data.sKey = sPropName;
294 pNewData->data.nType = JS_GLOBALDATA_TYPE_NULL;
296 m_arrayGlobalData.Add(pNewData);
300 FX_BOOL CJS_GlobalData::SetGlobalVariablePersistent(FX_LPCSTR propname, FX_BOOL bPersistent)
302 ASSERT(propname != NULL);
303 CFX_ByteString sPropName = propname;
305 sPropName.TrimLeft();
306 sPropName.TrimRight();
308 if (sPropName.GetLength() == 0) return FALSE;
310 if (CJS_GlobalData_Element* pData = GetGlobalVariable(sPropName))
312 pData->bPersistent = bPersistent;
319 FX_BOOL CJS_GlobalData::DeleteGlobalVariable(FX_LPCSTR propname)
321 ASSERT(propname != NULL);
322 CFX_ByteString sPropName = propname;
324 sPropName.TrimLeft();
325 sPropName.TrimRight();
327 if (sPropName.GetLength() == 0) return FALSE;
329 int nFind = FindGlobalVariable(sPropName);
333 delete m_arrayGlobalData.GetAt(nFind);
334 m_arrayGlobalData.RemoveAt(nFind);
341 FX_INT32 CJS_GlobalData::GetSize() const
343 return m_arrayGlobalData.GetSize();
346 CJS_GlobalData_Element* CJS_GlobalData::GetAt(int index) const
348 return m_arrayGlobalData.GetAt(index);
351 void CJS_GlobalData::LoadGlobalPersistentVariables()
353 FX_LPBYTE pBuffer = NULL;
354 FX_INT32 nLength = 0;
356 LoadFileBuffer(m_sFilePath, pBuffer, nLength);
358 CRYPT_ArcFourCryptBlock(pBuffer, nLength, JS_RC4KEY, sizeof(JS_RC4KEY));
362 FX_LPBYTE p = pBuffer;
363 FX_WORD wType = *((FX_WORD*)p);
364 p += sizeof(FX_WORD);
366 //FX_WORD wTemp = (FX_WORD)(('X' << 8) | 'F');
368 if (wType == (FX_WORD)(('X' << 8) | 'F'))
370 FX_WORD wVersion = *((FX_WORD*)p);
371 p += sizeof(FX_WORD);
373 ASSERT(wVersion <= 2);
375 FX_DWORD dwCount = *((FX_DWORD*)p);
376 p += sizeof(FX_DWORD);
378 FX_DWORD dwSize = *((FX_DWORD*)p);
379 p += sizeof(FX_DWORD);
381 if (dwSize == nLength - sizeof(FX_WORD) * 2 - sizeof(FX_DWORD)* 2)
383 for (FX_INT32 i=0,sz=dwCount; i<sz; i++)
385 if (p > pBuffer + nLength)
388 FX_DWORD dwNameLen = *((FX_DWORD*)p);
389 p += sizeof(FX_DWORD);
391 if (p + dwNameLen > pBuffer + nLength)
394 CFX_ByteString sEntry = CFX_ByteString(p, dwNameLen);
395 p += sizeof(char) * dwNameLen;
397 FX_WORD wDataType = *((FX_WORD*)p);
398 p += sizeof(FX_WORD);
402 case JS_GLOBALDATA_TYPE_NUMBER:
409 FX_DWORD dwData = *((FX_DWORD*)p);
410 p += sizeof(FX_DWORD);
416 dData = *((double*)p);
421 SetGlobalVariableNumber(sEntry, dData);
422 SetGlobalVariablePersistent(sEntry, TRUE);
425 case JS_GLOBALDATA_TYPE_BOOLEAN:
427 FX_WORD wData = *((FX_WORD*)p);
428 p += sizeof(FX_WORD);
429 SetGlobalVariableBoolean(sEntry, (bool)(wData == 1));
430 SetGlobalVariablePersistent(sEntry, TRUE);
433 case JS_GLOBALDATA_TYPE_STRING:
435 FX_DWORD dwLength = *((FX_DWORD*)p);
436 p += sizeof(FX_DWORD);
438 if (p + dwLength > pBuffer + nLength)
441 SetGlobalVariableString(sEntry, CFX_ByteString(p, dwLength));
442 SetGlobalVariablePersistent(sEntry, TRUE);
443 p += sizeof(char) * dwLength;
446 case JS_GLOBALDATA_TYPE_NULL:
448 SetGlobalVariableNull(sEntry);
449 SetGlobalVariablePersistent(sEntry, TRUE);
460 struct js_global_datafile_header
462 FX_WORD type; //FX ('X' << 8) | 'F'
463 FX_WORD version; //1.0
466 struct js_global_datafile_data
476 void CJS_GlobalData::SaveGlobalPersisitentVariables()
481 for (int i=0,sz=m_arrayGlobalData.GetSize(); i<sz; i++)
483 CJS_GlobalData_Element* pElement = m_arrayGlobalData.GetAt(i);
484 ASSERT(pElement != NULL);
486 if (pElement->bPersistent)
488 CFX_BinaryBuf sElement;
489 MakeByteString(pElement->data.sKey, &pElement->data, sElement);
491 if (sData.GetSize() + sElement.GetSize() > JS_MAXGLOBALDATA)
494 sData.AppendBlock(sElement.GetBuffer(), sElement.GetSize());
501 FX_WORD wType = (FX_WORD)(('X' << 8) | 'F');
502 sFile.AppendBlock(&wType, sizeof(FX_WORD));
503 FX_WORD wVersion = 2;
504 sFile.AppendBlock(&wVersion, sizeof(FX_WORD));
505 sFile.AppendBlock(&nCount, sizeof(FX_DWORD));
506 FX_DWORD dwSize = sData.GetSize();
507 sFile.AppendBlock(&dwSize, sizeof(FX_DWORD));
509 sFile.AppendBlock(sData.GetBuffer(), sData.GetSize());
511 CRYPT_ArcFourCryptBlock(sFile.GetBuffer(), sFile.GetSize(), JS_RC4KEY, sizeof(JS_RC4KEY));
512 WriteFileBuffer(m_sFilePath, (FX_LPCSTR)sFile.GetBuffer(), sFile.GetSize());
515 void CJS_GlobalData::LoadFileBuffer(FX_LPCWSTR sFilePath, FX_LPBYTE& pBuffer, FX_INT32& nLength)
520 void CJS_GlobalData::WriteFileBuffer(FX_LPCWSTR sFilePath, FX_LPCSTR pBuffer, FX_INT32 nLength)
525 void CJS_GlobalData::MakeByteString(const CFX_ByteString& name, CJS_KeyValue* pData, CFX_BinaryBuf& sData)
527 ASSERT(pData != NULL);
529 FX_WORD wType = (FX_WORD)pData->nType;
533 case JS_GLOBALDATA_TYPE_NUMBER:
535 FX_DWORD dwNameLen = (FX_DWORD)name.GetLength();
536 sData.AppendBlock(&dwNameLen, sizeof(FX_DWORD));
537 sData.AppendString(name);
539 sData.AppendBlock(&wType, sizeof(FX_WORD));
540 double dData = pData->dData;
541 sData.AppendBlock(&dData, sizeof(double));
544 case JS_GLOBALDATA_TYPE_BOOLEAN:
546 FX_DWORD dwNameLen = (FX_DWORD)name.GetLength();
547 sData.AppendBlock(&dwNameLen, sizeof(FX_DWORD));
548 sData.AppendString(name);
550 sData.AppendBlock(&wType, sizeof(FX_WORD));
551 FX_WORD wData = (FX_WORD)pData->bData;
552 sData.AppendBlock(&wData, sizeof(FX_WORD));
555 case JS_GLOBALDATA_TYPE_STRING:
557 FX_DWORD dwNameLen = (FX_DWORD)name.GetLength();
558 sData.AppendBlock(&dwNameLen, sizeof(FX_DWORD));
559 sData.AppendString(name);
561 sData.AppendBlock(&wType, sizeof(FX_WORD));
563 FX_DWORD dwDataLen = (FX_DWORD)pData->sData.GetLength();
564 sData.AppendBlock(&dwDataLen, sizeof(FX_DWORD));
565 sData.AppendString(pData->sData);
568 case JS_GLOBALDATA_TYPE_NULL:
570 FX_DWORD dwNameLen = (FX_DWORD)name.GetLength();
571 sData.AppendBlock(&dwNameLen, sizeof(FX_DWORD));
572 sData.AppendString(name);
574 sData.AppendBlock(&wType, sizeof(FX_DWORD));