kopia lustrzana https://gitlab.com/sane-project/backends
224 wiersze
6.1 KiB
C
224 wiersze
6.1 KiB
C
/* sane - Scanner Access Now Easy.
|
|
|
|
Copyright (C) 2019 Thierry HUCHARD <thierry@ordissimo.com>
|
|
|
|
This file is part of the SANE package.
|
|
|
|
SANE is free software; you can redistribute it and/or modify it under
|
|
the terms of the GNU General Public License as published by the Free
|
|
Software Foundation; either version 3 of the License, or (at your
|
|
option) any later version.
|
|
|
|
SANE 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 General Public License
|
|
for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with sane; see the file COPYING. If not, write to the Free
|
|
Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
This file implements a SANE backend for eSCL scanners. */
|
|
|
|
#define DEBUG_DECLARE_ONLY
|
|
#include "../include/sane/config.h"
|
|
|
|
#include "escl.h"
|
|
|
|
#include "../include/sane/sanei.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stddef.h>
|
|
#include <math.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#if(defined HAVE_POPPLER_GLIB)
|
|
#include <poppler/glib/poppler.h>
|
|
#endif
|
|
|
|
#include <setjmp.h>
|
|
|
|
|
|
#if(defined HAVE_POPPLER_GLIB)
|
|
|
|
#define INPUT_BUFFER_SIZE 4096
|
|
|
|
static unsigned char*
|
|
set_file_in_buffer(FILE *fp, int *size)
|
|
{
|
|
char buffer[1024] = { 0 };
|
|
unsigned char *data = (unsigned char *)calloc(1, sizeof(char));
|
|
int nx = 0;
|
|
|
|
while(!feof(fp))
|
|
{
|
|
int n = fread(buffer,sizeof(char),1024,fp);
|
|
unsigned char *t = realloc(data, nx + n + 1);
|
|
if (t == NULL) {
|
|
DBG(10, "not enough memory (realloc returned NULL)");
|
|
free(data);
|
|
return NULL;
|
|
}
|
|
data = t;
|
|
memcpy(&(data[nx]), buffer, n);
|
|
nx = nx + n;
|
|
data[nx] = 0;
|
|
}
|
|
*size = nx;
|
|
return data;
|
|
}
|
|
|
|
static unsigned char *
|
|
cairo_surface_to_pixels (cairo_surface_t *surface, int bps)
|
|
{
|
|
int cairo_width, cairo_height, cairo_rowstride;
|
|
unsigned char *data, *dst, *cairo_data;
|
|
unsigned int *src;
|
|
int x, y;
|
|
|
|
cairo_width = cairo_image_surface_get_width (surface);
|
|
cairo_height = cairo_image_surface_get_height (surface);
|
|
cairo_rowstride = cairo_image_surface_get_stride (surface);
|
|
cairo_data = cairo_image_surface_get_data (surface);
|
|
data = (unsigned char*)calloc(1, sizeof(unsigned char) * (cairo_height * cairo_width * bps));
|
|
|
|
for (y = 0; y < cairo_height; y++)
|
|
{
|
|
src = (unsigned int *) (cairo_data + y * cairo_rowstride);
|
|
dst = data + y * (cairo_width * bps);
|
|
for (x = 0; x < cairo_width; x++)
|
|
{
|
|
dst[0] = (*src >> 16) & 0xff;
|
|
dst[1] = (*src >> 8) & 0xff;
|
|
dst[2] = (*src >> 0) & 0xff;
|
|
dst += bps;
|
|
src++;
|
|
}
|
|
}
|
|
return data;
|
|
}
|
|
|
|
SANE_Status
|
|
get_PDF_data(capabilities_t *scanner, int *width, int *height, int *bps)
|
|
{
|
|
cairo_surface_t *cairo_surface = NULL;
|
|
cairo_t *cr;
|
|
PopplerPage *page;
|
|
PopplerDocument *doc;
|
|
double dw, dh;
|
|
int w, h, size = 0;
|
|
char *data = NULL;
|
|
unsigned char* surface = NULL;
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
|
|
|
|
data = (char*)set_file_in_buffer(scanner->tmp, &size);
|
|
if (!data) {
|
|
DBG(1, "Error : poppler_document_new_from_data");
|
|
status = SANE_STATUS_INVAL;
|
|
goto close_file;
|
|
}
|
|
doc = poppler_document_new_from_data(data,
|
|
size,
|
|
NULL,
|
|
NULL);
|
|
|
|
if (!doc) {
|
|
DBG(1, "Error : poppler_document_new_from_data");
|
|
status = SANE_STATUS_INVAL;
|
|
goto free_file;
|
|
}
|
|
|
|
page = poppler_document_get_page (doc, 0);
|
|
if (!page) {
|
|
DBG(1, "Error : poppler_document_get_page");
|
|
status = SANE_STATUS_INVAL;
|
|
goto free_doc;
|
|
}
|
|
|
|
poppler_page_get_size (page, &dw, &dh);
|
|
dw = (double)scanner->default_resolution * dw / 72.0;
|
|
dh = (double)scanner->default_resolution * dh / 72.0;
|
|
w = (int)ceil(dw);
|
|
h = (int)ceil(dh);
|
|
cairo_surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, w, h);
|
|
if (!cairo_surface) {
|
|
DBG(1, "Error : cairo_image_surface_create");
|
|
status = SANE_STATUS_INVAL;
|
|
goto free_page;
|
|
}
|
|
|
|
cr = cairo_create (cairo_surface);
|
|
if (!cairo_surface) {
|
|
DBG(1, "Error : cairo_create");
|
|
status = SANE_STATUS_INVAL;
|
|
goto free_surface;
|
|
}
|
|
cairo_scale (cr, (double)scanner->default_resolution / 72.0,
|
|
(double)scanner->default_resolution / 72.0);
|
|
cairo_save (cr);
|
|
poppler_page_render (page, cr);
|
|
cairo_restore (cr);
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_DEST_OVER);
|
|
cairo_set_source_rgb (cr, 1, 1, 1);
|
|
cairo_paint (cr);
|
|
|
|
int st = cairo_status(cr);
|
|
if (st)
|
|
{
|
|
DBG(1, "%s", cairo_status_to_string (st));
|
|
status = SANE_STATUS_INVAL;
|
|
goto destroy_cr;
|
|
}
|
|
|
|
*bps = 3;
|
|
|
|
DBG(1, "Escl Pdf : Image Size [%dx%d]\n", w, h);
|
|
|
|
surface = cairo_surface_to_pixels (cairo_surface, *bps);
|
|
if (!surface) {
|
|
status = SANE_STATUS_NO_MEM;
|
|
DBG(1, "Escl Pdf : Surface Memory allocation problem");
|
|
goto destroy_cr;
|
|
}
|
|
|
|
// If necessary, trim the image.
|
|
surface = escl_crop_surface(scanner, surface, w, h, *bps, width, height);
|
|
if (!surface) {
|
|
DBG(1, "Escl Pdf Crop: Surface Memory allocation problem");
|
|
status = SANE_STATUS_NO_MEM;
|
|
}
|
|
|
|
destroy_cr:
|
|
cairo_destroy (cr);
|
|
free_surface:
|
|
cairo_surface_destroy (cairo_surface);
|
|
free_page:
|
|
g_object_unref (page);
|
|
free_doc:
|
|
g_object_unref (doc);
|
|
free_file:
|
|
free(data);
|
|
close_file:
|
|
if (scanner->tmp)
|
|
fclose(scanner->tmp);
|
|
scanner->tmp = NULL;
|
|
return status;
|
|
}
|
|
#else
|
|
|
|
SANE_Status
|
|
get_PDF_data(capabilities_t __sane_unused__ *scanner,
|
|
int __sane_unused__ *width,
|
|
int __sane_unused__ *height,
|
|
int __sane_unused__ *bps)
|
|
{
|
|
return (SANE_STATUS_INVAL);
|
|
}
|
|
|
|
#endif
|