kopia lustrzana https://gitlab.com/sane-project/backends
847 wiersze
24 KiB
C
Executable File
847 wiersze
24 KiB
C
Executable File
/* sane - Scanner Access Now Easy.
|
|
|
|
ScanMaker 3840 Backend
|
|
Copyright (C) 2005-7 Earle F. Philhower, III
|
|
earle@ziplabel.com - http://www.ziplabel.com
|
|
|
|
This program 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 2 of the
|
|
License, or (at your option) any later version.
|
|
|
|
This program 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 this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston,
|
|
MA 02111-1307, USA.
|
|
|
|
As a special exception, the authors of SANE give permission for
|
|
additional uses of the libraries contained in this release of SANE.
|
|
|
|
The exception is that, if you link a SANE library with other files
|
|
to produce an executable, this does not by itself cause the
|
|
resulting executable to be covered by the GNU General Public
|
|
License. Your use of that executable is in no way restricted on
|
|
account of linking the SANE library code into it.
|
|
|
|
This exception does not, however, invalidate any other reasons why
|
|
the executable file might be covered by the GNU General Public
|
|
License.
|
|
|
|
If you submit changes to SANE to the maintainers to be included in
|
|
a subsequent release, you agree by submitting the changes that
|
|
those changes may be distributed with this exception intact.
|
|
|
|
If you write modifications of your own for SANE, it is your choice
|
|
whether to permit this exception to apply to your modifications.
|
|
If you do not wish that, delete this exception notice.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "../include/sane/config.h"
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include "../include/sane/sane.h"
|
|
#include "../include/sane/saneopts.h"
|
|
|
|
#define BACKENDNAME sm3840
|
|
#include "../include/sane/sanei_backend.h"
|
|
#include "../include/sane/sanei_usb.h"
|
|
#include "../include/sane/sanei_config.h"
|
|
|
|
#include "sm3840.h"
|
|
|
|
#include "sm3840_scan.c"
|
|
#include "sm3840_lib.c"
|
|
|
|
static double sm3840_unit_convert (SANE_Int val);
|
|
|
|
static int num_devices;
|
|
static SM3840_Device *first_dev;
|
|
static SM3840_Scan *first_handle;
|
|
static const SANE_Device **devlist = 0;
|
|
|
|
static const SANE_String_Const mode_list[] = {
|
|
SANE_VALUE_SCAN_MODE_GRAY,
|
|
SANE_VALUE_SCAN_MODE_COLOR,
|
|
SANE_VALUE_SCAN_MODE_LINEART,
|
|
SANE_VALUE_SCAN_MODE_HALFTONE,
|
|
0
|
|
};
|
|
|
|
static const SANE_Word resolution_list[] = {
|
|
4, 1200, 600, 300, 150
|
|
};
|
|
|
|
static const SANE_Word bpp_list[] = {
|
|
2, 8, 16
|
|
};
|
|
|
|
static const SANE_Range x_range = {
|
|
SANE_FIX (0),
|
|
SANE_FIX (215.91), /* 8.5 inches */
|
|
SANE_FIX (0)
|
|
};
|
|
|
|
static const SANE_Range y_range = {
|
|
SANE_FIX (0),
|
|
SANE_FIX (297.19), /* 11.7 inches */
|
|
SANE_FIX (0)
|
|
};
|
|
|
|
static const SANE_Range brightness_range = {
|
|
1,
|
|
4096,
|
|
1.0
|
|
};
|
|
|
|
static const SANE_Range contrast_range = {
|
|
SANE_FIX (0.1),
|
|
SANE_FIX (9.9),
|
|
SANE_FIX (0.1)
|
|
};
|
|
|
|
static const SANE_Range lamp_range = {
|
|
1,
|
|
15,
|
|
1
|
|
};
|
|
|
|
static const SANE_Range threshold_range = {
|
|
0,
|
|
255,
|
|
1
|
|
};
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
static int
|
|
min (int a, int b)
|
|
{
|
|
if (a < b)
|
|
return a;
|
|
else
|
|
return b;
|
|
}
|
|
|
|
SANE_Status
|
|
sane_read (SANE_Handle handle, SANE_Byte * buf, SANE_Int max_len,
|
|
SANE_Int * len)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
unsigned char c, d;
|
|
int i;
|
|
|
|
DBG (2, "+sane-read:%p %p %d %p\n", (unsigned char *) s, buf, max_len,
|
|
(unsigned char *) len);
|
|
DBG (2,
|
|
"+sane-read:remain:%lu offset:%lu linesleft:%d linebuff:%p linesread:%d\n",
|
|
(u_long)s->remaining, (u_long)s->offset, s->linesleft, s->line_buffer, s->linesread);
|
|
|
|
if (!s->scanning)
|
|
return SANE_STATUS_INVAL;
|
|
|
|
if (!s->remaining)
|
|
{
|
|
if (!s->linesleft)
|
|
{
|
|
*len = 0;
|
|
s->scanning = 0;
|
|
/* Move to home position */
|
|
reset_scanner ((p_usb_dev_handle)s->udev);
|
|
/* Send lamp timeout */
|
|
set_lamp_timer ((p_usb_dev_handle)s->udev, s->sm3840_params.lamp);
|
|
|
|
/* Free memory */
|
|
if (s->save_scan_line)
|
|
free (s->save_scan_line);
|
|
s->save_scan_line = NULL;
|
|
if (s->save_dpi1200_remap)
|
|
free (s->save_dpi1200_remap);
|
|
s->save_dpi1200_remap = NULL;
|
|
if (s->save_color_remap)
|
|
free (s->save_color_remap);
|
|
s->save_color_remap = NULL;
|
|
|
|
return SANE_STATUS_EOF;
|
|
}
|
|
|
|
record_line ((s->linesread == 0) ? 1 : 0,
|
|
(p_usb_dev_handle) s->udev,
|
|
s->line_buffer,
|
|
s->sm3840_params.dpi,
|
|
s->sm3840_params.scanpix,
|
|
s->sm3840_params.gray,
|
|
(s->sm3840_params.bpp == 16) ? 1 : 0,
|
|
&s->save_i,
|
|
&s->save_scan_line,
|
|
&s->save_dpi1200_remap, &s->save_color_remap);
|
|
s->remaining = s->sm3840_params.linelen;
|
|
s->offset = 0;
|
|
s->linesread++;
|
|
s->linesleft--;
|
|
}
|
|
|
|
/* Need to software emulate 1-bpp modes, simple threshold and error */
|
|
/* diffusion dither implemented. */
|
|
if (s->sm3840_params.lineart || s->sm3840_params.halftone)
|
|
{
|
|
d = 0;
|
|
for (i = 0; i < min (max_len * 8, s->remaining); i++)
|
|
{
|
|
d = d << 1;
|
|
if (s->sm3840_params.halftone)
|
|
{
|
|
c = (*(unsigned char *) (s->offset + s->line_buffer + i));
|
|
if (c + s->save_dither_err < 128)
|
|
{
|
|
d |= 1;
|
|
s->save_dither_err += c;
|
|
}
|
|
else
|
|
{
|
|
s->save_dither_err += c - 255;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((*(unsigned char *) (s->offset + s->line_buffer + i)) < s->threshold )
|
|
d |= 1;
|
|
}
|
|
if (i % 8 == 7)
|
|
*(buf++) = d;
|
|
}
|
|
*len = i / 8;
|
|
s->offset += i;
|
|
s->remaining -= i;
|
|
}
|
|
else
|
|
{
|
|
memcpy (buf, s->offset + s->line_buffer, min (max_len, s->remaining));
|
|
*len = min (max_len, s->remaining);
|
|
s->offset += min (max_len, s->remaining);
|
|
s->remaining -= min (max_len, s->remaining);
|
|
}
|
|
|
|
DBG (2, "-sane_read\n");
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
void
|
|
sane_cancel (SANE_Handle h)
|
|
{
|
|
SM3840_Scan *s = h;
|
|
|
|
DBG (2, "trying to cancel...\n");
|
|
if (s->scanning)
|
|
{
|
|
if (!s->cancelled)
|
|
{
|
|
/* Move to home position */
|
|
reset_scanner ((p_usb_dev_handle) s->udev);
|
|
/* Send lamp timeout */
|
|
set_lamp_timer ((p_usb_dev_handle) s->udev, s->sm3840_params.lamp);
|
|
|
|
/* Free memory */
|
|
if (s->save_scan_line)
|
|
free (s->save_scan_line);
|
|
s->save_scan_line = NULL;
|
|
if (s->save_dpi1200_remap)
|
|
free (s->save_dpi1200_remap);
|
|
s->save_dpi1200_remap = NULL;
|
|
if (s->save_color_remap)
|
|
free (s->save_color_remap);
|
|
s->save_color_remap = NULL;
|
|
|
|
s->scanning = 0;
|
|
s->cancelled = SANE_TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_start (SANE_Handle handle)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
SANE_Status status;
|
|
|
|
/* First make sure we have a current parameter set. Some of the
|
|
* parameters will be overwritten below, but that's OK. */
|
|
DBG (2, "sane_start\n");
|
|
status = sane_get_parameters (s, 0);
|
|
if (status != SANE_STATUS_GOOD)
|
|
return status;
|
|
DBG (1, "Got params again...\n");
|
|
|
|
s->scanning = SANE_TRUE;
|
|
s->cancelled = 0;
|
|
|
|
s->line_buffer = malloc (s->sm3840_params.linelen);
|
|
s->remaining = 0;
|
|
s->offset = 0;
|
|
s->linesleft = s->sm3840_params.scanlines;
|
|
s->linesread = 0;
|
|
|
|
s->save_i = 0;
|
|
s->save_scan_line = NULL;
|
|
s->save_dpi1200_remap = NULL;
|
|
s->save_color_remap = NULL;
|
|
|
|
s->save_dither_err = 0;
|
|
s->threshold = s->sm3840_params.threshold;
|
|
|
|
setup_scan ((p_usb_dev_handle) s->udev, &(s->sm3840_params));
|
|
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
static double
|
|
sm3840_unit_convert (SANE_Int val)
|
|
{
|
|
double d;
|
|
d = SANE_UNFIX (val);
|
|
d /= MM_PER_INCH;
|
|
return d;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_get_parameters (SANE_Handle handle, SANE_Parameters * params)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
|
|
DBG (2, "sane_get_parameters\n");
|
|
if (!s->scanning)
|
|
{
|
|
memset (&s->sane_params, 0, sizeof (s->sane_params));
|
|
/* Copy from options to sm3840_params */
|
|
s->sm3840_params.gray =
|
|
(!strcasecmp (s->value[OPT_MODE].s, SANE_VALUE_SCAN_MODE_GRAY)) ? 1 : 0;
|
|
s->sm3840_params.halftone =
|
|
(!strcasecmp (s->value[OPT_MODE].s, SANE_VALUE_SCAN_MODE_HALFTONE)) ? 1 : 0;
|
|
s->sm3840_params.lineart =
|
|
(!strcasecmp (s->value[OPT_MODE].s, SANE_VALUE_SCAN_MODE_LINEART)) ? 1 : 0;
|
|
|
|
s->sm3840_params.dpi = s->value[OPT_RESOLUTION].w;
|
|
s->sm3840_params.bpp = s->value[OPT_BIT_DEPTH].w;
|
|
s->sm3840_params.gain = SANE_UNFIX (s->value[OPT_CONTRAST].w);
|
|
s->sm3840_params.offset = s->value[OPT_BRIGHTNESS].w;
|
|
s->sm3840_params.lamp = s->value[OPT_LAMP_TIMEOUT].w;
|
|
s->sm3840_params.threshold = s->value[OPT_THRESHOLD].w;
|
|
|
|
if (s->sm3840_params.lineart || s->sm3840_params.halftone)
|
|
{
|
|
s->sm3840_params.gray = 1;
|
|
s->sm3840_params.bpp = 8;
|
|
}
|
|
|
|
s->sm3840_params.top = sm3840_unit_convert (s->value[OPT_TL_Y].w);
|
|
s->sm3840_params.left = sm3840_unit_convert (s->value[OPT_TL_X].w);
|
|
s->sm3840_params.width =
|
|
sm3840_unit_convert (s->value[OPT_BR_X].w) - s->sm3840_params.left;
|
|
s->sm3840_params.height =
|
|
sm3840_unit_convert (s->value[OPT_BR_Y].w) - s->sm3840_params.top;
|
|
|
|
/* Legalize and calculate pixel sizes */
|
|
prepare_params (&(s->sm3840_params));
|
|
|
|
/* Copy into sane_params */
|
|
s->sane_params.pixels_per_line = s->sm3840_params.scanpix;
|
|
s->sane_params.lines = s->sm3840_params.scanlines;
|
|
s->sane_params.format =
|
|
s->sm3840_params.gray ? SANE_FRAME_GRAY : SANE_FRAME_RGB;
|
|
s->sane_params.bytes_per_line = s->sm3840_params.linelen;
|
|
s->sane_params.depth = s->sm3840_params.bpp;
|
|
|
|
if (s->sm3840_params.lineart || s->sm3840_params.halftone)
|
|
{
|
|
s->sane_params.bytes_per_line += 7;
|
|
s->sane_params.bytes_per_line /= 8;
|
|
s->sane_params.depth = 1;
|
|
s->sane_params.pixels_per_line = s->sane_params.bytes_per_line * 8;
|
|
}
|
|
|
|
s->sane_params.last_frame = SANE_TRUE;
|
|
} /*!scanning */
|
|
|
|
if (params)
|
|
*params = s->sane_params;
|
|
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_control_option (SANE_Handle handle, SANE_Int option,
|
|
SANE_Action action, void *val, SANE_Int * info)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
SANE_Status status = 0;
|
|
SANE_Word cap;
|
|
DBG (2, "sane_control_option\n");
|
|
if (info)
|
|
*info = 0;
|
|
if (s->scanning)
|
|
return SANE_STATUS_DEVICE_BUSY;
|
|
if (option >= NUM_OPTIONS)
|
|
return SANE_STATUS_INVAL;
|
|
cap = s->options_list[option].cap;
|
|
if (!SANE_OPTION_IS_ACTIVE (cap))
|
|
return SANE_STATUS_INVAL;
|
|
if (action == SANE_ACTION_GET_VALUE)
|
|
{
|
|
DBG (1, "sane_control_option %d, get value\n", option);
|
|
switch (option)
|
|
{
|
|
/* word options: */
|
|
case OPT_RESOLUTION:
|
|
case OPT_BIT_DEPTH:
|
|
case OPT_TL_X:
|
|
case OPT_TL_Y:
|
|
case OPT_BR_X:
|
|
case OPT_BR_Y:
|
|
case OPT_NUM_OPTS:
|
|
case OPT_CONTRAST:
|
|
case OPT_BRIGHTNESS:
|
|
case OPT_LAMP_TIMEOUT:
|
|
case OPT_THRESHOLD:
|
|
*(SANE_Word *) val = s->value[option].w;
|
|
return (SANE_STATUS_GOOD);
|
|
/* string options: */
|
|
case OPT_MODE:
|
|
strcpy (val, s->value[option].s);
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
}
|
|
else if (action == SANE_ACTION_SET_VALUE)
|
|
{
|
|
DBG (1, "sane_control_option %d, set value\n", option);
|
|
if (!SANE_OPTION_IS_SETTABLE (cap))
|
|
return (SANE_STATUS_INVAL);
|
|
if (status != SANE_STATUS_GOOD)
|
|
return (status);
|
|
status = sanei_constrain_value (s->options_list + option, val, info);
|
|
switch (option)
|
|
{
|
|
/* (mostly) side-effect-free word options: */
|
|
case OPT_RESOLUTION:
|
|
case OPT_BIT_DEPTH:
|
|
case OPT_BR_X:
|
|
case OPT_BR_Y:
|
|
case OPT_TL_X:
|
|
case OPT_TL_Y:
|
|
if (info)
|
|
*info |= SANE_INFO_RELOAD_PARAMS;
|
|
/* fall through */
|
|
case OPT_NUM_OPTS:
|
|
case OPT_CONTRAST:
|
|
case OPT_BRIGHTNESS:
|
|
case OPT_LAMP_TIMEOUT:
|
|
case OPT_THRESHOLD:
|
|
s->value[option].w = *(SANE_Word *) val;
|
|
return (SANE_STATUS_GOOD);
|
|
case OPT_MODE:
|
|
if (s->value[option].s)
|
|
free (s->value[option].s);
|
|
s->value[option].s = strdup (val);
|
|
|
|
if (info)
|
|
*info |= SANE_INFO_RELOAD_PARAMS;
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
}
|
|
return (SANE_STATUS_INVAL);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
const SANE_Option_Descriptor *
|
|
sane_get_option_descriptor (SANE_Handle handle, SANE_Int option)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
DBG (2, "sane_get_option_descriptor\n");
|
|
if ((unsigned) option >= NUM_OPTIONS)
|
|
return (0);
|
|
return (&s->options_list[option]);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
void
|
|
sane_close (SANE_Handle handle)
|
|
{
|
|
SM3840_Scan *prev, *s;
|
|
|
|
DBG (2, "sane_close\n");
|
|
/* remove handle from list of open handles: */
|
|
prev = 0;
|
|
for (s = first_handle; s; s = s->next)
|
|
{
|
|
if (s == handle)
|
|
break;
|
|
prev = s;
|
|
}
|
|
if (!s)
|
|
{
|
|
DBG (1, "close: invalid handle %p\n", handle);
|
|
return; /* oops, not a handle we know about */
|
|
}
|
|
|
|
if (s->scanning)
|
|
{
|
|
sane_cancel (handle);
|
|
}
|
|
|
|
sanei_usb_close (s->udev);
|
|
|
|
if (s->line_buffer)
|
|
free (s->line_buffer);
|
|
if (s->save_scan_line)
|
|
free (s->save_scan_line);
|
|
if (s->save_dpi1200_remap)
|
|
free (s->save_dpi1200_remap);
|
|
if (s->save_color_remap)
|
|
free (s->save_color_remap);
|
|
|
|
if (prev)
|
|
prev->next = s->next;
|
|
else
|
|
first_handle = s;
|
|
free (handle);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
void
|
|
sane_exit (void)
|
|
{
|
|
SM3840_Device *next;
|
|
DBG (2, "sane_exit\n");
|
|
while (first_dev != NULL)
|
|
{
|
|
next = first_dev->next;
|
|
free (first_dev);
|
|
first_dev = next;
|
|
}
|
|
if (devlist)
|
|
free (devlist);
|
|
}
|
|
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize)
|
|
{
|
|
DBG_INIT ();
|
|
if (version_code)
|
|
*version_code = SANE_VERSION_CODE (SANE_CURRENT_MAJOR, V_MINOR, 0);
|
|
if (authorize)
|
|
DBG (2, "Unused authorize\n");
|
|
|
|
sanei_usb_init ();
|
|
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
|
|
static SANE_Status
|
|
add_sm_device (SANE_String_Const devname, SANE_String_Const modname)
|
|
{
|
|
SM3840_Device *dev;
|
|
|
|
dev = calloc (sizeof (*dev), 1);
|
|
if (!dev)
|
|
return (SANE_STATUS_NO_MEM);
|
|
|
|
memset (dev, 0, sizeof (*dev));
|
|
dev->sane.name = strdup (devname);
|
|
dev->sane.model = modname;
|
|
dev->sane.vendor = "Microtek";
|
|
dev->sane.type = "flatbed scanner";
|
|
++num_devices;
|
|
dev->next = first_dev;
|
|
first_dev = dev;
|
|
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
static SANE_Status
|
|
add_sm3840_device (SANE_String_Const devname)
|
|
{
|
|
return add_sm_device (devname, "ScanMaker 3840");
|
|
}
|
|
|
|
static SANE_Status
|
|
add_sm4800_device (SANE_String_Const devname)
|
|
{
|
|
return add_sm_device (devname, "ScanMaker 4800");
|
|
}
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only)
|
|
{
|
|
static const SANE_Device **devlist = 0;
|
|
SM3840_Device *dev;
|
|
int i;
|
|
|
|
DBG (3, "sane_get_devices (local_only = %d)\n", local_only);
|
|
|
|
while (first_dev)
|
|
{
|
|
dev = first_dev->next;
|
|
free (first_dev);
|
|
first_dev = dev;
|
|
}
|
|
first_dev = NULL;
|
|
num_devices = 0;
|
|
|
|
/* If we get enough scanners should use an array, but for now */
|
|
/* do it one-by-one... */
|
|
sanei_usb_find_devices (0x05da, 0x30d4, add_sm3840_device);
|
|
sanei_usb_find_devices (0x05da, 0x30cf, add_sm4800_device);
|
|
|
|
if (devlist)
|
|
free (devlist);
|
|
devlist = calloc ((num_devices + 1) * sizeof (devlist[0]), 1);
|
|
if (!devlist)
|
|
return SANE_STATUS_NO_MEM;
|
|
i = 0;
|
|
for (dev = first_dev; i < num_devices; dev = dev->next)
|
|
devlist[i++] = &dev->sane;
|
|
devlist[i++] = 0;
|
|
if (device_list)
|
|
*device_list = devlist;
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
static size_t
|
|
max_string_size (const SANE_String_Const strings[])
|
|
{
|
|
size_t size, max_size = 0;
|
|
int i;
|
|
for (i = 0; strings[i]; ++i)
|
|
{
|
|
size = strlen (strings[i]) + 1;
|
|
if (size > max_size)
|
|
max_size = size;
|
|
}
|
|
|
|
return (max_size);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
initialize_options_list (SM3840_Scan * s)
|
|
{
|
|
|
|
SANE_Int option;
|
|
DBG (2, "initialize_options_list\n");
|
|
for (option = 0; option < NUM_OPTIONS; ++option)
|
|
{
|
|
s->options_list[option].size = sizeof (SANE_Word);
|
|
s->options_list[option].cap =
|
|
SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
}
|
|
|
|
s->options_list[OPT_NUM_OPTS].name = SANE_NAME_NUM_OPTIONS;
|
|
s->options_list[OPT_NUM_OPTS].title = SANE_TITLE_NUM_OPTIONS;
|
|
s->options_list[OPT_NUM_OPTS].desc = SANE_DESC_NUM_OPTIONS;
|
|
s->options_list[OPT_NUM_OPTS].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_NUM_OPTS].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_NUM_OPTS].size = sizeof (SANE_Word);
|
|
s->options_list[OPT_NUM_OPTS].cap = SANE_CAP_SOFT_DETECT;
|
|
s->options_list[OPT_NUM_OPTS].constraint_type = SANE_CONSTRAINT_NONE;
|
|
s->value[OPT_NUM_OPTS].w = NUM_OPTIONS;
|
|
|
|
s->options_list[OPT_MODE].name = SANE_NAME_SCAN_MODE;
|
|
s->options_list[OPT_MODE].title = SANE_TITLE_SCAN_MODE;
|
|
s->options_list[OPT_MODE].desc = SANE_DESC_SCAN_MODE;
|
|
s->options_list[OPT_MODE].type = SANE_TYPE_STRING;
|
|
s->options_list[OPT_MODE].size = max_string_size (mode_list);
|
|
s->options_list[OPT_MODE].constraint_type = SANE_CONSTRAINT_STRING_LIST;
|
|
s->options_list[OPT_MODE].constraint.string_list = mode_list;
|
|
s->value[OPT_MODE].s = strdup (mode_list[1]);
|
|
|
|
s->options_list[OPT_RESOLUTION].name = SANE_NAME_SCAN_RESOLUTION;
|
|
s->options_list[OPT_RESOLUTION].title = SANE_TITLE_SCAN_RESOLUTION;
|
|
s->options_list[OPT_RESOLUTION].desc = SANE_DESC_SCAN_RESOLUTION;
|
|
s->options_list[OPT_RESOLUTION].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_RESOLUTION].unit = SANE_UNIT_DPI;
|
|
s->options_list[OPT_RESOLUTION].constraint_type = SANE_CONSTRAINT_WORD_LIST;
|
|
s->options_list[OPT_RESOLUTION].constraint.word_list = resolution_list;
|
|
s->value[OPT_RESOLUTION].w = 300;
|
|
|
|
s->options_list[OPT_BIT_DEPTH].name = SANE_NAME_BIT_DEPTH;
|
|
s->options_list[OPT_BIT_DEPTH].title = SANE_TITLE_BIT_DEPTH;
|
|
s->options_list[OPT_BIT_DEPTH].desc = SANE_DESC_BIT_DEPTH;
|
|
s->options_list[OPT_BIT_DEPTH].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_BIT_DEPTH].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_BIT_DEPTH].constraint_type = SANE_CONSTRAINT_WORD_LIST;
|
|
s->options_list[OPT_BIT_DEPTH].constraint.word_list = bpp_list;
|
|
s->value[OPT_BIT_DEPTH].w = 8;
|
|
|
|
s->options_list[OPT_TL_X].name = SANE_NAME_SCAN_TL_X;
|
|
s->options_list[OPT_TL_X].title = SANE_TITLE_SCAN_TL_X;
|
|
s->options_list[OPT_TL_X].desc = SANE_DESC_SCAN_TL_X;
|
|
s->options_list[OPT_TL_X].type = SANE_TYPE_FIXED;
|
|
s->options_list[OPT_TL_X].unit = SANE_UNIT_MM;
|
|
s->options_list[OPT_TL_X].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_TL_X].constraint.range = &x_range;
|
|
s->value[OPT_TL_X].w = s->options_list[OPT_TL_X].constraint.range->min;
|
|
s->options_list[OPT_TL_Y].name = SANE_NAME_SCAN_TL_Y;
|
|
s->options_list[OPT_TL_Y].title = SANE_TITLE_SCAN_TL_Y;
|
|
s->options_list[OPT_TL_Y].desc = SANE_DESC_SCAN_TL_Y;
|
|
s->options_list[OPT_TL_Y].type = SANE_TYPE_FIXED;
|
|
s->options_list[OPT_TL_Y].unit = SANE_UNIT_MM;
|
|
s->options_list[OPT_TL_Y].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_TL_Y].constraint.range = &y_range;
|
|
s->value[OPT_TL_Y].w = s->options_list[OPT_TL_Y].constraint.range->min;
|
|
s->options_list[OPT_BR_X].name = SANE_NAME_SCAN_BR_X;
|
|
s->options_list[OPT_BR_X].title = SANE_TITLE_SCAN_BR_X;
|
|
s->options_list[OPT_BR_X].desc = SANE_DESC_SCAN_BR_X;
|
|
s->options_list[OPT_BR_X].type = SANE_TYPE_FIXED;
|
|
s->options_list[OPT_BR_X].unit = SANE_UNIT_MM;
|
|
s->options_list[OPT_BR_X].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_BR_X].constraint.range = &x_range;
|
|
s->value[OPT_BR_X].w = s->options_list[OPT_BR_X].constraint.range->max;
|
|
s->options_list[OPT_BR_Y].name = SANE_NAME_SCAN_BR_Y;
|
|
s->options_list[OPT_BR_Y].title = SANE_TITLE_SCAN_BR_Y;
|
|
s->options_list[OPT_BR_Y].desc = SANE_DESC_SCAN_BR_Y;
|
|
s->options_list[OPT_BR_Y].type = SANE_TYPE_FIXED;
|
|
s->options_list[OPT_BR_Y].unit = SANE_UNIT_MM;
|
|
s->options_list[OPT_BR_Y].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_BR_Y].constraint.range = &y_range;
|
|
s->value[OPT_BR_Y].w = s->options_list[OPT_BR_Y].constraint.range->max;
|
|
|
|
s->options_list[OPT_CONTRAST].name = SANE_NAME_CONTRAST;
|
|
s->options_list[OPT_CONTRAST].title = SANE_TITLE_CONTRAST;
|
|
s->options_list[OPT_CONTRAST].desc = SANE_DESC_CONTRAST;
|
|
s->options_list[OPT_CONTRAST].type = SANE_TYPE_FIXED;
|
|
s->options_list[OPT_CONTRAST].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_CONTRAST].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_CONTRAST].constraint.range = &contrast_range;
|
|
s->value[OPT_CONTRAST].w = SANE_FIX (3.5);
|
|
|
|
s->options_list[OPT_BRIGHTNESS].name = SANE_NAME_BRIGHTNESS;
|
|
s->options_list[OPT_BRIGHTNESS].title = SANE_TITLE_BRIGHTNESS;
|
|
s->options_list[OPT_BRIGHTNESS].desc = SANE_DESC_BRIGHTNESS;
|
|
s->options_list[OPT_BRIGHTNESS].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_BRIGHTNESS].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_BRIGHTNESS].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_BRIGHTNESS].constraint.range = &brightness_range;
|
|
s->value[OPT_BRIGHTNESS].w = 1800;
|
|
|
|
s->options_list[OPT_LAMP_TIMEOUT].name = "lamp-timeout";
|
|
s->options_list[OPT_LAMP_TIMEOUT].title = SANE_I18N ("Lamp timeout");
|
|
s->options_list[OPT_LAMP_TIMEOUT].desc =
|
|
SANE_I18N ("Minutes until lamp is turned off after scan");
|
|
s->options_list[OPT_LAMP_TIMEOUT].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_LAMP_TIMEOUT].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_LAMP_TIMEOUT].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_LAMP_TIMEOUT].constraint.range = &lamp_range;
|
|
s->value[OPT_LAMP_TIMEOUT].w = 15;
|
|
|
|
s->options_list[OPT_THRESHOLD].name = "threshold";
|
|
s->options_list[OPT_THRESHOLD].title = SANE_I18N ("Threshold");
|
|
s->options_list[OPT_THRESHOLD].desc =
|
|
SANE_I18N ("Threshold value for lineart mode");
|
|
s->options_list[OPT_THRESHOLD].type = SANE_TYPE_INT;
|
|
s->options_list[OPT_THRESHOLD].unit = SANE_UNIT_NONE;
|
|
s->options_list[OPT_THRESHOLD].constraint_type = SANE_CONSTRAINT_RANGE;
|
|
s->options_list[OPT_THRESHOLD].constraint.range = &threshold_range;
|
|
s->value[OPT_THRESHOLD].w = 128;
|
|
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_open (SANE_String_Const devicename, SANE_Handle * handle)
|
|
{
|
|
SANE_Status status;
|
|
SM3840_Device *dev;
|
|
SM3840_Scan *s;
|
|
DBG (2, "sane_open\n");
|
|
|
|
/* Make sure we have first_dev */
|
|
sane_get_devices (NULL, 0);
|
|
if (devicename[0])
|
|
{
|
|
for (dev = first_dev; dev; dev = dev->next)
|
|
if (strcmp (dev->sane.name, devicename) == 0)
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
/* empty devicename -> use first device */
|
|
dev = first_dev;
|
|
}
|
|
DBG (2, "using device: %s %p\n", dev->sane.name, (unsigned char *) dev);
|
|
if (!dev)
|
|
return SANE_STATUS_INVAL;
|
|
s = calloc (sizeof (*s), 1);
|
|
if (!s)
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
status = sanei_usb_open (dev->sane.name, &(s->udev));
|
|
if (status != SANE_STATUS_GOOD)
|
|
return status;
|
|
|
|
initialize_options_list (s);
|
|
s->scanning = 0;
|
|
/* insert newly opened handle into list of open handles: */
|
|
s->next = first_handle;
|
|
first_handle = s;
|
|
*handle = s;
|
|
return (SANE_STATUS_GOOD);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_set_io_mode (SANE_Handle handle, SANE_Bool non_blocking)
|
|
{
|
|
SM3840_Scan *s = handle;
|
|
DBG (2, "sane_set_io_mode( %p, %d )\n", handle, non_blocking);
|
|
if (s->scanning)
|
|
{
|
|
if (non_blocking == SANE_FALSE)
|
|
return SANE_STATUS_GOOD;
|
|
else
|
|
return (SANE_STATUS_UNSUPPORTED);
|
|
}
|
|
else
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
SANE_Status
|
|
sane_get_select_fd (SANE_Handle handle, SANE_Int * fd)
|
|
{
|
|
DBG (2, "sane_get_select_fd( %p, %p )\n", (void *) handle, (void *) fd);
|
|
return SANE_STATUS_UNSUPPORTED;
|
|
}
|