summaryrefslogtreecommitdiffstats
path: root/kernel/kls_jbig/jbig2mem.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/kls_jbig/jbig2mem.cpp')
-rw-r--r--kernel/kls_jbig/jbig2mem.cpp270
1 files changed, 270 insertions, 0 deletions
diff --git a/kernel/kls_jbig/jbig2mem.cpp b/kernel/kls_jbig/jbig2mem.cpp
new file mode 100644
index 0000000..9ae1abf
--- /dev/null
+++ b/kernel/kls_jbig/jbig2mem.cpp
@@ -0,0 +1,270 @@
+/*
+ * jbgtopbm - JBIG to Portable Bitmap converter
+ *
+ * Markus Kuhn -- http://www.cl.cam.ac.uk/~mgk25/jbigkit/
+ *
+ * $Id: jbgtopbm.c,v 1.11 2004-06-11 15:17:49+01 mgk25 Exp $
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <ctype.h>
+#include <string.h>
+
+#include "jbig.h"
+
+double koeff = 1.0;
+
+void write_it(unsigned char *data, size_t len, void *file)
+{
+ int bb;
+ unsigned char cc;
+
+ FILE *ff = (FILE *)file;
+
+ for(size_t i = 0;i < len;i++)
+ {
+ bb = int(koeff * (*(data+i)));
+
+ if(bb > 255) cc = 255;
+ else cc = bb;
+
+ fwrite(&cc, 1, 1, ff);
+ fwrite(&cc, 1, 1, ff);
+ fwrite(&cc, 1, 1, ff);
+ }
+}
+
+int read_file(unsigned char **buf, size_t *buflen, size_t *len, FILE *f)
+{
+ if (*buflen == 0) {
+ *buflen = 4000;
+ *len = 0;
+ *buf = (unsigned char *) malloc(*buflen);
+ if (!*buf) {
+ fprintf(stderr, "Sorry, not enough memory available!\n");
+ return 0;
+ }
+ }
+ do {
+ *len += fread(*buf + *len, 1, *buflen - *len, f);
+ if (*len == *buflen) {
+ *buflen *= 2;
+ *buf = (unsigned char *) realloc(*buf, *buflen);
+ if (!*buf) {
+ fprintf(stderr, "Sorry, not enough memory available!\n");
+ return 0;
+ }
+ }
+ if (ferror(f)) {
+ perror("Problem while reading input file");
+ return 0;
+ }
+ } while (!feof(f));
+ *buflen = *len;
+ *buf = (unsigned char *) realloc(*buf, *buflen);
+ if (!*buf) {
+ fprintf(stderr, "Oops, realloc failed when shrinking buffer!\n");
+ return 0;
+ }
+
+ return 1;
+}
+
+int jbig2mem (char *argv[])
+{
+ FILE *fin = stdin, *fout = stdout;
+ const char *fnin = NULL, *fnout = NULL;
+ int i, result;
+ struct jbg_dec_state s;
+ unsigned char *buffer, *p;
+ size_t buflen, len, cnt;
+ unsigned long xmax = 4294967295UL, ymax = 4294967295UL, max;
+ int plane = -1, use_graycode = 1, multi = 0;
+
+printf("+JBIG %s, %s\n", argv[0], argv[1]);
+
+ buflen = 8000;
+ buffer = (unsigned char *) malloc(buflen);
+ if (!buffer) {
+ return 1;
+ }
+
+fnin = argv[0];
+fnout = argv[1];
+
+ fin = fopen(fnin, "rb");
+ if (!fin) {
+ free(buffer);
+ return 1;
+ }
+
+ fout = fopen(fnout, "wb");
+ if (!fout) {
+ fclose(fin);
+ free(buffer);
+ return 1;
+ }
+
+ /* send input file to decoder */
+ jbg_dec_init(&s);
+ jbg_dec_maxsize(&s, xmax, ymax);
+ /* read BIH first to check VLENGTH */
+ len = fread(buffer, 1, 20, fin);
+
+ if(len < 20)
+ {
+ fclose(fin);
+ fclose(fout);
+ remove(fnout);
+ return 1;
+ }
+
+ if (buffer[19] & JBG_VLENGTH) {
+ /* VLENGTH = 1 => we might encounter a NEWLEN, therefore read entire
+ * input file into memory and run two passes over it */
+ if(!read_file(&buffer, &buflen, &len, fin))
+ {
+ fclose(fin);
+ fclose(fout);
+ remove(fnout);
+ return 1;
+ }
+ /* scan for NEWLEN marker segments and update BIE header accordingly */
+ result = jbg_newlen(buffer, len);
+ /* feed data to decoder */
+ if (result == JBG_EOK) {
+ p = (unsigned char *) buffer;
+ result = JBG_EAGAIN;
+ while (len > 0 &&
+ (result == JBG_EAGAIN || (result == JBG_EOK && multi))) {
+ result = jbg_dec_in(&s, p, len, &cnt);
+ p += cnt;
+ len -= cnt;
+ }
+ }
+ } else {
+ /* VLENGTH = 0 => we can simply pass the input file directly to decoder */
+ result = JBG_EAGAIN;
+ do {
+ cnt = 0;
+ p = (unsigned char *) buffer;
+ while (len > 0 &&
+ (result == JBG_EAGAIN || (result == JBG_EOK && multi))) {
+ result = jbg_dec_in(&s, p, len, &cnt);
+ p += cnt;
+ len -= cnt;
+ }
+ if (!(result == JBG_EAGAIN || (result == JBG_EOK && multi)))
+ break;
+ len = fread(buffer, 1, buflen, fin);
+ } while (len > 0);
+
+ if (ferror(fin)) {
+ fclose(fin);
+ fclose(fout);
+ remove(fnout);
+ return 1;
+ }
+ }
+ if (result != JBG_EOK && result != JBG_EOK_INTR) {
+ fprintf(stderr, "Problem with input file '%s': %s\n",
+ fnin, jbg_strerror(result, JBG_EN));
+
+ fclose(fout);
+ remove(fnout);
+ return 1;
+ }
+ if (plane >= 0 && jbg_dec_getplanes(&s) <= plane) {
+ fprintf(stderr, "Image has only %d planes!\n", jbg_dec_getplanes(&s));
+ fclose(fout);
+ remove(fnout);
+ return 1;
+ }
+
+ if (jbg_dec_getplanes(&s) == 1 || plane >= 0)
+ {
+ int w, h, bpp = 24;
+
+ w = jbg_dec_getwidth(&s);
+ h = jbg_dec_getheight(&s);
+
+ fwrite(&w, sizeof(int), 1, fout);
+ fwrite(&h, sizeof(int), 1, fout);
+ fwrite(&bpp, sizeof(int), 1, fout);
+
+ unsigned char *d = jbg_dec_getimage(&s, plane < 0 ? 0 : plane), bt;
+
+ int G = 0, index;
+ bool brk = false;
+ unsigned char S1;
+
+ for(int f = 0;f < h;f++)
+ {
+ for(int i = 0, g = 0;;g++)
+ {
+ bt = *(d + G);
+ G++;
+
+ for(int F = 256;F >= 2;F /= 2)
+ {
+ S1 = (unsigned char)(F / 2);
+ index = (bt&S1) ? 0 : 255;
+
+ fwrite(&index, 1, 1, fout);
+ fwrite(&index, 1, 1, fout);
+ fwrite(&index, 1, 1, fout);
+
+ if(++i >= w)
+ {
+ brk = true;
+ break;
+ }
+ }
+
+ if(brk)
+ {
+ brk = false;
+ break;
+ }
+ }
+ }
+ } else {
+ /* write PGM output file */
+ if ((size_t) jbg_dec_getplanes(&s) > sizeof(unsigned long) * 8)
+ {
+ fprintf(stderr, "Image has too many planes (%d)!\n", jbg_dec_getplanes(&s));
+ fclose(fout);
+ jbg_dec_free(&s);
+ return 1;
+ }
+ max = 0;
+ for (i = jbg_dec_getplanes(&s); i > 0; i--)
+ max = (max << 1) | 1;
+
+ int w, h, bpp = 24;
+
+ w = jbg_dec_getwidth(&s);
+ h = jbg_dec_getheight(&s);
+
+ fwrite(&w, sizeof(int), 1, fout);
+ fwrite(&h, sizeof(int), 1, fout);
+ fwrite(&bpp, sizeof(int), 1, fout);
+
+ koeff = 255.0 / max;
+
+ jbg_dec_merge_planes(&s, use_graycode, write_it, fout);
+ }
+
+ /* check for file errors and close fout */
+ if (ferror(fout) || fclose(fout)) {
+ fprintf(stderr, "Problem while writing output file '%s", fnout);
+ perror("'");
+ jbg_dec_free(&s);
+ return 1;
+ }
+
+ jbg_dec_free(&s);
+
+ return 0;
+}