kopia lustrzana https://gitlab.com/sane-project/backends
416 wiersze
12 KiB
C
416 wiersze
12 KiB
C
/* sane - Scanner Access Now Easy.
|
|
Copyright (C) 2002 Frank Zago (fzago at austin dot rr dot com)
|
|
|
|
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.
|
|
*/
|
|
|
|
/*
|
|
$Id$
|
|
*/
|
|
|
|
/* Commands supported by the Sceptre S1200 scanner. */
|
|
#define SCSI_TEST_UNIT_READY 0x00
|
|
#define SCSI_GET_STATUS 0x02
|
|
#define SCSI_INQUIRY 0x12
|
|
#define SCSI_MODE_SELECT 0x15
|
|
#define SCSI_MODE_SENSE 0x1a
|
|
#define SCSI_SCAN 0x1b
|
|
#define SCSI_RECEIVE_DIAG 0x1c
|
|
#define SCSI_SEND_DIAG 0x1D
|
|
#define SCSI_SET_WINDOW 0x24
|
|
#define SCSI_READ_10 0x28
|
|
#define SCSI_REQUEST_SENSE 0x03
|
|
#define SCSI_SEND_10 0x2a
|
|
#define SCSI_GET_DATA_BUFFER_STATUS 0x34
|
|
|
|
typedef struct
|
|
{
|
|
unsigned char data[16];
|
|
int len;
|
|
}
|
|
CDB;
|
|
|
|
|
|
/* Set a specific bit depending on a boolean.
|
|
* MKSCSI_BIT(TRUE, 3) will generate 0x08. */
|
|
#define MKSCSI_BIT(bit, pos) ((bit)? 1<<(pos): 0)
|
|
|
|
/* Set a value in a range of bits.
|
|
* MKSCSI_I2B(5, 3, 5) will generate 0x28 */
|
|
#define MKSCSI_I2B(bits, pos_b, pos_e) ((bits) << (pos_b) & ((1<<((pos_e)-(pos_b)+1))-1))
|
|
|
|
/* Store an integer in 2, 3 or 4 byte in an array. */
|
|
#define Ito16(val, buf) { \
|
|
((unsigned char *)buf)[0] = ((val) >> 8) & 0xff; \
|
|
((unsigned char *)buf)[1] = ((val) >> 0) & 0xff; \
|
|
}
|
|
|
|
#define Ito24(val, buf) { \
|
|
((unsigned char *)buf)[0] = ((val) >> 16) & 0xff; \
|
|
((unsigned char *)buf)[1] = ((val) >> 8) & 0xff; \
|
|
((unsigned char *)buf)[2] = ((val) >> 0) & 0xff; \
|
|
}
|
|
|
|
#define Ito32(val, buf) { \
|
|
((unsigned char *)buf)[0] = ((val) >> 24) & 0xff; \
|
|
((unsigned char *)buf)[1] = ((val) >> 16) & 0xff; \
|
|
((unsigned char *)buf)[2] = ((val) >> 8) & 0xff; \
|
|
((unsigned char *)buf)[3] = ((val) >> 0) & 0xff; \
|
|
}
|
|
|
|
#define MKSCSI_TEST_UNIT_READY(cdb) \
|
|
cdb.data[0] = SCSI_TEST_UNIT_READY; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = 0; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_GET_STATUS(cdb, buflen) \
|
|
cdb.data[0] = SCSI_GET_STATUS; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = buflen; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_INQUIRY(cdb, buflen) \
|
|
cdb.data[0] = SCSI_INQUIRY; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = buflen; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_MODE_SELECT(cdb, pf, sp, buflen) \
|
|
cdb.data[0] = SCSI_MODE_SELECT; \
|
|
cdb.data[1] = MKSCSI_BIT(pf, 4) | MKSCSI_BIT(sp, 0); \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = buflen; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_MODE_SENSE(cdb, pc, page_code, buflen) \
|
|
cdb.data[0] = SCSI_MODE_SENSE; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = MKSCSI_I2B(pc, 6, 7) | MKSCSI_I2B(page_code, 0, 5); \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = buflen; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_SCAN(cdb) \
|
|
cdb.data[0] = SCSI_SCAN; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = 0; \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_RECEIVE_DIAG(cdb, pc, buflen) \
|
|
cdb.data[0] = SCSI_RECEIVE_DIAG; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = pc; \
|
|
cdb.data[3] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[4] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_SEND_DIAG(cdb, buflen) \
|
|
cdb.data[0] = SCSI_SEND_DIAG; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[4] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_SET_WINDOW(cdb, buflen) \
|
|
cdb.data[0] = SCSI_SET_WINDOW; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = 0; \
|
|
cdb.data[5] = 0; \
|
|
cdb.data[6] = (((buflen) >> 16) & 0xff); \
|
|
cdb.data[7] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[8] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[9] = 0; \
|
|
cdb.len = 10;
|
|
|
|
#define MKSCSI_READ_10(cdb, dtc, dtq, buflen) \
|
|
cdb.data[0] = SCSI_READ_10; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = (dtc); \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = (((dtq) >> 8) & 0xff); \
|
|
cdb.data[5] = (((dtq) >> 0) & 0xff); \
|
|
cdb.data[6] = (((buflen) >> 16) & 0xff); \
|
|
cdb.data[7] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[8] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[9] = 0; \
|
|
cdb.len = 10;
|
|
|
|
#define MKSCSI_REQUEST_SENSE(cdb, buflen) \
|
|
cdb.data[0] = SCSI_REQUEST_SENSE; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = (buflen); \
|
|
cdb.data[5] = 0; \
|
|
cdb.len = 6;
|
|
|
|
#define MKSCSI_SEND_10(cdb, dtc, dtq, buflen) \
|
|
cdb.data[0] = SCSI_SEND_10; \
|
|
cdb.data[1] = 0; \
|
|
cdb.data[2] = (dtc); \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = (((dtq) >> 8) & 0xff); \
|
|
cdb.data[5] = (((dtq) >> 0) & 0xff); \
|
|
cdb.data[6] = (((buflen) >> 16) & 0xff); \
|
|
cdb.data[7] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[8] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[9] = 0; \
|
|
cdb.len = 10;
|
|
|
|
#define MKSCSI_GET_DATA_BUFFER_STATUS(cdb, wait, buflen) \
|
|
cdb.data[0] = SCSI_GET_DATA_BUFFER_STATUS; \
|
|
cdb.data[1] = MKSCSI_BIT(wait, 0); \
|
|
cdb.data[2] = 0; \
|
|
cdb.data[3] = 0; \
|
|
cdb.data[4] = 0; \
|
|
cdb.data[5] = 0; \
|
|
cdb.data[6] = 0; \
|
|
cdb.data[7] = (((buflen) >> 8) & 0xff); \
|
|
cdb.data[8] = (((buflen) >> 0) & 0xff); \
|
|
cdb.data[9] = 0; \
|
|
cdb.len = 10;
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
#define MM_PER_INCH 25.4
|
|
#define length_quant SANE_UNFIX(SANE_FIX(MM_PER_INCH / 600))
|
|
#define mmToIlu(mm) ((mm) / length_quant)
|
|
#define iluToMm(ilu) ((ilu) * length_quant)
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
#define GAMMA_LENGTH 0x100 /* number of value per color */
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
enum Sceptre_Option
|
|
{
|
|
OPT_NUM_OPTS = 0,
|
|
|
|
OPT_MODE_GROUP,
|
|
OPT_MODE, /* scanner modes */
|
|
OPT_RESOLUTION, /* X and Y resolution */
|
|
|
|
OPT_GEOMETRY_GROUP,
|
|
OPT_TL_X, /* upper left X */
|
|
OPT_TL_Y, /* upper left Y */
|
|
OPT_BR_X, /* bottom right X */
|
|
OPT_BR_Y, /* bottom right Y */
|
|
|
|
OPT_ENHANCEMENT_GROUP,
|
|
OPT_CUSTOM_GAMMA, /* Use the custom gamma tables */
|
|
OPT_GAMMA_VECTOR_R, /* Custom Red gamma table */
|
|
OPT_GAMMA_VECTOR_G, /* Custom Green Gamma table */
|
|
OPT_GAMMA_VECTOR_B, /* Custom Blue Gamma table */
|
|
OPT_THRESHOLD, /* Threshold */
|
|
OPT_HALFTONE_PATTERN, /* Halftone pattern (1 to 4) */
|
|
|
|
OPT_PREVIEW, /* preview mode */
|
|
|
|
/* must come last: */
|
|
OPT_NUM_OPTIONS
|
|
};
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
* Scanner supported by this backend.
|
|
*/
|
|
struct scanners_supported
|
|
{
|
|
/* From scsi inquiry */
|
|
int scsi_type;
|
|
char scsi_vendor[9];
|
|
char scsi_product[17];
|
|
|
|
char *real_vendor;
|
|
char *real_product;
|
|
};
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
#define LINEART_STR SANE_I18N("Lineart")
|
|
#define HALFTONE_STR SANE_I18N("Halftone")
|
|
#define GRAY_STR SANE_I18N("Gray")
|
|
#define COLOR_STR SANE_I18N("Color")
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/* Define a scanner occurence. */
|
|
typedef struct Sceptre_Scanner
|
|
{
|
|
struct Sceptre_Scanner *next;
|
|
SANE_Device sane;
|
|
|
|
char *devicename;
|
|
int sfd; /* device handle */
|
|
|
|
/* Infos from inquiry. */
|
|
char scsi_type;
|
|
char scsi_vendor[9];
|
|
char scsi_product[17];
|
|
char scsi_version[5];
|
|
|
|
/* Scanner infos. */
|
|
SANE_Range x_range;
|
|
SANE_Range y_range;
|
|
SANE_Range resolution_range;
|
|
int scnum; /* index of that scanner in
|
|
* scanners_supported */
|
|
|
|
/* SCSI handling */
|
|
SANE_Byte *buffer; /* for SCSI transfer. */
|
|
size_t buffer_size; /* allocated size of buffer */
|
|
|
|
/* Scanning handling. */
|
|
int scanning; /* TRUE is a scan is running. */
|
|
int resolution; /* scan resolution */
|
|
int x_tl; /* X top left */
|
|
int y_tl; /* Y top left */
|
|
int x_br; /* X bottom right */
|
|
int y_br; /* Y bottom right */
|
|
int width; /* width of the scan area in mm */
|
|
int length; /* length of the scan area in mm */
|
|
|
|
enum
|
|
{
|
|
SCEPTRE_LINEART,
|
|
SCEPTRE_HALFTONE,
|
|
SCEPTRE_GRAYSCALE,
|
|
SCEPTRE_COLOR
|
|
}
|
|
scan_mode;
|
|
|
|
int depth; /* depth per color */
|
|
int halftone_param; /* haltone number, valid for SCEPTRE_HALFTONE */
|
|
|
|
size_t bytes_left; /* number of bytes left to give to the backend */
|
|
|
|
size_t real_bytes_left; /* number of bytes left the scanner will return. */
|
|
|
|
SANE_Byte *image; /* keep the raw image here */
|
|
size_t image_size; /* allocated size of image */
|
|
size_t image_begin; /* first significant byte in image */
|
|
size_t image_end; /* first free byte in image */
|
|
|
|
int color_shift; /* for color scan: number of lines to
|
|
* shift the colors. The higher the
|
|
* resolution, the higher this
|
|
* number. */
|
|
|
|
int raster_size; /* size of a raster */
|
|
int raster_num; /* for colour scan, current raster read */
|
|
int raster_real; /* real number of raster in the
|
|
* scan. This is necessary since I
|
|
* don't know how to reliably compute
|
|
* the number of lines */
|
|
|
|
int raster_ahead; /* max size of the incomplete lines */
|
|
|
|
int line; /* current line of the scan */
|
|
|
|
|
|
SANE_Parameters params;
|
|
|
|
/* Options */
|
|
SANE_Option_Descriptor opt[OPT_NUM_OPTIONS];
|
|
Option_Value val[OPT_NUM_OPTIONS];
|
|
|
|
/* Gamma table. 1 array per colour. */
|
|
SANE_Word gamma_R[GAMMA_LENGTH];
|
|
SANE_Word gamma_G[GAMMA_LENGTH];
|
|
SANE_Word gamma_B[GAMMA_LENGTH];
|
|
|
|
}
|
|
Sceptre_Scanner;
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/* Debug levels.
|
|
* Should be common to all backends. */
|
|
|
|
#define DBG_error0 0
|
|
#define DBG_error 1
|
|
#define DBG_sense 2
|
|
#define DBG_warning 3
|
|
#define DBG_inquiry 4
|
|
#define DBG_info 5
|
|
#define DBG_info2 6
|
|
#define DBG_proc 7
|
|
#define DBG_read 8
|
|
#define DBG_sane_init 10
|
|
#define DBG_sane_proc 11
|
|
#define DBG_sane_info 12
|
|
#define DBG_sane_option 13
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/* 32 bits from an array to an integer (eg ntohl). */
|
|
#define B32TOI(buf) \
|
|
((((unsigned char *)buf)[0] << 24) | \
|
|
(((unsigned char *)buf)[1] << 16) | \
|
|
(((unsigned char *)buf)[2] << 8) | \
|
|
(((unsigned char *)buf)[3] << 0))
|
|
|
|
/* 16 bits from an array to an integer (eg ntohs). */
|
|
#define B16TOI(buf) \
|
|
((((unsigned char *)buf)[0] << 8) | \
|
|
(((unsigned char *)buf)[1] << 0))
|