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.
4 
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6 
7 #include "core/include/fxcodec/fx_codec.h"
8 #include "core/include/fxge/fx_dib.h"
9 #include "codec_int.h"
10 #include "core/src/fxcodec/lgif/fx_gif.h"
11 struct FXGIF_Context {
12   gif_decompress_struct_p gif_ptr;
13   void* parent_ptr;
14   void* child_ptr;
15 
16   void* (*m_AllocFunc)(unsigned int);
17   void (*m_FreeFunc)(void*);
18 };
19 extern "C" {
_gif_alloc_func(unsigned int size)20 static void* _gif_alloc_func(unsigned int size) {
21   return FX_Alloc(char, size);
22 }
_gif_free_func(void * p)23 static void _gif_free_func(void* p) {
24   if (p != NULL) {
25     FX_Free(p);
26   }
27 }
28 };
_gif_error_data(gif_decompress_struct_p gif_ptr,const FX_CHAR * err_msg)29 static void _gif_error_data(gif_decompress_struct_p gif_ptr,
30                             const FX_CHAR* err_msg) {
31   FXSYS_strncpy((char*)gif_ptr->err_ptr, err_msg, GIF_MAX_ERROR_SIZE - 1);
32   longjmp(gif_ptr->jmpbuf, 1);
33 }
_gif_ask_buf_for_pal(gif_decompress_struct_p gif_ptr,int32_t pal_size)34 static uint8_t* _gif_ask_buf_for_pal(gif_decompress_struct_p gif_ptr,
35                                      int32_t pal_size) {
36   FXGIF_Context* p = (FXGIF_Context*)gif_ptr->context_ptr;
37   CCodec_GifModule* pModule = (CCodec_GifModule*)p->parent_ptr;
38   return pModule->AskLocalPaletteBufCallback(
39       p->child_ptr, _gif_get_frame_num(gif_ptr), pal_size);
40 }
_gif_record_current_position(gif_decompress_struct_p gif_ptr,FX_DWORD * cur_pos_ptr)41 static void _gif_record_current_position(gif_decompress_struct_p gif_ptr,
42                                          FX_DWORD* cur_pos_ptr) {
43   FXGIF_Context* p = (FXGIF_Context*)gif_ptr->context_ptr;
44   CCodec_GifModule* pModule = (CCodec_GifModule*)p->parent_ptr;
45   pModule->RecordCurrentPositionCallback(p->child_ptr, *cur_pos_ptr);
46 }
_gif_read_scanline(gif_decompress_struct_p gif_ptr,int32_t row_num,uint8_t * row_buf)47 static void _gif_read_scanline(gif_decompress_struct_p gif_ptr,
48                                int32_t row_num,
49                                uint8_t* row_buf) {
50   FXGIF_Context* p = (FXGIF_Context*)gif_ptr->context_ptr;
51   CCodec_GifModule* pModule = (CCodec_GifModule*)p->parent_ptr;
52   pModule->ReadScanlineCallback(p->child_ptr, row_num, row_buf);
53 }
_gif_get_record_position(gif_decompress_struct_p gif_ptr,FX_DWORD cur_pos,int32_t left,int32_t top,int32_t width,int32_t height,int32_t pal_num,void * pal_ptr,int32_t delay_time,FX_BOOL user_input,int32_t trans_index,int32_t disposal_method,FX_BOOL interlace)54 static FX_BOOL _gif_get_record_position(gif_decompress_struct_p gif_ptr,
55                                         FX_DWORD cur_pos,
56                                         int32_t left,
57                                         int32_t top,
58                                         int32_t width,
59                                         int32_t height,
60                                         int32_t pal_num,
61                                         void* pal_ptr,
62                                         int32_t delay_time,
63                                         FX_BOOL user_input,
64                                         int32_t trans_index,
65                                         int32_t disposal_method,
66                                         FX_BOOL interlace) {
67   FXGIF_Context* p = (FXGIF_Context*)gif_ptr->context_ptr;
68   CCodec_GifModule* pModule = (CCodec_GifModule*)p->parent_ptr;
69   return pModule->InputRecordPositionBufCallback(
70       p->child_ptr, cur_pos, FX_RECT(left, top, left + width, top + height),
71       pal_num, pal_ptr, delay_time, user_input, trans_index, disposal_method,
72       interlace);
73 }
Start(void * pModule)74 void* CCodec_GifModule::Start(void* pModule) {
75   FXGIF_Context* p = (FXGIF_Context*)FX_Alloc(uint8_t, sizeof(FXGIF_Context));
76   if (p == NULL) {
77     return NULL;
78   }
79   FXSYS_memset(p, 0, sizeof(FXGIF_Context));
80   p->m_AllocFunc = _gif_alloc_func;
81   p->m_FreeFunc = _gif_free_func;
82   p->gif_ptr = NULL;
83   p->parent_ptr = (void*)this;
84   p->child_ptr = pModule;
85   p->gif_ptr = _gif_create_decompress();
86   if (p->gif_ptr == NULL) {
87     FX_Free(p);
88     return NULL;
89   }
90   p->gif_ptr->context_ptr = (void*)p;
91   p->gif_ptr->err_ptr = m_szLastError;
92   p->gif_ptr->_gif_error_fn = _gif_error_data;
93   p->gif_ptr->_gif_ask_buf_for_pal_fn = _gif_ask_buf_for_pal;
94   p->gif_ptr->_gif_record_current_position_fn = _gif_record_current_position;
95   p->gif_ptr->_gif_get_row_fn = _gif_read_scanline;
96   p->gif_ptr->_gif_get_record_position_fn = _gif_get_record_position;
97   return p;
98 }
Finish(void * pContext)99 void CCodec_GifModule::Finish(void* pContext) {
100   FXGIF_Context* p = (FXGIF_Context*)pContext;
101   if (p != NULL) {
102     _gif_destroy_decompress(&p->gif_ptr);
103     p->m_FreeFunc(p);
104   }
105 }
ReadHeader(void * pContext,int * width,int * height,int * pal_num,void ** pal_pp,int * bg_index,CFX_DIBAttribute * pAttribute)106 int32_t CCodec_GifModule::ReadHeader(void* pContext,
107                                      int* width,
108                                      int* height,
109                                      int* pal_num,
110                                      void** pal_pp,
111                                      int* bg_index,
112                                      CFX_DIBAttribute* pAttribute) {
113   FXGIF_Context* p = (FXGIF_Context*)pContext;
114   if (setjmp(p->gif_ptr->jmpbuf)) {
115     return 0;
116   }
117   int32_t ret = _gif_read_header(p->gif_ptr);
118   if (ret != 1) {
119     return ret;
120   }
121   if (pAttribute) {
122   }
123   *width = p->gif_ptr->width;
124   *height = p->gif_ptr->height;
125   *pal_num = p->gif_ptr->global_pal_num;
126   *pal_pp = p->gif_ptr->global_pal_ptr;
127   *bg_index = p->gif_ptr->bc_index;
128   return 1;
129 }
LoadFrameInfo(void * pContext,int * frame_num)130 int32_t CCodec_GifModule::LoadFrameInfo(void* pContext, int* frame_num) {
131   FXGIF_Context* p = (FXGIF_Context*)pContext;
132   if (setjmp(p->gif_ptr->jmpbuf)) {
133     return 0;
134   }
135   int32_t ret = _gif_get_frame(p->gif_ptr);
136   if (ret != 1) {
137     return ret;
138   }
139   *frame_num = _gif_get_frame_num(p->gif_ptr);
140   return 1;
141 }
LoadFrame(void * pContext,int frame_num,CFX_DIBAttribute * pAttribute)142 int32_t CCodec_GifModule::LoadFrame(void* pContext,
143                                     int frame_num,
144                                     CFX_DIBAttribute* pAttribute) {
145   FXGIF_Context* p = (FXGIF_Context*)pContext;
146   if (setjmp(p->gif_ptr->jmpbuf)) {
147     return 0;
148   }
149   int32_t ret = _gif_load_frame(p->gif_ptr, frame_num);
150   if (ret == 1) {
151     if (pAttribute) {
152       pAttribute->m_nGifLeft =
153           p->gif_ptr->img_ptr_arr_ptr->GetAt(frame_num)->image_info_ptr->left;
154       pAttribute->m_nGifTop =
155           p->gif_ptr->img_ptr_arr_ptr->GetAt(frame_num)->image_info_ptr->top;
156       pAttribute->m_fAspectRatio = p->gif_ptr->pixel_aspect;
157       if (p->gif_ptr->cmt_data_ptr) {
158         const uint8_t* buf =
159             (const uint8_t*)p->gif_ptr->cmt_data_ptr->GetBuffer(0);
160         FX_DWORD len = p->gif_ptr->cmt_data_ptr->GetLength();
161         if (len > 21) {
162           uint8_t size = *buf++;
163           if (size) {
164             pAttribute->m_strAuthor = CFX_ByteString(buf, size);
165           } else {
166             pAttribute->m_strAuthor.Empty();
167           }
168           buf += size;
169           size = *buf++;
170           if (size == 20) {
171             FXSYS_memcpy(pAttribute->m_strTime, buf, size);
172           }
173         }
174       }
175     }
176   }
177   return ret;
178 }
GetAvailInput(void * pContext,uint8_t ** avial_buf_ptr)179 FX_DWORD CCodec_GifModule::GetAvailInput(void* pContext,
180                                          uint8_t** avial_buf_ptr) {
181   FXGIF_Context* p = (FXGIF_Context*)pContext;
182   return _gif_get_avail_input(p->gif_ptr, avial_buf_ptr);
183 }
Input(void * pContext,const uint8_t * src_buf,FX_DWORD src_size)184 void CCodec_GifModule::Input(void* pContext,
185                              const uint8_t* src_buf,
186                              FX_DWORD src_size) {
187   FXGIF_Context* p = (FXGIF_Context*)pContext;
188   _gif_input_buffer(p->gif_ptr, (uint8_t*)src_buf, src_size);
189 }
190