summaryrefslogtreecommitdiffstats
path: root/kernel/kls_sgi
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/kls_sgi')
-rw-r--r--kernel/kls_sgi/Makefile.am9
-rw-r--r--kernel/kls_sgi/fmt_codec_sgi.cpp341
-rw-r--r--kernel/kls_sgi/fmt_codec_sgi_defs.h50
3 files changed, 400 insertions, 0 deletions
diff --git a/kernel/kls_sgi/Makefile.am b/kernel/kls_sgi/Makefile.am
new file mode 100644
index 0000000..6284c4c
--- /dev/null
+++ b/kernel/kls_sgi/Makefile.am
@@ -0,0 +1,9 @@
+INCLUDES = -I../include
+
+pkglib_LTLIBRARIES = libkls_sgi.la
+
+libkls_sgi_la_SOURCES = fmt_codec_sgi.cpp fmt_codec_sgi_defs.h
+
+libkls_sgi_la_LDFLAGS = ${SQ_RELEASE}
+
+libkls_sgi_la_LIBADD = ${SQ_LOCAL_RPATH}
diff --git a/kernel/kls_sgi/fmt_codec_sgi.cpp b/kernel/kls_sgi/fmt_codec_sgi.cpp
new file mode 100644
index 0000000..0ec1ea4
--- /dev/null
+++ b/kernel/kls_sgi/fmt_codec_sgi.cpp
@@ -0,0 +1,341 @@
+/* This file is part of SQuirrel (http://ksquirrel.sf.net) libraries
+
+ 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 Library General Public
+ License 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
+ Library 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 "ksquirrel-libs/fmt_types.h"
+#include "ksquirrel-libs/fmt_utils.h"
+#include "ksquirrel-libs/fileio.h"
+#include "ksquirrel-libs/error.h"
+
+#include "fmt_codec_sgi_defs.h"
+#include "fmt_codec_sgi.h"
+
+#include "../xpm/codec_sgi.xpm"
+
+/*
+ *
+ * The SGI image file format is actually part of the SGI image library found on
+ * all Silicon Graphics machines. SGI image files may store black-and-white (.BW
+ * extension), color RGB (.RGB extension),
+ * or color RGB with alpha channel data (.RGBA extension)
+ * images. SGI image files may also have the generic extension .SGI as well.
+ *
+ */
+
+fmt_codec::fmt_codec() : fmt_codec_base()
+{}
+
+fmt_codec::~fmt_codec()
+{}
+
+void fmt_codec::options(codec_options *o)
+{
+ o->version = "0.9.4";
+ o->name = "SGI Format";
+ o->filter = "*.rgb *.rgba *.bw";
+ o->config = "";
+ o->mime = "\001\332.[\001\002]";
+ o->mimetype = "image/x-rgb";
+ o->pixmap = codec_sgi;
+ o->readable = true;
+ o->canbemultiple = false;
+ 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;
+
+ currentImage = -1;
+ starttab = NULL;
+ lengthtab = NULL;
+ channel[0] = channel[1] = channel[2] = channel[3] = NULL;
+
+ finfo.animated = false;
+
+ return SQE_OK;
+}
+
+s32 fmt_codec::read_next()
+{
+ currentImage++;
+
+ if(currentImage)
+ return SQE_NOTOK;
+
+ fmt_image image;
+
+ if(!frs.be_getshort(&sfh.Magik)) return SQE_R_BADFILE;
+ if(!frs.readK(&sfh.StorageFormat, 1)) return SQE_R_BADFILE;
+ if(!frs.readK(&sfh.bpc, 1)) return SQE_R_BADFILE;
+ if(!frs.be_getshort(&sfh.Dimensions)) return SQE_R_BADFILE;
+ if(!frs.be_getshort(&sfh.x)) return SQE_R_BADFILE;
+ if(!frs.be_getshort(&sfh.y)) return SQE_R_BADFILE;
+ if(!frs.be_getshort(&sfh.z)) return SQE_R_BADFILE;
+ if(!frs.be_getlong(&sfh.pixmin)) return SQE_R_BADFILE;
+ if(!frs.be_getlong(&sfh.pixmax)) return SQE_R_BADFILE;
+ if(!frs.be_getlong(&sfh.dummy)) return SQE_R_BADFILE;
+
+ if(!frs.readK(sfh.name, sizeof(sfh.name))) return SQE_R_BADFILE;
+
+ if(!frs.be_getlong(&sfh.ColormapID)) return SQE_R_BADFILE;
+
+ if(!frs.readK(&sfh.dummy2, sizeof(sfh.dummy2))) return SQE_R_BADFILE;
+
+ image.w = sfh.x;
+ image.h = sfh.y;
+ image.bpp = sfh.bpc * sfh.z * 8;
+
+ if(image.bpp == 32) image.hasalpha = true;
+
+ if(sfh.Magik != 474 || (sfh.StorageFormat != 0 && sfh.StorageFormat != 1) || (sfh.Dimensions != 1 && sfh.Dimensions != 2 && sfh.Dimensions != 3) || (sfh.bpc != 1 && sfh.bpc != 2))
+ return SQE_R_BADFILE;
+
+ if(sfh.bpc == 2 || sfh.ColormapID > 0)
+ return SQE_R_NOTSUPPORTED;
+
+ for(s32 i = 0;i < 4;i++)
+ {
+ channel[i] = new s8 [sfh.x];
+
+ if(!channel[i])
+ return SQE_R_NOMEMORY;
+ }
+
+ if(sfh.StorageFormat == 1)
+ {
+ s32 sz = sfh.y * sfh.z, i;
+ lengthtab = new u32 [sz];
+ starttab = new u32 [sz];
+
+ if(!lengthtab || !starttab)
+ return SQE_R_NOMEMORY;
+
+ frs.seekg(512, ios::beg);
+
+ for(i = 0;i < sz;i++)
+ if(!frs.be_getlong(&starttab[i]))
+ return SQE_R_BADFILE;
+
+ for(i = 0;i < sz;i++)
+ if(!frs.be_getlong(&lengthtab[i]))
+ return SQE_R_BADFILE;
+ }
+
+ rle_row = 0;
+
+ if(strlen(sfh.name))
+ {
+ fmt_metaentry mt;
+
+ mt.group = "Image Name";
+ mt.data = sfh.name;
+
+ addmeta(mt);
+ }
+
+ image.needflip = true;
+ image.compression = (sfh.StorageFormat ? "RLE" : "-");
+ image.colorspace = fmt_utils::colorSpaceByBpp(image.bpp);
+
+ finfo.image.push_back(image);
+
+ return SQE_OK;
+}
+
+s32 fmt_codec::read_next_pass()
+{
+ return SQE_OK;
+}
+
+s32 fmt_codec::read_scanline(RGBA *scan)
+{
+ const s32 sz = sfh.x;
+ s32 i = 0, j = 0;
+ s32 len;
+ fstream::pos_type pos;
+ fmt_image *im = image(currentImage);
+ fmt_utils::fillAlpha(scan, im->w);
+
+ s8 bt;
+
+ memset(channel[3], 255, sz);
+
+ switch(sfh.z)
+ {
+ case 1:
+ {
+ if(sfh.StorageFormat)
+ {
+ j = 0;
+
+ frs.seekg(starttab[rle_row], ios::beg);
+ len = lengthtab[rle_row];
+
+ for(;;)
+ {
+ s8 count;
+
+ if(!frs.readK(&bt, 1)) return SQE_R_BADFILE;
+ count = bt&0x7f;
+
+ if(!count) break;
+
+ if(bt & 0x80)
+ while(count--)
+ {
+ if(!frs.readK(&channel[0][j], 1)) return SQE_R_BADFILE;
+
+ j++;
+
+ if(!len--) goto ex1;
+ }
+ else
+ {
+ if(!frs.readK(&bt, 1)) return SQE_R_BADFILE;
+
+ if(!len--) goto ex1;
+
+ while(count--)
+ channel[0][j++] = bt;
+ }
+ }
+ ex1:
+ len = len; // some stuff: get rid of compile warning
+
+ rle_row++;
+ }
+ else
+ {
+ if(!frs.readK(channel[0], sz)) return SQE_R_BADFILE;
+ }
+
+ memcpy(channel[1], channel[0], sz);
+ memcpy(channel[2], channel[0], sz);
+ }
+ break;
+
+
+ case 3:
+ case 4:
+ {
+ if(sfh.StorageFormat)
+ {
+ for(i = 0;i < sfh.z;i++)
+ {
+ j = 0;
+
+ frs.seekg(starttab[rle_row + i*im->h], ios::beg);
+ len = lengthtab[rle_row + i*im->h];
+
+ for(;;)
+ {
+ s8 count;
+
+ if(!frs.readK(&bt, 1)) return SQE_R_BADFILE;
+
+ count = bt&0x7f;
+
+ if(!count) break;
+
+ if(bt & 0x80)
+ while(count--)
+ {
+ if(!frs.readK(&channel[i][j], 1)) return SQE_R_BADFILE;
+ j++;
+ if(!len--) goto ex;
+ }
+ else
+ {
+ if(!frs.readK(&bt, 1)) return SQE_R_BADFILE;
+
+ if(!len--) goto ex;
+
+ while(count--)
+ channel[i][j++] = bt;
+ }
+ }
+ ex:
+ len = len; // some stuff: get rid of compile warning
+ }
+ rle_row++;
+ }
+ else
+ {
+ if(!frs.readK(channel[0], sz)) return SQE_R_BADFILE;
+
+ pos = frs.tellg();
+ frs.seekg(im->w * (im->h - 1), ios::cur);
+ if(!frs.readK(channel[1], sz)) return SQE_R_BADFILE;
+
+ frs.seekg(im->w * (im->h - 1), ios::cur);
+ if(!frs.readK(channel[2], sz)) return SQE_R_BADFILE;
+
+ frs.seekg(im->w * (im->h - 1), ios::cur);
+ if(!frs.readK(channel[3], sz)) return SQE_R_BADFILE;
+
+ frs.seekg(pos);
+ }
+
+ }
+ break;
+ }
+
+ for(i = 0;i < sz;i++)
+ {
+ scan[i].r = channel[0][i];
+ scan[i].g = channel[1][i];
+ scan[i].b = channel[2][i];
+ scan[i].a = channel[3][i];
+ }
+
+ return SQE_OK;
+}
+
+void fmt_codec::read_close()
+{
+ frs.close();
+
+ delete [] starttab;
+ starttab = NULL;
+
+ delete [] lengthtab;
+ lengthtab = NULL;
+
+ for(s32 i = 0;i < 4;i++)
+ {
+ delete [] channel[i];
+ channel[i] = NULL;
+ }
+
+ finfo.meta.clear();
+ finfo.image.clear();
+}
+
+#include "fmt_codec_cd_func.h"
diff --git a/kernel/kls_sgi/fmt_codec_sgi_defs.h b/kernel/kls_sgi/fmt_codec_sgi_defs.h
new file mode 100644
index 0000000..3573d4e
--- /dev/null
+++ b/kernel/kls_sgi/fmt_codec_sgi_defs.h
@@ -0,0 +1,50 @@
+/* This file is part of SQuirrel (http://ksquirrel.sf.net) libraries
+
+ 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 Library General Public
+ License 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
+ Library 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_sgi
+#define KSQUIRREL_READ_IMAGE_sgi
+
+struct SGI_HEADER
+{
+ u16 Magik; /* should be 474 */
+ u8 StorageFormat; /* 1 == RLE, 0 = Verbatim */
+ u8 bpc; /* 1|2 */
+ u16 Dimensions; /* 1|2|3 1==1channle+1scanline, 2==1channle+some scanlines, 3==number of channels */
+ u16 x;
+ u16 y;
+ u16 z;
+ u32 pixmin;
+ u32 pixmax;
+ u32 dummy;
+ char name[80]; /* ascii string */
+ u32 ColormapID;
+
+ /* 0=Normal
+ 1=Dither. RGB==3:3:2 per byte, 1 channel
+ 2=Screen. Obsolete
+ 3=Colormap. Has ONLY colormap, nothing else.
+ */
+
+ u8 dummy2[404];
+
+}PACKED;
+
+#endif