summaryrefslogtreecommitdiffstats
path: root/kernel/kls_gif
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/kls_gif')
-rw-r--r--kernel/kls_gif/Makefile.am9
-rw-r--r--kernel/kls_gif/fmt_codec_gif.cpp516
-rw-r--r--kernel/kls_gif/fmt_codec_gif_defs.h32
3 files changed, 557 insertions, 0 deletions
diff --git a/kernel/kls_gif/Makefile.am b/kernel/kls_gif/Makefile.am
new file mode 100644
index 0000000..62f5ed5
--- /dev/null
+++ b/kernel/kls_gif/Makefile.am
@@ -0,0 +1,9 @@
+INCLUDES = -I../include
+
+pkglib_LTLIBRARIES = libkls_gif.la
+
+libkls_gif_la_SOURCES = fmt_codec_gif.cpp fmt_codec_gif_defs.h
+
+libkls_gif_la_LDFLAGS = ${SQ_RELEASE}
+
+libkls_gif_la_LIBADD = ${SQ_LOCAL_RPATH} ${SQ_GIFLIBS}
diff --git a/kernel/kls_gif/fmt_codec_gif.cpp b/kernel/kls_gif/fmt_codec_gif.cpp
new file mode 100644
index 0000000..80f2dbb
--- /dev/null
+++ b/kernel/kls_gif/fmt_codec_gif.cpp
@@ -0,0 +1,516 @@
+/* This file is part of the ksquirrel-libs (http://ksquirrel.sf.net)
+
+ Copyright (c) 2004,2005 Dmitry Baryshev <[email protected]>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License (LGPL) as published by the Free Software Foundation;
+ either version 2 of the License, or (at your option) any later
+ version.
+
+ This library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ as32 with this library; see the file COPYING. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+*/
+
+#include <iostream>
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+
+#include "ksquirrel-libs/fmt_types.h"
+#include "ksquirrel-libs/fileio.h"
+#include "ksquirrel-libs/error.h"
+#include "ksquirrel-libs/fmt_utils.h"
+
+#include "fmt_codec_gif_defs.h"
+
+extern "C" {
+#include "gif_lib.h"
+}
+
+#include "fmt_codec_gif.h"
+
+#include "../xpm/codec_gif.xpm"
+
+/*
+ *
+ * Originally designed to facilitate image transfer and online
+ * storage for use by CompuServe and its customers,
+ * GIF is primarily an exchange and storage
+ * format, although it is based on, and is supported by, many
+ * applications.
+ *
+ */
+
+static s32
+InterlacedOffset[] = { 0, 4, 2, 1 }, /* The way Interlaced image should. */
+InterlacedJumps[] = { 8, 8, 4, 2 }; /* be read - offsets and jumps... */
+
+fmt_codec::fmt_codec() : fmt_codec_base()
+{}
+
+fmt_codec::~fmt_codec()
+{}
+
+void fmt_codec::options(codec_options *o)
+{
+ o->version = "1.3.1";
+ o->name = "Compuserve GIF";
+ o->filter = "*.gif ";
+ o->config = "";
+ o->mime = "\x0047\x0049\x0046\x0038[\x0039\x0037]\x0061";
+ o->mimetype = "image/gif";
+ o->pixmap = codec_gif;
+ o->readable = true;
+ o->canbemultiple = true;
+ o->writestatic = false;
+ o->writeanimated = false;
+ o->needtempfile = false;
+}
+
+s32 fmt_codec::read_init(const std::string &file)
+{
+ frs.open(file.c_str(), ios::binary | ios::in);
+
+ if(!frs.good())
+ return SQE_R_NOFILE;
+
+ frs.close();
+
+ transIndex = -1;
+
+ Last = 0;
+ Lines = 0;
+ buf = 0;
+ saved = 0;
+
+ gif = DGifOpenFileName(file.c_str());
+
+ // for safety...
+ if(!gif)
+ return SQE_R_BADFILE;
+
+ linesz = gif->SWidth * sizeof(GifPixelType);
+
+ if((buf = (u8*)malloc(linesz)) == NULL)
+ return SQE_R_NOMEMORY;
+
+ if((saved = (RGBA *)calloc(linesz, sizeof(RGBA))) == NULL)
+ return SQE_R_NOMEMORY;
+
+ if(gif->SColorMap)
+ {
+ back.r = gif->SColorMap->Colors[gif->SBackGroundColor].Red;
+ back.g = gif->SColorMap->Colors[gif->SBackGroundColor].Green;
+ back.b = gif->SColorMap->Colors[gif->SBackGroundColor].Blue;
+ back.a = 255;
+ }
+ else
+ memset(&back, 0, sizeof(RGBA));
+
+ layer = -1;
+ line = 0;
+ curLine = 0;
+
+ Lines_h = gif->SHeight;
+ Lines = (RGBA **)malloc(Lines_h * sizeof(RGBA*));
+
+ if(!Lines)
+ return SQE_R_NOMEMORY;
+
+ for(s32 i = 0;i < Lines_h;i++)
+ Lines[i] = (RGBA *)0;
+
+ map = (gif->Image.ColorMap) ? gif->Image.ColorMap : gif->SColorMap;
+
+ Last = (RGBA **)malloc(gif->SHeight * sizeof(RGBA*));
+
+ if(!Last)
+ return SQE_R_NOMEMORY;
+
+ for(s32 i = 0;i < gif->SHeight;i++)
+ Last[i] = (RGBA *)0;
+
+ for(s32 i = 0;i < gif->SHeight;i++)
+ {
+ Last[i] = (RGBA *)calloc(gif->SWidth, sizeof(RGBA));
+
+ if(!Last[i])
+ return SQE_R_NOMEMORY;
+
+// for(s32 k = 0;k < gif->SWidth;k++)
+// memcpy(Last[i]+k, &back, sizeof(RGBA));
+ }
+
+ currentImage = -1;
+ lastDisposal = DISPOSAL_NO;
+
+ finfo.animated = false;
+
+ return SQE_OK;
+}
+
+s32 fmt_codec::read_next_pass()
+{
+ layer++;
+ currentPass++;
+ line = 0;
+ curLine = 0;
+
+ return SQE_OK;
+}
+
+s32 fmt_codec::read_next()
+{
+ bool foundExt = false;
+
+ currentImage++;
+
+ fmt_image image;
+
+ image.interlaced = gif->Image.Interlace;
+ image.passes = (gif->Image.Interlace) ? 4 : 1;
+
+ while(true)
+ {
+ if (DGifGetRecordType(gif, &record) == GIF_ERROR)
+ {
+ PrintGifError();
+ return SQE_R_BADFILE;
+ }
+
+ switch(record)
+ {
+ case IMAGE_DESC_RECORD_TYPE:
+ if(DGifGetImageDesc(gif) == GIF_ERROR)
+ {
+ PrintGifError();
+ return SQE_R_BADFILE;
+ }
+
+ if(!foundExt)
+ {
+ lastDisposal = disposal;
+ disposal = DISPOSAL_NO;
+ image.delay = 100;
+ transIndex = -1;
+ image.hasalpha = true;
+ }
+
+ lastRow = (currentImage) ? Row : gif->Image.Top;
+ lastCol = (currentImage) ? Col : gif->Image.Left;
+ Row = gif->Image.Top;
+ Col = gif->Image.Left;
+ image.w = gif->SWidth;
+ image.h = gif->SHeight;
+ lastWidth = (currentImage) ? Width : gif->Image.Width;
+ lastHeight = (currentImage) ? Height : gif->Image.Height;
+ Width = gif->Image.Width;
+ Height = gif->Image.Height;
+ image.bpp = 8;
+
+ curLine = 0;
+
+ if(gif->Image.Left + gif->Image.Width > gif->SWidth || gif->Image.Top + gif->Image.Height > gif->SHeight)
+ {
+ return SQE_R_BADFILE;
+ }
+ break;
+
+ case EXTENSION_RECORD_TYPE:
+ if(DGifGetExtension(gif, &ExtCode, &Extension) == GIF_ERROR)
+ {
+ PrintGifError();
+ return SQE_R_BADFILE;
+ }
+
+ if(!Extension)
+ break;
+
+ if(ExtCode == 249)
+ {
+ foundExt = true;
+
+ lastDisposal = disposal;
+ disposal = (Extension[1] >> 2) & 7;
+ bool b = Extension[1] & 1;
+ s32 u = (unsigned)*(Extension + 2);
+ image.delay = (!u) ? 100 : (u * 10);
+
+ if(b)
+ transIndex = Extension[4];
+
+ image.hasalpha = b;
+ }
+ else if(ExtCode == 254 && Extension[0])
+ {
+ fmt_metaentry mt;
+ s8 d[Extension[0]+1];
+
+ memcpy(d, (s8*)Extension+1, Extension[0]);
+ d[Extension[0]] = '\0';
+
+ for(s32 s = 0;s < Extension[0];s++)
+ if(d[s] == '\n')
+ d[s] = ' ';
+
+ mt.group = "Comment";
+ mt.data = d;
+
+ addmeta(mt);
+ }
+
+ while(Extension)
+ {
+ if(DGifGetExtensionNext(gif, &Extension) == GIF_ERROR)
+ {
+ PrintGifError();
+ return SQE_R_BADFILE;
+ }
+ }
+ break;
+
+ case TERMINATE_RECORD_TYPE:
+ return SQE_NOTOK;
+
+ default: ;
+ }
+
+ if(record == IMAGE_DESC_RECORD_TYPE)
+ {
+ if(currentImage >= 1)
+ finfo.animated = true;
+
+ map = (gif->Image.ColorMap) ? gif->Image.ColorMap : gif->SColorMap;
+
+ back.a = (transIndex != -1) ? 0 : 255;
+
+ for(s32 k = 0;k < gif->SWidth;k++)
+ memcpy(saved+k, &back, sizeof(RGBA));
+
+ image.compression = "LZW";
+ image.colorspace = "Color indexed";
+ image.interlaced = gif->Image.Interlace;
+ image.passes = (gif->Image.Interlace) ? 4 : 1;
+
+ finfo.image.push_back(image);
+
+ layer = -1;
+ currentPass = -1;
+
+ return SQE_OK;
+ }
+ }
+}
+
+s32 fmt_codec::read_scanline(RGBA *scan)
+{
+ fmt_image *im = image(currentImage);
+ fmt_utils::fillAlpha(scan, im->w);
+
+ if(curLine < Row || curLine >= Row + Height)
+ {
+ if(currentPass == im->passes-1)
+ {
+ memcpy(scan, Last[curLine], im->w * sizeof(RGBA));
+
+ if(lastDisposal == DISPOSAL_BACKGROUND)
+ if(curLine >= lastRow && curLine < lastRow+lastHeight)
+ {
+ memcpy(scan+lastCol, saved, lastWidth * sizeof(RGBA));
+ memcpy(Last[curLine], scan, im->w * sizeof(RGBA));
+ }
+ }
+
+ curLine++;
+
+ return SQE_OK;
+ }
+
+ curLine++;
+
+ s32 i;
+ s32 index;
+
+ if(gif->Image.Interlace)
+ {
+ memcpy(scan, Last[curLine-1], im->w * sizeof(RGBA));
+
+ if(line == 0)
+ j = InterlacedOffset[layer];
+
+ if(line == j)
+ {
+ if(DGifGetLine(gif, buf, Width) == GIF_ERROR)
+ {
+ PrintGifError();
+ memset(scan, 255, im->w * sizeof(RGBA));
+ return SQE_R_BADFILE;
+ }
+ else
+ {
+ j += InterlacedJumps[layer];
+
+ for(i = 0;i < Width;i++)
+ {
+ index = Col + i;
+
+ if(buf[i] == transIndex && transIndex != -1)
+ {
+ RGB rgb = *((RGB *)&(map->Colors[buf[i]]));
+
+ if(back == rgb && !currentImage)
+ (scan+index)->a = 0;
+ else if(back == rgb && lastDisposal != DISPOSAL_BACKGROUND && currentImage)
+ {
+ RGBA *t = &Last[curLine-1][index];
+ memcpy(scan+index, t, sizeof(RGBA));
+ }
+ else if(back == rgb && lastDisposal == DISPOSAL_BACKGROUND && currentImage)
+ {
+ (scan+index)->a = 0;
+ }
+ else if(currentImage)
+ {
+ RGBA *t = &Last[curLine-1][index];
+
+ if(lastDisposal == DISPOSAL_BACKGROUND)
+ {
+ memcpy(scan+index, &back, sizeof(RGBA));//(scan+index)->a=0;
+
+ if(t->a == 0)
+ (scan+index)->a=0;
+ }
+ }
+ }
+ else
+ {
+ memcpy(scan+index, &(map->Colors[buf[i]]), sizeof(RGB));
+ (scan+index)->a = 255;
+ }
+ }
+
+ Lines[line] = (RGBA*)realloc(Lines[line], im->w * sizeof(RGBA));
+
+ if(!Lines[line])
+ return SQE_R_NOMEMORY;
+
+ memcpy(Lines[line], scan, im->w * sizeof(RGBA));
+ }
+ } // if(line == j)
+ else
+ {
+ if(Lines[line])
+ memcpy(scan, Lines[line], im->w * sizeof(RGBA));
+ else
+ memset(scan, 255, im->w * sizeof(RGBA));
+ }
+
+ if(currentPass == im->passes-1)
+ memcpy(Last[curLine-1], scan, im->w * sizeof(RGBA));
+
+ line++;
+ }
+ else // !s32erlaced
+ {
+ if(DGifGetLine(gif, buf, Width) == GIF_ERROR)
+ {
+ memset(scan, 255, im->w * sizeof(RGBA));
+ PrintGifError();
+ return SQE_R_BADFILE;
+ }
+ else
+ {
+ memcpy(scan, Last[curLine-1], im->w * sizeof(RGBA));
+
+ if(lastDisposal == DISPOSAL_BACKGROUND)
+ {
+ if(curLine-1 >= lastRow && curLine-1 < lastRow+lastHeight)
+ memcpy(scan+lastCol, saved, lastWidth * sizeof(RGBA));
+ }
+
+ for(i = 0;i < Width;i++)
+ {
+ index = Col + i;
+
+ if(buf[i] == transIndex && transIndex != -1)
+ {
+ RGB rgb = *((RGB *)&(map->Colors[buf[i]]));
+
+ if(back == rgb && !currentImage)
+ (scan+index)->a = 0;
+ else if(back == rgb && lastDisposal != DISPOSAL_BACKGROUND && currentImage)
+ {
+ RGBA *t = &Last[curLine-1][index];
+ memcpy(scan+index, t, sizeof(RGBA));// = 255;
+ }
+ else if(back == rgb && lastDisposal == DISPOSAL_BACKGROUND && currentImage)
+ {
+ (scan+index)->a = 0;
+ }
+ else if(currentImage)
+ {
+ RGBA *t = &Last[curLine-1][index];
+
+ if(lastDisposal == DISPOSAL_BACKGROUND)
+ {
+ memcpy(scan+index, &back, sizeof(RGBA));//(scan+index)->a=0;
+
+ if(t->a == 0)
+ (scan+index)->a=0;
+ }
+ }
+ }// if transIndex
+ else
+ {
+ memcpy(scan+index, &(map->Colors[buf[i]]), sizeof(RGB));
+ (scan+index)->a = 255;
+ }
+ } // for
+
+ memcpy(Last[curLine-1], scan, im->w * sizeof(RGBA));
+ }
+ }
+
+ return SQE_OK;
+}
+
+void fmt_codec::read_close()
+{
+ if(buf) free(buf);
+ if(saved) free(saved);
+
+ if(Lines)
+ {
+ for(s32 i = 0;i < Lines_h;i++)
+ if(Lines[i])
+ free(Lines[i]);
+
+ free(Lines);
+ Lines = 0;
+ }
+
+ if(Last)
+ {
+ for(s32 i = 0;i < gif->SHeight;i++)
+ if(Last[i])
+ free(Last[i]);
+
+ free(Last);
+ Last = 0;
+ }
+
+ finfo.meta.clear();
+ finfo.image.clear();
+
+ if(gif) DGifCloseFile(gif);
+}
+
+#include "fmt_codec_cd_func.h"
diff --git a/kernel/kls_gif/fmt_codec_gif_defs.h b/kernel/kls_gif/fmt_codec_gif_defs.h
new file mode 100644
index 0000000..02b4f41
--- /dev/null
+++ b/kernel/kls_gif/fmt_codec_gif_defs.h
@@ -0,0 +1,32 @@
+/* This file is part of the ksquirrel-libs (http://ksquirrel.sf.net)
+
+ Copyright (c) 2004 Dmitry Baryshev <[email protected]>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License (LGPL) as published by the Free Software Foundation;
+ either version 2 of the License, or (at your option) any later
+ version.
+
+ This library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ as32 with this library; see the file COPYING. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+*/
+
+#ifndef KSQUIRREL_READ_IMAGE_gif
+#define KSQUIRREL_READ_IMAGE_gif
+
+#define SQ_NEED_OPERATOR_RGBA_RGB
+
+#define DISPOSAL_NO 0
+#define DISPOSAL_LEFT 1
+#define DISPOSAL_BACKGROUND 2
+#define DISPOSAL_PREVIOUS 3
+
+#endif