sane-project-backends/backend/epson.c

5153 wiersze
130 KiB
C

/*
epson.c - SANE library for Epson flatbed scanners.
based on Kazuhiro Sasayama previous
Work on epson.[ch] file from the SANE package.
original code taken from sane-0.71
Copyright (C) 1997 Hypercore Software Design, Ltd.
modifications
Copyright (C) 1998-1999 Christian Bucher <bucher@vernetzt.at>
Copyright (C) 1998-1999 Kling & Hautzinger GmbH
Copyright (C) 1999 Norihiko Sawa <sawa@yb3.so-net.ne.jp>
Copyright (C) 1999-2000 Karl Heinz Kremer <khk@khk.net>
Copyright (C) 2000 Mike Porter <mike@udel.edu> (mjp)
*/
#define SANE_EPSON_VERSION "SANE Epson Backend v0.1.36 - 2000-12-02"
/*
This file is part of the SANE package.
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. */
/*
2000-12-02 Read information about optical resolution and line
distance from scanner instead of hardcoded values.
Add support for color depth > 8 bits per channel.
2000-11-23 Display "Set Focus" control only for scanners that
can actually handle the command.
2000-11-19 Added support for the "set focus position" command,
this is necessary for the Expression1600.
2000-07-28 Changed #include <...> to #include "..." for the
sane/... include files.
2000-07-26 Fixed problem with Perfection610: The variable
s->color_shuffle_line was never correctly initialized
2000-06-28 When closing the scanner device the data that's
still in the scanner, waiting to be transferred
is flushed. This fixes the problem with scanimage -T
2000-06-13 Invert image when scanning negative with TPU,
Show film type only when TPU is selected
2000-06-13 Initialize optical_res to 0 (Dave Hill)
2000-06-07 Fix in sane_close() - found by Henning Meier-Geinitz
2000-06-01 Threshhold should only be active when scan depth
is 1 and halftoning is off. (mjp)
2000-05-28 Turned on scanner based color correction.
Dependancies between many options are now
being enforced. For instance, auto area seg
(AAS) should only be on when scan depth == 1.
Added some routines to active and deactivate
options. Routines report if option changed.
Help prevent extraneous option reloads. Split
sane_control_option in getvalue and setvalue.
Further split up setvalue into several different
routines. (mjp)
2000-05-21 In sane_close use close_scanner instead of just the
SCSI close function.
2000-05-20 ... finally fixed the problem with the 610
Added resolution_list to Epson_Device structure in
epson.h - this fixes a bug that caused problems when
more than one EPSON scanner was connected.
2000-05-13 Fixed the color problem with the Perfection 610. The few
lines with "garbage" at the beginning of the scan are not
yet removed.
2000-05-06 Added support for multiple EPSON scanners. At this time
this may not be bug free, but it's a start and it seems
to work well with just one scanner.
2000-04-06 Did some cleanup on the gamma correction part. The user
defined table is now initialized to gamma=1, the gamma
handling is also no longer depending on platform specific
tables (handled instead by pointers to the actual tables)
2000-03-27 Disable request for push button status
2000-03-22 Removed free() calls to static strings to remove
compile warnings. These were introduced to apparently
fix an OS/2 bug. It now turned out that they are not
necessary. The real fix was in the repository for a
long time (2000-01-25).
2000-03-19 Fixed problem with A4 level devices - they use the
line mode instead of the block mode. The routine to
handle this was screwed up pretty bad. Now I have
a solid version that handles all variations of line
mode (automatically deals with the order the color
lines are sent).
2000-03-06 Fixed occasional crash after warm up when the "in warmup
state" went away in between doing ESC G and getting the
extended status message.
2000-03-02 Code cleanup, disabled ZOOM until I have time to
deal with all the side effects.
2000-03-01 More D1 fixes. In the future I have to come up with
a more elegant solution to destinguish between different
function levels. The level > n does not work anymore with
D1.
Added support for "set threshold" and "set zoom".
2000-02-23 First stab at level D1 support, also added a test
for valid "set halftone" command to enable OPT_HALFTONE
2000-02-21 Check for "warming up" in after sane_start. This is
IMHO a horrible hack, but that's the only way without
a major redesign that will work. (KHK)
2000-02-20 Added some cleanup on error conditions in attach()
Use new sanei_config_read() instead of fgets() for
compatibility with OS/2 (Yuri Dario)
2000-02-19 Changed some "int" to "size_t" types
Removed "Preview Resolution"
Implemented resolution list as WORD_LIST instead of
a RANGE (KHK)
2000-02-11 Default scan source is always "Flatbed", regardless
of installed options. Corrected some typos. (KHK)
2000-02-03 Gamma curves now coupled with gamma correction menu.
Only when "User defined" is selected are the curves
selected. (Dave Hill)
Renamed "Contrast" to "Gamma Correction" (KHK)
2000-02-02 "Brown Paper Bag Release" Put the USB fix finally
into the CVS repository.
2000-02-01 Fixed problem with USB scanner not being recognized
because of hte changes to attach a few days ago. (KHK)
2000-01-29 fixed core dump with xscanimage by moving the gamma
curves to the standard interface (no longer advanced)
Removed pragma pack() from source code to make it
easier to compile on non-gcc compilers (KHK)
2000-01-26 fixed problem with resolution selection when using the
resolution list in xsane (KHK)
2000-01-25 moved the section where the device name is assigned
in attach. This avoids the core dump of frontend
applications when no scanner is found (Dave Hill)
2000-01-24 reorganization of SCSI related "helper" functions
started support for user defined color correction -
this is not yet available via the UI (Christian Bucher)
2000-01-24 Removed C++ style comments '//' (KHK)
*/
#ifdef _AIX
# include <lalloca.h> /* MUST come first for AIX! */
#endif
#include "sane/config.h"
#include <lalloca.h>
#include <limits.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include "sane/sane.h"
#include "sane/saneopts.h"
#include "sane/sanei_scsi.h"
/*
* NOTE: try to isolate scsi stuff in own section.
*
*/
#define TEST_UNIT_READY_COMMAND 0x00
#define READ_6_COMMAND 0x08
#define WRITE_6_COMMAND 0x0a
#define INQUIRY_COMMAND 0x12
#define TYPE_PROCESSOR 0x03
/*
*
*
*/
static SANE_Status inquiry ( int fd, int page_code, void * buf, size_t * buf_size) {
u_char cmd [ 6];
int status;
memset( cmd, 0, 6);
cmd[ 0] = INQUIRY_COMMAND;
cmd[ 2] = page_code;
cmd[ 4] = *buf_size > 255 ? 255 : *buf_size;
status = sanei_scsi_cmd( fd, cmd, sizeof cmd, buf, buf_size);
return status;
}
/*
*
*
*/
static int scsi_read ( int fd, void * buf, size_t buf_size, SANE_Status * status) {
u_char cmd [ 6];
memset( cmd, 0, 6);
cmd[ 0] = READ_6_COMMAND;
cmd[ 2] = buf_size >> 16;
cmd[ 3] = buf_size >> 8;
cmd[ 4] = buf_size;
if( SANE_STATUS_GOOD == ( *status = sanei_scsi_cmd( fd, cmd, sizeof( cmd), buf, &buf_size)))
return buf_size;
return 0;
}
/*
*
*
*/
static int scsi_write ( int fd, const void * buf, size_t buf_size, SANE_Status * status) {
u_char * cmd;
cmd = alloca( 6 + buf_size);
memset( cmd, 0, 6);
cmd[ 0] = WRITE_6_COMMAND;
cmd[ 2] = buf_size >> 16;
cmd[ 3] = buf_size >> 8;
cmd[ 4] = buf_size;
memcpy( cmd + 6, buf, buf_size);
if( SANE_STATUS_GOOD == ( *status = sanei_scsi_cmd( fd, cmd, 6 + buf_size, NULL, NULL)))
return buf_size;
return 0;
}
/*
*
*
*/
#include <sane/sanei_pio.h>
#include "epson.h"
#define BACKEND_NAME epson
#include <sane/sanei_backend.h>
#include <sane/sanei_config.h>
#define EPSON_CONFIG_FILE "epson.conf"
#ifndef PATH_MAX
# define PATH_MAX (1024)
#endif
#define walloc(x) ( x *) malloc( sizeof( x) )
#define walloca(x) ( x *) alloca( sizeof( x) )
#ifndef XtNumber
# define XtNumber(x) ( sizeof x / sizeof x [ 0] )
# define XtOffset(p_type,field) ((size_t)&(((p_type)NULL)->field))
# define XtOffsetOf(s_type,field) XtOffset(s_type*,field)
#endif
/* NOTE: you can find these codes with "man ascii". */
#define STX 0x02
#define ACK 0x06
#define NAK 0x15
#define CAN 0x18
#define ESC 0x1B
#define S_ACK "\006"
#define S_CAN "\030"
#define STATUS_FER 0x80 /* fatal error */
#define STATUS_AREA_END 0x20 /* area end */
#define STATUS_OPTION 0x10 /* option installed */
#define EXT_STATUS_FER 0x80 /* fatal error */
#define EXT_STATUS_FBF 0x40 /* flat bed scanner */
#define EXT_STATUS_WU 0x02 /* warming up */
#define EXT_STATUS_PB 0x01 /* scanner has a push button */
#define EXT_STATUS_IST 0x80 /* option detected */
#define EXT_STATUS_EN 0x40 /* option enabled */
#define EXT_STATUS_ERR 0x20 /* other error */
#define EXT_STATUS_PE 0x08 /* no paper */
#define EXT_STATUS_PJ 0x04 /* paper jam */
#define EXT_STATUS_OPN 0x02 /* cover open */
#define EPSON_LEVEL_A1 0
#define EPSON_LEVEL_A2 1
#define EPSON_LEVEL_B1 2
#define EPSON_LEVEL_B2 3
#define EPSON_LEVEL_B3 4
#define EPSON_LEVEL_B4 5
#define EPSON_LEVEL_B5 6
#define EPSON_LEVEL_B6 7
#define EPSON_LEVEL_B7 8
#define EPSON_LEVEL_B8 9
#define EPSON_LEVEL_F5 10
#define EPSON_LEVEL_D1 11
/* there is also a function level "A5", which I'm igoring here until somebody can
convince me that this is still needed. The A5 level was for the GT-300, which
was (is) a monochrome only scanner. So if somebody really wants to use this
scanner with SANE get in touch with me and we can work something out - khk */
#define EPSON_LEVEL_DEFAULT EPSON_LEVEL_B3
static EpsonCmdRec epson_cmd [ ] =
{
/*
* request identity
* | request identity2
* | | request status
* | | | request condition
* | | | | set color mode
* | | | | | start scanning
* | | | | | | set data format
* | | | | | | | set resolution
* | | | | | | | | set zoom
* | | | | | | | | | set scan area
* | | | | | | | | | | set brightness
* | | | | | | | | | | | set gamma
* | | | | | | | | | | | | set halftoning
* | | | | | | | | | | | | | set color correction
* | | | | | | | | | | | | | | initialize scanner
* | | | | | | | | | | | | | | | set speed
* | | | | | | | | | | | | | | | | set lcount
* | | | | | | | | | | | | | | | | | mirror image
* | | | | | | | | | | | | | | | | | | set gamma table
* | | | | | | | | | | | | | | | | | | | set outline emphasis
* | | | | | | | | | | | | | | | | | | | | set dither
* | | | | | | | | | | | | | | | | | | | | | set color correction coefficients
* | | | | | | | | | | | | | | | | | | | | | | request extension status
* | | | | | | | | | | | | | | | | | | | | | | | control an extension
* | | | | | | | | | | | | | | | | | | | | | | | | forward feed / eject
* | | | | | | | | | | | | | | | | | | | | | | | | | request push button status
* | | | | | | | | | | | | | | | | | | | | | | | | | | control auto area segmentation
* | | | | | | | | | | | | | | | | | | | | | | | | | | | set film type
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | set exposure time
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | set bay
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | set threshold
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | set focus position
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | request focus position
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |
* | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |
*/
{"A1",'I', 0 ,'F','S', 0 ,'G', 0 ,'R', 0 ,'A', 0 ,{ 0,0,0}, 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"A2",'I', 0 ,'F','S', 0 ,'G','D','R','H','A','L',{-3,3,0},'Z','B', 0 ,'@', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B1",'I', 0 ,'F','S','C','G','D','R', 0 ,'A', 0 ,{ 0,0,0}, 0 ,'B', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B2",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z','B', 0 ,'@', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B3",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z','B','M','@', 0 , 0 , 0 , 0 , 0 , 0 ,'m','f','e', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B4",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z','B','M','@','g','d', 0 ,'z','Q','b','m','f','e', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B5",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z','B','M','@','g','d','K','z','Q','b','m','f','e', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B6",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z','B','M','@','g','d','K','z','Q','b','m','f','e', 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 },
{"B7",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-4,3,0},'Z','B','M','@','g','d','K','z','Q','b','m','f','e','\f','!','s','N', 0 , 0 ,'t', 0 , 0 },
{"B8",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-4,3,0},'Z','B','M','@','g','d','K','z','Q','b','m','f','e', 0 ,'!','s','N', 0 , 0 , 0 ,'p','q'},
{"F5",'I', 0 ,'F','S','C','G','D','R','H','A','L',{-3,3,0},'Z', 0 ,'M','@','g','d','K','z','Q', 0 ,'m','f','e','\f', 0 , 0 ,'N','T','P', 0 , 0 , 0 },
{"D1",'I','i','F', 0 ,'C','G','D','R', 0 ,'A', 0 ,{ 0,0,0},'Z', 0 , 0 ,'@','g','d', 0 ,'z', 0 , 0 , 0 ,'f', 0 , 0 ,'!', 0 , 0 , 0 , 0 , 0 , 0 , 0 },
};
/*
* Definition of the mode_param struct, that is used to
* specify the valid parameters for the different scan modes.
* Depending on the capabilities of the scanner three different
* param sets are used: 8 bit, 12 bit and 16 bit per color channel
*/
struct mode_param {
int color;
int mode_flags;
int dropout_mask;
int depth;
};
static const struct mode_param mode_params_8 [ ] =
{ { 0, 0x00, 0x30, 1}
, { 0, 0x00, 0x30, 8}
, { 1, 0x02, 0x00, 8}
};
static const struct mode_param mode_params_12 [ ] =
{ { 0, 0x00, 0x30, 1}
, { 0, 0x00, 0x30, 8}
, { 0, 0x00, 0x30, 12}
, { 1, 0x03, 0x10, 8}
, { 1, 0x03, 0x10, 12}
};
static const struct mode_param mode_params_16 [ ] =
{ { 0, 0x00, 0x30, 1}
, { 0, 0x00, 0x30, 8}
, { 0, 0x00, 0x30, 16}
, { 1, 0x03, 0x10, 8}
, { 1, 0x03, 0x10, 16}
};
static const SANE_String_Const mode_list_8 [ ] =
{ "Binary"
, "Gray"
, "Color"
, NULL
};
static const SANE_String_Const mode_list_12 [ ] =
{ "Binary"
, "Gray (8 bit)"
, "Gray (12 bit)"
, "Color (24 bit)"
, "Color (36 bit)"
, NULL
};
static const SANE_String_Const mode_list_16 [ ] =
{ "Binary"
, "Gray (8 bit)"
, "Gray (16 bit)"
, "Color (24 bit)"
, "Color (42 bit)"
, NULL
};
static const SANE_String_Const * mode_list = NULL;
static const struct mode_param * mode_params = NULL;
/*
* Define the different scan sources:
*/
#define FBF_STR "Flatbed"
#define TPU_STR "Transparency Unit"
#define ADF_STR "Automatic Document Feeder"
/*
* source list need one dummy entry (save device settings is crashing).
* NOTE: no const - this list gets created while exploring the capabilities
* of the scanner.
*/
static SANE_String_Const source_list [ ] =
{ FBF_STR
, NULL
, NULL
, NULL
};
/* some defines to make handling the TPU easier: */
#define FILM_TYPE_POSITIVE (0)
#define FILM_TYPE_NEGATIVE (1)
static const SANE_String_Const film_list [ ] =
{ "Positive Film"
, "Negative Film"
, NULL
};
static const SANE_String_Const focus_list [] =
{
"Focus on glass",
"Focus 2.5mm above glass",
NULL
};
/*
* TODO: add some missing const.
*/
#define HALFTONE_NONE 0x01
#define HALFTONE_TET 0x03
static int halftone_params [ ] =
{ HALFTONE_NONE
, 0x00
, 0x10
, 0x20
, 0x80
, 0x90
, 0xa0
, 0xb0
, HALFTONE_TET
, 0xc0
, 0xd0
};
static const SANE_String_Const halftone_list [ ] =
{ "None"
, "Halftone A (Hard Tone)"
, "Halftone B (Soft Tone)"
, "Halftone C (Net Screen)"
, NULL
};
static const SANE_String_Const halftone_list_4 [ ] =
{ "None"
, "Halftone A (Hard Tone)"
, "Halftone B (Soft Tone)"
, "Halftone C (Net Screen)"
, "Dither A (4x4 Bayer)"
, "Dither B (4x4 Spiral)"
, "Dither C (4x4 Net Screen)"
, "Dither D (8x4 Net Screen)"
, NULL
};
static const SANE_String_Const halftone_list_7 [ ] =
{ "None"
, "Halftone A (Hard Tone)"
, "Halftone B (Soft Tone)"
, "Halftone C (Net Screen)"
, "Dither A (4x4 Bayer)"
, "Dither B (4x4 Spiral)"
, "Dither C (4x4 Net Screen)"
, "Dither D (8x4 Net Screen)"
, "Text Enhanced Technology"
, "Download pattern A"
, "Download pattern B"
, NULL
};
static int dropout_params [ ] =
{ 0x00 /* none */
, 0x10 /* red */
, 0x20 /* green */
, 0x30 /* blue */
};
static const SANE_String_Const dropout_list [ ] =
{ "None"
, "Red"
, "Green"
, "Blue"
, NULL
};
/*
* Color correction:
* One array for the actual parameters that get sent to the scanner (color_params[]),
* one array for the strings that get displayed in the user interface (color_list[])
* and one array to mark the user defined color correction (dolor_userdefined[]).
*/
static int color_params [ ] =
{ 0x00
, 0x01
, 0x10
, 0x20
, 0x40
, 0x80
};
static SANE_Bool color_userdefined [] =
{ SANE_FALSE
, SANE_TRUE
, SANE_FALSE
, SANE_FALSE
, SANE_FALSE
, SANE_FALSE
};
static const SANE_String_Const color_list [ ] =
{ "No Correction"
, "User defined"
, "Impact-dot printers"
, "Thermal printers"
, "Ink-jet printers"
, "CRT monitors"
, NULL
};
/*
* Gamma correction:
* The A and B level scanners work differently than the D level scanners, therefore
* I define two different sets of arrays, plus one set of variables that get set to
* the actally used params and list arrays at runtime.
*/
static int gamma_params_ab [ ] =
{ 0x01
, 0x03
, 0x00
, 0x10
, 0x20
};
static const SANE_String_Const gamma_list_ab [ ] =
{ "Default"
, "User defined"
, "High density printing"
, "Low density printing"
, "High contrast printing"
, NULL
};
static SANE_Bool gamma_userdefined_ab [ ] =
{
SANE_FALSE,
SANE_TRUE,
SANE_FALSE,
SANE_FALSE,
SANE_FALSE,
};
static int gamma_params_d [ ] =
{ 0x03
, 0x04
};
static const SANE_String_Const gamma_list_d [ ] =
{ "User defined (Gamma=1.0)"
, "User defined (Gamma=1.8)"
, NULL
};
static SANE_Bool gamma_userdefined_d [ ] =
{
SANE_TRUE,
SANE_TRUE
};
static SANE_Bool * gamma_userdefined;
static int * gamma_params;
/* Bay list:
* this is used for the FilmScan
*/
static const SANE_String_Const bay_list [ ] =
{ " 1 "
, " 2 "
, " 3 "
, " 4 "
, " 5 "
, " 6 "
, NULL
};
/*
* minimum, maximum, quantization.
*/
static const SANE_Range u8_range = { 0, 255, 0};
static const SANE_Range s8_range = { -127, 127, 0};
static const SANE_Range zoom_range = { 50, 200, 0};
/*
* The "switch_params" are used for several boolean choices
*/
static int switch_params [ ] =
{
0,
1
};
#define mirror_params switch_params
#define speed_params switch_params
#define film_params switch_params
static const SANE_Range outline_emphasis_range = { -2, 2, 0 };
/* static const SANE_Range gamma_range = { -2, 2, 0 }; */
struct qf_param {
SANE_Word tl_x;
SANE_Word tl_y;
SANE_Word br_x;
SANE_Word br_y;
};
/* gcc don't like to overwrite const field */
static /*const*/ struct qf_param qf_params [ ] =
{ { 0, 0, SANE_FIX( 120.0), SANE_FIX( 120.0) }
, { 0, 0, SANE_FIX( 148.5), SANE_FIX( 210.0) }
, { 0, 0, SANE_FIX( 210.0), SANE_FIX( 148.5) }
, { 0, 0, SANE_FIX( 215.9), SANE_FIX( 279.4) } /* 8.5" x 11" */
, { 0, 0, SANE_FIX( 210.0), SANE_FIX( 297.0) }
, { 0, 0, 0, 0}
};
static const SANE_String_Const qf_list [ ] =
{ "CD"
, "A5 portrait"
, "A5 landscape"
, "Letter"
, "A4"
, "max"
, NULL
};
/*
* this is now stored in the s->hw->resolution_list field
static SANE_Word * resolution_list = NULL;
*/
/*
*
*
*/
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;
}
typedef struct {
u_char code;
u_char status;
u_short count;
u_char buf [ 1];
} EpsonHdrRec, * EpsonHdr;
typedef struct {
u_char code;
u_char status;
u_short count;
u_char type;
u_char level;
u_char buf [ 1];
} EpsonIdentRec, * EpsonIdent;
typedef struct {
u_char code;
u_char status;
u_short count;
u_char buf [ 1];
} EpsonParameterRec, * EpsonParameter;
typedef struct {
u_char code;
u_char status;
u_char buf [ 4];
} EpsonDataRec, * EpsonData;
/*
*
*
*/
static EpsonHdr command ( Epson_Scanner * s, const u_char * cmd, size_t cmd_size, SANE_Status * status);
static SANE_Status get_identity_information(SANE_Handle handle);
static SANE_Status get_identity2_information(SANE_Handle handle);
static int send ( Epson_Scanner * s, const void *buf, size_t buf_size, SANE_Status * status);
static ssize_t receive ( Epson_Scanner * s, void *buf, ssize_t buf_size, SANE_Status * status);
static SANE_Status color_shuffle(SANE_Handle handle, int *new_length);
static SANE_Status request_focus_position(SANE_Handle handle, u_char * position);
/*
*
*
*/
static int send ( Epson_Scanner * s, const void *buf, size_t buf_size, SANE_Status * status) {
DBG( 3, "send buf, size = %lu\n", ( u_long) buf_size);
#if 1
{
size_t k;
const u_char * s = buf;
for( k = 0; k < buf_size; k++) {
DBG( 125, "buf[%u] %02x %c\n", k, s[ k], isprint( s[ k]) ? s[ k] : '.');
}
}
#endif
if( s->hw->connection == SANE_EPSON_SCSI) {
return scsi_write( s->fd, buf, buf_size, status);
} else if (s->hw->connection == SANE_EPSON_PIO) {
size_t n;
if( buf_size == ( n = sanei_pio_write( s->fd, buf, buf_size)))
*status = SANE_STATUS_GOOD;
else
*status = SANE_STATUS_INVAL;
return n;
} else if (s->hw->connection == SANE_EPSON_USB) {
size_t n;
if( buf_size == ( n = write( s->fd, buf, buf_size)))
*status = SANE_STATUS_GOOD;
else
*status = SANE_STATUS_INVAL;
return n;
}
return SANE_STATUS_INVAL;
/* never reached */
}
/*
*
*
*/
static ssize_t receive ( Epson_Scanner * s, void *buf, ssize_t buf_size, SANE_Status * status) {
ssize_t n = 0;
if( s->hw->connection == SANE_EPSON_SCSI)
{
n = scsi_read( s->fd, buf, buf_size, status);
}
else if ( s->hw->connection == SANE_EPSON_PIO)
{
if( buf_size == ( n = sanei_pio_read( s->fd, buf, buf_size)))
*status = SANE_STATUS_GOOD;
else
*status = SANE_STATUS_INVAL;
}
else if (s->hw->connection == SANE_EPSON_USB)
{
/* only report an error if we don't read anything */
if( 0 < ( n = read( s->fd, buf, buf_size)))
*status = SANE_STATUS_GOOD;
else
{
if (n < 0)
{
DBG(0, "error in receive - status = %d\n", errno);
}
*status = SANE_STATUS_INVAL;
}
}
DBG( 7, "receive buf, expected = %lu, got = %d\n", ( u_long) buf_size, n);
#if 1
if (n > 0)
{
ssize_t k;
const u_char * s = buf;
for( k = 0; k < n; k++) {
DBG( 127, "buf[%u] %02x %c\n", k, s[ k], isprint( s[ k]) ? s[ k] : '.');
}
}
#endif
return n;
}
/*
*
*
*/
static SANE_Status expect_ack ( Epson_Scanner * s) {
u_char result [ 1];
size_t len;
SANE_Status status;
len = sizeof result;
receive( s, result, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
if( ACK != result[ 0])
return SANE_STATUS_INVAL;
return SANE_STATUS_GOOD;
}
/*
*
*
*/
static SANE_Status set_cmd ( Epson_Scanner * s, u_char cmd, int val) {
SANE_Status status;
u_char params [ 2];
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
params[ 0] = ESC;
params[ 1] = cmd;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
params[ 0] = val;
send( s, params, 1, &status);
status = expect_ack( s);
return status;
}
/*
*
*
*/
#define set_focus_position(s,v) set_cmd( s,(s)->hw->cmd->set_focus_position,v)
#define set_color_mode(s,v) set_cmd( s,(s)->hw->cmd->set_color_mode,v)
#define set_data_format(s,v) set_cmd( s,(s)->hw->cmd->set_data_format, v)
#define set_halftoning(s,v) set_cmd( s,(s)->hw->cmd->set_halftoning, v)
#define set_gamma(s,v) set_cmd( s,(s)->hw->cmd->set_gamma, v)
#define set_color_correction(s,v) set_cmd( s,(s)->hw->cmd->set_color_correction, v)
#define set_lcount(s,v) set_cmd( s,(s)->hw->cmd->set_lcount, v)
#define set_bright(s,v) set_cmd( s,(s)->hw->cmd->set_bright, v)
#define mirror_image(s,v) set_cmd( s,(s)->hw->cmd->mirror_image, v)
#define set_speed(s,v) set_cmd( s,(s)->hw->cmd->set_speed, v)
#define set_outline_emphasis(s,v) set_cmd( s,(s)->hw->cmd->set_outline_emphasis, v)
#define control_auto_area_segmentation(s,v) set_cmd( s,(s)->hw->cmd->control_auto_area_segmentation, v)
#define set_film_type(s,v) set_cmd( s,(s)->hw->cmd->set_film_type, v)
#define set_exposure_time(s,v) set_cmd( s,(s)->hw->cmd->set_exposure_time, v)
#define set_bay(s,v) set_cmd( s,(s)->hw->cmd->set_bay, v)
#define set_threshold(s,v) set_cmd( s,(s)->hw->cmd->set_threshold, v)
/*#define (s,v) set_cmd( s,(s)->hw->cmd->, v) */
static SANE_Status set_zoom ( Epson_Scanner * s, int x_zoom, int y_zoom)
{
SANE_Status status;
u_char cmd[2];
u_char params[2];
if( ! s->hw->cmd->set_zoom)
return SANE_STATUS_GOOD;
cmd[0] = ESC;
cmd[1] = s->hw->cmd->set_zoom;
send( s, cmd, 2, &status);
status = expect_ack( s);
if( status != SANE_STATUS_GOOD)
return status;
params[ 0] = x_zoom;
params[ 1] = y_zoom;
send( s, params, 2, &status);
status = expect_ack( s);
return status;
}
static SANE_Status set_resolution ( Epson_Scanner * s, int xres, int yres) {
SANE_Status status;
u_char params[4];
if( ! s->hw->cmd->set_resolution)
return SANE_STATUS_GOOD;
params[0] = ESC;
params[1] = s->hw->cmd->set_resolution;
send( s, params, 2, &status);
status = expect_ack( s);
if( status != SANE_STATUS_GOOD)
return status;
params[ 0] = xres;
params[ 1] = xres >> 8;
params[ 2] = yres;
params[ 3] = yres >> 8;
send( s, params, 4, &status);
status = expect_ack( s);
return status;
}
/*
*
*
*/
static SANE_Status set_scan_area ( Epson_Scanner * s, int x, int y, int width, int height) {
SANE_Status status;
u_char params[ 8];
if( ! s->hw->cmd->set_scan_area) {
return SANE_STATUS_GOOD;
}
params[0] = ESC;
params[1] = s->hw->cmd->set_scan_area;
send( s, params, 2, &status);
status = expect_ack( s);
if( status != SANE_STATUS_GOOD)
return status;
params[ 0] = x;
params[ 1] = x >> 8;
params[ 2] = y;
params[ 3] = y >> 8;
params[ 4] = width;
params[ 5] = width >> 8;
params[ 6] = height;
params[ 7] = height >> 8;
DBG( 1, "%p %d %d %d %d\n", ( void *) s, x, y, width, height);
send( s, params, 8, &status);
status = expect_ack( s);
return status;
}
/*
*
*
*/
static SANE_Status set_color_correction_coefficients ( Epson_Scanner * s) {
SANE_Status status;
u_char cmd = s->hw->cmd->set_color_correction_coefficients;
u_char params [ 2];
const int length = 9;
signed char cct [ 9];
DBG( 1, "set_color_correction_coefficients: starting.\n" );
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
params[ 0] = ESC;
params[ 1] = cmd;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
cct[ 0] = s->val[ OPT_CCT_1].w;
cct[ 1] = s->val[ OPT_CCT_2].w;
cct[ 2] = s->val[ OPT_CCT_3].w;
cct[ 3] = s->val[ OPT_CCT_4].w;
cct[ 4] = s->val[ OPT_CCT_5].w;
cct[ 5] = s->val[ OPT_CCT_6].w;
cct[ 6] = s->val[ OPT_CCT_7].w;
cct[ 7] = s->val[ OPT_CCT_8].w;
cct[ 8] = s->val[ OPT_CCT_9].w;
DBG( 1, "set_color_correction_coefficients: %d,%d,%d %d,%d,%d %d,%d,%d.\n",
cct[0], cct[1], cct[2], cct[3],
cct[4], cct[5], cct[6], cct[7], cct[8] );
send( s, cct, length, &status);
status = expect_ack( s);
DBG( 1, "set_color_correction_coefficients: ending=%d.\n", status );
return status;
}
/*
*
*
*/
static SANE_Status set_gamma_table ( Epson_Scanner * s) {
SANE_Status status;
u_char cmd = s->hw->cmd->set_gamma_table;
u_char params [ 2];
const int length = 257;
u_char gamma [ 257];
int n;
DBG( 1, "set_gamma_table: starting.\n" );
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
params[ 0] = ESC;
params[ 1] = cmd;
/*
Print the gamma tables before sending them to the scanner.
*/
if (DBG_LEVEL > 10) {
int c, i, j;
DBG (1, "set_gamma_table()\n");
for (c=0; c<4; c++) {
for (i=0; i<256; i+= 16) {
DBG (1, "Gamma Table[%d][%d] ", c, i);
for (j=0; j<16; j++) {
DBG (1, " %02x", s->gamma_table[c][i+j]);
}
DBG (1, "\n");
}
}
}
/*
* TODO: &status in send make no sense like that.
*/
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
gamma[ 0] = 'm';
for( n = 0; n < 256; ++n) {
gamma[ n + 1] = s->gamma_table[ 0] [ n];
}
send( s, gamma, length, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
gamma[ 0] = 'r';
for( n = 0; n < 256; ++n) {
gamma[ n + 1] = s->gamma_table[ 1] [ n];
}
send( s, gamma, length, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
gamma[ 0] = 'g';
for( n = 0; n < 256; ++n) {
gamma[ n + 1] = s->gamma_table[ 2] [ n];
}
send( s, gamma, length, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) )
return status;
gamma[ 0] = 'b';
for( n = 0; n < 256; ++n) {
gamma[ n + 1] = s->gamma_table[ 3] [ n];
}
send( s, gamma, length, &status);
status = expect_ack( s);
DBG( 1, "set_gamma_table: complete = %d.\n", status );
return status;
}
/*
*
*
*/
static SANE_Status check_ext_status ( Epson_Scanner * s) {
SANE_Status status;
u_char cmd = s->hw->cmd->request_extension_status;
u_char params [ 2];
u_char * buf;
EpsonHdr head;
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
params[ 0] = ESC;
params[ 1] = cmd;
if( NULL == ( head = ( EpsonHdr) command( s, params, 2, &status) ) ) {
DBG( 0, "Extended status flag request failed\n");
return status;
}
buf = &head->buf[ 0];
if( buf[ 0] & EXT_STATUS_WU) {
DBG( 10, "option: warming up\n");
status = SANE_STATUS_DEVICE_BUSY;
}
if( buf[ 0] & EXT_STATUS_FER) {
DBG( 0, "option: fatal error\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_ERR) {
DBG( 0, "ADF: other error\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_PE) {
DBG( 0, "ADF: no paper\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_PJ) {
DBG( 0, "ADF: paper jam\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_OPN) {
DBG( 0, "ADF: cover open\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 6] & EXT_STATUS_ERR) {
DBG( 0, "TPU: other error\n");
status = SANE_STATUS_INVAL;
}
return status;
}
/*
*
*
*/
#if 0
static SANE_Status reset ( Epson_Scanner * s) {
SANE_Status status;
u_char param[2];
if( ! s->hw->cmd->initialize_scanner)
return SANE_STATUS_GOOD;
param[0] = ESC;
param[1] = s->hw->cmd->initialize_scanner;
send (s, param, 2, &status);
status = expect_ack( s);
return status;
}
#endif
/*
*
*
*/
static void close_scanner ( Epson_Scanner * s) {
if( s->hw->connection == SANE_EPSON_SCSI)
sanei_scsi_close( s->fd);
else if ( s->hw->connection == SANE_EPSON_PIO)
sanei_pio_close( s->fd);
else if ( s->hw->connection == SANE_EPSON_USB)
close( s->fd);
return;
}
/*
*
*
*/
static SANE_Status open_scanner ( Epson_Scanner * s) {
SANE_Status status = 0;
DBG(5, "open_scanner()\n");
if( s->hw->connection == SANE_EPSON_SCSI) {
if( SANE_STATUS_GOOD != ( status = sanei_scsi_open( s->hw->sane.name, &s->fd, NULL, NULL))) {
DBG( 1, "sane_start: %s open failed: %s\n", s->hw->sane.name, sane_strstatus( status));
return status;
}
} else if ( s->hw->connection == SANE_EPSON_PIO) {
if( SANE_STATUS_GOOD != ( status = sanei_pio_open( s->hw->sane.name, &s->fd))) {
DBG( 1, "sane_start: %s open failed: %s\n", s->hw->sane.name, sane_strstatus( status));
return status;
}
} else if (s->hw->connection == SANE_EPSON_USB) {
int flags;
#ifdef _O_RDWR
flags = _O_RDWR;
#else
flags = O_RDWR;
#endif
#ifdef _O_EXCL
flags |= _O_EXCL;
#else
flags |= O_EXCL;
#endif
#ifdef _O_BINARY
flags |= _O_BINARY;
#endif
#ifdef O_BINARY
flags |= O_BINARY;
#endif
s->fd = open(s->hw->sane.name, flags);
if (s->fd < 0) {
DBG( 1, "sane_start: %s open failed: %s\n",
s->hw->sane.name, strerror(errno));
status = (errno == EACCES) ? SANE_STATUS_ACCESS_DENIED
: SANE_STATUS_INVAL;
return status;
}
}
return status;
}
/*
*
*
*/
static SANE_Status eject ( Epson_Scanner * s) {
SANE_Status status;
u_char params [ 2];
u_char cmd = s->hw->cmd->eject;
DBG(5, "eject()\n");
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
if( SANE_STATUS_GOOD != ( status = open_scanner( s)))
return status;
params[ 0] = cmd;
send( s, params, 1, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) ) {
close_scanner( s);
return status;
}
close_scanner( s);
return status;
}
/*
*
*
*/
#ifdef OBSOLETE
static Epson_Device dummy_dev =
{ { NULL, "Epson", NULL, "flatbed scanner" }
};
#endif
static int num_devices= 0; /* number of EPSON scanners attached to backend */
static Epson_Device *first_dev = NULL; /* first EPSON scanner in list */
static Epson_Scanner *first_handle = NULL;
static EpsonHdr command ( Epson_Scanner * s, const u_char * cmd,
size_t cmd_size, SANE_Status * status)
{
EpsonHdr head;
u_char * buf;
if( NULL == ( head = walloc( EpsonHdrRec))) {
*status = SANE_STATUS_NO_MEM;
return ( EpsonHdr) 0;
}
send( s, cmd, cmd_size, status);
if( SANE_STATUS_GOOD != *status)
return ( EpsonHdr) 0;
buf = ( u_char *) head;
if( s->hw->connection == SANE_EPSON_SCSI)
{
receive( s, buf, 4, status);
buf += 4;
}
else if (s->hw->connection == SANE_EPSON_USB)
{
int bytes_read;
bytes_read = receive( s, buf, 4, status);
buf += bytes_read;
}
else
{
receive( s, buf, 1, status);
buf += 1;
}
if( SANE_STATUS_GOOD != *status)
return ( EpsonHdr) 0;
DBG( 4, "code %02x\n", (int) head->code);
switch (head->code)
{
default:
if( 0 == head->code)
DBG( 1, "Incompatible printer port (probably bi/directional)\n");
else if( cmd[cmd_size - 1] == head->code)
DBG( 1, "Incompatible printer port (probably not bi/directional)\n");
DBG( 2, "Illegal response of scanner for command: %02x\n", head->code);
break;
case NAK:
/* fall through */
case ACK:
break; /* no need to read any more data after ACK or NAK */
case STX:
if( s->hw->connection == SANE_EPSON_SCSI)
{
/* nope */
}
else if (s->hw->connection == SANE_EPSON_USB)
{
/* we've already read the complete data */
}
else
{
receive (s, buf, 3, status);
/* buf += 3; */
}
if( SANE_STATUS_GOOD != *status)
return (EpsonHdr) 0;
DBG( 4, "status %02x\n", (int) head->status);
DBG( 4, "count %d\n", (int) head->count);
if( NULL == (head = realloc (head, sizeof (EpsonHdrRec) + head->count)))
{
*status = SANE_STATUS_NO_MEM;
return (EpsonHdr) 0;
}
buf = head->buf;
receive (s, buf, head->count, status);
if( SANE_STATUS_GOOD != *status)
return (EpsonHdr) 0;
break;
}
return head;
}
/*
* static SANE_Status attach()
*
* Attach one device with name *dev_name to the backend.
*/
static SANE_Status attach ( const char * dev_name, Epson_Device * * devp) {
SANE_Status status;
Epson_Scanner * s = walloca( Epson_Scanner);
char * str;
struct Epson_Device * dev;
DBG(1, "%s\n", SANE_EPSON_VERSION);
DBG(5, "attach(%s)\n", dev_name);
for (dev = first_dev; dev; dev = dev->next)
{
if (strcmp(dev->sane.name, dev_name) == 0)
{
if (devp)
{
*devp = dev;
}
return SANE_STATUS_GOOD;
}
}
dev = malloc(sizeof(*dev));
if (!dev)
{
return SANE_STATUS_NO_MEM;
}
/*
* set dummy values.
*/
s->hw = dev;
s->hw->sane.name = NULL;
s->hw->sane.type = "flatbed scanner";
s->hw->sane.vendor = "Epson";
s->hw->sane.model = NULL;
s->hw->optical_res = 0; /* just to have it initialized */
s->hw->color_shuffle = SANE_FALSE;
s->hw->extension = SANE_FALSE;
s->hw->use_extension = SANE_FALSE;
s->hw->cmd = &epson_cmd[EPSON_LEVEL_DEFAULT]; /* use default function level */
s->hw->connection = SANE_EPSON_NODEV; /* no device configured yet */
DBG( 3, "attach: opening %s\n", dev_name);
s->hw->last_res = s->hw->last_res_preview = 0; /* set resolution to safe values */
/*
* decide if interface is USB, SCSI or parallel.
*/
/*
* if the config file contains a line "usb /dev/usbscanner", then handle this
* here and use the USB device from now on.
*/
if (strncmp(dev_name, SANE_EPSON_CONFIG_USB, strlen(SANE_EPSON_CONFIG_USB)) == 0) {
/* we have a match for the USB string and adjust the device name */
dev_name += strlen(SANE_EPSON_CONFIG_USB);
dev_name = sanei_config_skip_whitespace(dev_name);
s->hw->connection = SANE_EPSON_USB;
}
/*
* if the config file contains a line "pio 0xXXX", then handle this case here
* and use the PIO (parallel interface) device from now on.
*/
else if (strncmp(dev_name, SANE_EPSON_CONFIG_PIO, strlen(SANE_EPSON_CONFIG_PIO)) == 0) {
/* we have a match for the PIO string and adjust the device name */
dev_name += strlen(SANE_EPSON_CONFIG_PIO);
dev_name = sanei_config_skip_whitespace(dev_name);
s->hw->connection = SANE_EPSON_PIO;
}
else { /* legacy mode */
char * end;
strtol( dev_name, &end, 0);
if( ( end == dev_name) || *end) {
s->hw->connection = SANE_EPSON_SCSI;
} else {
s->hw->connection = SANE_EPSON_PIO;
}
}
if (s->hw->connection == SANE_EPSON_NODEV)
{
/*
With the current code this can neve happen, because
the routine to handle the legacy mode will always
return a SCSI device. If this gets changed however,
here is the test to return with an error code.
*/
return SANE_STATUS_INVAL;
}
/*
* if interface is SCSI do an inquiry.
*/
if( s->hw->connection == SANE_EPSON_SCSI) {
#define INQUIRY_BUF_SIZE 36
u_char buf[ INQUIRY_BUF_SIZE + 1];
size_t buf_size = INQUIRY_BUF_SIZE;
if( SANE_STATUS_GOOD != ( status = sanei_scsi_open( dev_name, &s->fd, NULL, NULL))) {
DBG( 1, "attach: open failed: %s\n", sane_strstatus( status));
return status;
}
DBG( 3, "attach: sending INQUIRY\n");
/* buf_size = sizeof buf; */
if( SANE_STATUS_GOOD != ( status = inquiry( s->fd, 0, buf, &buf_size))) {
DBG( 1, "attach: inquiry failed: %s\n", sane_strstatus( status));
close_scanner( s);
return status;
}
buf[ INQUIRY_BUF_SIZE] = 0;
DBG( 1, ">%s<\n", buf + 8);
/*
* For USB and PIO scanners this will be done later, once
* we have communication established with the device.
*/
if( buf[ 0] != TYPE_PROCESSOR
|| strncmp( buf + 8, "EPSON", 5) != 0
|| (strncmp( buf + 16, "SCANNER ", 8) != 0
&& strncmp( buf + 14, "SCANNER ", 8) != 0
&& strncmp( buf + 14, "Perfection", 10) != 0
&& strncmp( buf + 16, "Perfection", 10) != 0
&& strncmp( buf + 16, "Expression", 10) != 0))
{
DBG( 1, "attach: device doesn't look like an Epson scanner\n");
close_scanner( s);
return SANE_STATUS_INVAL;
}
/*
* else parallel or USB.
*/
} else if (s->hw->connection == SANE_EPSON_PIO) {
if( SANE_STATUS_GOOD != ( status = sanei_pio_open( dev_name, &s->fd))) {
DBG( 1, "dev_open: %s: can't open %s as a parallel-port device\n",
sane_strstatus( status), dev_name);
return status;
}
} else if (s->hw->connection == SANE_EPSON_USB) {
int flags;
#ifdef _O_RDWR
flags = _O_RDWR;
#else
flags = O_RDWR;
#endif
#ifdef _O_EXCL
flags |= _O_EXCL;
#else
flags |= O_EXCL;
#endif
#ifdef _O_BINARY
flags |= _O_BINARY;
#endif
#ifdef O_BINARY
flags |= O_BINARY;
#endif
s->fd = open(dev_name, flags);
if (s->fd < 0) {
DBG( 1, "sane_start: %s open (USB) failed: %s\n",
dev_name, strerror(errno));
status = (errno == EACCES) ? SANE_STATUS_ACCESS_DENIED
: SANE_STATUS_INVAL;
return status;
}
}
/*
* Initialize (ESC @).
*/
/* NOTE: disabled cause of batch use of scanimage with ADF. */
#if 1
{
void * ptr;
u_char param[2];
param[0] = ESC;
param[1] = s->hw->cmd->initialize_scanner;
ptr = command( s, param, 2, &status);
free(ptr);
}
#endif
/*
* Identification Request (ESC I).
*/
if (s->hw->cmd->request_identity != 0)
{
status = get_identity_information(s);
if (status != SANE_STATUS_GOOD)
return status;
} /* request identity */
/*
* Check for "Request Identity 2" command. If this command is available
* get the information from the scanner and store it in dev
*/
if (s->hw->cmd->request_identity2 != 0)
{
get_identity2_information(s);
if (status != SANE_STATUS_GOOD)
return status;
} /* request identity 2 */
/*
* Check for the max. supported color depth.
*/
if (set_data_format(s, 16) == SANE_STATUS_GOOD)
{
s->hw->maxDepth = 16;
mode_list = mode_list_16;
mode_params = mode_params_16;
}
else if (set_data_format(s, 12) == SANE_STATUS_GOOD)
{
s->hw->maxDepth = 12;
mode_list = mode_list_12;
mode_params = mode_params_12;
}
else
{
s->hw->maxDepth = 8;
mode_list = mode_list_8;
mode_params = mode_params_8;
}
DBG( 5, "Max. supported color depth = %d\n", s->hw->maxDepth);
/*
* Check for "request focus position" command. If this command is
* supported, then the scanner does also support the "set focus
* position" command.
*/
if (request_focus_position(s, &s->currentFocusPosition) == SANE_STATUS_GOOD)
{
DBG( 1, "Enabling 'Set Focus' support\n");
s->hw->focusSupport = SANE_TRUE;
s->opt[ OPT_FOCUS].cap &= ~SANE_CAP_INACTIVE;
/* reflect the current focus position in the GUI */
if (s->currentFocusPosition < 0x4C)
{
/* focus on glass */
s->val[OPT_FOCUS].w = 0;
}
else
{
/* focus 2.5mm above glass */
s->val[OPT_FOCUS].w = 1;
}
}
else
{
DBG( 1, "Disabling 'Set Focus' support\n");
s->hw->focusSupport = SANE_FALSE;
s->opt[OPT_FOCUS].cap |= SANE_CAP_INACTIVE;
s->val[OPT_FOCUS].w = 0; /* on glass - just in case */
}
/*
* Set defaults for no extension.
*/
dev->x_range = &dev->fbf_x_range;
dev->y_range = &dev->fbf_y_range;
/*
* Extended status flag request (ESC f).
* this also requests the scanner device name from the the scanner
*/
#if 0
if( SANE_TRUE == dev->extension)
#endif
/*
* because we are also using the device name from this command,
* we have to run this block even if the scanner does not report
* an extension. The extensions are only reported if the ADF or
* the TPU are actually detected.
*/
{
u_char * buf;
u_char params[2];
EpsonHdr head;
SANE_String_Const * source_list_add = source_list;
params[0] = ESC;
params[1] = s->hw->cmd->request_extension_status;
if( NULL == ( head = ( EpsonHdr) command( s, params, 2, &status) ) ) {
DBG( 0, "Extended status flag request failed\n");
return status;
}
buf = &head->buf[ 0];
/*
* FBF
*/
*source_list_add++ = FBF_STR;
/*
* ADF
*/
if( buf[ 1] & EXT_STATUS_IST) {
DBG( 1, "ADF detected\n");
if( buf[ 1] & EXT_STATUS_EN) {
DBG( 1, "ADF is enabled\n");
dev->x_range = &dev->adf_x_range;
dev->y_range = &dev->adf_y_range;
}
dev->adf_x_range.min = 0;
dev->adf_x_range.max = SANE_FIX( ( buf[ 3] << 8 | buf[ 2]) * 25.4 / dev->dpi_range.max);
dev->adf_x_range.quant = 0;
dev->adf_y_range.min = 0;
dev->adf_y_range.max = SANE_FIX( ( buf[ 5] << 8 | buf[ 4]) * 25.4 / dev->dpi_range.max);
dev->adf_y_range.quant = 0;
DBG( 5, "adf tlx %f tly %f brx %f bry %f [mm]\n"
, SANE_UNFIX( dev->adf_x_range.min)
, SANE_UNFIX( dev->adf_y_range.min)
, SANE_UNFIX( dev->adf_x_range.max)
, SANE_UNFIX( dev->adf_y_range.max)
);
*source_list_add++ = ADF_STR;
dev->ADF = SANE_TRUE;
}
/*
* TPU
*/
if( buf[ 6] & EXT_STATUS_IST) {
DBG( 1, "TPU detected\n");
if( buf[ 6] & EXT_STATUS_EN) {
DBG( 1, "TPU is enabled\n");
dev->x_range = &dev->tpu_x_range;
dev->y_range = &dev->tpu_y_range;
}
dev->tpu_x_range.min = 0;
dev->tpu_x_range.max = SANE_FIX( ( buf[ 8] << 8 | buf[ 7]) * 25.4 / dev->dpi_range.max);
dev->tpu_x_range.quant = 0;
dev->tpu_y_range.min = 0;
dev->tpu_y_range.max = SANE_FIX( ( buf[ 10] << 8 | buf[ 9]) * 25.4 / dev->dpi_range.max);
dev->tpu_y_range.quant = 0;
DBG( 5, "tpu tlx %f tly %f brx %f bry %f [mm]\n"
, SANE_UNFIX( dev->tpu_x_range.min)
, SANE_UNFIX( dev->tpu_y_range.min)
, SANE_UNFIX( dev->tpu_x_range.max)
, SANE_UNFIX( dev->tpu_y_range.max)
);
*source_list_add++ = TPU_STR;
dev->TPU = SANE_TRUE;
}
*source_list_add = NULL;
/*
* Get the device name and copy it to dummy_dev.sane.model
* The device name starts at buf[0x1A] and is up to 16 bytes long
* We are overwriting whatever was set previously!
*/
{
#define DEVICE_NAME_LEN (16)
char device_name[DEVICE_NAME_LEN + 1];
char *end_ptr;
int len;
/* make sure that the end of string is marked */
device_name[DEVICE_NAME_LEN] = '\0';
/* copy the string to an area where we can work with it */
memcpy(device_name, buf + 0x1A, DEVICE_NAME_LEN);
end_ptr = strchr(device_name, ' ');
if (end_ptr != NULL)
{
*end_ptr = '\0';
}
len = strlen(device_name);
str = malloc( len + 1);
str[len] = '\0';
/* finally copy the device name to the structure */
dev->sane.model = ( char *) memcpy( str, device_name, len);
}
}
/*
* Set values for quick format "max" entry.
*/
qf_params[ XtNumber( qf_params) - 1].tl_x = dev->x_range->min;
qf_params[ XtNumber( qf_params) - 1].tl_y = dev->y_range->min;
qf_params[ XtNumber( qf_params) - 1].br_x = dev->x_range->max;
qf_params[ XtNumber( qf_params) - 1].br_y = dev->y_range->max;
/*
*
*/
#if 0
{
u_char * buf;
EpsonHdr head;
#define PUSH_BUTTON_STATUS_EN 0x01
if( NULL == ( head = ( EpsonHdr) command( s, "\033!", 2, &status) ) ) {
DBG( 0, "Request the push button status failed\n");
return status;
}
buf = &head->buf[ 0];
if( buf[ 0] & PUSH_BUTTON_STATUS_EN)
DBG( 1, "Push button was pressed\n");
}
#endif
/*
* now we can finally set the device name
*/
str = malloc( strlen( dev_name) + 1);
dev->sane.name = strcpy( str, dev_name);
close_scanner( s);
++num_devices;
dev->next = first_dev;
first_dev = dev;
if (devp)
{
*devp = dev;
}
return SANE_STATUS_GOOD;
}
/*
*
*/
static SANE_Status attach_one ( const char *dev) {
return attach( dev, 0);
}
/*
*
*/
SANE_Status sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize) {
size_t len;
FILE *fp;
authorize = authorize; /* get rid of the compiler warning */
DBG_INIT ();
#if defined PACKAGE && defined VERSION
DBG( 2, "sane_init: " PACKAGE " " VERSION "\n");
#endif
if( version_code != NULL)
*version_code = SANE_VERSION_CODE (V_MAJOR, V_MINOR, 0);
/* default to /dev/scanner instead of insisting on config file */
if( (fp = sanei_config_open (EPSON_CONFIG_FILE)))
{
char line[PATH_MAX];
while (sanei_config_read (line, sizeof (line), fp))
{
DBG( 4, "sane_init, >%s<\n", line);
if( line[0] == '#') /* ignore line comments */
continue;
len = strlen (line);
if( line[len - 1] == '\n')
line[--len] = '\0';
if( !len)
continue; /* ignore empty lines */
DBG( 4, "sane_init, >%s<\n", line);
sanei_config_attach_matching_devices (line, attach_one);
}
fclose (fp);
}
/* read the option section and assign the connection type to the
scanner structure - which we don't have at this time. So I have
to come up with something :-) */
return SANE_STATUS_GOOD;
}
/*
* void sane_exit(void)
*
* Clean up the list of attached scanners.
*/
void sane_exit ( void) {
Epson_Device *dev, *next;
for (dev = first_dev; dev; dev = next)
{
next = dev->next;
free((char *) dev->sane.name);
free((char *) dev->sane.model);
free(dev);
}
}
/*
*
*
*/
SANE_Status sane_get_devices ( const SANE_Device * * * device_list, SANE_Bool local_only)
{
static const SANE_Device **devlist = 0;
Epson_Device *dev;
int i;
DBG(5, "sane_get_devices()\n");
local_only = local_only; /* just to get rid of the compiler warning */
if (devlist)
{
free (devlist);
}
devlist = malloc((num_devices + 1) * sizeof(devlist[0]));
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;
*device_list = devlist;
return SANE_STATUS_GOOD;
}
/*
*
*
*/
static SANE_Status init_options ( Epson_Scanner * s) {
int i;
for( i = 0; i < NUM_OPTIONS; ++i) {
s->opt[ i].size = sizeof( SANE_Word);
s->opt[ i].cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
}
s->opt[ OPT_NUM_OPTS].title = SANE_TITLE_NUM_OPTIONS;
s->opt[ OPT_NUM_OPTS].desc = SANE_DESC_NUM_OPTIONS;
s->opt[ OPT_NUM_OPTS].cap = SANE_CAP_SOFT_DETECT;
s->val[ OPT_NUM_OPTS].w = NUM_OPTIONS;
/* "Scan Mode" group: */
s->opt[ OPT_MODE_GROUP].title = "Scan Mode";
s->opt[ OPT_MODE_GROUP].desc = "";
s->opt[ OPT_MODE_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_MODE_GROUP].cap = 0;
/* scan mode */
s->opt[ OPT_MODE].name = SANE_NAME_SCAN_MODE;
s->opt[ OPT_MODE].title = SANE_TITLE_SCAN_MODE;
s->opt[ OPT_MODE].desc = SANE_DESC_SCAN_MODE;
s->opt[ OPT_MODE].type = SANE_TYPE_STRING;
s->opt[ OPT_MODE].size = max_string_size(mode_list);
s->opt[ OPT_MODE].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_MODE].constraint.string_list = mode_list;
s->val[ OPT_MODE].w = 0; /* Binary */
/* halftone */
s->opt[ OPT_HALFTONE].name = SANE_NAME_HALFTONE;
s->opt[ OPT_HALFTONE].title = SANE_TITLE_HALFTONE;
s->opt[ OPT_HALFTONE].desc = "Selects the halftone.";
s->opt[ OPT_HALFTONE].type = SANE_TYPE_STRING;
s->opt[ OPT_HALFTONE].size = max_string_size(halftone_list_7);
s->opt[ OPT_HALFTONE].constraint_type = SANE_CONSTRAINT_STRING_LIST;
if( s->hw->level >= 7)
s->opt[ OPT_HALFTONE].constraint.string_list = halftone_list_7;
else if( s->hw->level >= 4)
s->opt[ OPT_HALFTONE].constraint.string_list = halftone_list_4;
else
s->opt[ OPT_HALFTONE].constraint.string_list = halftone_list;
s->val[ OPT_HALFTONE].w = 1; /* Halftone A */
if( ! s->hw->cmd->set_halftoning) {
s->opt[ OPT_HALFTONE].cap |= SANE_CAP_INACTIVE;
}
/* dropout */
s->opt[ OPT_DROPOUT].name = "dropout";
s->opt[ OPT_DROPOUT].title = "Dropout";
s->opt[ OPT_DROPOUT].desc = "Selects the dropout.";
s->opt[ OPT_DROPOUT].type = SANE_TYPE_STRING;
s->opt[ OPT_DROPOUT].size = max_string_size(dropout_list);
s->opt[ OPT_DROPOUT].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_DROPOUT].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_DROPOUT].constraint.string_list = dropout_list;
s->val[ OPT_DROPOUT].w = 0; /* None */
/* brightness */
s->opt[ OPT_BRIGHTNESS].name = SANE_NAME_BRIGHTNESS;
s->opt[ OPT_BRIGHTNESS].title = SANE_TITLE_BRIGHTNESS;
s->opt[ OPT_BRIGHTNESS].desc = "Selects the brightness.";
s->opt[ OPT_BRIGHTNESS].type = SANE_TYPE_INT;
s->opt[ OPT_BRIGHTNESS].unit = SANE_UNIT_NONE;
s->opt[ OPT_BRIGHTNESS].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_BRIGHTNESS].constraint.range = &s->hw->cmd->bright_range;
s->val[ OPT_BRIGHTNESS].w = 0; /* Normal */
if( ! s->hw->cmd->set_bright) {
s->opt[ OPT_BRIGHTNESS].cap |= SANE_CAP_INACTIVE;
}
/* sharpness */
s->opt[ OPT_SHARPNESS].name = "sharpness";
s->opt[ OPT_SHARPNESS].title = "Sharpness";
s->opt[ OPT_SHARPNESS].desc = "";
s->opt[ OPT_SHARPNESS].type = SANE_TYPE_INT;
s->opt[ OPT_SHARPNESS].unit = SANE_UNIT_NONE;
s->opt[ OPT_SHARPNESS].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_SHARPNESS].constraint.range = &outline_emphasis_range;
s->val[ OPT_SHARPNESS].w = 0; /* Normal */
if( ! s->hw->cmd->set_outline_emphasis) {
s->opt[ OPT_SHARPNESS].cap |= SANE_CAP_INACTIVE;
}
/* gamma */
s->opt[ OPT_GAMMA_CORRECTION].name = SANE_NAME_GAMMA_CORRECTION;
s->opt[ OPT_GAMMA_CORRECTION].title = SANE_TITLE_GAMMA_CORRECTION;
s->opt[ OPT_GAMMA_CORRECTION].desc = SANE_DESC_GAMMA_CORRECTION;
s->opt[ OPT_GAMMA_CORRECTION].type = SANE_TYPE_STRING;
s->opt[ OPT_GAMMA_CORRECTION].constraint_type = SANE_CONSTRAINT_STRING_LIST;
/*
* special handling for D1 function level - at this time I'm not
* testing for D1, I'm just assuming that all D level scanners will
* behave the same way. This has to be confirmed with the next D-level
* scanner
*/
if (s->hw->cmd->level[0] == 'D')
{
s->opt[ OPT_GAMMA_CORRECTION].size = max_string_size(gamma_list_d);
s->opt[ OPT_GAMMA_CORRECTION].constraint.string_list = gamma_list_d;
s->val[ OPT_GAMMA_CORRECTION].w = 1; /* Default */
gamma_userdefined = gamma_userdefined_d;
gamma_params = gamma_params_d;
}
else
{
s->opt[ OPT_GAMMA_CORRECTION].size = max_string_size(gamma_list_ab);
s->opt[ OPT_GAMMA_CORRECTION].constraint.string_list = gamma_list_ab;
s->val[ OPT_GAMMA_CORRECTION].w = 0; /* Default */
gamma_userdefined = gamma_userdefined_ab;
gamma_params = gamma_params_ab;
}
if( ! s->hw->cmd->set_gamma) {
s->opt[ OPT_GAMMA_CORRECTION].cap |= SANE_CAP_INACTIVE;
}
/* gamma vector */
s->opt[ OPT_GAMMA_VECTOR].name = SANE_NAME_GAMMA_VECTOR;
s->opt[ OPT_GAMMA_VECTOR].title = SANE_TITLE_GAMMA_VECTOR;
s->opt[ OPT_GAMMA_VECTOR].desc = SANE_DESC_GAMMA_VECTOR;
s->opt[ OPT_GAMMA_VECTOR].type = SANE_TYPE_INT;
s->opt[ OPT_GAMMA_VECTOR].unit = SANE_UNIT_NONE;
s->opt[ OPT_GAMMA_VECTOR].size = 256 * sizeof (SANE_Word);
s->opt[ OPT_GAMMA_VECTOR].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_GAMMA_VECTOR].constraint.range = &u8_range;
s->val[ OPT_GAMMA_VECTOR].wa = &s->gamma_table [ 0] [ 0];
/* red gamma vector */
s->opt[ OPT_GAMMA_VECTOR_R].name = SANE_NAME_GAMMA_VECTOR_R;
s->opt[ OPT_GAMMA_VECTOR_R].title = SANE_TITLE_GAMMA_VECTOR_R;
s->opt[ OPT_GAMMA_VECTOR_R].desc = SANE_DESC_GAMMA_VECTOR_R;
s->opt[ OPT_GAMMA_VECTOR_R].type = SANE_TYPE_INT;
s->opt[ OPT_GAMMA_VECTOR_R].unit = SANE_UNIT_NONE;
s->opt[ OPT_GAMMA_VECTOR_R].size = 256 * sizeof (SANE_Word);
s->opt[ OPT_GAMMA_VECTOR_R].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_GAMMA_VECTOR_R].constraint.range = &u8_range;
s->val[ OPT_GAMMA_VECTOR_R].wa = &s->gamma_table [ 1] [ 0];
/* green gamma vector */
s->opt[ OPT_GAMMA_VECTOR_G].name = SANE_NAME_GAMMA_VECTOR_G;
s->opt[ OPT_GAMMA_VECTOR_G].title = SANE_TITLE_GAMMA_VECTOR_G;
s->opt[ OPT_GAMMA_VECTOR_G].desc = SANE_DESC_GAMMA_VECTOR_G;
s->opt[ OPT_GAMMA_VECTOR_G].type = SANE_TYPE_INT;
s->opt[ OPT_GAMMA_VECTOR_G].unit = SANE_UNIT_NONE;
s->opt[ OPT_GAMMA_VECTOR_G].size = 256 * sizeof (SANE_Word);
s->opt[ OPT_GAMMA_VECTOR_G].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_GAMMA_VECTOR_G].constraint.range = &u8_range;
s->val[ OPT_GAMMA_VECTOR_G].wa = &s->gamma_table [ 2] [ 0];
/* red gamma vector */
s->opt[ OPT_GAMMA_VECTOR_B].name = SANE_NAME_GAMMA_VECTOR_B;
s->opt[ OPT_GAMMA_VECTOR_B].title = SANE_TITLE_GAMMA_VECTOR_B;
s->opt[ OPT_GAMMA_VECTOR_B].desc = SANE_DESC_GAMMA_VECTOR_B;
s->opt[ OPT_GAMMA_VECTOR_B].type = SANE_TYPE_INT;
s->opt[ OPT_GAMMA_VECTOR_B].unit = SANE_UNIT_NONE;
s->opt[ OPT_GAMMA_VECTOR_B].size = 256 * sizeof (SANE_Word);
s->opt[ OPT_GAMMA_VECTOR_B].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_GAMMA_VECTOR_B].constraint.range = &u8_range;
s->val[ OPT_GAMMA_VECTOR_B].wa = &s->gamma_table [ 3] [ 0];
if (gamma_userdefined[s->val[ OPT_GAMMA_CORRECTION].w] == SANE_TRUE )
{
s->opt[ OPT_GAMMA_VECTOR].cap &= ~SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_R].cap &= ~SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_G].cap &= ~SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_B].cap &= ~SANE_CAP_INACTIVE;
} else {
s->opt[ OPT_GAMMA_VECTOR].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_R].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_G].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_GAMMA_VECTOR_B].cap |= SANE_CAP_INACTIVE;
}
/* initialize the Gamma tables */
memset(&s->gamma_table[0], 0, 256 * sizeof(SANE_Word));
memset(&s->gamma_table[1], 0, 256 * sizeof(SANE_Word));
memset(&s->gamma_table[2], 0, 256 * sizeof(SANE_Word));
memset(&s->gamma_table[3], 0, 256 * sizeof(SANE_Word));
for (i = 0 ; i < 256 ; i++)
{
s->gamma_table[0][i] = i;
s->gamma_table[1][i] = i;
s->gamma_table[2][i] = i;
s->gamma_table[3][i] = i;
}
/* color correction */
s->opt[ OPT_COLOR_CORRECTION].name = "color-correction";
s->opt[ OPT_COLOR_CORRECTION].title = "Color correction";
s->opt[ OPT_COLOR_CORRECTION].desc = "Sets the color correction table for the selected output device.";
s->opt[ OPT_COLOR_CORRECTION].type = SANE_TYPE_STRING;
s->opt[ OPT_COLOR_CORRECTION].size = 32;
s->opt[ OPT_COLOR_CORRECTION].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_COLOR_CORRECTION].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_COLOR_CORRECTION].constraint.string_list = color_list;
s->val[ OPT_COLOR_CORRECTION].w = 5; /* scanner default: CRT monitors */
if( ! s->hw->cmd->set_color_correction) {
s->opt[ OPT_COLOR_CORRECTION].cap |= SANE_CAP_INACTIVE;
}
/* resolution */
s->opt[ OPT_RESOLUTION].name = SANE_NAME_SCAN_RESOLUTION;
s->opt[ OPT_RESOLUTION].title = SANE_TITLE_SCAN_RESOLUTION;
s->opt[ OPT_RESOLUTION].desc = SANE_DESC_SCAN_RESOLUTION;
s->opt[ OPT_RESOLUTION].type = SANE_TYPE_INT;
s->opt[ OPT_RESOLUTION].unit = SANE_UNIT_DPI;
s->opt[ OPT_RESOLUTION].constraint_type = SANE_CONSTRAINT_WORD_LIST;
s->opt[ OPT_RESOLUTION].constraint.word_list = s->hw->resolution_list;
s->val[ OPT_RESOLUTION].w = s->hw->dpi_range.min;
/* threshold */
s->opt[ OPT_THRESHOLD].name = SANE_NAME_THRESHOLD;
s->opt[ OPT_THRESHOLD].title = SANE_TITLE_THRESHOLD;
s->opt[ OPT_THRESHOLD].desc = SANE_DESC_THRESHOLD;
s->opt[ OPT_THRESHOLD].type = SANE_TYPE_INT;
s->opt[ OPT_THRESHOLD].unit = SANE_UNIT_NONE;
s->opt[ OPT_THRESHOLD].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_THRESHOLD].constraint.range = &u8_range;
s->val[ OPT_THRESHOLD].w = 0x80;
if( ! s->hw->cmd->set_threshold) {
s->opt[ OPT_THRESHOLD].cap |= SANE_CAP_INACTIVE;
}
s->opt[ OPT_CCT_GROUP].title = "Color correction coefficients";
s->opt[ OPT_CCT_GROUP].desc = "Matrix multiplication of RGB";
s->opt[ OPT_CCT_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_CCT_GROUP].cap = SANE_CAP_ADVANCED;
/* color correction coefficients */
s->opt[ OPT_CCT_1].name = "cct-1";
s->opt[ OPT_CCT_2].name = "cct-2";
s->opt[ OPT_CCT_3].name = "cct-3";
s->opt[ OPT_CCT_4].name = "cct-4";
s->opt[ OPT_CCT_5].name = "cct-5";
s->opt[ OPT_CCT_6].name = "cct-6";
s->opt[ OPT_CCT_7].name = "cct-7";
s->opt[ OPT_CCT_8].name = "cct-8";
s->opt[ OPT_CCT_9].name = "cct-9";
s->opt[ OPT_CCT_1].title = "Green";
s->opt[ OPT_CCT_2].title = "Shift green to red";
s->opt[ OPT_CCT_3].title = "Shift green to blue";
s->opt[ OPT_CCT_4].title = "Shift red to green";
s->opt[ OPT_CCT_5].title = "Red";
s->opt[ OPT_CCT_6].title = "Shift red to blue";
s->opt[ OPT_CCT_7].title = "Shift blue to green";
s->opt[ OPT_CCT_8].title = "Shift blue to red";
s->opt[ OPT_CCT_9].title = "Blue";
s->opt[ OPT_CCT_1].desc = "Controls green level";
s->opt[ OPT_CCT_2].desc = "Adds to red based on green level";
s->opt[ OPT_CCT_3].desc = "Adds to blue based on green level";
s->opt[ OPT_CCT_4].desc = "Adds to green based on red level";
s->opt[ OPT_CCT_5].desc = "Controls red level";
s->opt[ OPT_CCT_6].desc = "Adds to blue based on red level";
s->opt[ OPT_CCT_7].desc = "Adds to green based on blue level";
s->opt[ OPT_CCT_8].desc = "Adds to red based on blue level";
s->opt[ OPT_CCT_9].desc = "Control blue level";
s->opt[ OPT_CCT_1].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_2].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_3].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_4].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_5].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_6].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_7].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_8].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_9].type = SANE_TYPE_INT;
s->opt[ OPT_CCT_1].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_2].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_3].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_4].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_5].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_6].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_7].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_8].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_9].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_CCT_1].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_2].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_3].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_4].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_5].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_6].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_7].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_8].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_9].unit = SANE_UNIT_NONE;
s->opt[ OPT_CCT_1].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_2].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_3].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_4].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_5].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_6].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_7].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_8].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_9].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_CCT_1].constraint.range = &s8_range;
s->opt[ OPT_CCT_2].constraint.range = &s8_range;
s->opt[ OPT_CCT_3].constraint.range = &s8_range;
s->opt[ OPT_CCT_4].constraint.range = &s8_range;
s->opt[ OPT_CCT_5].constraint.range = &s8_range;
s->opt[ OPT_CCT_6].constraint.range = &s8_range;
s->opt[ OPT_CCT_7].constraint.range = &s8_range;
s->opt[ OPT_CCT_8].constraint.range = &s8_range;
s->opt[ OPT_CCT_9].constraint.range = &s8_range;
s->val[ OPT_CCT_1].w = 32;
s->val[ OPT_CCT_2].w = 0;
s->val[ OPT_CCT_3].w = 0;
s->val[ OPT_CCT_4].w = 0;
s->val[ OPT_CCT_5].w = 32;
s->val[ OPT_CCT_6].w = 0;
s->val[ OPT_CCT_7].w = 0;
s->val[ OPT_CCT_8].w = 0;
s->val[ OPT_CCT_9].w = 32;
if( ! s->hw->cmd->set_color_correction_coefficients)
{
s->opt[ OPT_CCT_1].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_2].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_3].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_4].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_5].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_6].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_7].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_8].cap |= SANE_CAP_INACTIVE;
s->opt[ OPT_CCT_9].cap |= SANE_CAP_INACTIVE;
}
/* "Advanced" group: */
s->opt[ OPT_ADVANCED_GROUP].title = "Advanced";
s->opt[ OPT_ADVANCED_GROUP].desc = "";
s->opt[ OPT_ADVANCED_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_ADVANCED_GROUP].cap = SANE_CAP_ADVANCED;
/* mirror */
s->opt[ OPT_MIRROR].name = "mirror";
s->opt[ OPT_MIRROR].title = "Mirror image";
s->opt[ OPT_MIRROR].desc = "Mirror the image.";
s->opt[ OPT_MIRROR].type = SANE_TYPE_BOOL;
s->val[ OPT_MIRROR].w = SANE_FALSE;
if( ! s->hw->cmd->mirror_image) {
s->opt[ OPT_MIRROR].cap |= SANE_CAP_INACTIVE;
}
/* speed */
s->opt[ OPT_SPEED].name = "speed";
s->opt[ OPT_SPEED].title = "Speed";
s->opt[ OPT_SPEED].desc = "";
s->opt[ OPT_SPEED].type = SANE_TYPE_BOOL;
s->val[ OPT_SPEED].w = SANE_FALSE;
if( ! s->hw->cmd->set_speed) {
s->opt[ OPT_SPEED].cap |= SANE_CAP_INACTIVE;
}
/* preview speed */
s->opt[ OPT_PREVIEW_SPEED].name = "preview-speed";
s->opt[ OPT_PREVIEW_SPEED].title = "Speed";
s->opt[ OPT_PREVIEW_SPEED].desc = "";
s->opt[ OPT_PREVIEW_SPEED].type = SANE_TYPE_BOOL;
s->val[ OPT_PREVIEW_SPEED].w = SANE_FALSE;
if( ! s->hw->cmd->set_speed) {
s->opt[ OPT_PREVIEW_SPEED].cap |= SANE_CAP_INACTIVE;
}
/* auto area segmentation */
s->opt[ OPT_AAS].name = "auto-area-segmentation";
s->opt[ OPT_AAS].title = "Auto area segmentation";
s->opt[ OPT_AAS].desc = "";
s->opt[ OPT_AAS].type = SANE_TYPE_BOOL;
s->val[ OPT_AAS].w = SANE_TRUE;
if( ! s->hw->cmd->control_auto_area_segmentation) {
s->opt[ OPT_AAS].cap |= SANE_CAP_INACTIVE;
}
/* zoom */
s->opt[ OPT_ZOOM].name = "zoom";
s->opt[ OPT_ZOOM].title = "Zoom";
s->opt[ OPT_ZOOM].desc = "Defines the zoom factor the scanner will use";
s->opt[ OPT_ZOOM].type = SANE_TYPE_INT;
s->opt[ OPT_ZOOM].unit = SANE_UNIT_NONE;
s->opt[ OPT_ZOOM].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_ZOOM].constraint.range = &zoom_range;
s->val[ OPT_ZOOM].w = 100;
/* if( ! s->hw->cmd->set_zoom) */ {
s->opt[ OPT_ZOOM].cap |= SANE_CAP_INACTIVE;
}
/* "Preview settings" group: */
s->opt[ OPT_PREVIEW_GROUP].title = SANE_TITLE_PREVIEW;
s->opt[ OPT_PREVIEW_GROUP].desc = "";
s->opt[ OPT_PREVIEW_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_PREVIEW_GROUP].cap = SANE_CAP_ADVANCED;
/* preview */
s->opt[ OPT_PREVIEW].name = SANE_NAME_PREVIEW;
s->opt[ OPT_PREVIEW].title = SANE_TITLE_PREVIEW;
s->opt[ OPT_PREVIEW].desc = SANE_DESC_PREVIEW;
s->opt[ OPT_PREVIEW].type = SANE_TYPE_BOOL;
s->val[ OPT_PREVIEW].w = SANE_FALSE;
/* "Geometry" group: */
s->opt[ OPT_GEOMETRY_GROUP].title = "Geometry";
s->opt[ OPT_GEOMETRY_GROUP].desc = "";
s->opt[ OPT_GEOMETRY_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_GEOMETRY_GROUP].cap = SANE_CAP_ADVANCED;
/* top-left x */
s->opt[ OPT_TL_X].name = SANE_NAME_SCAN_TL_X;
s->opt[ OPT_TL_X].title = SANE_TITLE_SCAN_TL_X;
s->opt[ OPT_TL_X].desc = SANE_DESC_SCAN_TL_X;
s->opt[ OPT_TL_X].type = SANE_TYPE_FIXED;
s->opt[ OPT_TL_X].unit = SANE_UNIT_MM;
s->opt[ OPT_TL_X].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_TL_X].constraint.range = s->hw->x_range;
s->val[ OPT_TL_X].w = 0;
/* top-left y */
s->opt[ OPT_TL_Y].name = SANE_NAME_SCAN_TL_Y;
s->opt[ OPT_TL_Y].title = SANE_TITLE_SCAN_TL_Y;
s->opt[ OPT_TL_Y].desc = SANE_DESC_SCAN_TL_Y;
s->opt[ OPT_TL_Y].type = SANE_TYPE_FIXED;
s->opt[ OPT_TL_Y].unit = SANE_UNIT_MM;
s->opt[ OPT_TL_Y].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_TL_Y].constraint.range = s->hw->y_range;
s->val[ OPT_TL_Y].w = 0;
/* bottom-right x */
s->opt[ OPT_BR_X].name = SANE_NAME_SCAN_BR_X;
s->opt[ OPT_BR_X].title = SANE_TITLE_SCAN_BR_X;
s->opt[ OPT_BR_X].desc = SANE_DESC_SCAN_BR_X;
s->opt[ OPT_BR_X].type = SANE_TYPE_FIXED;
s->opt[ OPT_BR_X].unit = SANE_UNIT_MM;
s->opt[ OPT_BR_X].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_BR_X].constraint.range = s->hw->x_range;
s->val[ OPT_BR_X].w = s->hw->x_range->max;
/* bottom-right y */
s->opt[ OPT_BR_Y].name = SANE_NAME_SCAN_BR_Y;
s->opt[ OPT_BR_Y].title = SANE_TITLE_SCAN_BR_Y;
s->opt[ OPT_BR_Y].desc = SANE_DESC_SCAN_BR_Y;
s->opt[ OPT_BR_Y].type = SANE_TYPE_FIXED;
s->opt[ OPT_BR_Y].unit = SANE_UNIT_MM;
s->opt[ OPT_BR_Y].constraint_type = SANE_CONSTRAINT_RANGE;
s->opt[ OPT_BR_Y].constraint.range = s->hw->y_range;
s->val[ OPT_BR_Y].w = s->hw->y_range->max;
/* Quick format */
s->opt[ OPT_QUICK_FORMAT].name = "quick-format";
s->opt[ OPT_QUICK_FORMAT].title = "Quick format";
s->opt[ OPT_QUICK_FORMAT].desc = "";
s->opt[ OPT_QUICK_FORMAT].type = SANE_TYPE_STRING;
s->opt[ OPT_QUICK_FORMAT].size = max_string_size(qf_list);
s->opt[ OPT_QUICK_FORMAT].cap |= SANE_CAP_ADVANCED;
s->opt[ OPT_QUICK_FORMAT].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_QUICK_FORMAT].constraint.string_list = qf_list;
s->val[ OPT_QUICK_FORMAT].w = XtNumber( qf_params) - 1; /* max */
/* "Optional equipment" group: */
s->opt[ OPT_EQU_GROUP].title = "Optional equipment";
s->opt[ OPT_EQU_GROUP].desc = "";
s->opt[ OPT_EQU_GROUP].type = SANE_TYPE_GROUP;
s->opt[ OPT_EQU_GROUP].cap = SANE_CAP_ADVANCED;
/* source */
s->opt[ OPT_SOURCE].name = SANE_NAME_SCAN_SOURCE;
s->opt[ OPT_SOURCE].title = SANE_TITLE_SCAN_SOURCE;
s->opt[ OPT_SOURCE].desc = SANE_DESC_SCAN_SOURCE;
s->opt[ OPT_SOURCE].type = SANE_TYPE_STRING;
s->opt[ OPT_SOURCE].size = max_string_size(source_list);
s->opt[ OPT_SOURCE].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_SOURCE].constraint.string_list = source_list;
if( ! s->hw->extension) {
s->opt[ OPT_SOURCE].cap |= SANE_CAP_INACTIVE;
}
s->val[ OPT_SOURCE].w = 0; /* always use Flatbed as default */
/* film type */
s->opt[ OPT_FILM_TYPE].name = "film-type";
s->opt[ OPT_FILM_TYPE].title = "Film type";
s->opt[ OPT_FILM_TYPE].desc = "";
s->opt[ OPT_FILM_TYPE].type = SANE_TYPE_STRING;
s->opt[ OPT_FILM_TYPE].size = max_string_size(film_list);
s->opt[ OPT_FILM_TYPE].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_FILM_TYPE].constraint.string_list = film_list;
s->val[ OPT_FILM_TYPE].w = 0;
s->opt[ OPT_FILM_TYPE].cap |= SANE_CAP_INACTIVE; /* default is inactive */
/* focus position */
s->opt[ OPT_FOCUS].name = SANE_EPSON_FOCUS_NAME;
s->opt[ OPT_FOCUS].title = SANE_EPSON_FOCUS_TITLE;
s->opt[ OPT_FOCUS].desc = SANE_EPSON_FOCUS_DESC;
s->opt[ OPT_FOCUS].type = SANE_TYPE_STRING;
s->opt[ OPT_FOCUS].size = max_string_size(focus_list);
s->opt[ OPT_FOCUS].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_FOCUS].constraint.string_list = focus_list;
s->val[ OPT_FOCUS].w = 0;
s->opt[ OPT_FOCUS].cap |= SANE_CAP_ADVANCED;
if (s->hw->focusSupport == SANE_TRUE)
{
s->opt[ OPT_FOCUS].cap &= ~SANE_CAP_INACTIVE;
}
else
{
s->opt[ OPT_FOCUS].cap |= SANE_CAP_INACTIVE;
}
#if 0
if( ( ! s->hw->TPU) && ( ! s->hw->cmd->set_bay) ) { /* Hack: Using set_bay to indicate. */
s->opt[ OPT_FILM_TYPE].cap |= SANE_CAP_INACTIVE;
}
#endif
/* forward feed / eject */
s->opt[ OPT_EJECT].name = "eject";
s->opt[ OPT_EJECT].title = "Eject";
s->opt[ OPT_EJECT].desc = "Eject the sheet in the ADF";
s->opt[ OPT_EJECT].type = SANE_TYPE_BUTTON;
if( ( ! s->hw->ADF) && ( ! s->hw->cmd->set_bay) ) { /* Hack: Using set_bay to indicate. */
s->opt[ OPT_EJECT].cap |= SANE_CAP_INACTIVE;
}
/* auto forward feed / eject */
s->opt[ OPT_AUTO_EJECT].name = "auto-eject";
s->opt[ OPT_AUTO_EJECT].title = "Auto eject";
s->opt[ OPT_AUTO_EJECT].desc = "Eject document after scanning";
s->opt[ OPT_AUTO_EJECT].type = SANE_TYPE_BOOL;
s->val[ OPT_AUTO_EJECT].w = SANE_FALSE;
if( ! s->hw->ADF) { /* Hack: Using set_bay to indicate. */
s->opt[ OPT_AUTO_EJECT].cap |= SANE_CAP_INACTIVE;
}
/* select bay */
s->opt[ OPT_BAY].name = "bay";
s->opt[ OPT_BAY].title = "Bay";
s->opt[ OPT_BAY].desc = "select bay to scan";
s->opt[ OPT_BAY].type = SANE_TYPE_STRING;
s->opt[ OPT_BAY].size = max_string_size(bay_list);
s->opt[ OPT_BAY].constraint_type = SANE_CONSTRAINT_STRING_LIST;
s->opt[ OPT_BAY].constraint.string_list = bay_list;
s->val[ OPT_BAY].w = 0; /* Bay 1 */
if( ! s->hw->cmd->set_bay) {
s->opt[ OPT_BAY].cap |= SANE_CAP_INACTIVE;
}
#if 0
/* button test */
s->opt[ OPT_FORMAT_MAX].name = "max.";
s->opt[ OPT_FORMAT_MAX].title = "max.";
s->opt[ OPT_FORMAT_MAX].desc = "";
s->opt[ OPT_FORMAT_MAX].type = SANE_TYPE_BUTTON;
s->opt[ OPT_FORMAT_MAX].unit = SANE_UNIT_NONE;
s->opt[ OPT_FORMAT_MAX].constraint_type = SANE_CONSTRAINT_NONE;
s->opt[ OPT_FORMAT_MAX].constraint.range = NULL;
#endif
return SANE_STATUS_GOOD;
}
/*
*
*
*/
SANE_Status sane_open ( SANE_String_Const devicename, SANE_Handle * handle)
{
Epson_Device *dev;
Epson_Scanner *s;
SANE_Status status;
DBG(5, "sane_open(%s)\n", devicename);
/* search for device */
if (devicename[0])
{
for (dev = first_dev; dev; dev = dev->next)
{
if (strcmp(dev->sane.name, devicename) == 0)
{
break;
}
}
if (!dev)
{
status = attach(devicename, &dev);
if (status != SANE_STATUS_GOOD)
{
return status;
}
}
}
else
{
dev = first_dev;
}
if (!dev)
{
return SANE_STATUS_INVAL;
}
s = calloc(sizeof( Epson_Scanner), 1);
if (!s)
{
return SANE_STATUS_NO_MEM;
}
s->fd = -1;
s->hw = dev;
init_options( s);
/* insert newly opened handle into list of open handles */
s->next = first_handle;
first_handle = s;
*handle = ( SANE_Handle) s;
return SANE_STATUS_GOOD;
}
/*
*
*
*/
void sane_close ( SANE_Handle handle)
{
Epson_Scanner *s, *prev;
/*
* Test if there is still data pending from
* the scanner. If so, then do a cancel
*/
s = (Epson_Scanner *) handle;
/*
* If the s->ptr pointer is not NULL, then a scan operation
* was started and if s->eof is FALSE, it was not finished.
*/
if (!s->eof && s->ptr != NULL)
{
u_char * dummy;
int len;
SANE_Status status;
/* malloc one line */
dummy = malloc (s->params.bytes_per_line);
if (dummy == NULL)
{
DBG (0, "Out of memory\n");
return;
}
else
{
/* there is still data to read from the scanner */
s->canceling = SANE_TRUE;
status = sane_read(s, dummy, s->params.bytes_per_line, &len);
while (!s->eof &&
SANE_STATUS_CANCELLED != sane_read(s, dummy, s->params.bytes_per_line, &len))
{
/* empty body, the while condition does the processing */
}
}
}
/* 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 (0x%p)\n", handle);
return;
}
if (prev)
prev->next = s->next;
else
first_handle = s->next;
if (s->fd != -1)
close_scanner(s);
free(s);
}
/*
*
*
*/
const SANE_Option_Descriptor *
sane_get_option_descriptor ( SANE_Handle handle, SANE_Int option)
{
Epson_Scanner *s = (Epson_Scanner *) handle;
if( option < 0 || option >= NUM_OPTIONS)
return NULL;
return( s->opt + option );
}
/*
*
*
*/
static const SANE_String_Const *
search_string_list (const SANE_String_Const * list, SANE_String value)
{
while( *list != NULL && strcmp( value, *list) != 0) {
++list;
}
return( (*list == NULL) ? NULL : list );
}
/*
*
*
*/
/*
Activate, deactivate an option. Subroutines so we can add
debugging info if we want. The change flag is set to TRUE
if we changed an option. If we did not change an option,
then the value of the changed flag is not modified.
*/
static void sane_activate( Epson_Scanner * s, SANE_Int option,
SANE_Bool * change )
{
if (!SANE_OPTION_IS_ACTIVE( s->opt[ option ].cap )) {
s->opt[ option ].cap &= ~SANE_CAP_INACTIVE;
*change = SANE_TRUE;
}
}
static void sane_deactivate( Epson_Scanner * s, SANE_Int option,
SANE_Bool * change )
{
if (SANE_OPTION_IS_ACTIVE(s->opt[ option ].cap)) {
s->opt[ option ].cap |= SANE_CAP_INACTIVE;
*change = SANE_TRUE;
}
}
static void sane_optstate( SANE_Bool state, Epson_Scanner * s,
SANE_Int option, SANE_Bool * change )
{
if (state) {
sane_activate( s, option, change );
} else {
sane_deactivate( s, option, change );
}
}
/**
End of sane_activate, sane_deactivate, sane_optstate.
**/
static SANE_Status getvalue( SANE_Handle handle,
SANE_Int option,
void * value)
{
Epson_Scanner * s = (Epson_Scanner *) handle;
SANE_Option_Descriptor * sopt = &(s->opt[ option ]);
Option_Value * sval = &(s->val[ option ]);
switch (option) {
case OPT_GAMMA_VECTOR:
case OPT_GAMMA_VECTOR_R:
case OPT_GAMMA_VECTOR_G:
case OPT_GAMMA_VECTOR_B:
memcpy( value, sval->wa, sopt->size );
break;
case OPT_NUM_OPTS:
case OPT_RESOLUTION:
case OPT_TL_X:
case OPT_TL_Y:
case OPT_BR_X:
case OPT_BR_Y:
case OPT_MIRROR:
case OPT_SPEED:
case OPT_PREVIEW_SPEED:
case OPT_AAS:
case OPT_PREVIEW:
case OPT_BRIGHTNESS:
case OPT_SHARPNESS:
case OPT_AUTO_EJECT:
case OPT_CCT_1:
case OPT_CCT_2:
case OPT_CCT_3:
case OPT_CCT_4:
case OPT_CCT_5:
case OPT_CCT_6:
case OPT_CCT_7:
case OPT_CCT_8:
case OPT_CCT_9:
case OPT_THRESHOLD:
case OPT_ZOOM:
*((SANE_Word *) value) = sval->w;
break;
case OPT_MODE:
case OPT_HALFTONE:
case OPT_DROPOUT:
case OPT_QUICK_FORMAT:
case OPT_SOURCE:
case OPT_FILM_TYPE:
case OPT_GAMMA_CORRECTION:
case OPT_COLOR_CORRECTION:
case OPT_BAY:
case OPT_FOCUS:
strcpy( (char *) value, sopt->constraint.string_list[sval->w]);
break;
#if 0
case OPT_MODEL:
strcpy( value, sval->s);
break;
#endif
default:
return SANE_STATUS_INVAL;
}
return SANE_STATUS_GOOD;
}
/**
End of getvalue.
**/
static void handle_depth_halftone( Epson_Scanner * s, SANE_Bool * reload)
/*
This routine handles common options between OPT_MODE and
OPT_HALFTONE. These options are TET (a HALFTONE mode), AAS
- auto area segmentation, and threshold. Apparently AAS
is some method to differentiate between text and photos.
Or something like that.
AAS is available when the scan color depth is 1 and the
halftone method is not TET.
Threshold is available when halftone is NONE, and depth is 1.
*/
{
int hti = s->val[ OPT_HALFTONE ].w;
int mdi = s->val[ OPT_MODE ].w;
SANE_Bool aas = SANE_FALSE;
SANE_Bool thresh= SANE_FALSE;
if (!s->hw->cmd->control_auto_area_segmentation) return;
if (mode_params[ mdi ].depth == 1) {
if (halftone_params[ hti ] != HALFTONE_TET) {
aas = SANE_TRUE;
}
if (halftone_params[ hti ] == HALFTONE_NONE) {
thresh = SANE_TRUE;
}
}
sane_optstate( aas, s, OPT_AAS, reload );
sane_optstate( thresh, s, OPT_THRESHOLD, reload );
}
/**
End of handle_depth_halftone.
**/
static void handle_resolution( Epson_Scanner * s,
SANE_Int option, void * value )
{
int n, k = 0, f;
int min_d = s->hw->res_list[ s->hw->res_list_size - 1];
SANE_Int v = *(SANE_Word *) value;
SANE_Int best = v;
int * last = ( OPT_RESOLUTION == option) ?
&s->hw->last_res :
&s->hw->last_res_preview;
/*
We don't assume the list is sorted. Search the list of
resolutions, looking for the value closest to the one we want.
*/
for( n = 0; n < s->hw->res_list_size; n++) {
int d = abs( v - s->hw->res_list[ n]);
if( d < min_d) {
min_d = d;
k = n;
best = s->hw->res_list[ n];
}
}
/*
Problem: does not reach all values cause of scroll bar resolution.
If the requested resolution is not actually available, search
search the list of resolutions to see if the last resolution we
used is in the list. If the last resolution is in the list, and
it is not near the to the closest possible value to the user's
request, then use the one next to the last one - taking into
account the direction the user appears to be going. Or something
like that. I did not write this code.
*/
if( (v != best) && *last) {
for( f = 0; f < s->hw->res_list_size; f++)
if( *last == s->hw->res_list[ f])
break;
if( f != k && f != k - 1 && f != k + 1) {
if( k > f) {
best = s->hw->res_list[ f + 1];
} else if ( k < f) {
best = s->hw->res_list[ f - 1];
}
}
}
*last = best;
s->val[ option ].w = (SANE_Word) best;
DBG(3, "Selected resolution %d dpi\n", best);
}
/**
End of handle_resolution.
**/
static void handle_source( Epson_Scanner * s, SANE_Int optindex,
char * value )
/*
Handles setting the source (flatbed, transparency adapter (TPU),
or auto document feeder (ADF)).
For newer scanners it also sets the focus according to the
glass / TPU settings.
*/
{
int force_max = SANE_FALSE;
SANE_Bool dummy;
s->focusOnGlass = SANE_TRUE; /* this is the default */
if (s->val[ OPT_SOURCE ].w == optindex) return;
s->val[ OPT_SOURCE ].w = optindex;
if( s->val[ OPT_TL_X ].w == s->hw->x_range->min
&& s->val[ OPT_TL_Y ].w == s->hw->y_range->min
&& s->val[ OPT_BR_X ].w == s->hw->x_range->max
&& s->val[ OPT_BR_Y ].w == s->hw->y_range->max
) {
force_max = SANE_TRUE;
}
if( ! strcmp( ADF_STR, value) ) {
s->hw->x_range = &s->hw->adf_x_range;
s->hw->y_range = &s->hw->adf_y_range;
s->hw->use_extension = SANE_TRUE;
/* disable film type option */
s->opt[ OPT_FILM_TYPE].cap &= ~SANE_CAP_INACTIVE;
s->val[ OPT_FOCUS].w = 0;
} else if( ! strcmp( TPU_STR, value) ) {
s->hw->x_range = &s->hw->tpu_x_range;
s->hw->y_range = &s->hw->tpu_y_range;
s->hw->use_extension = SANE_TRUE;
/* enable film type option */
s->opt[ OPT_FILM_TYPE].cap |= SANE_CAP_INACTIVE;
s->val[ OPT_FOCUS].w = 1;
s->focusOnGlass = SANE_FALSE;
} else {
s->hw->x_range = &s->hw->fbf_x_range;
s->hw->y_range = &s->hw->fbf_y_range;
s->hw->use_extension = SANE_FALSE;
/* disable film type option */
s->opt[ OPT_FILM_TYPE].cap &= ~SANE_CAP_INACTIVE;
s->val[ OPT_FOCUS].w = 0;
}
qf_params[ XtNumber(qf_params)-1 ].tl_x = s->hw->x_range->min;
qf_params[ XtNumber(qf_params)-1 ].tl_y = s->hw->y_range->min;
qf_params[ XtNumber(qf_params)-1 ].br_x = s->hw->x_range->max;
qf_params[ XtNumber(qf_params)-1 ].br_y = s->hw->y_range->max;
s->opt[ OPT_BR_X ].constraint.range = s->hw->x_range;
s->opt[ OPT_BR_Y ].constraint.range = s->hw->y_range;
if( s->val[ OPT_TL_X].w < s->hw->x_range->min || force_max)
s->val[ OPT_TL_X].w = s->hw->x_range->min;
if( s->val[ OPT_TL_Y].w < s->hw->y_range->min || force_max)
s->val[ OPT_TL_Y].w = s->hw->y_range->min;
if( s->val[ OPT_BR_X].w > s->hw->x_range->max || force_max)
s->val[ OPT_BR_X].w = s->hw->x_range->max;
if( s->val[ OPT_BR_Y].w > s->hw->y_range->max || force_max)
s->val[ OPT_BR_Y].w = s->hw->y_range->max;
sane_optstate( s->hw->TPU && s->hw->use_extension,
s, OPT_FILM_TYPE, &dummy );
sane_optstate( s->hw->ADF && s->hw->use_extension,
s, OPT_AUTO_EJECT, &dummy );
sane_optstate( s->hw->ADF && s->hw->use_extension,
s, OPT_EJECT, &dummy );
#if 0
BAY is part of the filmscan device. We are not sure
if we are really going to support this device in this
code. Is there an online manual for it?
sane_optstate( s->hw->ADF && s->hw->use_extension,
s, OPT_BAY, &reload );
#endif
}
/**
End of handle_source.
**/
static SANE_Status setvalue( SANE_Handle handle,
SANE_Int option,
void * value,
SANE_Int * info)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Option_Descriptor * sopt = &(s->opt[ option ]);
Option_Value * sval = &(s->val[ option ]);
SANE_Status status;
const SANE_String_Const * optval;
int optindex;
SANE_Bool reload = SANE_FALSE;
status = sanei_constrain_value( sopt, value, info);
if( status != SANE_STATUS_GOOD) return status;
optval = NULL;
optindex = 0;
if( sopt->constraint_type == SANE_CONSTRAINT_STRING_LIST) {
optval = search_string_list( sopt->constraint.string_list,
(char *) value);
if( optval == NULL) return SANE_STATUS_INVAL;
optindex = optval - sopt->constraint.string_list;
}
switch (option) {
case OPT_GAMMA_VECTOR:
case OPT_GAMMA_VECTOR_R:
case OPT_GAMMA_VECTOR_G:
case OPT_GAMMA_VECTOR_B:
memcpy( sval->wa, value, sopt->size); /* Word arrays */
break;
case OPT_CCT_1:
case OPT_CCT_2:
case OPT_CCT_3:
case OPT_CCT_4:
case OPT_CCT_5:
case OPT_CCT_6:
case OPT_CCT_7:
case OPT_CCT_8:
case OPT_CCT_9:
sval->w = *((SANE_Word *) value); /* Simple values */
break;
case OPT_DROPOUT:
case OPT_FILM_TYPE:
case OPT_BAY:
case OPT_FOCUS:
sval->w = optindex; /* Simple lists */
break;
case OPT_EJECT:
/* return eject( s ); */
eject( s );
break;
case OPT_RESOLUTION:
handle_resolution( s, option, value );
break;
case OPT_TL_X:
case OPT_TL_Y:
case OPT_BR_X:
case OPT_BR_Y:
sval->w = *((SANE_Word *) value);
DBG( 1, "set = %f\n", SANE_UNFIX( sval->w));
if (NULL != info) *info |= SANE_INFO_RELOAD_PARAMS;
break;
case OPT_SOURCE:
handle_source( s, optindex, (char *) value );
reload = SANE_TRUE;
break;
case OPT_MODE:
{
SANE_Bool ic = mode_params[ optindex ].color; /* IsColor? */
SANE_Bool iccu = /* Is color correction a user type? */
color_userdefined[ s->val[ OPT_COLOR_CORRECTION ].w ];
sval->w = optindex;
if (s->hw->cmd->set_halftoning != 0) {
sane_optstate( mode_params[ optindex ].depth == 1,
s, OPT_HALFTONE, &reload );
}
sane_optstate( !ic, s, OPT_DROPOUT, &reload );
if (s->hw->cmd->set_color_correction) {
sane_optstate( ic, s, OPT_COLOR_CORRECTION, &reload );
}
if (s->hw->cmd->set_color_correction_coefficients) {
sane_optstate( ic && iccu, s, OPT_CCT_1, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_2, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_3, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_4, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_5, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_6, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_7, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_8, &reload );
sane_optstate( ic && iccu, s, OPT_CCT_9, &reload );
}
handle_depth_halftone( s, &reload );
reload = SANE_TRUE;
break;
}
case OPT_HALFTONE:
sval->w = optindex;
handle_depth_halftone( s, &reload );
break;
sval->w = optindex;
break;
case OPT_COLOR_CORRECTION:
{
SANE_Bool f = color_userdefined[ optindex ];
sval->w = optindex;
sane_optstate( f, s, OPT_CCT_1, &reload );
sane_optstate( f, s, OPT_CCT_2, &reload );
sane_optstate( f, s, OPT_CCT_3, &reload );
sane_optstate( f, s, OPT_CCT_4, &reload );
sane_optstate( f, s, OPT_CCT_5, &reload );
sane_optstate( f, s, OPT_CCT_6, &reload );
sane_optstate( f, s, OPT_CCT_7, &reload );
sane_optstate( f, s, OPT_CCT_8, &reload );
sane_optstate( f, s, OPT_CCT_9, &reload );
break;
}
case OPT_GAMMA_CORRECTION:
{
SANE_Bool f = gamma_userdefined[ optindex ];
sval->w = optindex;
sane_optstate( f, s, OPT_GAMMA_VECTOR, &reload );
sane_optstate( f, s, OPT_GAMMA_VECTOR_R, &reload );
sane_optstate( f, s, OPT_GAMMA_VECTOR_G, &reload );
sane_optstate( f, s, OPT_GAMMA_VECTOR_B, &reload );
sane_optstate( !f, s, OPT_BRIGHTNESS, &reload ); /* Note... */
break;
}
case OPT_MIRROR:
case OPT_SPEED:
case OPT_PREVIEW_SPEED:
case OPT_AAS:
case OPT_PREVIEW: /* needed? */
case OPT_BRIGHTNESS:
case OPT_SHARPNESS:
case OPT_AUTO_EJECT:
case OPT_THRESHOLD:
case OPT_ZOOM:
sval->w = *(( SANE_Word *) value);
break;
case OPT_QUICK_FORMAT:
sval->w = optindex;
s->val[ OPT_TL_X ].w = qf_params[ sval->w ].tl_x;
s->val[ OPT_TL_Y ].w = qf_params[ sval->w ].tl_y;
s->val[ OPT_BR_X ].w = qf_params[ sval->w ].br_x;
s->val[ OPT_BR_Y ].w = qf_params[ sval->w ].br_y;
if( s->val[ OPT_TL_X ].w < s->hw->x_range->min)
s->val[ OPT_TL_X ].w = s->hw->x_range->min;
if( s->val[ OPT_TL_Y ].w < s->hw->y_range->min)
s->val[ OPT_TL_Y ].w = s->hw->y_range->min;
if( s->val[ OPT_BR_X ].w > s->hw->x_range->max)
s->val[ OPT_BR_X ].w = s->hw->x_range->max;
if( s->val[ OPT_BR_Y ].w > s->hw->y_range->max)
s->val[ OPT_BR_Y ].w = s->hw->y_range->max;
reload = SANE_TRUE;
break;
default:
return SANE_STATUS_INVAL;
}
if (reload && info != NULL) {
*info |= SANE_INFO_RELOAD_OPTIONS | SANE_INFO_RELOAD_PARAMS;
}
return SANE_STATUS_GOOD;
}
/**
End of setvalue.
**/
SANE_Status sane_control_option ( SANE_Handle handle,
SANE_Int option,
SANE_Action action,
void * value, SANE_Int * info)
{
if( option < 0 || option >= NUM_OPTIONS)
return SANE_STATUS_INVAL;
if( info != NULL) *info = 0;
switch (action) {
case SANE_ACTION_GET_VALUE:
return( getvalue( handle, option, value ) );
case SANE_ACTION_SET_VALUE:
return( setvalue( handle, option, value, info ) );
default:
return SANE_STATUS_INVAL;
}
return SANE_STATUS_GOOD;
}
/*
* sane_get_parameters()
*
* This function is part of the SANE API and gets called when the front end
* requests information aobut the scan configuration (e.g. color depth, mode,
* bytes and pixels per line, number of lines. This information is returned
* in the SANE_Parameters structure.
*
*/
SANE_Status sane_get_parameters ( SANE_Handle handle, SANE_Parameters * params)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
int ndpi;
int bytes_per_pixel;
DBG(5, "sane_get_parameters()\n");
memset( &s->params, 0, sizeof( SANE_Parameters));
ndpi = s->val[ OPT_RESOLUTION].w;
s->params.pixels_per_line = SANE_UNFIX( s->val[ OPT_BR_X].w - s->val[ OPT_TL_X].w) / 25.4 * ndpi;
s->params.lines = SANE_UNFIX( s->val[ OPT_BR_Y].w - s->val[ OPT_TL_Y].w) / 25.4 * ndpi;
DBG( 3, "Preview = %d\n", s->val[OPT_PREVIEW].w);
DBG( 3, "Resolution = %d\n", s->val[OPT_RESOLUTION].w);
DBG( 1, "get para %p %p tlx %f tly %f brx %f bry %f [mm]\n"
, ( void * ) s
, ( void * ) s->val
, SANE_UNFIX( s->val[ OPT_TL_X].w)
, SANE_UNFIX( s->val[ OPT_TL_Y].w)
, SANE_UNFIX( s->val[ OPT_BR_X].w)
, SANE_UNFIX( s->val[ OPT_BR_Y].w)
);
/* Calculate bytes_per_pixel and bytes_per_line for
* any color depths.
*
* The color depth is stored in mode_params.depth:
*/
s->params.depth = mode_params[ s->val[ OPT_MODE].w].depth;
if (s->params.depth > 8)
{
if (s->val[OPT_PREVIEW].w) /* for preview the frontends can only handle 8 bits */
s->params.depth = 8;
else
s->params.depth = 16; /* the frontends can only handle 8 or 16 bits for gray or color */
}
bytes_per_pixel = s->params.depth / 8;
if (s->params.depth % 8) /* just in case ... */
{
bytes_per_pixel++;
}
/* pixels_per_line seems to be rounded to the next 8bit boundary */
s->params.pixels_per_line = s->params.pixels_per_line & ~7;
s->params.last_frame = SANE_TRUE;
if( mode_params[ s->val[ OPT_MODE].w].color) {
s->params.format = SANE_FRAME_RGB;
s->params.bytes_per_line = 3 * s->params.pixels_per_line * bytes_per_pixel;
} else {
s->params.format = SANE_FRAME_GRAY;
s->params.bytes_per_line = s->params.pixels_per_line * s->params.depth / 8;
}
if( NULL != params)
*params = s->params;
return SANE_STATUS_GOOD;
}
/*
* sane_start()
*
* This function is part of the SANE API and gets called from the front end to
* start the scan process.
*
*/
SANE_Status sane_start ( SANE_Handle handle)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Status status;
const struct mode_param * mparam;
u_char params[4];
int ndpi;
int left, top;
int lcount;
int i, j; /* loop counter */
open_scanner( s);
/*
* There is some undocumented special with TPU enable/disable.
* TPU power ESC e status
* on 0 NAK
* on 1 ACK
* off 0 ACK
* off 1 NAK
*
* probably it make no sense to scan with TPU powered on and source flatbed, cause light
* will come from both sides.
*/
if( s->hw->extension) {
u_char * buf;
EpsonHdr head;
DBG( 1, "use extension = %d\n", s->hw->use_extension);
params[0] = ESC;
params[1] = s->hw->cmd->control_an_extension;
if( NULL == ( head = ( EpsonHdr) command( s, params, 2, &status) ) ) {
DBG( 0, "control of an extension failed\n");
return status;
}
params[ 0] = s->hw->use_extension; /* 1: effective, 0: ineffective */
send( s, params, 1, &status); /* to make (in)effective an extension unit*/
status = expect_ack ( s);
if( SANE_STATUS_GOOD != status) {
DBG( 0, "Probably you have to power %s your TPU\n"
, s->hw->use_extension ? "on" : "off");
DBG( 0, "Also you have to restart sane, cause it gives a ....\n");
DBG( 0, "about the return code I'm sending.\n");
return status;
}
params[0] = ESC;
params[1] = s->hw->cmd->request_extension_status;
if( NULL == ( head = ( EpsonHdr) command( s, params, 2, &status) ) ) {
DBG( 0, "Extended status flag request failed\n");
return status;
}
buf = &head->buf[ 0];
if( buf[ 0] & EXT_STATUS_FER) {
DBG( 0, "option: fatal error\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_ERR) {
DBG( 0, "ADF: other error\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_PE) {
DBG( 0, "ADF: no paper\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_PJ) {
DBG( 0, "ADF: paper jam\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 1] & EXT_STATUS_OPN) {
DBG( 0, "ADF: cover open\n");
status = SANE_STATUS_INVAL;
}
if( buf[ 6] & EXT_STATUS_ERR) {
DBG( 0, "TPU: other error\n");
status = SANE_STATUS_INVAL;
}
if( SANE_STATUS_GOOD != status) {
close_scanner( s);
return status;
}
/*
* set the focus position according to the extension used:
* if the TPU is selected, then focus 2.5mm above the glass,
* otherwise focus on the glass
*/
if (s->hw->focusSupport == SANE_TRUE)
{
if (s->val[ OPT_FOCUS].w == 0)
{
DBG( 1, "Setting focus to glass surface\n");
set_focus_position(s, 0x40);
}
else
{
DBG( 1, "Setting focus to 2.5mm above glass\n");
set_focus_position(s, 0x59);
}
}
}
/*
* Set the color depth. If a preview is requested, then always
* scan in 8bit mode if grayscale or color mode was selected. We
* are not testing for any of these modes, if preview is selected,
* and the color depth was more than 8 bits, then we just reset it
* to 8 bits.
*/
mparam = mode_params + s->val[ OPT_MODE].w;
if (s->val[OPT_PREVIEW].w && mparam->depth > 8)
status = set_data_format( s, 8);
else
status = set_data_format( s, mparam->depth);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_data_format failed: %s\n", sane_strstatus( status));
return status;
}
/*
* The byte sequence mode was introduced in B5, for B34 we need line sequence mode
*/
if( (s->hw->cmd->level[0] == 'D' ||
(s->hw->cmd->level[0] == 'B' && s->hw->level >= 5)) && mparam->mode_flags == 0x02)
{
status = set_color_mode( s, 0x13);
}
else
{
status = set_color_mode( s , mparam->mode_flags
| ( mparam->dropout_mask
& dropout_params[ s->val[ OPT_DROPOUT].w]
)
);
}
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_color_mode failed: %s\n", sane_strstatus( status));
return status;
}
if( s->hw->cmd->set_halftoning && SANE_OPTION_IS_ACTIVE( s->opt[ OPT_HALFTONE].cap) ) {
status = set_halftoning( s, halftone_params[ s->val[ OPT_HALFTONE].w]);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_halftoning failed: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_BRIGHTNESS].cap) ) {
status = set_bright( s, s->val[OPT_BRIGHTNESS].w);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_bright failed: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_MIRROR].cap) ) {
status = mirror_image( s, mirror_params[ s->val[ OPT_MIRROR].w]);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: mirror_image failed: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_SPEED].cap) ) {
if( s->val[ OPT_PREVIEW].w)
status = set_speed( s, speed_params[ s->val[ OPT_PREVIEW_SPEED].w]);
else
status = set_speed( s, speed_params[ s->val[ OPT_SPEED].w]);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_speed failed: %s\n", sane_strstatus( status));
return status;
}
}
/*
* use of speed_params is ok here since they are false and true.
* NOTE: I think I should throw that "params" stuff as long w is already the value.
*/
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_AAS].cap) ) {
status = control_auto_area_segmentation( s, speed_params[ s->val[ OPT_AAS].w]);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: control_auto_area_segmentation failed: %s\n", sane_strstatus( status));
return status;
}
}
s->invert_image = SANE_FALSE; /* default to not inverting the image */
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_FILM_TYPE].cap) ) {
s->invert_image = (s->val[ OPT_FILM_TYPE].w == FILM_TYPE_NEGATIVE);
status = set_film_type( s, film_params[ s->val[ OPT_FILM_TYPE].w]);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_film_type failed: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_BAY].cap) ) {
status = set_bay( s, s->val[ OPT_BAY].w);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_bay: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_SHARPNESS].cap) ) {
status = set_outline_emphasis( s, s->val[ OPT_SHARPNESS].w);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_outline_emphasis failed: %s\n", sane_strstatus( status));
return status;
}
}
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_GAMMA_CORRECTION].cap) ) {
int val;
if (s->hw->cmd->level[0] == 'D')
{
/*
* The D1 level has only the two user defined gamma
* settings.
*/
val = gamma_params[ s->val[ OPT_GAMMA_CORRECTION].w];
}
else
{
val = gamma_params[ s->val[ OPT_GAMMA_CORRECTION].w];
/*
* If "Default" is selected then determine the actual value
* to send to the scanner: If bilevel mode, just send the
* value from the table (0x01), for grayscale or color mode
* add one and send 0x02.
*/
if( s->val[ OPT_GAMMA_CORRECTION].w <= 1) {
val += mparam->depth == 1 ? 0 : 1;
}
}
DBG( 1, "sane_start: set_gamma( s, 0x%x ).\n", val);
status = set_gamma( s, val);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_gamma failed: %s\n", sane_strstatus( status));
return status;
}
}
if (gamma_userdefined[s->val[ OPT_GAMMA_CORRECTION].w])
{ /* user defined. */
status = set_gamma_table( s);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_gamma_table failed: %s\n", sane_strstatus (status));
return status;
}
}
/*
* TODO: think about if SANE_OPTION_IS_ACTIVE is a good criteria to send commands.
*/
if( SANE_OPTION_IS_ACTIVE( s->opt[ OPT_COLOR_CORRECTION].cap) ) {
int val = color_params[ s->val[ OPT_COLOR_CORRECTION].w];
DBG( 1, "sane_start: set_color_correction( s, 0x%x )\n", val );
status = set_color_correction( s, val);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_color_correction failed: %s\n", sane_strstatus( status));
return status;
}
}
if( 1 == s->val[ OPT_COLOR_CORRECTION].w) { /* user defined. */
status = set_color_correction_coefficients( s);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_color_correction_coefficients failed: %s\n", sane_strstatus( status));
return status;
}
}
if (s->hw->cmd->set_threshold != 0 && SANE_OPTION_IS_ACTIVE( s->opt[ OPT_THRESHOLD].cap))
{
status = set_threshold(s, s->val[ OPT_THRESHOLD].w);
if (SANE_STATUS_GOOD != status)
{
DBG(1, "sane_start: set_threshold(%d) failed: %s\n",
s->val[ OPT_THRESHOLD].w, sane_strstatus(status));
return status;
}
}
ndpi = s->val[ OPT_RESOLUTION].w;
status = set_resolution( s, ndpi, ndpi);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_resolution(%d, %d) failed: %s\n",
ndpi, ndpi, sane_strstatus( status));
return status;
}
status = sane_get_parameters( handle, NULL);
if( status != SANE_STATUS_GOOD)
return status;
/* set the zoom */
if (s->hw->cmd->set_zoom != 0 && SANE_OPTION_IS_ACTIVE( s->opt[ OPT_ZOOM].cap))
{
status = set_zoom(s, s->val[ OPT_ZOOM].w, s->val[ OPT_ZOOM].w);
if (status != SANE_STATUS_GOOD)
{
DBG(1, "sane_start: set_zoom(%d) failed: %s\n",
s->val[ OPT_ZOOM].w, sane_strstatus(status));
return status;
}
}
/*
* Now s->params is initialized.
*/
/*
* in file:frontend/preview.c
*
* The preview strategy is as follows:
*
* 1) A preview always acquires an image that covers the entire
* scan surface. This is necessary so the user can see not
* only what is, but also what isn't selected.
*/
left = SANE_UNFIX( s->val[OPT_TL_X].w) / 25.4 * ndpi + 0.5;
top = SANE_UNFIX( s->val[OPT_TL_Y].w) / 25.4 * ndpi + 0.5;
/*
* Calculate correction for line_distance in D1 scanner:
* Start line_distance lines earlier and add line_distance lines at the end
*
* Because the actual line_distance is not yet calculated we have to do this
* first. !!!
*/
s->hw->color_shuffle = SANE_FALSE;
s->current_output_line = 0;
s->lines_written = 0;
s->color_shuffle_line = 0;
if ((s->hw->optical_res != 0) && (mparam->depth == 8) && (mparam->mode_flags != 0))
{
s->line_distance = s->hw->max_line_distance * ndpi / s->hw->optical_res;
if (s->line_distance != 0)
{
s->hw->color_shuffle = SANE_TRUE;
}
else
s->hw->color_shuffle = SANE_FALSE;
}
/* modify the scan area */
if (s->hw->color_shuffle == SANE_TRUE)
{
#if 0
top -= s->line_distance;
if (top < 0)
{
top = 0;
}
#endif
s->params.lines += 2* s->line_distance;
}
if (SANE_UNFIX( s->val[ OPT_BR_Y].w) / 25.4 * ndpi < s->params.lines)
{
s->params.lines = (int) SANE_UNFIX(s->val[OPT_BR_Y].w) / 25.4 * ndpi + 0.5;
}
status = set_scan_area( s, left, top, s->params.pixels_per_line, s->params.lines);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_scan_area failed: %s\n", sane_strstatus( status));
return status;
}
s->block = SANE_FALSE;
lcount = 1;
/*
* The set line count commands needs to be sent for certain scanners in
* color mode. The D1 level requires it, we are however only testing for
* 'D' and not for the actual numeric level.
*/
if( ( (s->hw->cmd->level[0] == 'B') &&
( (s->hw->level >= 5) || ( (s->hw->level >= 4) && (! mode_params[s->val[ OPT_MODE].w].color))))
|| ( s->hw->cmd->level[0] == 'D') )
{
s->block = SANE_TRUE;
lcount = sanei_scsi_max_request_size / s->params.bytes_per_line;
if( lcount > 255)
lcount = 255;
if( lcount == 0)
return SANE_STATUS_NO_MEM;
status = set_lcount( s, lcount);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: set_lcount(%d) failed: %s\n",
lcount, sane_strstatus (status));
return status;
}
}
if( SANE_TRUE == s->hw->extension) { /* make sure if any errors */
/* TODO */
u_char result[ 4]; /* with an extension */
u_char * buf;
size_t len;
params[0] = ESC;
params[1] = s->hw->cmd->request_extension_status;
send( s, params, 2, &status); /* send ESC f (request extension status) */
if( SANE_STATUS_GOOD != status)
return status;
len = 4; /* receive header */
receive( s, result, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = result[ 3] << 8 | result[ 2];
buf = alloca( len);
receive( s, buf, len, &status); /* reveive actual status data */
if( buf[ 0] & 0x80) {
close_scanner( s);
return SANE_STATUS_INVAL;
}
}
/*
* for debug purpose
* check scanner conditions
*/
#if 1
if (s->hw->cmd->request_condition != 0)
{
u_char result [ 4];
u_char * buf;
size_t len;
params[0] = ESC;
params[1] = s->hw->cmd->request_condition;
send( s, params, 2, &status); /* send request condition */
if( SANE_STATUS_GOOD != status)
return status;
len = 4;
receive( s, result, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = result[ 3] << 8 | result[ 2];
buf = alloca( len);
receive( s, buf, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
#if 0
DBG(10, "SANE_START: length=%d\n", len);
for (i = 1; i <= len; i++) {
DBG(10, "SANE_START: %d: %c\n", i, buf[i-1]);
}
#endif
DBG( 5, "SANE_START: color: %d\n", (int) buf[1]);
DBG( 5, "SANE_START: resolution (x, y): (%d, %d)\n",
(int) (buf[4]<<8|buf[3]), (int) (buf[6]<<8|buf[5]));
DBG( 5, "SANE_START: area[dots] (x-offset, y-offset), (x-range, y-range): (%d, %d), (%d, %d)\n",
(int) (buf[9]<<8|buf[8]), (int) (buf[11]<<8|buf[10]),
(int) (buf[13]<<8|buf[12]), (int) (buf[15]<<8|buf[14]));
DBG( 5, "SANE_START: data format: %d\n", (int) buf[17]);
DBG( 5, "SANE_START: halftone: %d\n", (int) buf[19]);
DBG( 5, "SANE_START: brightness: %d\n", (int) buf[21]);
DBG( 5, "SANE_START: gamma: %d\n", (int) buf[23]);
DBG( 5, "SANE_START: zoom[percentage] (x, y): (%d, %d)\n", (int) buf[26], (int) buf[25]);
DBG( 5, "SANE_START: color correction: %d\n", (int) buf[28]);
DBG( 5, "SANE_START: outline emphasis: %d\n", (int) buf[30]);
DBG( 5, "SANE_START: read mode: %d\n", (int) buf[32]);
DBG( 5, "SANE_START: mirror image: %d\n", (int) buf[34]);
DBG( 5, "SANE_START: (new B6 or B7 command ESC s): %d\n", (int) buf[36]);
DBG( 5, "SANE_START: (new B6 or B7 command ESC t): %d\n", (int) buf[38]);
DBG( 5, "SANE_START: line counter: %d\n", (int) buf[40]);
DBG( 5, "SANE_START: extension control: %d\n", (int) buf[42]);
DBG( 5, "SANE_START: (new B6 or B7 command ESC N): %d\n", (int) buf[44]);
}
#endif
/* set the retry count to 0 */
s->retry_count = 0;
if (s->hw->color_shuffle == SANE_TRUE)
{
/* initialize the line buffers */
for (i = 0; i < s->line_distance * 2 + 1; i++)
{
if (s->line_buffer[i] != NULL)
free(s->line_buffer[i]);
s->line_buffer[i] = malloc(s->params.bytes_per_line);
if (s->line_buffer[i] == NULL)
{
/* free the memory we've malloced so far */
for (j = 0; j < i; j++)
{
free(s->line_buffer[j]);
s->line_buffer[j] = NULL;
}
return SANE_STATUS_NO_MEM;
}
}
}
params[0] = ESC;
params[1] = s->hw->cmd->start_scanning;
send( s, params, 2, &status);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "sane_start: start failed: %s\n", sane_strstatus( status));
return status;
}
s->eof = SANE_FALSE;
s->buf = realloc( s->buf, lcount * s->params.bytes_per_line);
s->ptr = s->end = s->buf;
s->canceling = SANE_FALSE;
return SANE_STATUS_GOOD;
} /* sane_start */
/*
*
* TODO: clean up the eject and direct cmd mess.
*/
SANE_Status sane_auto_eject ( Epson_Scanner * s) {
if( s->hw->ADF && s->hw->use_extension && s->val[ OPT_AUTO_EJECT].w) { /* sequence! */
SANE_Status status;
u_char params [ 1];
u_char cmd = s->hw->cmd->eject;
if( ! cmd)
return SANE_STATUS_UNSUPPORTED;
params[ 0] = cmd;
send( s, params, 1, &status);
if( SANE_STATUS_GOOD != ( status = expect_ack( s) ) ) {
return status;
}
}
return SANE_STATUS_GOOD;
}
/*
*
*
*/
static SANE_Status read_data_block ( Epson_Scanner * s, EpsonDataRec * result) {
SANE_Status status;
u_char param[3];
receive( s, result, s->block ? 6 : 4, &status);
if( SANE_STATUS_GOOD != status)
return status;
if( STX != result->code) {
DBG( 1, "code %02x\n", ( int) result->code);
DBG( 1, "error, expected STX\n");
return SANE_STATUS_INVAL;
}
if( result->status & STATUS_FER) {
DBG( 1, "fatal error - Status = %02x\n", result->status);
status = check_ext_status( s);
/*
* Hack Alert!!!
* If the status is SANE_STATUS_DEVICE_BUSY then we need to
* re-issue the command again. We can assume that the command that
* caused this problem was ESC G, so in a loop with a sleep 1 we
* are testing this over and over and over again, until the lamp
* "thinks" it is ready.
*
* TODO: Store the last command and execute what was actually used
* as the last command. For all situations this error may occur
* ESC G is very very likely to be the command in question, but
* we better make sure that this is the case.
*
*/
/*
* let's safe some stack space: If this is not the first go around,
* then just return the status and let the loop handle this - otherwise
* we would run this function recursively.
*/
if ((status == SANE_STATUS_DEVICE_BUSY && s->retry_count > 0) ||
(status == SANE_STATUS_GOOD && s->retry_count > 0))
{
return SANE_STATUS_DEVICE_BUSY; /* return busy even if we just read OK
so that the following loop can end
gracefully */
}
#define SANE_EPSON_MAX_RETRIES (61)
while (status == SANE_STATUS_DEVICE_BUSY) {
if (s->retry_count > SANE_EPSON_MAX_RETRIES)
{
DBG(0, "Max retry count exceeded (%d)\n", s->retry_count);
return SANE_STATUS_INVAL;
}
sleep(1); /* wait one second for the next attempt */
DBG(1, "retrying ESC G - %d\n", ++(s->retry_count));
param[0] = ESC;
param[1] = s->hw->cmd->start_scanning;
send( s, param, 2, &status);
if( SANE_STATUS_GOOD != status) {
DBG( 1, "read_data_block: start failed: %s\n", sane_strstatus( status));
return status;
}
status = read_data_block(s, result);
}
}
return status;
}
/*
*
*
*/
#define GET_COLOR(x) ((x.status>>2) & 0x03)
SANE_Status sane_read ( SANE_Handle handle, SANE_Byte * data, SANE_Int max_length, SANE_Int * length) {
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Status status;
int index = 0;
SANE_Bool reorder = SANE_FALSE;
int i; /* loop counter */
int bytes_to_process = 0;
START_READ:
DBG( 5, "sane_read: begin\n");
if( s->ptr == s->end)
{
EpsonDataRec result;
size_t buf_len;
if( s->eof) {
if (s->hw->color_shuffle)
{
DBG(1, "Written %d lines after color shuffle\n", s->lines_written);
DBG(1, "Lines requested: %d\n", s->params.lines);
}
free( s->buf);
s->buf = NULL;
sane_auto_eject( s);
close_scanner( s);
s->fd = -1;
*length = 0;
/*
* free the line-buffers
*/
for (i = 0; i< s->line_distance; i++)
{
if (s->line_buffer[i] != NULL)
{
free(s->line_buffer[i]);
s->line_buffer[i] = NULL;
}
}
return SANE_STATUS_EOF;
}
DBG( 5, "sane_read: begin scan1\n");
if( SANE_STATUS_GOOD != ( status = read_data_block( s, &result)))
return status;
buf_len = result.buf[ 1] << 8 | result.buf[ 0];
DBG( 5, "sane_read: buf len = %lu\n", (u_long) buf_len);
if( s->block)
{
buf_len *= ( result.buf[ 3] << 8 | result.buf[ 2]);
DBG( 5, "sane_read: buf len (adjusted) = %lu\n", (u_long) buf_len);
}
if( !s->block && SANE_FRAME_RGB == s->params.format) {
/*
* Read color data in line mode
*/
/*
* read the first color line - the number of bytes to read
* is already known (from last call to read_data_block()
* We determine where to write the line from the color information
* in the data block. At the end we want the order RGB, but the
* way the data is delivered does not guarantee this - actually it's
* most likely that the order is GRB if it's not RGB!
*/
switch (GET_COLOR(result))
{
case 1 : index = 1;
break;
case 2 : index = 0;
break;
case 3 : index = 2;
break;
}
receive( s, s->buf + index * s->params.pixels_per_line, buf_len, &status);
if( SANE_STATUS_GOOD != status)
return status;
/*
* send the ACK signal to the scanner in order to make
* it ready for the next data block.
*/
send( s, S_ACK, 1, &status);
/*
* ... and request the next data block
*/
if( SANE_STATUS_GOOD != ( status = read_data_block( s, &result)))
return status;
buf_len = result.buf[ 1] << 8 | result.buf[ 0];
/*
* this should never happen, because we are already in
* line mode, but it does not hurt to check ...
*/
if( s->block)
buf_len *= ( result.buf[ 3] << 8 | result.buf[ 2]);
DBG( 5, "sane_read: buf len2 = %lu\n", (u_long) buf_len);
switch (GET_COLOR(result))
{
case 1 : index = 1;
break;
case 2 : index = 0;
break;
case 3 : index = 2;
break;
}
receive( s, s->buf + index * s->params.pixels_per_line, buf_len, &status);
if( SANE_STATUS_GOOD != status)
return status;
send( s, S_ACK, 1, &status);
/*
* ... and the last data block
*/
if( SANE_STATUS_GOOD != ( status = read_data_block( s, &result)))
return status;
buf_len = result.buf[ 1] << 8 | result.buf[ 0];
if( s->block)
buf_len *= ( result.buf[ 3] << 8 | result.buf[ 2]);
DBG( 5, "sane_read: buf len3 = %lu\n", (u_long) buf_len);
switch (GET_COLOR(result))
{
case 1 : index = 1;
break;
case 2 : index = 0;
break;
case 3 : index = 2;
break;
}
receive( s, s->buf + index * s->params.pixels_per_line, buf_len, &status);
if( SANE_STATUS_GOOD != status)
return status;
} else {
/*
* Read data in block mode
*/
/* do we have to reorder the data ? */
if (GET_COLOR(result) == 0x01)
{
reorder = SANE_TRUE;
}
receive( s, s->buf, buf_len, &status);
bytes_to_process = buf_len;
if( SANE_STATUS_GOOD != status)
return status;
}
if( result.status & STATUS_AREA_END)
s->eof = SANE_TRUE;
else {
if( s->canceling) {
send( s, S_CAN, 1, &status);
expect_ack( s);
free( s->buf);
s->buf = NULL;
sane_auto_eject( s);
close_scanner( s);
s->fd = -1;
*length = 0;
/*
* free the line-buffers
*/
for (i = 0; i< s->line_distance; i++)
{
if (s->line_buffer[i] != NULL)
{
free(s->line_buffer[i]);
s->line_buffer[i] = NULL;
}
}
return SANE_STATUS_CANCELLED;
} else
send( s, S_ACK, 1, &status);
}
s->end = s->buf + buf_len;
s->ptr = s->buf;
/*
* if we have to re-order the color components (GRB->RGB) we
* are doing this here:
*/
if (reorder)
{
SANE_Byte *ptr;
for (ptr = s->buf; ptr < s->end; ptr++)
{
if (s->params.depth > 8)
{
/* R->G G->R */
SANE_Word tmp;
tmp = (SANE_Word) *ptr;
(SANE_Word) *ptr = (SANE_Word) *(ptr+2); /* G */
(SANE_Word) *(ptr+2) = tmp; /* R */
/* B stays the same */
ptr += 6; /* go to next */
}
else
{
/* R->G G->R */
SANE_Byte tmp;
tmp = *ptr;
*ptr = *(ptr+1); /* G */
*(ptr+1) = tmp; /* R */
/* B stays the same */
ptr += 3; /* go to next */
}
}
}
/*
* Do the color_shuffle if everything else is correct - at this time
* most of the stuff is hardcoded for the Perfection 610
*/
if (s->hw->cmd->level[0] == 'D' && SANE_FRAME_RGB == s->params.format && s->block)
{
int new_length = 0;
status = color_shuffle(s, &new_length);
/* !!!
* If no bytes are returned, check if the scanner is already done, if so,
* we'll probably just return, but if there is more data to process get
* the next batch.
*/
if (new_length == 0 && s->end != s->ptr)
{
goto START_READ;
}
s->end = s->buf + new_length;
s->ptr = s->buf;
}
DBG( 5, "sane_read: begin scan2\n");
}
/*
* copy the image data to the data memory area
*/
if( ! s->block && SANE_FRAME_RGB == s->params.format) {
max_length /= 3;
if( max_length > s->end - s->ptr)
max_length = s->end - s->ptr;
*length = 3 * max_length;
if (s->invert_image == SANE_TRUE)
{
while( max_length-- != 0) {
/* invert the three values */
*data++ = (u_char) ~(s->ptr[ 0]);
*data++ = (u_char) ~(s->ptr[ s->params.pixels_per_line]);
*data++ = (u_char) ~(s->ptr[ 2 * s->params.pixels_per_line]);
++s->ptr;
}
}
else
{
while( max_length-- != 0) {
*data++ = s->ptr[ 0];
*data++ = s->ptr[ s->params.pixels_per_line];
*data++ = s->ptr[ 2 * s->params.pixels_per_line];
++s->ptr;
}
}
} else {
if( max_length > s->end - s->ptr)
max_length = s->end - s->ptr;
*length = max_length;
if( 1 == s->params.depth) {
if (s->invert_image == SANE_TRUE)
{
while( max_length-- != 0)
*data++ = *s->ptr++;
}
else
{
while( max_length-- != 0)
*data++ = ~*s->ptr++;
}
} else {
if (s->invert_image == SANE_TRUE)
{
int i;
for (i = 0 ; i < max_length; i++)
{
data[i] = (u_char) ~(s->ptr[i]);
}
}
else
{
memcpy( data, s->ptr, max_length);
}
s->ptr += max_length;
}
}
DBG( 5, "sane_read: end\n");
return SANE_STATUS_GOOD;
}
static SANE_Status
color_shuffle(SANE_Handle handle, int *new_length)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Byte *buf = s->buf;
int length = s->end - s->buf;
if (s->hw->color_shuffle == SANE_TRUE)
{
SANE_Byte * data_ptr; /* ptr to data to process */
SANE_Byte * data_end; /* ptr to end of processed data */
SANE_Byte * out_data_ptr; /* ptr to memory when writing data */
int i; /* loop counter */
/*
* It looks like we are dealing with a scanner that has an odd way
* of dealing with colors... The red and blue scan lines are shifted
* up or down by a certain number of lines relative to the green line.
*/
DBG( 5, "sane_read: color_shuffle\n");
/*
* Initialize the variables we are going to use for the
* copying of the data. data_ptr is the pointer to
* the currently worked on scan line. data_end is the
* end of the data area as calculated from adding *length
* to the start of data.
* out_data_ptr is used when writing out the processed data
* and always points to the beginning of the next line to
* write.
*/
data_ptr = out_data_ptr = buf;
data_end = data_ptr + length;
/*
* The image data is in *buf, we know that the buffer contains s->end - s->buf ( = length)
* bytes of data. The width of one line is in s->params.bytes_per_line
*/
/*
* The buffer area is supposed to have a number of full scan
* lines, let's test if this is the case.
*/
if (length % s->params.bytes_per_line != 0)
{
DBG(0, "ERROR in size of buffer: %d / %d\n",
length, s->params.bytes_per_line);
return SANE_STATUS_INVAL;
}
while (data_ptr < data_end)
{
SANE_Byte * source_ptr, *dest_ptr;
int loop;
/* copy the green information into the current line */
source_ptr = data_ptr + 1;
dest_ptr = s->line_buffer[s->color_shuffle_line] +1;
for (i=0; i< s->params.bytes_per_line /3; i++)
{
*dest_ptr = *source_ptr;
dest_ptr += 3;
source_ptr += 3;
}
/* copy the red information n lines back */
if (s->color_shuffle_line >= s->line_distance)
{
source_ptr = data_ptr + 2;
dest_ptr = s->line_buffer[s->color_shuffle_line - s->line_distance] + 2;
/* while (source_ptr < s->line_buffer[s->color_shuffle_line] + s->params.bytes_per_line) */
for (loop=0; loop < s->params.bytes_per_line / 3 ; loop++)
{
*dest_ptr = *source_ptr;
dest_ptr += 3;
source_ptr += 3;
}
}
/* copy the blue information n lines forward */
source_ptr = data_ptr;
dest_ptr = s->line_buffer[s->color_shuffle_line + s->line_distance];
/* while (source_ptr < s->line_buffer[s->color_shuffle_line] + s->params.bytes_per_line) */
for (loop=0; loop < s->params.bytes_per_line / 3 ; loop++)
{
*dest_ptr = *source_ptr;
dest_ptr += 3;
source_ptr += 3;
}
data_ptr += s->params.bytes_per_line;
if (s->color_shuffle_line == s->line_distance)
{
/*
* we just finished the line in line_buffer[0] - write it to the
* output buffer and continue.
*/
/*
* The ouput buffer ist still "buf", but because we are
* only overwriting from the beginning of the memory area
* we are not interfering with the "still to shuffle" data
* in the same area.
*/
/*
* Strip the first and last n lines and limit to
*/
if ((s->current_output_line >= s->line_distance) &&
(s->current_output_line < s->params.lines + s->line_distance))
{
memcpy(out_data_ptr, s->line_buffer[0], s->params.bytes_per_line);
out_data_ptr += s->params.bytes_per_line;
s->lines_written++;
}
s->current_output_line++;
/*
* Now remove the 0-entry and move all other
* lines up by one. There are 2*line_distance + 1
* buffers, * therefore the loop has to run from 0
* to * 2*line_distance, and because we want to
* copy every n+1st entry to n the loop runs
* from - to 2*line_distance-1!
*/
free(s->line_buffer[0]);
for (i = 0; i < s->line_distance*2; i++)
{
s->line_buffer[i] = s->line_buffer[i+1];
}
/*
* and create one new buffer at the end
*/
s->line_buffer[s->line_distance*2] = malloc(s->params.bytes_per_line);
if (s->line_buffer[s->line_distance*2] == NULL)
{
int i;
for (i=0; i<s->line_distance*2; i++)
{
free(s->line_buffer[i]);
s->line_buffer[i] = NULL;
}
return SANE_STATUS_NO_MEM;
}
} else {
s->color_shuffle_line++; /* increase the buffer number */
}
}
/*
* At this time we've used up all the new data from the scanner, some of
* it is still in the line_buffers, but we are ready to return some of it
* to the front end software. To do so we have to adjust the size of the
* data area and the *new_length variable.
*/
*new_length = out_data_ptr - buf;
}
return SANE_STATUS_GOOD;
}
/*
* static SANE_Status get_identity_information ( SANE_Handle handle)
*
* Request Identity information from scanner and fill in information
* into dev and/or scanner structures.
*/
static SANE_Status
get_identity_information(SANE_Handle handle)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
Epson_Device * dev = s->hw;
EpsonIdent ident;
u_char param[3];
SANE_Status status;
u_char * buf;
DBG(5, "get_identity_information()\n");
if( ! s->hw->cmd->request_identity)
return SANE_STATUS_INVAL;
param[0] = ESC;
param[1] = s->hw->cmd->request_identity;
param[2] = '\0';
if( NULL == ( ident = ( EpsonIdent) command( s, param, 2, &status) ) ) {
DBG( 0, "ident failed\n");
return SANE_STATUS_INVAL;
}
DBG( 1, "type %3c 0x%02x\n", ident->type, ident->type);
DBG( 1, "level %3c 0x%02x\n", ident->level, ident->level);
{
char * force = getenv( "SANE_EPSON_CMD_LVL");
if( force) {
ident->type = force[ 0];
ident->level = force[ 1];
DBG( 1, "type %3c 0x%02x\n", ident->type, ident->type);
DBG( 1, "level %3c 0x%02x\n", ident->level, ident->level);
DBG( 1, "forced\n");
}
}
/*
* check if option equipment is installed.
*/
if( ident->status & STATUS_OPTION) {
DBG( 1, "option equipment is installed\n");
dev->extension = SANE_TRUE;
} else {
DBG( 1, "no option equipment installed\n");
dev->extension = SANE_FALSE;
}
dev->TPU = SANE_FALSE;
dev->ADF = SANE_FALSE;
/*
* set command type and level.
*/
{
int n;
for( n = 0; n < NELEMS( epson_cmd); n++)
if( ! strncmp( &ident->type, epson_cmd[ n].level, 2) )
break;
if( n < NELEMS( epson_cmd) ) {
dev->cmd = &epson_cmd[ n];
} else {
dev->cmd = &epson_cmd[EPSON_LEVEL_DEFAULT];
DBG( 0, "Unknown type %c or level %c, using %s\n",
ident->buf[0], ident->buf[1], dev->cmd->level);
}
s->hw->level = dev->cmd->level[ 1] - '0';
} /* set comand type and level */
/*
* Setting available resolutions and xy ranges for sane frontend.
*/
s->hw->res_list_size = 0;
s->hw->res_list = ( SANE_Int *) calloc( s->hw->res_list_size, sizeof( SANE_Int));
if( NULL == s->hw->res_list) {
DBG( 0, "out of memory\n");
return SANE_STATUS_NO_MEM;
}
{
int n, k;
int x = 0, y = 0;
for( n = ident->count, buf = ident->buf; n; n -= k, buf += k) {
switch (*buf) {
case 'R':
{
int val = buf[ 2] << 8 | buf[ 1];
s->hw->res_list_size++;
s->hw->res_list = ( SANE_Int *) realloc( s->hw->res_list, s->hw->res_list_size * sizeof( SANE_Int));
if( NULL == s->hw->res_list) {
DBG( 0, "out of memory\n");
return SANE_STATUS_NO_MEM;
}
s->hw->res_list[ s->hw->res_list_size - 1] = ( SANE_Int) val;
DBG( 1, "resolution (dpi): %d\n", val);
k = 3;
continue;
}
case 'A':
{
x = buf[ 2] << 8 | buf[ 1];
y = buf[ 4] << 8 | buf[ 3];
DBG( 1, "maximum scan area: x %d y %d\n", x, y);
k = 5;
continue;
}
default:
break;
} /* case */
break;
} /* for */
dev->dpi_range.min = s->hw->res_list[ 0];
dev->dpi_range.max = s->hw->res_list[ s->hw->res_list_size - 1];
dev->dpi_range.quant = 0;
dev->fbf_x_range.min = 0;
dev->fbf_x_range.max = SANE_FIX( x * 25.4 / dev->dpi_range.max);
dev->fbf_x_range.quant = 0;
dev->fbf_y_range.min = 0;
dev->fbf_y_range.max = SANE_FIX( y * 25.4 / dev->dpi_range.max);
dev->fbf_y_range.quant = 0;
DBG( 5, "fbf tlx %f tly %f brx %f bry %f [mm]\n"
, SANE_UNFIX( dev->fbf_x_range.min)
, SANE_UNFIX( dev->fbf_y_range.min)
, SANE_UNFIX( dev->fbf_x_range.max)
, SANE_UNFIX( dev->fbf_y_range.max)
);
}
/*
* Copy the resolution list to the resolution_list array so that the frontend can
* display the correct values
*/
s->hw->resolution_list = malloc( (s->hw->res_list_size +1) * sizeof(SANE_Word));
if (s->hw->resolution_list == NULL)
{
DBG( 0, "out of memory\n");
return SANE_STATUS_NO_MEM;
}
*(s->hw->resolution_list) = s->hw->res_list_size;
memcpy(&(s->hw->resolution_list[1]), s->hw->res_list, s->hw->res_list_size * sizeof(SANE_Word));
return SANE_STATUS_GOOD;
} /* request identity */
/*
* static SANE_Status get_identity2_information ( SANE_Handle handle)
*
* Request Identity2 information from scanner and fill in information
* into dev and/or scanner structures.
*/
static SANE_Status
get_identity2_information(SANE_Handle handle)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Status status;
int len;
u_char param[3];
u_char result[4];
u_char *buf;
DBG(5, "get_identity2_information()\n");
if (s->hw->cmd->request_identity2 == 0)
return SANE_STATUS_UNSUPPORTED;
param[0] = ESC;
param[1] = s->hw->cmd->request_identity2;
param[2] = '\0';
send( s, param, 2, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = 4; /* receive header */
receive( s, result, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = result[ 3] << 8 | result[ 2];
buf = alloca( len);
receive( s, buf, len, &status); /* reveive actual status data */
if( buf[ 0] & 0x80) {
close_scanner( s);
return SANE_STATUS_INVAL;
}
/* the first two bytes of the buffer contain the optical resolution */
s->hw->optical_res = buf[1] << 8 | buf[0];
/*
* the 4th and 5th byte contain the line distance. Both values have to
* be identical, otherwise this software can not handle this scanner.
*/
if (buf[4] != buf[5])
{
close_scanner(s);
return SANE_STATUS_INVAL;
}
s->hw->max_line_distance = buf[4];
return SANE_STATUS_GOOD;
}
/*
* void sane_cancel(SANE_Handle handle)
*
* Set the cancel flag to true. The next time the backend requests data
* from the scanner the CAN message will be sent.
*/
void sane_cancel ( SANE_Handle handle) {
Epson_Scanner * s = ( Epson_Scanner *) handle;
if( s->buf != NULL)
s->canceling = SANE_TRUE;
}
static SANE_Status
request_focus_position(SANE_Handle handle, u_char * position)
{
Epson_Scanner * s = ( Epson_Scanner *) handle;
SANE_Status status;
int len;
u_char param[3];
u_char result[4];
u_char *buf;
DBG(5, "request_focus_position()\n");
if (s->hw->cmd->request_focus_position == 0)
return SANE_STATUS_UNSUPPORTED;
param[0] = ESC;
param[1] = s->hw->cmd->request_focus_position;
param[2] = '\0';
send( s, param, 2, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = 4; /* receive header */
receive( s, result, len, &status);
if( SANE_STATUS_GOOD != status)
return status;
len = result[ 3] << 8 | result[ 2];
buf = alloca( len);
receive( s, buf, len, &status); /* reveive actual status data */
*position = buf[ 1];
DBG(1, "Focus position = 0x%x\n", buf[1]);
return SANE_STATUS_GOOD;
}
/**********************************************************************************/
/*
* SANE_Status sane_set_io_mode()
*
* not supported - for asynchronous I/O
*/
SANE_Status sane_set_io_mode ( SANE_Handle handle, SANE_Bool non_blocking)
{
/* get rid of compiler warning */
handle = handle;
non_blocking = non_blocking;
return SANE_STATUS_UNSUPPORTED;
}
/*
* SANE_Status sane_get_select_fd()
*
* not supported - for asynchronous I/O
*/
SANE_Status sane_get_select_fd ( SANE_Handle handle, SANE_Int * fd)
{
/* get rid of compiler warnings */
handle = handle;
fd = fd;
return SANE_STATUS_UNSUPPORTED;
}