1
0
mirror of https://github.com/VCMP-SqMod/SqMod.git synced 2025-03-04 03:07:29 +01:00
SqMod/vendor/POCO/PDF/src/hpdf_dict.c
Sandu Liviu Catalin 4a6bfc086c Major plugin refactor and cleanup.
Switched to POCO library for unified platform/library interface.
Deprecated the external module API. It was creating more problems than solving.
Removed most built-in libraries in favor of system libraries for easier maintenance.
Cleaned and secured code with help from static analyzers.
2021-01-30 08:51:39 +02:00

519 lines
14 KiB
C

/*
* << Haru Free PDF Library >> -- hpdf_dict.c
*
* URL: http://libharu.org
*
* Copyright (c) 1999-2006 Takeshi Kanno <takeshi_kanno@est.hi-ho.ne.jp>
* Copyright (c) 2007-2009 Antony Dovgal <tony@daylessday.org>
*
* Permission to use, copy, modify, distribute and sell this software
* and its documentation for any purpose is hereby granted without fee,
* provided that the above copyright notice appear in all copies and
* that both that copyright notice and this permission notice appear
* in supporting documentation.
* It is provided "as is" without express or implied warranty.
*
*/
#include "hpdf_conf.h"
#include "hpdf_utils.h"
#include "hpdf_objects.h"
HPDF_DictElement
GetElement (HPDF_Dict dict,
const char *key);
/*--------------------------------------------------------------------------*/
HPDF_Dict
HPDF_Dict_New (HPDF_MMgr mmgr)
{
HPDF_Dict obj;
obj = (HPDF_Dict)HPDF_GetMem (mmgr, sizeof(HPDF_Dict_Rec));
if (obj) {
HPDF_MemSet (obj, 0, sizeof(HPDF_Dict_Rec));
obj->header.obj_class = HPDF_OCLASS_DICT;
obj->mmgr = mmgr;
obj->error = mmgr->error;
obj->list = HPDF_List_New (mmgr, HPDF_DEF_ITEMS_PER_BLOCK);
obj->filter = HPDF_STREAM_FILTER_NONE;
if (!obj->list) {
HPDF_FreeMem (mmgr, obj);
obj = NULL;
}
}
return obj;
}
HPDF_Dict
HPDF_DictStream_New (HPDF_MMgr mmgr,
HPDF_Xref xref)
{
HPDF_Dict obj;
HPDF_Number length;
HPDF_STATUS ret = 0;
obj = HPDF_Dict_New (mmgr);
if (!obj)
return NULL;
/* only stream object is added to xref automatically */
ret += HPDF_Xref_Add (xref, obj);
if (ret != HPDF_OK)
return NULL;
length = HPDF_Number_New (mmgr, 0);
if (!length)
return NULL;
ret = HPDF_Xref_Add (xref, length);
if (ret != HPDF_OK)
return NULL;
ret = HPDF_Dict_Add (obj, "Length", length);
if (ret != HPDF_OK)
return NULL;
obj->stream = HPDF_MemStream_New (mmgr, HPDF_STREAM_BUF_SIZ);
if (!obj->stream)
return NULL;
return obj;
}
void
HPDF_Dict_Free (HPDF_Dict dict)
{
HPDF_UINT i;
if (!dict)
return;
if (dict->free_fn)
dict->free_fn (dict);
for (i = 0; i < dict->list->count; i++) {
HPDF_DictElement element =
(HPDF_DictElement)HPDF_List_ItemAt (dict->list, i);
if (element) {
HPDF_Obj_Free (dict->mmgr, element->value);
HPDF_FreeMem (dict->mmgr, element);
}
}
if (dict->stream)
HPDF_Stream_Free (dict->stream);
HPDF_List_Free (dict->list);
dict->header.obj_class = 0;
HPDF_FreeMem (dict->mmgr, dict);
}
HPDF_STATUS
HPDF_Dict_Add_FilterParams(HPDF_Dict dict, HPDF_Dict filterParam)
{
HPDF_Array paramArray;
/* prepare params object */
paramArray = HPDF_Dict_GetItem (dict, "DecodeParms",
HPDF_OCLASS_ARRAY);
if(paramArray==NULL) {
paramArray = HPDF_Array_New (dict->mmgr);
if (!paramArray)
return HPDF_Error_GetCode (dict->error);
/* add parameters */
HPDF_Dict_Add(dict, "DecodeParms", paramArray);
}
HPDF_Array_Add(paramArray, filterParam);
return HPDF_OK;
}
HPDF_STATUS
HPDF_Dict_Write (HPDF_Dict dict,
HPDF_Stream stream,
HPDF_Encrypt e)
{
HPDF_UINT i;
HPDF_STATUS ret;
ret = HPDF_Stream_WriteStr (stream, "<<\012");
if (ret != HPDF_OK)
return ret;
if (dict->before_write_fn) {
if ((ret = dict->before_write_fn (dict)) != HPDF_OK)
return ret;
}
/* encrypt-dict must not be encrypted. */
if (dict->header.obj_class == (HPDF_OCLASS_DICT | HPDF_OSUBCLASS_ENCRYPT))
e = NULL;
if (dict->stream) {
/* set filter element */
if (dict->filter == HPDF_STREAM_FILTER_NONE)
HPDF_Dict_RemoveElement (dict, "Filter");
else {
HPDF_Array array = HPDF_Dict_GetItem (dict, "Filter",
HPDF_OCLASS_ARRAY);
if (!array) {
array = HPDF_Array_New (dict->mmgr);
if (!array)
return HPDF_Error_GetCode (dict->error);
ret = HPDF_Dict_Add (dict, "Filter", array);
if (ret != HPDF_OK)
return ret;
}
HPDF_Array_Clear (array);
#ifndef LIBHPDF_HAVE_NOZLIB
if (dict->filter & HPDF_STREAM_FILTER_FLATE_DECODE)
HPDF_Array_AddName (array, "FlateDecode");
#endif /* LIBHPDF_HAVE_NOZLIB */
if (dict->filter & HPDF_STREAM_FILTER_DCT_DECODE)
HPDF_Array_AddName (array, "DCTDecode");
if(dict->filter & HPDF_STREAM_FILTER_CCITT_DECODE)
HPDF_Array_AddName (array, "CCITTFaxDecode");
if(dict->filterParams!=NULL)
{
HPDF_Dict_Add_FilterParams(dict, dict->filterParams);
}
}
}
for (i = 0; i < dict->list->count; i++) {
HPDF_DictElement element =
(HPDF_DictElement)HPDF_List_ItemAt (dict->list, i);
HPDF_Obj_Header *header = (HPDF_Obj_Header *)(element->value);
if (!element->value)
return HPDF_SetError (dict->error, HPDF_INVALID_OBJECT, 0);
if (header->obj_id & HPDF_OTYPE_HIDDEN) {
HPDF_PTRACE((" HPDF_Dict_Write obj=%p skipped obj_id=0x%08X\n",
element->value, (HPDF_UINT)header->obj_id));
} else {
ret = HPDF_Stream_WriteEscapeName (stream, element->key);
if (ret != HPDF_OK)
return ret;
ret = HPDF_Stream_WriteChar (stream, ' ');
if (ret != HPDF_OK)
return ret;
ret = HPDF_Obj_Write (element->value, stream, e);
if (ret != HPDF_OK)
return ret;
ret = HPDF_Stream_WriteStr (stream, "\012");
if (ret != HPDF_OK)
return ret;
}
}
if (dict->write_fn) {
if ((ret = dict->write_fn (dict, stream)) != HPDF_OK)
return ret;
}
if ((ret = HPDF_Stream_WriteStr (stream, ">>")) != HPDF_OK)
return ret;
if (dict->stream) {
HPDF_UINT32 strptr;
HPDF_Number length;
/* get "length" element */
length = (HPDF_Number)HPDF_Dict_GetItem (dict, "Length",
HPDF_OCLASS_NUMBER);
if (!length)
return HPDF_SetError (dict->error,
HPDF_DICT_STREAM_LENGTH_NOT_FOUND, 0);
/* "length" element must be indirect-object */
if (!(length->header.obj_id & HPDF_OTYPE_INDIRECT)) {
return HPDF_SetError (dict->error, HPDF_DICT_ITEM_UNEXPECTED_TYPE,
0);
}
if ((ret = HPDF_Stream_WriteStr (stream, "\012stream\015\012")) /* Acrobat 8.15 requires both \r and \n here */
!= HPDF_OK)
return ret;
strptr = stream->size;
if (e)
HPDF_Encrypt_Reset (e);
if ((ret = HPDF_Stream_WriteToStream (dict->stream, stream,
dict->filter, e)) != HPDF_OK)
return ret;
HPDF_Number_SetValue (length, stream->size - strptr);
ret = HPDF_Stream_WriteStr (stream, "\012endstream");
}
/* 2006.08.13 add. */
if (dict->after_write_fn) {
if ((ret = dict->after_write_fn (dict)) != HPDF_OK)
return ret;
}
return ret;
}
HPDF_STATUS
HPDF_Dict_Add (HPDF_Dict dict,
const char *key,
void *obj)
{
HPDF_Obj_Header *header;
HPDF_STATUS ret = HPDF_OK;
HPDF_DictElement element;
if (!obj) {
if (HPDF_Error_GetCode (dict->error) == HPDF_OK)
return HPDF_SetError (dict->error, HPDF_INVALID_OBJECT, 0);
else
return HPDF_INVALID_OBJECT;
}
header = (HPDF_Obj_Header *)obj;
if (header->obj_id & HPDF_OTYPE_DIRECT)
return HPDF_SetError (dict->error, HPDF_INVALID_OBJECT, 0);
if (!key) {
HPDF_Obj_Free (dict->mmgr, obj);
return HPDF_SetError (dict->error, HPDF_INVALID_OBJECT, 0);
}
if (dict->list->count >= HPDF_LIMIT_MAX_DICT_ELEMENT) {
HPDF_PTRACE((" HPDF_Dict_Add exceed limitatin of dict count(%d)\n",
HPDF_LIMIT_MAX_DICT_ELEMENT));
HPDF_Obj_Free (dict->mmgr, obj);
return HPDF_SetError (dict->error, HPDF_DICT_COUNT_ERR, 0);
}
/* check whether there is an object which has same name */
element = GetElement (dict, key);
if (element) {
HPDF_Obj_Free (dict->mmgr, element->value);
element->value = NULL;
} else {
element = (HPDF_DictElement)HPDF_GetMem (dict->mmgr,
sizeof(HPDF_DictElement_Rec));
if (!element) {
/* cannot create element object */
if (!(header->obj_id & HPDF_OTYPE_INDIRECT))
HPDF_Obj_Free (dict->mmgr, obj);
return HPDF_Error_GetCode (dict->error);
}
HPDF_StrCpy (element->key, key, element->key +
HPDF_LIMIT_MAX_NAME_LEN + 1);
element->value = NULL;
ret = HPDF_List_Add (dict->list, element);
if (ret != HPDF_OK) {
if (!(header->obj_id & HPDF_OTYPE_INDIRECT))
HPDF_Obj_Free (dict->mmgr, obj);
HPDF_FreeMem (dict->mmgr, element);
return HPDF_Error_GetCode (dict->error);
}
}
if (header->obj_id & HPDF_OTYPE_INDIRECT) {
HPDF_Proxy proxy = HPDF_Proxy_New (dict->mmgr, obj);
if (!proxy)
return HPDF_Error_GetCode (dict->error);
element->value = proxy;
proxy->header.obj_id |= HPDF_OTYPE_DIRECT;
} else {
element->value = obj;
header->obj_id |= HPDF_OTYPE_DIRECT;
}
return ret;
}
HPDF_STATUS
HPDF_Dict_AddName (HPDF_Dict dict,
const char *key,
const char *value)
{
HPDF_Name name = HPDF_Name_New (dict->mmgr, value);
if (!name)
return HPDF_Error_GetCode (dict->error);
return HPDF_Dict_Add (dict, key, name);
}
HPDF_STATUS
HPDF_Dict_AddNumber (HPDF_Dict dict,
const char *key,
HPDF_INT32 value)
{
HPDF_Number number = HPDF_Number_New (dict->mmgr, value);
if (!number)
return HPDF_Error_GetCode (dict->error);
return HPDF_Dict_Add (dict, key, number);
}
HPDF_STATUS
HPDF_Dict_AddReal (HPDF_Dict dict,
const char *key,
HPDF_REAL value)
{
HPDF_Real real = HPDF_Real_New (dict->mmgr, value);
if (!real)
return HPDF_Error_GetCode (dict->error);
return HPDF_Dict_Add (dict, key, real);
}
HPDF_STATUS
HPDF_Dict_AddBoolean (HPDF_Dict dict,
const char *key,
HPDF_BOOL value)
{
HPDF_Boolean obj = HPDF_Boolean_New (dict->mmgr, value);
if (!obj)
return HPDF_Error_GetCode (dict->error);
return HPDF_Dict_Add (dict, key, obj);
}
void*
HPDF_Dict_GetItem (HPDF_Dict dict,
const char *key,
HPDF_UINT16 obj_class)
{
HPDF_DictElement element = GetElement (dict, key);
void *obj;
if (element && HPDF_StrCmp(key, element->key) == 0) {
HPDF_Obj_Header *header = (HPDF_Obj_Header *)element->value;
if (header->obj_class == HPDF_OCLASS_PROXY) {
HPDF_Proxy p = element->value;
header = (HPDF_Obj_Header *)p->obj;
obj = p->obj;
} else
obj = element->value;
if ((header->obj_class & HPDF_OCLASS_ANY) != obj_class) {
HPDF_PTRACE((" HPDF_Dict_GetItem dict=%p key=%s obj_class=0x%08X\n",
dict, key, (HPDF_UINT)header->obj_class));
HPDF_SetError (dict->error, HPDF_DICT_ITEM_UNEXPECTED_TYPE, 0);
return NULL;
}
return obj;
}
return NULL;
}
HPDF_DictElement
GetElement (HPDF_Dict dict,
const char *key)
{
HPDF_UINT i;
for (i = 0; i < dict->list->count; i++) {
HPDF_DictElement element =
(HPDF_DictElement)HPDF_List_ItemAt (dict->list, i);
if (HPDF_StrCmp (key, element->key) == 0)
return element;
}
return NULL;
}
HPDF_STATUS
HPDF_Dict_RemoveElement (HPDF_Dict dict,
const char *key)
{
HPDF_UINT i;
for (i = 0; i < dict->list->count; i++) {
HPDF_DictElement element =
(HPDF_DictElement)HPDF_List_ItemAt (dict->list, i);
if (HPDF_StrCmp (key, element->key) == 0) {
HPDF_List_Remove (dict->list, element);
HPDF_Obj_Free (dict->mmgr, element->value);
HPDF_FreeMem (dict->mmgr, element);
return HPDF_OK;
}
}
return HPDF_DICT_ITEM_NOT_FOUND;
}
const char*
HPDF_Dict_GetKeyByObj (HPDF_Dict dict,
void *obj)
{
HPDF_UINT i;
for (i = 0; i < dict->list->count; i++) {
HPDF_Obj_Header *header;
HPDF_DictElement element =
(HPDF_DictElement)HPDF_List_ItemAt (dict->list, i);
header = (HPDF_Obj_Header *)(element->value);
if (header->obj_class == HPDF_OCLASS_PROXY) {
HPDF_Proxy p = element->value;
if (p->obj == obj)
return element->key;
} else {
if (element->value == obj)
return element->key;
}
}
return NULL;
}