diff options
Diffstat (limited to 'backend/canon_dr.c')
-rw-r--r-- | backend/canon_dr.c | 8133 |
1 files changed, 8133 insertions, 0 deletions
diff --git a/backend/canon_dr.c b/backend/canon_dr.c new file mode 100644 index 0000000..594aab2 --- /dev/null +++ b/backend/canon_dr.c @@ -0,0 +1,8133 @@ +/* sane - Scanner Access Now Easy. + + This file is part of the SANE package, and implements a SANE backend + for various Canon DR-series scanners. + + Copyright (C) 2008-2010 m. allan noah + + Yabarana Corp. www.yabarana.com provided significant funding + EvriChart, Inc. www.evrichart.com provided funding and loaned equipment + Canon, USA. www.usa.canon.com loaned equipment + HPrint hprint.com.br provided funding and testing for DR-2510 support + Stone-IT www.stone-it.com provided funding for DR-2010 and DR-2050 support + + -------------------------------------------------------------------------- + + 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. + + -------------------------------------------------------------------------- + + The source code is divided in sections which you can easily find by + searching for the tag "@@". + + Section 1 - Init & static stuff + Section 2 - sane_init, _get_devices, _open & friends + Section 3 - sane_*_option functions + Section 4 - sane_start, _get_param, _read & friends + Section 5 - calibration functions + Section 6 - sane_close functions + Section 7 - misc functions + Section 8 - image processing functions + + Changes: + v1 2008-10-29, MAN + - initial version + v2 2008-11-04, MAN + - round scanlines to even bytes + - spin RS and usb_clear_halt code into new function + - update various scsi payloads + - calloc out block so it gets set to 0 initially + v3 2008-11-07, MAN + - back window uses id 1 + - add option and functions to read/send page counter + - add rif option + v4 2008-11-11, MAN + - eject document when sane_read() returns EOF + v5 2008-11-25, MAN + - remove EOF ejection code + - add SSM and GSM commands + - add dropout, doublefeed, and jpeg compression options + - disable adf backside + - fix adf duplex + - read two extra lines (ignore errors) at end of image + - only send scan command at beginning of batch + - fix bug in hexdump with 0 length string + - DR-7580 support + v6 2008-11-29, MAN + - fix adf simplex + - rename ssm_duplex to ssm_buffer + - add --buffer option + - reduce inter-page commands when buffering is enabled + - improve sense_handler output + - enable counter option + - drop unused code + v7 2008-11-29, MAN + - jpeg support (size rounding and header overwrite) + - call object_position(load) between pages even if buffering is on + - use request sense info bytes on short scsi reads + - byte swap color BGR to RGB + - round image width down, not up + - round image height down to even # of lines + - always transfer even # of lines per block + - scsi and jpeg don't require reading extra lines to reach EOF + - rename buffer option to buffermode to avoid conflict with scanimage + - send ssm_do and ssm_df during sane_start + - improve sense_handler output + v8 2008-12-07, MAN + - rename read/send_counter to read/send_panel + - enable control panel during init + - add options for all buttons + - call TUR twice in wait_scanner(), even if first succeeds + - disable rif + - enable brightness/contrast/threshold options + v9 2008-12-07, MAN + - add rollerdeskew and stapledetect options + - add rollerdeskew and stapledetect bits to ssm_df() + v10 2008-12-10, MAN + - add all documented request sense codes to sense_handler() + - fix color jpeg (remove unneeded BGR to RGB swapping code) + - add macros for LUT data + v11 2009-01-10, MAN + - send_panel() can disable too + - add cancel() to send d8 command + - call cancel() only after final read from scanner + - stop button reqests cancel + v12 2009-01-21, MAN + - dont export private symbols + v13 2009-03-06, MAN + - new vendor ID for recent machines + - add usb ids for several new machines + v14 2009-03-07, MAN + - remove HARD_SELECT from counter (Legitimate, but API violation) + - attach to CR-series scanners as well + v15 2009-03-15, MAN + - add byte-oriented duplex interlace code + - add RRGGBB color interlace code + - add basic support for DR-2580C + v16 2009-03-20, MAN + - add more unknown setwindow bits + - add support for 16 byte status packets + - clean do_usb_cmd error handling (call reset more often) + - add basic support for DR-2050C, DR-2080C, DR-2510C + v17 2009-03-20, MAN + - set status packet size from config file + v18 2009-03-21, MAN + - rewrite config file parsing to reset options after each scanner + - add config options for vendor, model, version + - dont call inquiry if those 3 options are set + - remove default config file from code + - add initial gray deinterlacing code for DR-2510C + - rename do_usb_reset to do_usb_clear + v19 2009-03-22, MAN + - pad gray deinterlacing area for DR-2510C + - override tl_x and br_x for fixed width scanners + v20 2009-03-23, MAN + - improved macros for inquiry and set window + - shorten inquiry vpd length to match windows driver + - remove status-length config option + - add padded-read config option + - rewrite do_usb_cmd to pad reads and calloc/copy buffers + v21 2009-03-24, MAN + - correct rgb padding macro + - skip send_panel and ssm_df commands for DR-20xx scanners + v22 2009-03-25, MAN + - add deinterlacing code for DR-2510C in duplex and color + v23 2009-03-27, MAN + - rewrite all image data processing code + - handle more image interlacing formats + - re-enable binary mode on some scanners + - limit some machines to full-width scanning + v24 2009-04-02, MAN + - fix DR-2510C duplex deinterlacing code + - rewrite sane_read helpers to read until EOF + - update sane_start for scanners that dont use object_position + - dont call sanei_usb_clear_halt() if device is not open + - increase default buffer size to 4 megs + - set buffermode on by default + - hide modes and resolutions that DR-2510C lies about + - read_panel() logs front-end access to sensors instead of timing + - rewrite do_usb_cmd() to use remainder from RS info + v25 2009-04-12, MAN + - disable SANE_FRAME_JPEG + v26 2009-04-14, MAN (SANE 1.0.20) + - return cmd status for reads on sensors + - allow rs to adjust read length for all bad status responses + v27 2009-05-08, MAN + - bug fix in read_panel() + - initialize vars in do_usb_cmd() + - set buffermode off by default + - clear page counter during init and sane_start() + - eject previous page during init and sane_start() + - improved SSM_BUFF macros + - moved set_window() to after ssm-*() + - add coarse calibration (AFE offset/gain & per-channel exposure) + - add fine calibration (per-cell offset/gain) + - free image and fine cal buffers in sane_close() + - compare page counter of small scanners only in non-buffered mode + - add back-side gray mirroring code for DR-2580C + v28 2009-05-20, MAN + - use average instead of min/max for fine offset and gain + - rewrite supported resolution list as x and y arrays + - merge x and y resolution options into single option + - move scan params into two new structs, s->u and s->s + - sane_get_parameters() just returns values from s->u + - dont call wait_scanner() in object_position() + - dont call ssm_*() from option handler + - refactor sane_start() + - read_from_buffer() can workaround missing res, modes and cropping + - set most DR-2xxx machines to use the read_from_buffer workarounds + - set default threshold to 90 + - add option for button #3 of some machines + - don't eject paper during init + - add DR-2010 quirks + - switch counter to HARD_SELECT, not SOFT + v29 2009-06-01, MAN + - split coarse and fine cal to run independently + - add side option + - reset scan params to user request if calibration fails + - better handling of sane_cancel + - better handling of errors during sane_start and sane_read + v30 2009-06-17, MAN + - add fine cal support for machines with internal buffer (2050/2080) + - support fixed-width machines that require even bytes per scanline + - pad end of scan with gray if scanner stops prematurely + - better handling of errors during calibration + - cleanup canceling debug messages + - remove old cancel() prototype + - small sleep before clearing usb halt condition + v31 2009-06-29, MAN + - reduce default buffer size to 2 megs + v32 2009-07-21, MAN + - crop/resample image data before buffering, not after + - shink image buffers to size of output image, not input + - correct some debug message + - better handling of EOF + - add intermediate param struct to existing user and scan versions + v33 2009-07-23, MAN + - add software brightness/contrast for dumb scanners + - add blocking mode to allow full-page manipulation options to run + - add swdespeck option and support code + - add swdeskew and swcrop options (disabled) + v34 2009-07-28, MAN + - add simplified Hough transform based deskewing code + - add extremity detecting cropping code + - use per-model background color to fill corners after deskew + - request and chop extra scanlines instead of rounding down + - remove padding dumb scanners add to top of front side + - sane_get_params uses intermediate struct instead of user struct + - if scanner stops, clone the last line until the end of buffer + - reset some intermediate params between duplex sides + v35 2010-02-09, MAN (SANE 1.0.21) + - cleanup #includes and copyright + - add SANE_I18N to static strings + - don't fail if scsi buffer is too small + v36 2011-01-03, MAN + - initial support for DR-3080 and DR-5060 + - add code to clamp scan width to an arbitrary byte width boundary + - add code to prevent setting of brightness/threshold/contrast + - don't send dropout color command on non-color scanners + - initial support for DR-7090C + - update credits + v37 2011-01-26, MAN (SANE 1.0.22) + - don't center window when using flatbed + - improve request sense error messages + - enable flatbed for all known models + v38 2011-07-06, MAN + - initial support for DR-5020 + - use ppl_mod instead of Bpl_mod, apply to all modes + - invert logic of read_panel tracking + - add ability to disable read_panel() + - automatically disable read/send_panel if unsupported + v39 2011-11-01, MAN + - DR-2580C pads the backside of duplex scans + v40 2012-11-01, MAN + - initial DR-9050C, DR-7550C, DR-6050C and DR-3010C support + v41 2013-07-31, MAN + - initial P-208 and P-215 support + - bug fix for calibration of scanners with duplex_offset + - allow duplex_offset to be controlled from config file + + SANE FLOW DIAGRAM + + - sane_init() : initialize backend + . - sane_get_devices() : query list of scanner devices + . - sane_open() : open a particular scanner device + . . - sane_set_io_mode : set blocking mode + . . - sane_get_select_fd : get scanner fd + . . + . . - sane_get_option_descriptor() : get option information + . . - sane_control_option() : change option values + . . - sane_get_parameters() : returns estimated scan parameters + . . - (repeat previous 3 functions) + . . + . . - sane_start() : start image acquisition + . . - sane_get_parameters() : returns actual scan parameters + . . - sane_read() : read image data (from pipe) + . . (sane_read called multiple times; after sane_read returns EOF, + . . loop may continue with sane_start which may return a 2nd page + . . when doing duplex scans, or load the next page from the ADF) + . . + . . - sane_cancel() : cancel operation + . - sane_close() : close opened scanner device + - sane_exit() : terminate use of backend + +*/ + +/* + * @@ Section 1 - Init + */ + +#include "../include/sane/config.h" + +#include <string.h> /*memcpy...*/ +#include <ctype.h> /*isspace*/ +#include <math.h> /*tan*/ +#include <unistd.h> /*usleep*/ + +#include "../include/sane/sanei_backend.h" +#include "../include/sane/sanei_scsi.h" +#include "../include/sane/sanei_usb.h" +#include "../include/sane/saneopts.h" +#include "../include/sane/sanei_config.h" + +#include "canon_dr-cmd.h" +#include "canon_dr.h" + +#define DEBUG 1 +#define BUILD 41 + +/* values for SANE_DEBUG_CANON_DR env var: + - errors 5 + - function trace 10 + - function detail 15 + - get/setopt cmds 20 + - scsi/usb trace 25 + - scsi/usb detail 30 + - useless noise 35 +*/ + +/* ------------------------------------------------------------------------- */ +#define STRING_FLATBED SANE_I18N("Flatbed") +#define STRING_ADFFRONT SANE_I18N("ADF Front") +#define STRING_ADFBACK SANE_I18N("ADF Back") +#define STRING_ADFDUPLEX SANE_I18N("ADF Duplex") + +#define STRING_LINEART SANE_VALUE_SCAN_MODE_LINEART +#define STRING_HALFTONE SANE_VALUE_SCAN_MODE_HALFTONE +#define STRING_GRAYSCALE SANE_VALUE_SCAN_MODE_GRAY +#define STRING_COLOR SANE_VALUE_SCAN_MODE_COLOR + +#define STRING_RED SANE_I18N("Red") +#define STRING_GREEN SANE_I18N("Green") +#define STRING_BLUE SANE_I18N("Blue") +#define STRING_EN_RED SANE_I18N("Enhance Red") +#define STRING_EN_GREEN SANE_I18N("Enhance Green") +#define STRING_EN_BLUE SANE_I18N("Enhance Blue") + +#define STRING_NONE SANE_I18N("None") +#define STRING_JPEG SANE_I18N("JPEG") + +/* Also set via config file. */ +static int global_buffer_size; +static int global_buffer_size_default = 2 * 1024 * 1024; +static int global_padded_read; +static int global_padded_read_default = 0; +static int global_duplex_offset; +static int global_duplex_offset_default = 0; +static char global_vendor_name[9]; +static char global_model_name[17]; +static char global_version_name[5]; + +/* + * used by attach* and sane_get_devices + * a ptr to a null term array of ptrs to SANE_Device structs + * a ptr to a single-linked list of scanner structs + */ +static const SANE_Device **sane_devArray = NULL; +static struct scanner *scanner_devList = NULL; + +/* + * @@ Section 2 - SANE & scanner init code + */ + +/* + * Called by SANE initially. + * + * From the SANE spec: + * This function must be called before any other SANE function can be + * called. The behavior of a SANE backend is undefined if this + * function is not called first. The version code of the backend is + * returned in the value pointed to by version_code. If that pointer + * is NULL, no version code is returned. Argument authorize is either + * a pointer to a function that is invoked when the backend requires + * authentication for a specific resource or NULL if the frontend does + * not support authentication. + */ +SANE_Status +sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize) +{ + authorize = authorize; /* get rid of compiler warning */ + + DBG_INIT (); + DBG (10, "sane_init: start\n"); + + if (version_code) + *version_code = SANE_VERSION_CODE (SANE_CURRENT_MAJOR, V_MINOR, BUILD); + + DBG (5, "sane_init: canon_dr backend %d.%d.%d, from %s\n", + SANE_CURRENT_MAJOR, V_MINOR, BUILD, PACKAGE_STRING); + + DBG (10, "sane_init: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * Called by SANE to find out about supported devices. + * + * From the SANE spec: + * This function can be used to query the list of devices that are + * available. If the function executes successfully, it stores a + * pointer to a NULL terminated array of pointers to SANE_Device + * structures in *device_list. The returned list is guaranteed to + * remain unchanged and valid until (a) another call to this function + * is performed or (b) a call to sane_exit() is performed. This + * function can be called repeatedly to detect when new devices become + * available. If argument local_only is true, only local devices are + * returned (devices directly attached to the machine that SANE is + * running on). If it is false, the device list includes all remote + * devices that are accessible to the SANE library. + * + * SANE does not require that this function is called before a + * sane_open() call is performed. A device name may be specified + * explicitly by a user which would make it unnecessary and + * undesirable to call this function first. + */ +/* + * Read the config file, find scanners with help from sanei_* + * and store in global device structs + */ +SANE_Status +sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only) +{ + SANE_Status ret = SANE_STATUS_GOOD; + struct scanner * s; + struct scanner * prev = NULL; + char line[PATH_MAX]; + const char *lp; + FILE *fp; + int num_devices=0; + int i=0; + + local_only = local_only; /* get rid of compiler warning */ + + DBG (10, "sane_get_devices: start\n"); + + /* mark all existing scanners as missing, attach_one will remove mark */ + for (s = scanner_devList; s; s = s->next) { + s->missing = 1; + } + + sanei_usb_init(); + + /* reset globals before reading the file */ + default_globals(); + + fp = sanei_config_open (CANON_DR_CONFIG_FILE); + + if (fp) { + + DBG (15, "sane_get_devices: reading config file %s\n", + CANON_DR_CONFIG_FILE); + + while (sanei_config_read (line, PATH_MAX, fp)) { + + lp = line; + + /* ignore comments */ + if (*lp == '#') + continue; + + /* skip empty lines */ + if (*lp == 0) + continue; + + if (!strncmp ("option", lp, 6) && isspace (lp[6])) { + + lp += 6; + lp = sanei_config_skip_whitespace (lp); + + /* BUFFERSIZE: > 4K */ + if (!strncmp (lp, "buffer-size", 11) && isspace (lp[11])) { + + int buf; + lp += 11; + lp = sanei_config_skip_whitespace (lp); + buf = atoi (lp); + + if (buf < 4096) { + DBG (5, "sane_get_devices: config option \"buffer-size\" " + "(%d) is < 4096, ignoring!\n", buf); + continue; + } + + if (buf > global_buffer_size_default) { + DBG (5, "sane_get_devices: config option \"buffer-size\" " + "(%d) is > %d, scanning problems may result\n", buf, + global_buffer_size_default); + } + + DBG (15, "sane_get_devices: setting \"buffer-size\" to %d\n", + buf); + + global_buffer_size = buf; + } + + /* PADDED READ: we clamp to 0 or 1 */ + else if (!strncmp (lp, "padded-read", 11) && isspace (lp[11])) { + + int buf; + lp += 11; + lp = sanei_config_skip_whitespace (lp); + buf = atoi (lp); + + if (buf < 0) { + DBG (5, "sane_get_devices: config option \"padded-read\" " + "(%d) is < 0, ignoring!\n", buf); + continue; + } + + if (buf > 1) { + DBG (5, "sane_get_devices: config option \"padded-read\" " + "(%d) is > 1, ignoring!\n", buf); + continue; + } + + DBG (15, "sane_get_devices: setting \"padded-read\" to %d\n", + buf); + + global_padded_read = buf; + } + + /* DUPLEXOFFSET: < 1200 */ + else if (!strncmp (lp, "duplex-offset", 13) && isspace (lp[13])) { + + int buf; + lp += 13; + lp = sanei_config_skip_whitespace (lp); + buf = atoi (lp); + + if (buf > 1200) { + DBG (5, "sane_get_devices: config option \"duplex-offset\" " + "(%d) is > 1200, ignoring!\n", buf); + continue; + } + + if (buf < 0) { + DBG (5, "sane_get_devices: config option \"duplex-offset\" " + "(%d) is < 0, ignoring!\n", buf); + continue; + } + + DBG (15, "sane_get_devices: setting \"duplex-offset\" to %d\n", + buf); + + global_duplex_offset = buf; + } + + /* VENDOR: we ingest up to 8 bytes */ + else if (!strncmp (lp, "vendor-name", 11) && isspace (lp[11])) { + + lp += 11; + lp = sanei_config_skip_whitespace (lp); + strncpy(global_vendor_name, lp, 8); + global_vendor_name[8] = 0; + + DBG (15, "sane_get_devices: setting \"vendor-name\" to %s\n", + global_vendor_name); + } + + /* MODEL: we ingest up to 16 bytes */ + else if (!strncmp (lp, "model-name", 10) && isspace (lp[10])) { + + lp += 10; + lp = sanei_config_skip_whitespace (lp); + strncpy(global_model_name, lp, 16); + global_model_name[16] = 0; + + DBG (15, "sane_get_devices: setting \"model-name\" to %s\n", + global_model_name); + } + + /* VERSION: we ingest up to 4 bytes */ + else if (!strncmp (lp, "version-name", 12) && isspace (lp[12])) { + + lp += 12; + lp = sanei_config_skip_whitespace (lp); + strncpy(global_version_name, lp, 4); + global_version_name[4] = 0; + + DBG (15, "sane_get_devices: setting \"version-name\" to %s\n", + global_version_name); + } + + else { + DBG (5, "sane_get_devices: config option \"%s\" unrecognized " + "- ignored.\n", lp); + } + } + else if ((strncmp ("usb", lp, 3) == 0) && isspace (lp[3])) { + DBG (15, "sane_get_devices: looking for '%s'\n", lp); + sanei_usb_attach_matching_devices(lp, attach_one_usb); + + /* re-default these after reading the usb line */ + default_globals(); + } + else if ((strncmp ("scsi", lp, 4) == 0) && isspace (lp[4])) { + DBG (15, "sane_get_devices: looking for '%s'\n", lp); + sanei_config_attach_matching_devices (lp, attach_one_scsi); + + /* re-default these after reading the scsi line */ + default_globals(); + } + else{ + DBG (5, "sane_get_devices: config line \"%s\" unrecognized - " + "ignored.\n", lp); + } + } + fclose (fp); + } + + else { + DBG (5, "sane_get_devices: missing required config file '%s'!\n", + CANON_DR_CONFIG_FILE); + } + + /*delete missing scanners from list*/ + for (s = scanner_devList; s;) { + if(s->missing){ + DBG (5, "sane_get_devices: missing scanner %s\n",s->device_name); + + /*splice s out of list by changing pointer in prev to next*/ + if(prev){ + prev->next = s->next; + free(s); + s=prev->next; + } + /*remove s from head of list, using prev to cache it*/ + else{ + prev = s; + s = s->next; + free(prev); + prev=NULL; + + /*reset head to next s*/ + scanner_devList = s; + } + } + else{ + prev = s; + s=prev->next; + } + } + + for (s = scanner_devList; s; s=s->next) { + DBG (15, "sane_get_devices: found scanner %s\n",s->device_name); + num_devices++; + } + + DBG (15, "sane_get_devices: found %d scanner(s)\n",num_devices); + + if (sane_devArray) + free (sane_devArray); + + sane_devArray = calloc (num_devices + 1, sizeof (SANE_Device*)); + if (!sane_devArray) + return SANE_STATUS_NO_MEM; + + for (s = scanner_devList; s; s=s->next) { + sane_devArray[i++] = (SANE_Device *)&s->sane; + } + sane_devArray[i] = 0; + + if(device_list){ + *device_list = sane_devArray; + } + + DBG (10, "sane_get_devices: finish\n"); + + return ret; +} + +/* callbacks used by sane_get_devices */ +static SANE_Status +attach_one_scsi (const char *device_name) +{ + return attach_one(device_name,CONNECTION_SCSI); +} + +static SANE_Status +attach_one_usb (const char *device_name) +{ + return attach_one(device_name,CONNECTION_USB); +} + +/* build the scanner struct and link to global list + * unless struct is already loaded, then pretend + */ +static SANE_Status +attach_one (const char *device_name, int connType) +{ + struct scanner *s; + int ret; + + DBG (10, "attach_one: start\n"); + DBG (15, "attach_one: looking for '%s'\n", device_name); + + for (s = scanner_devList; s; s = s->next) { + if (strcmp (s->device_name, device_name) == 0){ + DBG (10, "attach_one: already attached!\n"); + s->missing = 0; + return SANE_STATUS_GOOD; + } + } + + /* build a scanner struct to hold it */ + if ((s = calloc (sizeof (*s), 1)) == NULL) + return SANE_STATUS_NO_MEM; + + /* config file settings */ + s->buffer_size = global_buffer_size; + s->padded_read = global_padded_read; + s->duplex_offset = global_duplex_offset; + + /* copy the device name */ + strcpy (s->device_name, device_name); + + /* connect the fd */ + s->connection = connType; + s->fd = -1; + ret = connect_fd(s); + if(ret != SANE_STATUS_GOOD){ + free (s); + return ret; + } + + /* query the device to load its vendor/model/version, */ + /* if config file doesn't give all three */ + if ( !strlen(global_vendor_name) + || !strlen(global_model_name) + || !strlen(global_version_name) + ){ + ret = init_inquire (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: inquiry failed\n"); + return ret; + } + } + + /* override any inquiry settings with those from config file */ + if(strlen(global_vendor_name)) + strcpy(s->vendor_name, global_vendor_name); + if(strlen(global_model_name)) + strcpy(s->model_name, global_model_name); + if(strlen(global_version_name)) + strcpy(s->version_name, global_version_name); + + /* load detailed specs/capabilities from the device */ + /* if a model cannot support inquiry vpd, this function will die */ + ret = init_vpd (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: vpd failed\n"); + return ret; + } + + /* clean up the scanner struct based on model */ + /* this is the big piece of model specific code */ + ret = init_model (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: model failed\n"); + return ret; + } + + /* enable/read the buttons */ + ret = init_panel (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: model failed\n"); + return ret; + } + + /* sets SANE option 'values' to good defaults */ + ret = init_user (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: user failed\n"); + return ret; + } + + ret = init_options (s); + if (ret != SANE_STATUS_GOOD) { + disconnect_fd(s); + free (s); + DBG (5, "attach_one: options failed\n"); + return ret; + } + + /* load strings into sane_device struct */ + s->sane.name = s->device_name; + s->sane.vendor = s->vendor_name; + s->sane.model = s->model_name; + s->sane.type = "scanner"; + + /* change name in sane_device struct if scanner has serial number + ret = init_serial (s); + if (ret == SANE_STATUS_GOOD) { + s->sane.name = s->serial_name; + } + else{ + DBG (5, "attach_one: serial number unsupported?\n"); + } + */ + + /* we close the connection, so that another backend can talk to scanner */ + disconnect_fd(s); + + /* store this scanner in global vars */ + s->next = scanner_devList; + scanner_devList = s; + + DBG (10, "attach_one: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * connect the fd in the scanner struct + */ +static SANE_Status +connect_fd (struct scanner *s) +{ + SANE_Status ret; + int buffer_size = s->buffer_size; + + DBG (10, "connect_fd: start\n"); + + if(s->fd > -1){ + DBG (5, "connect_fd: already open\n"); + ret = SANE_STATUS_GOOD; + } + else if (s->connection == CONNECTION_USB) { + DBG (15, "connect_fd: opening USB device (%s)\n", s->device_name); + ret = sanei_usb_open (s->device_name, &(s->fd)); + if(!ret){ + ret = sanei_usb_clear_halt(s->fd); + } + } + else { + DBG (15, "connect_fd: opening SCSI device (%s)\n", s->device_name); + ret = sanei_scsi_open_extended (s->device_name, &(s->fd), sense_handler, s, + &s->buffer_size); + if(!ret && buffer_size != s->buffer_size){ + DBG (5, "connect_fd: cannot get requested buffer size (%d/%d)\n", + buffer_size, s->buffer_size); + } + } + + if(ret == SANE_STATUS_GOOD){ + + /* first generation usb scanners can get flaky if not closed + * properly after last use. very first commands sent to device + * must be prepared to correct this- see wait_scanner() */ + ret = wait_scanner(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "connect_fd: could not wait_scanner\n"); + disconnect_fd(s); + } + + } + else{ + DBG (5, "connect_fd: could not open device: %d\n", ret); + } + + DBG (10, "connect_fd: finish\n"); + + return ret; +} + +/* + * This routine will check if a certain device is a Canon scanner + * It also copies interesting data from INQUIRY into the handle structure + */ +static SANE_Status +init_inquire (struct scanner *s) +{ + int i; + SANE_Status ret; + + unsigned char cmd[INQUIRY_len]; + size_t cmdLen = INQUIRY_len; + + unsigned char in[INQUIRY_std_len]; + size_t inLen = INQUIRY_std_len; + + DBG (10, "init_inquire: start\n"); + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, INQUIRY_code); + set_IN_return_size (cmd, inLen); + set_IN_evpd (cmd, 0); + set_IN_page_code (cmd, 0); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + + if (ret != SANE_STATUS_GOOD){ + DBG (10, "init_inquire: failed: %d\n", ret); + return ret; + } + + if (get_IN_periph_devtype (in) != IN_periph_devtype_scanner){ + DBG (5, "The device at '%s' is not a scanner.\n", s->device_name); + return SANE_STATUS_INVAL; + } + + get_IN_vendor (in, s->vendor_name); + get_IN_product (in, s->model_name); + get_IN_version (in, s->version_name); + + s->vendor_name[8] = 0; + s->model_name[16] = 0; + s->version_name[4] = 0; + + /* gobble trailing spaces */ + for (i = 7; s->vendor_name[i] == ' ' && i >= 0; i--) + s->vendor_name[i] = 0; + for (i = 15; s->model_name[i] == ' ' && i >= 0; i--) + s->model_name[i] = 0; + for (i = 3; s->version_name[i] == ' ' && i >= 0; i--) + s->version_name[i] = 0; + + /*check for vendor name*/ + if (strcmp ("CANON", s->vendor_name)) { + DBG (5, "The device at '%s' is reported to be made by '%s'\n", + s->device_name, s->vendor_name); + DBG (5, "This backend only supports Canon products.\n"); + return SANE_STATUS_INVAL; + } + + /*check for model name*/ + if (strncmp ("DR", s->model_name, 2) + && strncmp ("CR", s->model_name, 2) + && strncmp ("P-", s->model_name, 2) + ) { + DBG (5, "The device at '%s' is reported to be a '%s'\n", + s->device_name, s->model_name); + DBG (5, "This backend only supports Canon P-, CR & DR-series products.\n"); + return SANE_STATUS_INVAL; + } + + DBG (15, "init_inquire: Found %s scanner %s version %s at %s\n", + s->vendor_name, s->model_name, s->version_name, s->device_name); + + DBG (10, "init_inquire: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * Use INQUIRY VPD to setup more detail about the scanner + */ +static SANE_Status +init_vpd (struct scanner *s) +{ + SANE_Status ret; + + unsigned char cmd[INQUIRY_len]; + size_t cmdLen = INQUIRY_len; + + unsigned char in[INQUIRY_vpd_len]; + size_t inLen = INQUIRY_vpd_len; + + DBG (10, "init_vpd: start\n"); + + /* get EVPD */ + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, INQUIRY_code); + set_IN_return_size (cmd, inLen); + set_IN_evpd (cmd, 1); + set_IN_page_code (cmd, 0xf0); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + + DBG (15, "init_vpd: length=%0x\n",get_IN_page_length (in)); + + /* This scanner supports vital product data. + * Use this data to set dpi-lists etc. */ + if (ret == SANE_STATUS_GOOD || ret == SANE_STATUS_EOF) { + + DBG (15, "standard options\n"); + + s->basic_x_res = get_IN_basic_x_res (in); + DBG (15, " basic x res: %d dpi\n",s->basic_x_res); + + s->basic_y_res = get_IN_basic_y_res (in); + DBG (15, " basic y res: %d dpi\n",s->basic_y_res); + + s->step_x_res = get_IN_step_x_res (in); + DBG (15, " step x res: %d dpi\n", s->step_x_res); + + s->step_y_res = get_IN_step_y_res (in); + DBG (15, " step y res: %d dpi\n", s->step_y_res); + + s->max_x_res = get_IN_max_x_res (in); + DBG (15, " max x res: %d dpi\n", s->max_x_res); + + s->max_y_res = get_IN_max_y_res (in); + DBG (15, " max y res: %d dpi\n", s->max_y_res); + + s->min_x_res = get_IN_min_x_res (in); + DBG (15, " min x res: %d dpi\n", s->min_x_res); + + s->min_y_res = get_IN_min_y_res (in); + DBG (15, " min y res: %d dpi\n", s->min_y_res); + + /* some scanners list B&W resolutions. */ + s->std_res_x[DPI_60] = get_IN_std_res_60 (in); + s->std_res_y[DPI_60] = s->std_res_x[DPI_60]; + DBG (15, " 60 dpi: %d\n", s->std_res_x[DPI_60]); + + s->std_res_x[DPI_75] = get_IN_std_res_75 (in); + s->std_res_y[DPI_75] = s->std_res_x[DPI_75]; + DBG (15, " 75 dpi: %d\n", s->std_res_x[DPI_75]); + + s->std_res_x[DPI_100] = get_IN_std_res_100 (in); + s->std_res_y[DPI_100] = s->std_res_x[DPI_100]; + DBG (15, " 100 dpi: %d\n", s->std_res_x[DPI_100]); + + s->std_res_x[DPI_120] = get_IN_std_res_120 (in); + s->std_res_y[DPI_120] = s->std_res_x[DPI_120]; + DBG (15, " 120 dpi: %d\n", s->std_res_x[DPI_120]); + + s->std_res_x[DPI_150] = get_IN_std_res_150 (in); + s->std_res_y[DPI_150] = s->std_res_x[DPI_150]; + DBG (15, " 150 dpi: %d\n", s->std_res_x[DPI_150]); + + s->std_res_x[DPI_160] = get_IN_std_res_160 (in); + s->std_res_y[DPI_160] = s->std_res_x[DPI_160]; + DBG (15, " 160 dpi: %d\n", s->std_res_x[DPI_160]); + + s->std_res_x[DPI_180] = get_IN_std_res_180 (in); + s->std_res_y[DPI_180] = s->std_res_x[DPI_180]; + DBG (15, " 180 dpi: %d\n", s->std_res_x[DPI_180]); + + s->std_res_x[DPI_200] = get_IN_std_res_200 (in); + s->std_res_y[DPI_200] = s->std_res_x[DPI_200]; + DBG (15, " 200 dpi: %d\n", s->std_res_x[DPI_200]); + + s->std_res_x[DPI_240] = get_IN_std_res_240 (in); + s->std_res_y[DPI_240] = s->std_res_x[DPI_240]; + DBG (15, " 240 dpi: %d\n", s->std_res_x[DPI_240]); + + s->std_res_x[DPI_300] = get_IN_std_res_300 (in); + s->std_res_y[DPI_300] = s->std_res_x[DPI_300]; + DBG (15, " 300 dpi: %d\n", s->std_res_x[DPI_300]); + + s->std_res_x[DPI_320] = get_IN_std_res_320 (in); + s->std_res_y[DPI_320] = s->std_res_x[DPI_320]; + DBG (15, " 320 dpi: %d\n", s->std_res_x[DPI_320]); + + s->std_res_x[DPI_400] = get_IN_std_res_400 (in); + s->std_res_y[DPI_400] = s->std_res_x[DPI_400]; + DBG (15, " 400 dpi: %d\n", s->std_res_x[DPI_400]); + + s->std_res_x[DPI_480] = get_IN_std_res_480 (in); + s->std_res_y[DPI_480] = s->std_res_x[DPI_480]; + DBG (15, " 480 dpi: %d\n", s->std_res_x[DPI_480]); + + s->std_res_x[DPI_600] = get_IN_std_res_600 (in); + s->std_res_y[DPI_600] = s->std_res_x[DPI_600]; + DBG (15, " 600 dpi: %d\n", s->std_res_x[DPI_600]); + + s->std_res_x[DPI_800] = get_IN_std_res_800 (in); + s->std_res_y[DPI_800] = s->std_res_x[DPI_800]; + DBG (15, " 800 dpi: %d\n", s->std_res_x[DPI_800]); + + s->std_res_x[DPI_1200] = get_IN_std_res_1200 (in); + s->std_res_y[DPI_1200] = s->std_res_x[DPI_1200]; + DBG (15, " 1200 dpi: %d\n", s->std_res_x[DPI_1200]); + + /* maximum window width and length are reported in basic units.*/ + s->max_x = get_IN_window_width(in) * 1200 / s->basic_x_res; + DBG(15, " max width: %d (%2.2f in)\n",s->max_x,(float)s->max_x/1200); + + s->max_y = get_IN_window_length(in) * 1200 / s->basic_y_res; + DBG(15, " max length: %d (%2.2f in)\n",s->max_y,(float)s->max_y/1200); + + DBG (15, " AWD: %d\n", get_IN_awd(in)); + DBG (15, " CE Emphasis: %d\n", get_IN_ce_emphasis(in)); + DBG (15, " C Emphasis: %d\n", get_IN_c_emphasis(in)); + DBG (15, " High quality: %d\n", get_IN_high_quality(in)); + + /* known modes FIXME more here? */ + s->can_grayscale = get_IN_multilevel (in); + DBG (15, " grayscale: %d\n", s->can_grayscale); + + s->can_halftone = get_IN_half_tone (in); + DBG (15, " halftone: %d\n", s->can_halftone); + + s->can_monochrome = get_IN_monochrome (in); + DBG (15, " monochrome: %d\n", s->can_monochrome); + + s->can_overflow = get_IN_overflow(in); + DBG (15, " overflow: %d\n", s->can_overflow); + } + /*FIXME no vpd, set some defaults? */ + else{ + DBG (5, "init_vpd: Your scanner does not support VPD?\n"); + DBG (5, "init_vpd: Please contact kitno455 at gmail dot com\n"); + DBG (5, "init_vpd: with details of your scanner model.\n"); + } + + DBG (10, "init_vpd: finish\n"); + + return ret; +} + +/* + * get model specific info that is not in vpd, and correct + * errors in vpd data. struct is already initialized to 0. + */ +static SANE_Status +init_model (struct scanner *s) +{ + + DBG (10, "init_model: start\n"); + + s->reverse_by_mode[MODE_LINEART] = 1; + s->reverse_by_mode[MODE_HALFTONE] = 1; + s->reverse_by_mode[MODE_GRAYSCALE] = 0; + s->reverse_by_mode[MODE_COLOR] = 0; + + s->always_op = 1; + s->has_df = 1; + s->has_btc = 1; + s->has_counter = 1; + s->has_adf = 1; + s->has_duplex = 1; + s->has_buffer = 1; + s->can_read_panel = 1; + s->can_write_panel = 1; + s->has_ssm = 1; + + s->brightness_steps = 255; + s->contrast_steps = 255; + s->threshold_steps = 255; + + s->ppl_mod = 1; + s->bg_color = 0xee; + + /* assume these are same as adf, override below */ + s->valid_x = s->max_x; + s->max_x_fb = s->max_x; + s->max_y_fb = s->max_y; + + /* generic settings missing from vpd */ + if (strstr (s->model_name,"C")){ + s->can_color = 1; + } + + /* specific settings missing from vpd */ + if (strstr (s->model_name,"DR-9080") + || strstr (s->model_name,"DR-7580")){ +#ifdef SANE_FRAME_JPEG + s->has_comp_JPEG = 1; +#endif + s->rgb_format = 2; + } + + else if (strstr (s->model_name,"DR-7090")){ + s->has_flatbed = 1; + } + + else if (strstr (s->model_name,"DR-9050") + || strstr (s->model_name,"DR-7550") + || strstr (s->model_name,"DR-6050")){ + + /*missing*/ + s->std_res_x[DPI_100]=1; + s->std_res_y[DPI_100]=1; + s->std_res_x[DPI_150]=1; + s->std_res_y[DPI_150]=1; + s->std_res_x[DPI_200]=1; + s->std_res_y[DPI_200]=1; + s->std_res_x[DPI_240]=1; + s->std_res_y[DPI_240]=1; + s->std_res_x[DPI_300]=1; + s->std_res_y[DPI_300]=1; + s->std_res_x[DPI_400]=1; + s->std_res_y[DPI_400]=1; + s->std_res_x[DPI_600]=1; + s->std_res_y[DPI_600]=1; + + /*weirdness*/ + s->has_ssm = 0; + s->has_ssm2 = 1; + } + + else if (strstr (s->model_name,"DR-4080") + || strstr (s->model_name,"DR-4580") + || strstr (s->model_name,"DR-7080")){ + s->has_flatbed = 1; + } + + else if (strstr (s->model_name,"DR-2580")){ + s->invert_tly = 1; + s->rgb_format = 1; + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_RRGGBB; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_rRgGbB; + s->gray_interlace[SIDE_BACK] = GRAY_INTERLACE_gG; + s->duplex_interlace = DUPLEX_INTERLACE_FBFB; + s->need_ccal = 1; + s->need_fcal = 1; + /*s->duplex_offset = 432; now set in config file*/ + s->duplex_offset_side = SIDE_BACK; + + /*lies*/ + s->can_halftone=0; + s->can_monochrome=0; + } + + else if (strstr (s->model_name,"DR-2510") + || strstr (s->model_name,"DR-2010") + ){ + s->rgb_format = 1; + s->always_op = 0; + s->unknown_byte2 = 0x80; + s->fixed_width = 1; + s->valid_x = 8.5 * 1200; + s->gray_interlace[SIDE_FRONT] = GRAY_INTERLACE_2510; + s->gray_interlace[SIDE_BACK] = GRAY_INTERLACE_2510; + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_2510; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_2510; + s->duplex_interlace = DUPLEX_INTERLACE_2510; + /*s->duplex_offset = 400; now set in config file*/ + s->need_ccal = 1; + s->need_fcal = 1; + s->sw_lut = 1; + /*s->invert_tly = 1;*/ + + /*only in Y direction, so we trash them in X*/ + s->std_res_x[DPI_100]=0; + s->std_res_x[DPI_150]=0; + s->std_res_x[DPI_200]=0; + s->std_res_x[DPI_240]=0; + s->std_res_x[DPI_400]=0; + + /*lies*/ + s->can_halftone=0; + s->can_monochrome=0; + } + + /* copied from 2510, possibly incorrect */ + else if (strstr (s->model_name,"DR-3010")){ + s->rgb_format = 1; + s->always_op = 0; + s->unknown_byte2 = 0x80; + s->fixed_width = 1; + s->valid_x = 8.5 * 1200; + s->gray_interlace[SIDE_FRONT] = GRAY_INTERLACE_2510; + s->gray_interlace[SIDE_BACK] = GRAY_INTERLACE_2510; + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_2510; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_2510; + s->duplex_interlace = DUPLEX_INTERLACE_2510; + /*s->duplex_offset = 400; now set in config file*/ + s->need_ccal = 1; + s->need_fcal = 1; + s->sw_lut = 1; + s->invert_tly = 1; + + /*only in Y direction, so we trash them in X*/ + s->std_res_x[DPI_100]=0; + s->std_res_x[DPI_150]=0; + s->std_res_x[DPI_200]=0; + s->std_res_x[DPI_240]=0; + s->std_res_x[DPI_400]=0; + + /*lies*/ + s->can_halftone=0; + s->can_monochrome=0; + } + + else if (strstr (s->model_name,"DR-2050") + || strstr (s->model_name,"DR-2080")){ + s->can_write_panel = 0; + s->has_df = 0; + s->fixed_width = 1; + s->even_Bpl = 1; + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_RRGGBB; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_RRGGBB; + s->duplex_interlace = DUPLEX_INTERLACE_FBFB; + s->need_fcal_buffer = 1; + s->bg_color = 0x08; + /*s->duplex_offset = 840; now set in config file*/ + s->sw_lut = 1; + + /*lies*/ + s->can_halftone=0; + s->can_monochrome=0; + } + + else if (strstr (s->model_name,"DR-3080")){ + s->can_write_panel = 0; + s->has_df = 0; + s->has_btc = 0; + } + + else if (strstr (s->model_name,"DR-5060F")){ + s->can_write_panel = 0; + s->has_df = 0; + s->has_btc = 0; + s->ppl_mod = 32; + s->reverse_by_mode[MODE_LINEART] = 0; + s->reverse_by_mode[MODE_HALFTONE] = 0; + } + + else if (strstr (s->model_name,"DR-5020")){ + s->can_read_panel = 0; + s->can_write_panel = 0; + s->has_df = 0; + s->has_btc = 0; + s->ppl_mod = 32; + s->reverse_by_mode[MODE_LINEART] = 0; + s->reverse_by_mode[MODE_HALFTONE] = 0; + } + + else if (strstr (s->model_name, "P-208")) { + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_RRGGBB; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_rRgGbB; + s->gray_interlace[SIDE_BACK] = GRAY_INTERLACE_gG; + s->duplex_interlace = DUPLEX_INTERLACE_FBFB; + s->need_ccal = 1; + s->invert_tly = 1; + s->can_color = 1; + s->unknown_byte2 = 0x88; + s->rgb_format = 1; + s->has_ssm_pay_head_len = 1; + s->ppl_mod = 8; + s->ccal_version = 3; + } + + else if (strstr (s->model_name, "P-215")) { + s->color_interlace[SIDE_FRONT] = COLOR_INTERLACE_rRgGbB; + s->color_interlace[SIDE_BACK] = COLOR_INTERLACE_RRGGBB; + s->gray_interlace[SIDE_FRONT] = GRAY_INTERLACE_gG; + s->duplex_interlace = DUPLEX_INTERLACE_FBFB; + s->need_ccal = 1; + s->invert_tly = 1; + s->can_color = 1; + s->unknown_byte2 = 0x88; + s->rgb_format = 1; + s->has_ssm_pay_head_len = 1; + s->ppl_mod = 8; + s->ccal_version = 3; + } + + DBG (10, "init_model: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * This function enables the buttons and preloads the current panel values + */ +static SANE_Status +init_panel (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "init_panel: start\n"); + + ret = read_panel(s,0); + if(ret){ + DBG (5, "init_panel: disabling read_panel\n"); + s->can_read_panel = 0; + ret = SANE_STATUS_GOOD; + } + + s->panel_enable_led = 1; + s->panel_counter = 0; + ret = send_panel(s); + if(ret){ + DBG (5, "init_panel: disabling send_panel\n"); + s->can_write_panel = 0; + ret = SANE_STATUS_GOOD; + } + + DBG (10, "init_panel: finish\n"); + + return ret; +} + +/* + * set good default user values. + * struct is already initialized to 0. + */ +static SANE_Status +init_user (struct scanner *s) +{ + + DBG (10, "init_user: start\n"); + + /* source */ + if(s->has_flatbed) + s->u.source = SOURCE_FLATBED; + else if(s->has_adf) + s->u.source = SOURCE_ADF_FRONT; + + /* scan mode */ + if(s->can_monochrome) + s->u.mode=MODE_LINEART; + else if(s->can_halftone) + s->u.mode=MODE_HALFTONE; + else if(s->can_grayscale) + s->u.mode=MODE_GRAYSCALE; + else if(s->can_color) + s->u.mode=MODE_COLOR; + + /*x and y res*/ + s->u.dpi_x = s->basic_x_res; + s->u.dpi_y = s->basic_x_res; + + /* page width US-Letter */ + s->u.page_x = 8.5 * 1200; + if(s->u.page_x > s->valid_x){ + s->u.page_x = s->valid_x; + } + + /* page height US-Letter */ + s->u.page_y = 11 * 1200; + if(s->u.page_y > s->max_y){ + s->u.page_y = s->max_y; + } + + /* bottom-right x */ + s->u.br_x = s->u.page_x; + + /* bottom-right y */ + s->u.br_y = s->u.page_y; + + s->threshold = 90; + s->compress_arg = 50; + + DBG (10, "init_user: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * This function presets the "option" array to blank + */ +static SANE_Status +init_options (struct scanner *s) +{ + int i; + + DBG (10, "init_options: start\n"); + + memset (s->opt, 0, sizeof (s->opt)); + for (i = 0; i < NUM_OPTIONS; ++i) { + s->opt[i].name = "filler"; + s->opt[i].size = sizeof (SANE_Word); + s->opt[i].cap = SANE_CAP_INACTIVE; + } + + /* go ahead and setup the first opt, because + * frontend may call control_option on it + * before calling get_option_descriptor + */ + s->opt[OPT_NUM_OPTS].name = SANE_NAME_NUM_OPTIONS; + 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].type = SANE_TYPE_INT; + s->opt[OPT_NUM_OPTS].cap = SANE_CAP_SOFT_DETECT; + + DBG (10, "init_options: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * From the SANE spec: + * This function is used to establish a connection to a particular + * device. The name of the device to be opened is passed in argument + * name. If the call completes successfully, a handle for the device + * is returned in *h. As a special case, specifying a zero-length + * string as the device requests opening the first available device + * (if there is such a device). + */ +SANE_Status +sane_open (SANE_String_Const name, SANE_Handle * handle) +{ + struct scanner *dev = NULL; + struct scanner *s = NULL; + SANE_Status ret; + + DBG (10, "sane_open: start\n"); + + if(scanner_devList){ + DBG (15, "sane_open: searching currently attached scanners\n"); + } + else{ + DBG (15, "sane_open: no scanners currently attached, attaching\n"); + + ret = sane_get_devices(NULL,0); + if(ret != SANE_STATUS_GOOD){ + return ret; + } + } + + if(name[0] == 0){ + DBG (15, "sane_open: no device requested, using default\n"); + s = scanner_devList; + } + else{ + DBG (15, "sane_open: device %s requested\n", name); + + for (dev = scanner_devList; dev; dev = dev->next) { + if (strcmp (dev->sane.name, name) == 0 + || strcmp (dev->device_name, name) == 0) { /*always allow sanei devname*/ + s = dev; + break; + } + } + } + + if (!s) { + DBG (5, "sane_open: no device found\n"); + return SANE_STATUS_INVAL; + } + + DBG (15, "sane_open: device %s found\n", s->sane.name); + + *handle = s; + + /* connect the fd so we can talk to scanner */ + ret = connect_fd(s); + if(ret != SANE_STATUS_GOOD){ + return ret; + } + + DBG (10, "sane_open: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * @@ Section 3 - SANE Options functions + */ + +/* + * Returns the options we know. + * + * From the SANE spec: + * This function is used to access option descriptors. The function + * returns the option descriptor for option number n of the device + * represented by handle h. Option number 0 is guaranteed to be a + * valid option. Its value is an integer that specifies the number of + * options that are available for device handle h (the count includes + * option 0). If n is not a valid option index, the function returns + * NULL. The returned option descriptor is guaranteed to remain valid + * (and at the returned address) until the device is closed. + */ +const SANE_Option_Descriptor * +sane_get_option_descriptor (SANE_Handle handle, SANE_Int option) +{ + struct scanner *s = handle; + int i; + SANE_Option_Descriptor *opt = &s->opt[option]; + + DBG (20, "sane_get_option_descriptor: %d\n", option); + + if ((unsigned) option >= NUM_OPTIONS) + return NULL; + + /* "Mode" group -------------------------------------------------------- */ + if(option==OPT_STANDARD_GROUP){ + opt->name = SANE_NAME_STANDARD; + opt->title = SANE_TITLE_STANDARD; + opt->desc = SANE_DESC_STANDARD; + opt->type = SANE_TYPE_GROUP; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + /* source */ + if(option==OPT_SOURCE){ + i=0; + if(s->has_flatbed){ + s->source_list[i++]=STRING_FLATBED; + } + if(s->has_adf){ + s->source_list[i++]=STRING_ADFFRONT; + + if(s->has_back){ + s->source_list[i++]=STRING_ADFBACK; + } + if(s->has_duplex){ + s->source_list[i++]=STRING_ADFDUPLEX; + } + } + s->source_list[i]=NULL; + + opt->name = SANE_NAME_SCAN_SOURCE; + opt->title = SANE_TITLE_SCAN_SOURCE; + opt->desc = SANE_DESC_SCAN_SOURCE; + opt->type = SANE_TYPE_STRING; + opt->constraint_type = SANE_CONSTRAINT_STRING_LIST; + opt->constraint.string_list = s->source_list; + opt->size = maxStringSize (opt->constraint.string_list); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* scan mode */ + if(option==OPT_MODE){ + i=0; + if(s->can_monochrome || s->can_grayscale || s->can_color){ + s->mode_list[i++]=STRING_LINEART; + } + if(s->can_halftone){ + s->mode_list[i++]=STRING_HALFTONE; + } + if(s->can_grayscale || s->can_color){ + s->mode_list[i++]=STRING_GRAYSCALE; + } + if(s->can_color){ + s->mode_list[i++]=STRING_COLOR; + } + s->mode_list[i]=NULL; + + opt->name = SANE_NAME_SCAN_MODE; + opt->title = SANE_TITLE_SCAN_MODE; + opt->desc = SANE_DESC_SCAN_MODE; + opt->type = SANE_TYPE_STRING; + opt->constraint_type = SANE_CONSTRAINT_STRING_LIST; + opt->constraint.string_list = s->mode_list; + opt->size = maxStringSize (opt->constraint.string_list); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* resolution */ + /* some scanners only support fixed res + * build a list of possible choices */ + /* we actually only look at the y resolution choices, + * and interpolate the image data as required for limited x resolutions */ + if(option==OPT_RES){ + i=0; + if(s->std_res_y[DPI_60] && s->max_y_res >= 60 && s->min_y_res <= 60){ + s->res_list[++i] = 60; + } + if(s->std_res_y[DPI_75] && s->max_y_res >= 75 && s->min_y_res <= 75){ + s->res_list[++i] = 75; + } + if(s->std_res_y[DPI_100] && s->max_y_res >= 100 && s->min_y_res <= 100){ + s->res_list[++i] = 100; + } + if(s->std_res_y[DPI_120] && s->max_y_res >= 120 && s->min_y_res <= 120){ + s->res_list[++i] = 120; + } + if(s->std_res_y[DPI_150] && s->max_y_res >= 150 && s->min_y_res <= 150){ + s->res_list[++i] = 150; + } + if(s->std_res_y[DPI_160] && s->max_y_res >= 160 && s->min_y_res <= 160){ + s->res_list[++i] = 160; + } + if(s->std_res_y[DPI_180] && s->max_y_res >= 180 && s->min_y_res <= 180){ + s->res_list[++i] = 180; + } + if(s->std_res_y[DPI_200] && s->max_y_res >= 200 && s->min_y_res <= 200){ + s->res_list[++i] = 200; + } + if(s->std_res_y[DPI_240] && s->max_y_res >= 240 && s->min_y_res <= 240){ + s->res_list[++i] = 240; + } + if(s->std_res_y[DPI_300] && s->max_y_res >= 300 && s->min_y_res <= 300){ + s->res_list[++i] = 300; + } + if(s->std_res_y[DPI_320] && s->max_y_res >= 320 && s->min_y_res <= 320){ + s->res_list[++i] = 320; + } + if(s->std_res_y[DPI_400] && s->max_y_res >= 400 && s->min_y_res <= 400){ + s->res_list[++i] = 400; + } + if(s->std_res_y[DPI_480] && s->max_y_res >= 480 && s->min_y_res <= 480){ + s->res_list[++i] = 480; + } + if(s->std_res_y[DPI_600] && s->max_y_res >= 600 && s->min_y_res <= 600){ + s->res_list[++i] = 600; + } + if(s->std_res_y[DPI_800] && s->max_y_res >= 800 && s->min_y_res <= 800){ + s->res_list[++i] = 800; + } + if(s->std_res_y[DPI_1200] && s->max_y_res >= 1200 && s->min_y_res <= 1200){ + s->res_list[++i] = 1200; + } + s->res_list[0] = i; + + opt->name = SANE_NAME_SCAN_RESOLUTION; + opt->title = SANE_TITLE_SCAN_RESOLUTION; + opt->desc = SANE_DESC_SCAN_RESOLUTION; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_DPI; + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + + if(s->step_y_res){ + s->res_range.min = s->min_y_res; + s->res_range.max = s->max_y_res; + s->res_range.quant = s->step_y_res; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->res_range; + } + else{ + opt->constraint_type = SANE_CONSTRAINT_WORD_LIST; + opt->constraint.word_list = s->res_list; + } + } + + /* "Geometry" group ---------------------------------------------------- */ + if(option==OPT_GEOMETRY_GROUP){ + opt->name = SANE_NAME_GEOMETRY; + opt->title = SANE_TITLE_GEOMETRY; + opt->desc = SANE_DESC_GEOMETRY; + opt->type = SANE_TYPE_GROUP; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + /* top-left x */ + if(option==OPT_TL_X){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->tl_x_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_x); + s->tl_x_range.max = SCANNER_UNIT_TO_FIXED_MM(get_page_width(s)); + s->tl_x_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_SCAN_TL_X; + opt->title = SANE_TITLE_SCAN_TL_X; + opt->desc = SANE_DESC_SCAN_TL_X; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &(s->tl_x_range); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* top-left y */ + if(option==OPT_TL_Y){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->tl_y_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_y); + s->tl_y_range.max = SCANNER_UNIT_TO_FIXED_MM(get_page_height(s)); + s->tl_y_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_SCAN_TL_Y; + opt->title = SANE_TITLE_SCAN_TL_Y; + opt->desc = SANE_DESC_SCAN_TL_Y; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &(s->tl_y_range); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* bottom-right x */ + if(option==OPT_BR_X){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->br_x_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_x); + s->br_x_range.max = SCANNER_UNIT_TO_FIXED_MM(get_page_width(s)); + s->br_x_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_SCAN_BR_X; + opt->title = SANE_TITLE_SCAN_BR_X; + opt->desc = SANE_DESC_SCAN_BR_X; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &(s->br_x_range); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* bottom-right y */ + if(option==OPT_BR_Y){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->br_y_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_y); + s->br_y_range.max = SCANNER_UNIT_TO_FIXED_MM(get_page_height(s)); + s->br_y_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_SCAN_BR_Y; + opt->title = SANE_TITLE_SCAN_BR_Y; + opt->desc = SANE_DESC_SCAN_BR_Y; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &(s->br_y_range); + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + + /* page width */ + if(option==OPT_PAGE_WIDTH){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->paper_x_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_x); + s->paper_x_range.max = SCANNER_UNIT_TO_FIXED_MM(s->valid_x); + s->paper_x_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_PAGE_WIDTH; + opt->title = SANE_TITLE_PAGE_WIDTH; + opt->desc = SANE_DESC_PAGE_WIDTH; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->paper_x_range; + + if(s->has_adf){ + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + if(s->u.source == SOURCE_FLATBED){ + opt->cap |= SANE_CAP_INACTIVE; + } + } + else{ + opt->cap = SANE_CAP_INACTIVE; + } + } + + /* page height */ + if(option==OPT_PAGE_HEIGHT){ + /* values stored in 1200 dpi units */ + /* must be converted to MM for sane */ + s->paper_y_range.min = SCANNER_UNIT_TO_FIXED_MM(s->min_y); + s->paper_y_range.max = SCANNER_UNIT_TO_FIXED_MM(s->max_y); + s->paper_y_range.quant = MM_PER_UNIT_FIX; + + opt->name = SANE_NAME_PAGE_HEIGHT; + opt->title = SANE_TITLE_PAGE_HEIGHT; + opt->desc = SANE_DESC_PAGE_HEIGHT; + opt->type = SANE_TYPE_FIXED; + opt->unit = SANE_UNIT_MM; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->paper_y_range; + + if(s->has_adf){ + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + if(s->u.source == SOURCE_FLATBED){ + opt->cap |= SANE_CAP_INACTIVE; + } + } + else{ + opt->cap = SANE_CAP_INACTIVE; + } + } + + /* "Enhancement" group ------------------------------------------------- */ + if(option==OPT_ENHANCEMENT_GROUP){ + opt->name = SANE_NAME_ENHANCEMENT; + opt->title = SANE_TITLE_ENHANCEMENT; + opt->desc = SANE_DESC_ENHANCEMENT; + opt->type = SANE_TYPE_GROUP; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + /* brightness */ + if(option==OPT_BRIGHTNESS){ + opt->name = SANE_NAME_BRIGHTNESS; + opt->title = SANE_TITLE_BRIGHTNESS; + opt->desc = SANE_DESC_BRIGHTNESS; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->brightness_range; + s->brightness_range.quant=1; + + /* some have hardware brightness (always 0 to 255?) */ + /* some use LUT or GT (-127 to +127)*/ + if (s->brightness_steps){ + s->brightness_range.min=-127; + s->brightness_range.max=127; + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + else{ + opt->cap = SANE_CAP_INACTIVE; + } + } + + /* contrast */ + if(option==OPT_CONTRAST){ + opt->name = SANE_NAME_CONTRAST; + opt->title = SANE_TITLE_CONTRAST; + opt->desc = SANE_DESC_CONTRAST; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->contrast_range; + s->contrast_range.quant=1; + + /* some have hardware contrast (always 0 to 255?) */ + /* some use LUT or GT (-127 to +127)*/ + if (s->contrast_steps){ + s->contrast_range.min=-127; + s->contrast_range.max=127; + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + else { + opt->cap = SANE_CAP_INACTIVE; + } + } + + /*threshold*/ + if(option==OPT_THRESHOLD){ + opt->name = SANE_NAME_THRESHOLD; + opt->title = SANE_TITLE_THRESHOLD; + opt->desc = SANE_DESC_THRESHOLD; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->threshold_range; + s->threshold_range.min=0; + s->threshold_range.max=s->threshold_steps; + s->threshold_range.quant=1; + + if (s->threshold_steps){ + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + if(s->u.mode != MODE_LINEART){ + opt->cap |= SANE_CAP_INACTIVE; + } + } + else { + opt->cap = SANE_CAP_INACTIVE; + } + } + + if(option==OPT_RIF){ + opt->name = "rif"; + opt->title = "RIF"; + opt->desc = "Reverse image format"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + if (s->has_rif) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /* "Advanced" group ------------------------------------------------------ */ + if(option==OPT_ADVANCED_GROUP){ + opt->name = SANE_NAME_ADVANCED; + opt->title = SANE_TITLE_ADVANCED; + opt->desc = SANE_DESC_ADVANCED; + opt->type = SANE_TYPE_GROUP; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + /*image compression*/ + if(option==OPT_COMPRESS){ + i=0; + s->compress_list[i++]=STRING_NONE; + + if(s->has_comp_JPEG){ + s->compress_list[i++]=STRING_JPEG; + } + + s->compress_list[i]=NULL; + + opt->name = "compression"; + opt->title = "Compression"; + opt->desc = "Enable compressed data. May crash your front-end program"; + opt->type = SANE_TYPE_STRING; + opt->constraint_type = SANE_CONSTRAINT_STRING_LIST; + opt->constraint.string_list = s->compress_list; + opt->size = maxStringSize (opt->constraint.string_list); + + if (i > 1){ + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + if (s->u.mode != MODE_COLOR && s->u.mode != MODE_GRAYSCALE){ + opt->cap |= SANE_CAP_INACTIVE; + } + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*image compression arg*/ + if(option==OPT_COMPRESS_ARG){ + + opt->name = "compression-arg"; + opt->title = "Compression argument"; + opt->desc = "Level of JPEG compression. 1 is small file, 100 is large file."; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->compress_arg_range; + s->compress_arg_range.quant=1; + + if(s->has_comp_JPEG){ + s->compress_arg_range.min=0; + s->compress_arg_range.max=100; + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + + if(s->compress != COMP_JPEG){ + opt->cap |= SANE_CAP_INACTIVE; + } + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*double feed by length*/ + if(option==OPT_DF_LENGTH){ + opt->name = "df-length"; + opt->title = "DF by length"; + opt->desc = "Detect double feeds by comparing document lengths"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_NONE; + + if (1) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*double feed by thickness */ + if(option==OPT_DF_THICKNESS){ + + opt->name = "df-thickness"; + opt->title = "DF by thickness"; + opt->desc = "Detect double feeds using thickness sensor"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_NONE; + + if (1){ + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*deskew by roller*/ + if(option==OPT_ROLLERDESKEW){ + opt->name = "rollerdeskew"; + opt->title = "Roller deskew"; + opt->desc = "Request scanner to correct skewed pages mechanically"; + opt->type = SANE_TYPE_BOOL; + if (1) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*deskew by software*/ + if(option==OPT_SWDESKEW){ + opt->name = "swdeskew"; + opt->title = "Software deskew"; + opt->desc = "Request driver to rotate skewed pages digitally"; + opt->type = SANE_TYPE_BOOL; + if (1) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*software despeckle radius*/ + if(option==OPT_SWDESPECK){ + + opt->name = "swdespeck"; + opt->title = "Software despeckle diameter"; + opt->desc = "Maximum diameter of lone dots to remove from scan"; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->swdespeck_range; + s->swdespeck_range.quant=1; + + if(1){ + s->swdespeck_range.min=0; + s->swdespeck_range.max=9; + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*crop by software*/ + if(option==OPT_SWCROP){ + opt->name = "swcrop"; + opt->title = "Software crop"; + opt->desc = "Request driver to remove border from pages digitally"; + opt->type = SANE_TYPE_BOOL; + if (1) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*staple detection*/ + if(option==OPT_STAPLEDETECT){ + opt->name = "stapledetect"; + opt->title = "Staple detect"; + opt->desc = "Request scanner to halt if stapled pages are detected"; + opt->type = SANE_TYPE_BOOL; + if (1) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*dropout color front*/ + if(option==OPT_DROPOUT_COLOR_F){ + s->do_color_list[0] = STRING_NONE; + s->do_color_list[1] = STRING_RED; + s->do_color_list[2] = STRING_GREEN; + s->do_color_list[3] = STRING_BLUE; + s->do_color_list[4] = STRING_EN_RED; + s->do_color_list[5] = STRING_EN_GREEN; + s->do_color_list[6] = STRING_EN_BLUE; + s->do_color_list[7] = NULL; + + opt->name = "dropout-front"; + opt->title = "Dropout color front"; + opt->desc = "One-pass scanners use only one color during gray or binary scanning, useful for colored paper or ink"; + opt->type = SANE_TYPE_STRING; + opt->constraint_type = SANE_CONSTRAINT_STRING_LIST; + opt->constraint.string_list = s->do_color_list; + opt->size = maxStringSize (opt->constraint.string_list); + + if (1){ + opt->cap = SANE_CAP_SOFT_SELECT|SANE_CAP_SOFT_DETECT|SANE_CAP_ADVANCED; + if(s->u.mode == MODE_COLOR) + opt->cap |= SANE_CAP_INACTIVE; + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*dropout color back*/ + if(option==OPT_DROPOUT_COLOR_B){ + s->do_color_list[0] = STRING_NONE; + s->do_color_list[1] = STRING_RED; + s->do_color_list[2] = STRING_GREEN; + s->do_color_list[3] = STRING_BLUE; + s->do_color_list[4] = STRING_EN_RED; + s->do_color_list[5] = STRING_EN_GREEN; + s->do_color_list[6] = STRING_EN_BLUE; + s->do_color_list[7] = NULL; + + opt->name = "dropout-back"; + opt->title = "Dropout color back"; + opt->desc = "One-pass scanners use only one color during gray or binary scanning, useful for colored paper or ink"; + opt->type = SANE_TYPE_STRING; + opt->constraint_type = SANE_CONSTRAINT_STRING_LIST; + opt->constraint.string_list = s->do_color_list; + opt->size = maxStringSize (opt->constraint.string_list); + + if (1){ + opt->cap = SANE_CAP_SOFT_SELECT|SANE_CAP_SOFT_DETECT|SANE_CAP_ADVANCED; + if(s->u.mode == MODE_COLOR) + opt->cap |= SANE_CAP_INACTIVE; + } + else + opt->cap = SANE_CAP_INACTIVE; + } + + /*buffer mode*/ + if(option==OPT_BUFFERMODE){ + opt->name = "buffermode"; + opt->title = "Buffer mode"; + opt->desc = "Request scanner to read pages async into internal memory"; + opt->type = SANE_TYPE_BOOL; + if (s->has_buffer) + opt->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_SIDE){ + opt->name = "side"; + opt->title = "Duplex side"; + opt->desc = "Tells which side (0=front, 1=back) of a duplex scan the next call to sane_read will return."; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->size = sizeof(SANE_Word); + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + /* "Sensor" group ------------------------------------------------------ */ + if(option==OPT_SENSOR_GROUP){ + opt->name = SANE_NAME_SENSORS; + opt->title = SANE_TITLE_SENSORS; + opt->desc = SANE_DESC_SENSORS; + opt->type = SANE_TYPE_GROUP; + opt->constraint_type = SANE_CONSTRAINT_NONE; + } + + if(option==OPT_START){ + opt->name = "start"; + opt->title = "Start/1 button"; + opt->desc = "Big green or small 1 button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_STOP){ + opt->name = "stop"; + opt->title = "Stop/2 button"; + opt->desc = "Small orange or small 2 button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_BUTT3){ + opt->name = "button-3"; + opt->title = "3 button"; + opt->desc = "Small 3 button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_NEWFILE){ + opt->name = "newfile"; + opt->title = "New File button"; + opt->desc = "New File button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_COUNTONLY){ + opt->name = "countonly"; + opt->title = "Count Only button"; + opt->desc = "Count Only button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_BYPASSMODE){ + opt->name = "bypassmode"; + opt->title = "Bypass Mode button"; + opt->desc = "Bypass Mode button"; + opt->type = SANE_TYPE_BOOL; + opt->unit = SANE_UNIT_NONE; + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + if(!s->can_read_panel) + opt->cap = SANE_CAP_INACTIVE; + } + + if(option==OPT_COUNTER){ + opt->name = "counter"; + opt->title = "Counter"; + opt->desc = "Scan counter"; + opt->type = SANE_TYPE_INT; + opt->unit = SANE_UNIT_NONE; + opt->constraint_type = SANE_CONSTRAINT_RANGE; + opt->constraint.range = &s->counter_range; + s->counter_range.min=0; + s->counter_range.max=500; + s->counter_range.quant=1; + + if (s->can_read_panel && s->has_counter) + opt->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_HARD_SELECT | SANE_CAP_ADVANCED; + else + opt->cap = SANE_CAP_INACTIVE; + } + + return opt; +} + +/** + * Gets or sets an option value. + * + * From the SANE spec: + * This function is used to set or inquire the current value of option + * number n of the device represented by handle h. The manner in which + * the option is controlled is specified by parameter action. The + * possible values of this parameter are described in more detail + * below. The value of the option is passed through argument val. It + * is a pointer to the memory that holds the option value. The memory + * area pointed to by v must be big enough to hold the entire option + * value (determined by member size in the corresponding option + * descriptor). + * + * The only exception to this rule is that when setting the value of a + * string option, the string pointed to by argument v may be shorter + * since the backend will stop reading the option value upon + * encountering the first NUL terminator in the string. If argument i + * is not NULL, the value of *i will be set to provide details on how + * well the request has been met. + */ +SANE_Status +sane_control_option (SANE_Handle handle, SANE_Int option, + SANE_Action action, void *val, SANE_Int * info) +{ + struct scanner *s = (struct scanner *) handle; + SANE_Int dummy = 0; + SANE_Status ret = SANE_STATUS_GOOD; + + /* Make sure that all those statements involving *info cannot break (better + * than having to do "if (info) ..." everywhere!) + */ + if (info == 0) + info = &dummy; + + if (option >= NUM_OPTIONS) { + DBG (5, "sane_control_option: %d too big\n", option); + return SANE_STATUS_INVAL; + } + + if (!SANE_OPTION_IS_ACTIVE (s->opt[option].cap)) { + DBG (5, "sane_control_option: %d inactive\n", option); + return SANE_STATUS_INVAL; + } + + /* + * SANE_ACTION_GET_VALUE: We have to find out the current setting and + * return it in a human-readable form (often, text). + */ + if (action == SANE_ACTION_GET_VALUE) { + SANE_Word * val_p = (SANE_Word *) val; + + DBG (20, "sane_control_option: get value for '%s' (%d)\n", s->opt[option].name,option); + + switch (option) { + + case OPT_NUM_OPTS: + *val_p = NUM_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_SOURCE: + if(s->u.source == SOURCE_FLATBED){ + strcpy (val, STRING_FLATBED); + } + else if(s->u.source == SOURCE_ADF_FRONT){ + strcpy (val, STRING_ADFFRONT); + } + else if(s->u.source == SOURCE_ADF_BACK){ + strcpy (val, STRING_ADFBACK); + } + else if(s->u.source == SOURCE_ADF_DUPLEX){ + strcpy (val, STRING_ADFDUPLEX); + } + return SANE_STATUS_GOOD; + + case OPT_MODE: + if(s->u.mode == MODE_LINEART){ + strcpy (val, STRING_LINEART); + } + else if(s->u.mode == MODE_HALFTONE){ + strcpy (val, STRING_HALFTONE); + } + else if(s->u.mode == MODE_GRAYSCALE){ + strcpy (val, STRING_GRAYSCALE); + } + else if(s->u.mode == MODE_COLOR){ + strcpy (val, STRING_COLOR); + } + return SANE_STATUS_GOOD; + + case OPT_RES: + *val_p = s->u.dpi_x; + return SANE_STATUS_GOOD; + + case OPT_TL_X: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.tl_x); + return SANE_STATUS_GOOD; + + case OPT_TL_Y: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.tl_y); + return SANE_STATUS_GOOD; + + case OPT_BR_X: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.br_x); + return SANE_STATUS_GOOD; + + case OPT_BR_Y: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.br_y); + return SANE_STATUS_GOOD; + + case OPT_PAGE_WIDTH: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.page_x); + return SANE_STATUS_GOOD; + + case OPT_PAGE_HEIGHT: + *val_p = SCANNER_UNIT_TO_FIXED_MM(s->u.page_y); + return SANE_STATUS_GOOD; + + case OPT_BRIGHTNESS: + *val_p = s->brightness; + return SANE_STATUS_GOOD; + + case OPT_CONTRAST: + *val_p = s->contrast; + return SANE_STATUS_GOOD; + + case OPT_THRESHOLD: + *val_p = s->threshold; + return SANE_STATUS_GOOD; + + case OPT_RIF: + *val_p = s->rif; + return SANE_STATUS_GOOD; + + /* Advanced Group */ + case OPT_COMPRESS: + if(s->compress == COMP_JPEG){ + strcpy (val, STRING_JPEG); + } + else{ + strcpy (val, STRING_NONE); + } + return SANE_STATUS_GOOD; + + case OPT_COMPRESS_ARG: + *val_p = s->compress_arg; + return SANE_STATUS_GOOD; + + case OPT_DF_LENGTH: + *val_p = s->df_length; + return SANE_STATUS_GOOD; + + case OPT_DF_THICKNESS: + *val_p = s->df_thickness; + return SANE_STATUS_GOOD; + + case OPT_ROLLERDESKEW: + *val_p = s->rollerdeskew; + return SANE_STATUS_GOOD; + + case OPT_SWDESKEW: + *val_p = s->swdeskew; + return SANE_STATUS_GOOD; + + case OPT_SWDESPECK: + *val_p = s->swdespeck; + return SANE_STATUS_GOOD; + + case OPT_SWCROP: + *val_p = s->swcrop; + return SANE_STATUS_GOOD; + + case OPT_STAPLEDETECT: + *val_p = s->stapledetect; + return SANE_STATUS_GOOD; + + case OPT_DROPOUT_COLOR_F: + switch (s->dropout_color_f) { + case COLOR_NONE: + strcpy (val, STRING_NONE); + break; + case COLOR_RED: + strcpy (val, STRING_RED); + break; + case COLOR_GREEN: + strcpy (val, STRING_GREEN); + break; + case COLOR_BLUE: + strcpy (val, STRING_BLUE); + break; + case COLOR_EN_RED: + strcpy (val, STRING_EN_RED); + break; + case COLOR_EN_GREEN: + strcpy (val, STRING_EN_GREEN); + break; + case COLOR_EN_BLUE: + strcpy (val, STRING_EN_BLUE); + break; + } + return SANE_STATUS_GOOD; + + case OPT_DROPOUT_COLOR_B: + switch (s->dropout_color_b) { + case COLOR_NONE: + strcpy (val, STRING_NONE); + break; + case COLOR_RED: + strcpy (val, STRING_RED); + break; + case COLOR_GREEN: + strcpy (val, STRING_GREEN); + break; + case COLOR_BLUE: + strcpy (val, STRING_BLUE); + break; + case COLOR_EN_RED: + strcpy (val, STRING_EN_RED); + break; + case COLOR_EN_GREEN: + strcpy (val, STRING_EN_GREEN); + break; + case COLOR_EN_BLUE: + strcpy (val, STRING_EN_BLUE); + break; + } + return SANE_STATUS_GOOD; + + case OPT_BUFFERMODE: + *val_p = s->buffermode; + return SANE_STATUS_GOOD; + + case OPT_SIDE: + *val_p = s->side; + return SANE_STATUS_GOOD; + + /* Sensor Group */ + case OPT_START: + ret = read_panel(s,OPT_START); + *val_p = s->panel_start; + return ret; + + case OPT_STOP: + ret = read_panel(s,OPT_STOP); + *val_p = s->panel_stop; + return ret; + + case OPT_BUTT3: + ret = read_panel(s,OPT_BUTT3); + *val_p = s->panel_butt3; + return ret; + + case OPT_NEWFILE: + ret = read_panel(s,OPT_NEWFILE); + *val_p = s->panel_new_file; + return ret; + + case OPT_COUNTONLY: + ret = read_panel(s,OPT_COUNTONLY); + *val_p = s->panel_count_only; + return ret; + + case OPT_BYPASSMODE: + ret = read_panel(s,OPT_BYPASSMODE); + *val_p = s->panel_bypass_mode; + return ret; + + case OPT_COUNTER: + ret = read_panel(s,OPT_COUNTER); + *val_p = s->panel_counter; + return ret; + + } + } + else if (action == SANE_ACTION_SET_VALUE) { + int tmp; + SANE_Word val_c; + SANE_Status status; + + DBG (20, "sane_control_option: set value for '%s' (%d)\n", s->opt[option].name,option); + + if ( s->started ) { + DBG (5, "sane_control_option: cant set, device busy\n"); + return SANE_STATUS_DEVICE_BUSY; + } + + if (!SANE_OPTION_IS_SETTABLE (s->opt[option].cap)) { + DBG (5, "sane_control_option: not settable\n"); + return SANE_STATUS_INVAL; + } + + status = sanei_constrain_value (s->opt + option, val, info); + if (status != SANE_STATUS_GOOD) { + DBG (5, "sane_control_option: bad value\n"); + return status; + } + + /* may have been changed by constrain, so dont copy until now */ + val_c = *(SANE_Word *)val; + + /* + * Note - for those options which can assume one of a list of + * valid values, we can safely assume that they will have + * exactly one of those values because that's what + * sanei_constrain_value does. Hence no "else: invalid" branches + * below. + */ + switch (option) { + + /* Mode Group */ + case OPT_SOURCE: + if (!strcmp (val, STRING_ADFFRONT)) { + tmp = SOURCE_ADF_FRONT; + } + else if (!strcmp (val, STRING_ADFBACK)) { + tmp = SOURCE_ADF_BACK; + } + else if (!strcmp (val, STRING_ADFDUPLEX)) { + tmp = SOURCE_ADF_DUPLEX; + } + else{ + tmp = SOURCE_FLATBED; + } + + if (s->u.source == tmp) + return SANE_STATUS_GOOD; + + s->u.source = tmp; + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_MODE: + if (!strcmp (val, STRING_LINEART)) { + tmp = MODE_LINEART; + } + else if (!strcmp (val, STRING_HALFTONE)) { + tmp = MODE_HALFTONE; + } + else if (!strcmp (val, STRING_GRAYSCALE)) { + tmp = MODE_GRAYSCALE; + } + else{ + tmp = MODE_COLOR; + } + + if (tmp == s->u.mode) + return SANE_STATUS_GOOD; + + s->u.mode = tmp; + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_RES: + + if (s->u.dpi_x == val_c && s->u.dpi_y == val_c) + return SANE_STATUS_GOOD; + + s->u.dpi_x = val_c; + s->u.dpi_y = val_c; + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + /* Geometry Group */ + case OPT_TL_X: + if (s->u.tl_x == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.tl_x = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_TL_Y: + if (s->u.tl_y == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.tl_y = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_BR_X: + if (s->u.br_x == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.br_x = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_BR_Y: + if (s->u.br_y == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.br_y = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_PAGE_WIDTH: + if (s->u.page_x == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.page_x = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + case OPT_PAGE_HEIGHT: + if (s->u.page_y == FIXED_MM_TO_SCANNER_UNIT(val_c)) + return SANE_STATUS_GOOD; + + s->u.page_y = FIXED_MM_TO_SCANNER_UNIT(val_c); + + *info |= SANE_INFO_RELOAD_OPTIONS; + return SANE_STATUS_GOOD; + + /* Enhancement Group */ + case OPT_BRIGHTNESS: + s->brightness = val_c; + return SANE_STATUS_GOOD; + + case OPT_CONTRAST: + s->contrast = val_c; + return SANE_STATUS_GOOD; + + case OPT_THRESHOLD: + s->threshold = val_c; + return SANE_STATUS_GOOD; + + case OPT_RIF: + s->rif = val_c; + return SANE_STATUS_GOOD; + + /* Advanced Group */ + case OPT_COMPRESS: + if (!strcmp (val, STRING_JPEG)) { + s->compress = COMP_JPEG; + } + else{ + s->compress = COMP_NONE; + } + return SANE_STATUS_GOOD; + + case OPT_COMPRESS_ARG: + s->compress_arg = val_c; + return SANE_STATUS_GOOD; + + case OPT_DF_LENGTH: + s->df_length = val_c; + return SANE_STATUS_GOOD; + + case OPT_DF_THICKNESS: + s->df_thickness = val_c; + return SANE_STATUS_GOOD; + + case OPT_ROLLERDESKEW: + s->rollerdeskew = val_c; + return SANE_STATUS_GOOD; + + case OPT_SWDESKEW: + s->swdeskew = val_c; + return SANE_STATUS_GOOD; + + case OPT_SWDESPECK: + s->swdespeck = val_c; + return SANE_STATUS_GOOD; + + case OPT_SWCROP: + s->swcrop = val_c; + return SANE_STATUS_GOOD; + + case OPT_STAPLEDETECT: + s->stapledetect = val_c; + return SANE_STATUS_GOOD; + + case OPT_DROPOUT_COLOR_F: + if (!strcmp(val, STRING_NONE)) + s->dropout_color_f = COLOR_NONE; + else if (!strcmp(val, STRING_RED)) + s->dropout_color_f = COLOR_RED; + else if (!strcmp(val, STRING_GREEN)) + s->dropout_color_f = COLOR_GREEN; + else if (!strcmp(val, STRING_BLUE)) + s->dropout_color_f = COLOR_BLUE; + else if (!strcmp(val, STRING_EN_RED)) + s->dropout_color_f = COLOR_EN_RED; + else if (!strcmp(val, STRING_EN_GREEN)) + s->dropout_color_f = COLOR_EN_GREEN; + else if (!strcmp(val, STRING_EN_BLUE)) + s->dropout_color_f = COLOR_EN_BLUE; + return SANE_STATUS_GOOD; + + case OPT_DROPOUT_COLOR_B: + if (!strcmp(val, STRING_NONE)) + s->dropout_color_b = COLOR_NONE; + else if (!strcmp(val, STRING_RED)) + s->dropout_color_b = COLOR_RED; + else if (!strcmp(val, STRING_GREEN)) + s->dropout_color_b = COLOR_GREEN; + else if (!strcmp(val, STRING_BLUE)) + s->dropout_color_b = COLOR_BLUE; + else if (!strcmp(val, STRING_EN_RED)) + s->dropout_color_b = COLOR_EN_RED; + else if (!strcmp(val, STRING_EN_GREEN)) + s->dropout_color_b = COLOR_EN_GREEN; + else if (!strcmp(val, STRING_EN_BLUE)) + s->dropout_color_b = COLOR_EN_BLUE; + return SANE_STATUS_GOOD; + + case OPT_BUFFERMODE: + s->buffermode = val_c; + return SANE_STATUS_GOOD; + + } + } /* else */ + + return SANE_STATUS_INVAL; +} + +static SANE_Status +ssm_buffer (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[SET_SCAN_MODE_len]; + size_t cmdLen = SET_SCAN_MODE_len; + + unsigned char out[SSM_PAY_len]; + size_t outLen = SSM_PAY_len; + + DBG (10, "ssm_buffer: start\n"); + + if(!s->has_ssm){ + DBG (10, "ssm_buffer: unsupported\n"); + return ret; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SET_SCAN_MODE_code); + set_SSM_pf(cmd, 1); + set_SSM_pay_len(cmd, outLen); + + memset(out,0,outLen); + if(s->has_ssm_pay_head_len){ + set_SSM_pay_head_len(out, SSM_PAY_HEAD_len); + } + set_SSM_page_code(out, SM_pc_buffer); + set_SSM_page_len(out, SSM_PAGE_len); + + if(s->s.source == SOURCE_ADF_DUPLEX){ + set_SSM_BUFF_duplex(out, 1); + } + else if(s->s.source == SOURCE_FLATBED){ + set_SSM_BUFF_fb(out, 1); + } + if(s->buffermode){ + set_SSM_BUFF_async(out, 1); + } + if(0){ + set_SSM_BUFF_ald(out, 1); + } + if(0){ + set_SSM_BUFF_unk(out,1); + } + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + DBG (10, "ssm_buffer: finish\n"); + + return ret; +} + +static SANE_Status +ssm_df (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[SET_SCAN_MODE_len]; + size_t cmdLen = SET_SCAN_MODE_len; + + unsigned char out[SSM_PAY_len]; + size_t outLen = SSM_PAY_len; + + DBG (10, "ssm_df: start\n"); + + if(!s->has_ssm || !s->has_df){ + DBG (10, "ssm_df: unsupported, finishing\n"); + return ret; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SET_SCAN_MODE_code); + set_SSM_pf(cmd, 1); + set_SSM_pay_len(cmd, outLen); + + memset(out,0,outLen); + if(s->has_ssm_pay_head_len){ + set_SSM_pay_head_len(out, SSM_PAY_HEAD_len); + } + set_SSM_page_code(out, SM_pc_df); + set_SSM_page_len(out, SSM_PAGE_len); + + /* deskew by roller */ + if(s->rollerdeskew){ + set_SSM_DF_deskew_roll(out, 1); + } + + /* staple detection */ + if(s->stapledetect){ + set_SSM_DF_staple(out, 1); + } + + /* thickness */ + if(s->df_thickness){ + set_SSM_DF_thick(out, 1); + } + + /* length */ + if(s->df_length){ + set_SSM_DF_len(out, 1); + } + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + DBG (10, "ssm_df: finish\n"); + + return ret; +} + +static SANE_Status +ssm_do (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[SET_SCAN_MODE_len]; + size_t cmdLen = SET_SCAN_MODE_len; + + unsigned char out[SSM_PAY_len]; + size_t outLen = SSM_PAY_len; + + DBG (10, "ssm_do: start\n"); + + if(!s->has_ssm || !s->can_color){ + DBG (10, "ssm_do: unsupported, finishing\n"); + return ret; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SET_SCAN_MODE_code); + set_SSM_pf(cmd, 1); + set_SSM_pay_len(cmd, outLen); + + memset(out,0,outLen); + if(s->has_ssm_pay_head_len){ + set_SSM_pay_head_len(out, SSM_PAY_HEAD_len); + } + set_SSM_page_code(out, SM_pc_dropout); + set_SSM_page_len(out, SSM_PAGE_len); + + set_SSM_DO_unk1(out, 0x03); + + switch(s->dropout_color_f){ + case COLOR_RED: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_do(out,SSM_DO_red); + break; + case COLOR_GREEN: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_do(out,SSM_DO_green); + break; + case COLOR_BLUE: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_do(out,SSM_DO_blue); + break; + case COLOR_EN_RED: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_en(out,SSM_DO_red); + break; + case COLOR_EN_GREEN: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_en(out,SSM_DO_green); + break; + case COLOR_EN_BLUE: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_f_en(out,SSM_DO_blue); + break; + } + + switch(s->dropout_color_b){ + case COLOR_RED: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_do(out,SSM_DO_red); + break; + case COLOR_GREEN: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_do(out,SSM_DO_green); + break; + case COLOR_BLUE: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_do(out,SSM_DO_blue); + break; + case COLOR_EN_RED: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_en(out,SSM_DO_red); + break; + case COLOR_EN_GREEN: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_en(out,SSM_DO_green); + break; + case COLOR_EN_BLUE: + set_SSM_DO_unk2(out, 0x05); + set_SSM_DO_b_en(out,SSM_DO_blue); + break; + } + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + DBG (10, "ssm_do: finish\n"); + + return ret; +} + +/* used by recent scanners. meaning of payloads unknown */ +static SANE_Status +ssm2 (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[SET_SCAN_MODE2_len]; + size_t cmdLen = SET_SCAN_MODE2_len; + + unsigned char out[SSM2_PAY_len]; + size_t outLen = SSM2_PAY_len; + + DBG (10, "ssm2:start\n"); + + if(!s->has_ssm2){ + DBG (10, "ssm2: unsupported, finishing\n"); + return ret; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SET_SCAN_MODE2_code); + /*FIXME: set this correctly */ + set_SSM2_page_code(cmd, 0); + set_SSM2_pay_len(cmd, outLen); + + /*FIXME: set these correctly */ + memset(out,0,outLen); + set_SSM2_unk(out, 0); + set_SSM2_unk2(out, 0); + set_SSM2_unk3(out, 0); + + /* + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + );*/ + + DBG (10, "ssm2: finish\n"); + + return ret; +} + +static SANE_Status +read_panel(struct scanner *s,SANE_Int option) +{ + SANE_Status ret=SANE_STATUS_GOOD; + + unsigned char cmd[READ_len]; + size_t cmdLen = READ_len; + + unsigned char in[R_PANEL_len]; + size_t inLen = R_PANEL_len; + + DBG (10, "read_panel: start %d\n", option); + + if(!s->can_read_panel){ + DBG (10, "read_panel: unsupported, finishing\n"); + return ret; + } + + /* only run this if frontend has read previous value + * or if the caller does not want the data stored */ + if (!option || !s->hw_read[option-OPT_START]) { + + DBG (15, "read_panel: running\n"); + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, READ_code); + set_R_datatype_code (cmd, SR_datatype_panel); + set_R_xfer_length (cmd, inLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + + if (ret == SANE_STATUS_GOOD || ret == SANE_STATUS_EOF) { + /*set flags indicating there is data to read*/ + if(option) + memset(s->hw_read,1,sizeof(s->hw_read)); + + s->panel_start = get_R_PANEL_start(in); + s->panel_stop = get_R_PANEL_stop(in); + s->panel_butt3 = get_R_PANEL_butt3(in); + s->panel_new_file = get_R_PANEL_new_file(in); + s->panel_count_only = get_R_PANEL_count_only(in); + s->panel_bypass_mode = get_R_PANEL_bypass_mode(in); + s->panel_enable_led = get_R_PANEL_enable_led(in); + s->panel_counter = get_R_PANEL_counter(in); + ret = SANE_STATUS_GOOD; + } + } + + if(option) + s->hw_read[option-OPT_START] = 0; + + DBG (10, "read_panel: finish %d\n",s->panel_counter); + + return ret; +} + +static SANE_Status +send_panel(struct scanner *s) +{ + SANE_Status ret=SANE_STATUS_GOOD; + + unsigned char cmd[SEND_len]; + size_t cmdLen = SEND_len; + + unsigned char out[S_PANEL_len]; + size_t outLen = S_PANEL_len; + + DBG (10, "send_panel: start\n"); + + if(!s->can_write_panel){ + DBG (10, "send_panel: unsupported, finishing\n"); + return ret; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SEND_code); + set_S_xfer_datatype (cmd, SR_datatype_panel); + set_S_xfer_length (cmd, outLen); + + memset(out,0,outLen); + set_S_PANEL_enable_led(out,s->panel_enable_led); + set_S_PANEL_counter(out,s->panel_counter); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + if (ret == SANE_STATUS_EOF) { + ret = SANE_STATUS_GOOD; + } + + DBG (10, "send_panel: finish %d\n", ret); + + return ret; +} + +/* + * @@ Section 4 - SANE scanning functions + */ +/* + * Called by SANE to retrieve information about the type of data + * that the current scan will return. + * + * From the SANE spec: + * This function is used to obtain the current scan parameters. The + * returned parameters are guaranteed to be accurate between the time + * a scan has been started (sane_start() has been called) and the + * completion of that request. Outside of that window, the returned + * values are best-effort estimates of what the parameters will be + * when sane_start() gets invoked. + * + * Calling this function before a scan has actually started allows, + * for example, to get an estimate of how big the scanned image will + * be. The parameters passed to this function are the handle h of the + * device for which the parameters should be obtained and a pointer p + * to a parameter structure. + */ +SANE_Status +sane_get_parameters (SANE_Handle handle, SANE_Parameters * params) +{ + SANE_Status ret = SANE_STATUS_GOOD; + struct scanner *s = (struct scanner *) handle; + + DBG (10, "sane_get_parameters: start\n"); + + if(!s->started){ + ret = update_params(s,0); + if(ret){ + DBG (5, "sane_get_parameters: up error, returning %d\n", ret); + return ret; + } + } + + /* this backend only sends single frame images */ + params->last_frame = 1; + + params->format = s->i.format; + params->lines = s->i.height; + params->depth = s->i.bpp; + if(params->depth == 24) params->depth = 8; + params->pixels_per_line = s->i.width; + params->bytes_per_line = s->i.Bpl; + + DBG(15,"sane_get_parameters: x: max=%d, page=%d, gpw=%d, res=%d\n", + s->valid_x, s->i.page_x, get_page_width(s), s->i.dpi_x); + + DBG(15,"sane_get_parameters: y: max=%d, page=%d, gph=%d, res=%d\n", + s->max_y, s->i.page_y, get_page_height(s), s->i.dpi_y); + + DBG(15,"sane_get_parameters: area: tlx=%d, brx=%d, tly=%d, bry=%d\n", + s->i.tl_x, s->i.br_x, s->i.tl_y, s->i.br_y); + + DBG (15, "sane_get_parameters: params: ppl=%d, Bpl=%d, lines=%d\n", + params->pixels_per_line, params->bytes_per_line, params->lines); + + DBG (15, "sane_get_parameters: params: format=%d, depth=%d, last=%d\n", + params->format, params->depth, params->last_frame); + + DBG (10, "sane_get_parameters: finish\n"); + + return ret; +} + +SANE_Status +update_params(struct scanner *s, int calib) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "update_params: start\n"); + + s->u.width = (s->u.br_x - s->u.tl_x) * s->u.dpi_x / 1200; + s->u.height = (s->u.br_y - s->u.tl_y) * s->u.dpi_y / 1200; + + if (s->u.mode == MODE_COLOR) { + s->u.format = SANE_FRAME_RGB; + s->u.bpp = 24; + } + else if (s->u.mode == MODE_GRAYSCALE) { + s->u.format = SANE_FRAME_GRAY; + s->u.bpp = 8; + } + else { + s->u.format = SANE_FRAME_GRAY; + s->u.bpp = 1; + + /* round down to byte boundary */ + s->u.width -= s->u.width % 8; + } + + /* round down to pixel boundary for some scanners */ + s->u.width -= s->u.width % s->ppl_mod; + +#ifdef SANE_FRAME_JPEG + /* jpeg requires 8x8 squares */ + if(s->compress == COMP_JPEG && s->u.mode >= MODE_GRAYSCALE){ + s->u.format = SANE_FRAME_JPEG; + s->u.width -= s->u.width % 8; + s->u.height -= s->u.height % 8; + } +#endif + + s->u.Bpl = s->u.width * s->u.bpp / 8; + s->u.valid_Bpl = s->u.Bpl; + s->u.valid_width = s->u.width; + + DBG (15, "update_params: user params: w:%d h:%d m:%d f:%d b:%d\n", + s->u.width, s->u.height, s->u.mode, s->u.format, s->u.bpp); + DBG (15, "update_params: user params: B:%d vB:%d vw:%d\n", + s->u.Bpl, s->u.valid_Bpl, s->u.valid_width); + DBG (15, "update_params: user params: x b:%d t:%d d:%d y b:%d t:%d d:%d\n", + s->u.br_x, s->u.tl_x, s->u.dpi_x, s->u.br_y, s->u.tl_y, s->u.dpi_y); + + /* some scanners are limited in their valid scan params + * make a second version of the params struct, but + * override the user's values with what the scanner can actually do */ + + memcpy(&s->s,&s->u,sizeof(struct img_params)); + + /*********** missing modes (move up to valid one) **************/ + if(s->s.mode == MODE_LINEART && !s->can_monochrome){ + s->s.mode = MODE_GRAYSCALE; + s->s.format = SANE_FRAME_GRAY; + s->s.bpp = 8; + } + if(s->s.mode == MODE_GRAYSCALE && !s->can_grayscale){ + s->s.mode = MODE_COLOR; + s->s.format = SANE_FRAME_RGB; + s->s.bpp = 24; + } + if(s->s.mode == MODE_COLOR && !s->can_color){ + DBG (5, "update_params: no valid mode\n"); + return SANE_STATUS_INVAL; + } + + /********** missing resolutions (move up to valid one) *********/ + if(!s->step_x_res){ + int i; + for(i=0;i<DPI_1200;i++){ + + /* this res is smaller or invalid, skip it */ + if(s->s.dpi_x > dpi_list[i] || !s->std_res_x[i]) + continue; + + /* same & valid res, done */ + if(s->s.dpi_x == dpi_list[i]) + break; + + /* different & valid res, switch */ + s->s.dpi_x = dpi_list[i]; + break; + } + + if(i > DPI_1200){ + DBG (5, "update_params: no dpi\n"); + return SANE_STATUS_INVAL; + } + } + + /*********** weird scan area (increase to valid one) *********/ + if(s->fixed_width){ + s->s.tl_x = 0; + s->s.br_x = s->max_x; + s->s.page_x = s->max_x; + } + + /*recalculate new params*/ + s->s.width = (s->s.br_x - s->s.tl_x) * s->s.dpi_x / 1200; + + /* round down to byte boundary */ + if(s->s.mode < MODE_GRAYSCALE){ + s->s.width -= s->s.width % 8; + } + + /* round down to pixel boundary for some scanners */ + s->s.width -= s->s.width % s->ppl_mod; + + s->s.valid_width = s->s.width; + s->s.valid_Bpl = s->s.valid_width * s->s.bpp / 8; + + /* some machines (DR-2050) require even bytes per scanline */ + /* increase width and Bpl, but not valid_width and valid_Bpl */ + if(s->even_Bpl && (s->s.width % 2)){ + s->s.width++; + } + + s->s.Bpl = s->s.width * s->s.bpp / 8; + + /* figure out how many valid bytes per line (2510 is padded) */ + if(s->color_interlace[SIDE_FRONT] == COLOR_INTERLACE_2510){ + s->s.valid_Bpl = s->s.Bpl*11/12; + s->s.valid_width = s->s.width*11/12; + } + + /* some scanners need longer scans because front/back is offset */ + if(s->u.source == SOURCE_ADF_DUPLEX && s->duplex_offset && !calib) + s->s.height = (s->u.br_y-s->u.tl_y+s->duplex_offset) * s->u.dpi_y / 1200; + + /* round lines up to even number */ + s->s.height += s->s.height % 2; + + DBG (15, "update_params: scan params: w:%d h:%d m:%d f:%d b:%d\n", + s->s.width, s->s.height, s->s.mode, s->s.format, s->s.bpp); + DBG (15, "update_params: scan params: B:%d vB:%d vw:%d\n", + s->s.Bpl, s->s.valid_Bpl, s->s.valid_width); + DBG (15, "update_params: scan params: x b:%d t:%d d:%d y b:%d t:%d d:%d\n", + s->s.br_x, s->s.tl_x, s->s.dpi_x, s->s.br_y, s->s.tl_y, s->s.dpi_y); + + /* make a third (intermediate) version of the params struct, + * currently identical to the user's params. this is what + * we actually will send back to the user (though buffer_xxx + * functions might change these values after this runs) */ + + /* calibration code needs the data just as it comes from the scanner */ + if(calib) + memcpy(&s->i,&s->s,sizeof(struct img_params)); + /* normal scans need the data cleaned for presentation to the user */ + else{ + memcpy(&s->i,&s->u,sizeof(struct img_params)); + /*dumb scanners pad the top of front page in duplex*/ + if(s->i.source == SOURCE_ADF_DUPLEX) + s->i.skip_lines[s->duplex_offset_side] = s->duplex_offset * s->i.dpi_y / 1200; + } + + DBG (15, "update_params: i params: w:%d h:%d m:%d f:%d b:%d\n", + s->i.width, s->i.height, s->i.mode, s->i.format, s->i.bpp); + DBG (15, "update_params: i params: B:%d vB:%d vw:%d\n", + s->i.Bpl, s->i.valid_Bpl, s->i.valid_width); + DBG (15, "update_params: i params: x b:%d t:%d d:%d y b:%d t:%d d:%d\n", + s->i.br_x, s->i.tl_x, s->i.dpi_x, s->i.br_y, s->i.tl_y, s->i.dpi_y); + + DBG (10, "update_params: finish\n"); + return ret; +} + +/* reset image size parameters after buffer_xxx functions changed them */ +SANE_Status +update_i_params(struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "update_i_params: start\n"); + + s->i.width = s->u.width; + s->i.Bpl = s->u.Bpl; + + DBG (10, "update_i_params: finish\n"); + return ret; +} + +/* + * Called by SANE when a page acquisition operation is to be started. + * commands: set window, object pos, and scan + * + * this will be called between sides of a duplex scan, + * and at the start of each page of an adf batch. + * hence, we spend alot of time playing with s->started, etc. + */ +SANE_Status +sane_start (SANE_Handle handle) +{ + struct scanner *s = handle; + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "sane_start: start\n"); + DBG (15, "started=%d, side=%d, source=%d\n", + s->started, s->side, s->u.source); + + /* undo any prior sane_cancel calls */ + s->cancelled=0; + + /* protect this block from sane_cancel */ + s->reading=1; + + /* not finished with current side, error */ + if (s->started && !s->u.eof[s->side]) { + DBG(5,"sane_start: previous transfer not finished?"); + return SANE_STATUS_INVAL; + } + + /* batch start? inititalize struct and scanner */ + if(!s->started){ + + /* load side marker */ + if(s->u.source == SOURCE_ADF_BACK){ + s->side = SIDE_BACK; + } + else{ + s->side = SIDE_FRONT; + } + + /* eject paper leftover*/ + if(object_position (s, SANE_FALSE)){ + DBG (5, "sane_start: ERROR: cannot eject page\n"); + } + + /* wait for scanner to finish eject */ + ret = wait_scanner (s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot wait scanner\n"); + goto errors; + } + + /* load the brightness/contrast lut with linear slope for calibration */ + ret = load_lut (s->lut, 8, 8, 0, 255, 0, 0); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load lut\n"); + goto errors; + } + + /* AFE cal */ + if((ret = calibrate_AFE(s))){ + DBG (5, "sane_start: ERROR: cannot cal afe\n"); + goto errors; + } + + /* fine cal */ + if((ret = calibrate_fine(s))){ + DBG (5, "sane_start: ERROR: cannot cal fine\n"); + goto errors; + } + + if((ret = calibrate_fine_buffer(s))){ + DBG (5, "sane_start: ERROR: cannot cal fine from buffer\n"); + goto errors; + } + + /* reset the page counter after calibration */ + s->panel_counter = 0; + s->prev_page = 0; + if(send_panel(s)){ + DBG (5, "sane_start: ERROR: cannot send panel\n"); + } + + /* load our own private copy of scan params */ + ret = update_params(s,0); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot update_params\n"); + goto errors; + } + + /* set window command */ + ret = set_window(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot set window\n"); + goto errors; + } + + /* buffer/duplex/ald command */ + ret = ssm_buffer(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot ssm buffer\n"); + goto errors; + } + + /* dropout color command */ + ret = ssm_do(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot ssm do\n"); + goto errors; + } + + /* double feed detection command */ + ret = ssm_df(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot ssm df\n"); + goto errors; + } + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot clean_params\n"); + goto errors; + } + + /* make large buffers to hold the images */ + ret = image_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load buffers\n"); + goto errors; + } + + /* load the brightness/contrast lut with user choices */ + ret = load_lut (s->lut, 8, 8, 0, 255, s->contrast, s->brightness); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load lut\n"); + goto errors; + } + + /* grab next page */ + ret = object_position (s, SANE_TRUE); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load page\n"); + goto errors; + } + + /* wait for scanner to finish load */ + ret = wait_scanner (s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot wait scanner\n"); + goto errors; + } + + /* start scanning */ + ret = start_scan (s,0); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot start_scan\n"); + goto errors; + } + + s->started = 1; + } + + /* stuff done for subsequent images */ + else{ + + /* duplex needs to switch sides */ + if(s->s.source == SOURCE_ADF_DUPLEX){ + s->side = !s->side; + } + + /* reset the intermediate params */ + ret = update_i_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot update_i_params\n"); + goto errors; + } + + /* set clean defaults with new sheet of paper */ + /* dont reset the transfer vars on backside of duplex page */ + /* otherwise buffered back page will be lost */ + /* ingest paper with adf (no-op for fb) */ + /* dont call object pos or scan on back side of duplex scan */ + if(s->side == SIDE_FRONT || s->s.source == SOURCE_ADF_BACK){ + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot clean_params\n"); + goto errors; + } + + /* big scanners and small ones in non-buff mode: OP to detect paper */ + if(s->always_op || !s->buffermode){ + ret = object_position (s, SANE_TRUE); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load page\n"); + goto errors; + } + + /* user wants unbuffered scans */ + /* send scan command */ + if(!s->buffermode){ + ret = start_scan (s,0); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot start_scan\n"); + goto errors; + } + } + } + + /* small, buffering scanners check for more pages by reading counter */ + else{ + ret = read_panel (s, OPT_COUNTER); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot load page\n"); + goto errors; + } + if(s->prev_page == s->panel_counter){ + DBG (5, "sane_start: same counter (%d) no paper?\n",s->prev_page); + ret = SANE_STATUS_NO_DOCS; + goto errors; + } + DBG (5, "sane_start: diff counter (%d/%d)\n", + s->prev_page,s->panel_counter); + } + } + } + + /* reset jpeg params on each page */ + s->jpeg_stage=JPEG_STAGE_NONE; + s->jpeg_ff_offset=0; + + DBG (15, "started=%d, side=%d, source=%d\n", + s->started, s->side, s->u.source); + + /* certain options require the entire image to + * be collected from the scanner before we can + * tell the user the size of the image. the sane + * API has no way to inform the frontend of this, + * so we block and buffer. yuck */ + if( (s->swdeskew || s->swdespeck || s->swcrop) +#ifdef SANE_FRAME_JPEG + && s->s.format != SANE_FRAME_JPEG +#endif + ){ + + /* get image */ + while(!s->s.eof[s->side] && !ret){ + SANE_Int len = 0; + ret = sane_read((SANE_Handle)s, NULL, 0, &len); + } + + /* check for errors */ + if (ret != SANE_STATUS_GOOD) { + DBG (5, "sane_start: ERROR: cannot buffer image\n"); + goto errors; + } + + DBG (5, "sane_start: OK: done buffering\n"); + + /* finished buffering, adjust image as required */ + if(s->swdeskew){ + buffer_deskew(s,s->side); + } + if(s->swcrop){ + buffer_crop(s,s->side); + } + if(s->swdespeck){ + buffer_despeck(s,s->side); + } + + } + + ret = check_for_cancel(s); + s->reading = 0; + + DBG (10, "sane_start: finish %d\n", ret); + return ret; + + errors: + DBG (10, "sane_start: error %d\n", ret); + s->started = 0; + s->cancelled = 0; + s->reading = 0; + return ret; +} + +/* + * cleans params for new scan + */ +static SANE_Status +clean_params (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "clean_params: start\n"); + + s->u.eof[0]=0; + s->u.eof[1]=0; + s->u.bytes_sent[0]=0; + s->u.bytes_sent[1]=0; + s->u.bytes_tot[0]=0; + s->u.bytes_tot[1]=0; + + s->i.eof[0]=0; + s->i.eof[1]=0; + s->i.bytes_sent[0]=0; + s->i.bytes_sent[1]=0; + s->i.bytes_tot[0]=0; + s->i.bytes_tot[1]=0; + + s->s.eof[0]=0; + s->s.eof[1]=0; + s->s.bytes_sent[0]=0; + s->s.bytes_sent[1]=0; + s->s.bytes_tot[0]=0; + s->s.bytes_tot[1]=0; + + /* store the number of front bytes */ + if ( s->u.source != SOURCE_ADF_BACK ) + s->u.bytes_tot[SIDE_FRONT] = s->u.Bpl * s->u.height; + + if ( s->i.source != SOURCE_ADF_BACK ) + s->i.bytes_tot[SIDE_FRONT] = s->i.Bpl * s->i.height; + + if ( s->s.source != SOURCE_ADF_BACK ) + s->s.bytes_tot[SIDE_FRONT] = s->s.Bpl * s->s.height; + + /* store the number of back bytes */ + if ( s->u.source == SOURCE_ADF_DUPLEX || s->u.source == SOURCE_ADF_BACK ) + s->u.bytes_tot[SIDE_BACK] = s->u.Bpl * s->u.height; + + if ( s->i.source == SOURCE_ADF_DUPLEX || s->i.source == SOURCE_ADF_BACK ) + s->i.bytes_tot[SIDE_BACK] = s->i.Bpl * s->i.height; + + if ( s->s.source == SOURCE_ADF_DUPLEX || s->s.source == SOURCE_ADF_BACK ) + s->s.bytes_tot[SIDE_BACK] = s->s.Bpl * s->s.height; + + DBG (10, "clean_params: finish\n"); + + return ret; +} + +/* + * frees/callocs buffers to hold the scan data + */ +static SANE_Status +image_buffers (struct scanner *s, int setup) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int side; + + DBG (10, "image_buffers: start\n"); + + for(side=0;side<2;side++){ + + /* free current buffer */ + if (s->buffers[side]) { + DBG (15, "image_buffers: free buffer %d.\n",side); + free(s->buffers[side]); + s->buffers[side] = NULL; + } + + /* build new buffer if asked */ + if(s->i.bytes_tot[side] && setup){ + s->buffers[side] = calloc (1,s->i.bytes_tot[side]); + if (!s->buffers[side]) { + DBG (5, "image_buffers: Error, no buffer %d.\n",side); + return SANE_STATUS_NO_MEM; + } + } + } + + DBG (10, "image_buffers: finish\n"); + + return ret; +} + +/* + * This routine issues a SCSI SET WINDOW command to the scanner, using the + * values currently in the s->s param structure. + */ +static SANE_Status +set_window (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + /* The command specifies the number of bytes in the data phase + * the data phase has a header, followed by 1 window desc block + * the header specifies the number of bytes in 1 window desc block + */ + + unsigned char cmd[SET_WINDOW_len]; + size_t cmdLen = SET_WINDOW_len; + + unsigned char out[SW_header_len + SW_desc_len]; + size_t outLen = SW_header_len + SW_desc_len; + + unsigned char * header = out; /*header*/ + unsigned char * desc1 = out + SW_header_len; /*descriptor*/ + + DBG (10, "set_window: start\n"); + + /*build the payload*/ + memset(out,0,outLen); + + /* set window desc size in header */ + set_WPDB_wdblen(header, SW_desc_len); + + /* init the window block */ + if (s->s.source == SOURCE_ADF_BACK) { + set_WD_wid (desc1, WD_wid_back); + } + else{ + set_WD_wid (desc1, WD_wid_front); + } + + set_WD_Xres (desc1, s->s.dpi_x); + set_WD_Yres (desc1, s->s.dpi_y); + + /* some machines need max width */ + if(s->fixed_width){ + set_WD_ULX (desc1, 0); + set_WD_width (desc1, s->max_x); + } + + /* or they align left */ + else if(s->u.source == SOURCE_FLATBED){ + set_WD_ULX (desc1, s->s.tl_x); + set_WD_width (desc1, s->s.width * 1200/s->s.dpi_x); + } + + /* or we have to center the window ourselves */ + else{ + set_WD_ULX (desc1, (s->max_x - s->s.page_x) / 2 + s->s.tl_x); + set_WD_width (desc1, s->s.width * 1200/s->s.dpi_x); + } + + /* some models require that the tly value be inverted? */ + if(s->invert_tly) + set_WD_ULY (desc1, ~s->s.tl_y); + else + set_WD_ULY (desc1, s->s.tl_y); + + set_WD_length (desc1, s->s.height * 1200/s->s.dpi_y); + + if(s->has_btc){ + /*convert our common -127 to +127 range into HW's range + *FIXME: this code assumes hardware range of 0-255 */ + set_WD_brightness (desc1, s->brightness+128); + + set_WD_threshold (desc1, s->threshold); + + /*convert our common -127 to +127 range into HW's range + *FIXME: this code assumes hardware range of 0-255 */ + set_WD_contrast (desc1, s->contrast+128); + } + + set_WD_composition (desc1, s->s.mode); + + if(s->s.bpp == 24) + set_WD_bitsperpixel (desc1, 8); + else + set_WD_bitsperpixel (desc1, s->s.bpp); + + if(s->s.mode == MODE_HALFTONE){ + /*set_WD_ht_type(desc1, s->ht_type); + set_WD_ht_pattern(desc1, s->ht_pattern);*/ + } + + set_WD_rif (desc1, s->rif); + set_WD_rgb(desc1, s->rgb_format); + set_WD_padding(desc1, s->padding); + + /*FIXME: what is this? */ + set_WD_reserved2(desc1, s->unknown_byte2); + + set_WD_compress_type(desc1, COMP_NONE); + set_WD_compress_arg(desc1, 0); + +#ifdef SANE_FRAME_JPEG + /* some scanners support jpeg image compression, for color/gs only */ + if(s->s.format == SANE_FRAME_JPEG){ + set_WD_compress_type(desc1, COMP_JPEG); + set_WD_compress_arg(desc1, s->compress_arg); + } +#endif + + /*build the command*/ + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SET_WINDOW_code); + set_SW_xferlen(cmd, outLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + if (!ret && s->s.source == SOURCE_ADF_DUPLEX) { + set_WD_wid (desc1, WD_wid_back); + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + } + + DBG (10, "set_window: finish\n"); + + return ret; +} + +/* + * Issues the SCSI OBJECT POSITION command if an ADF is in use. + */ +static SANE_Status +object_position (struct scanner *s, int i_load) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[OBJECT_POSITION_len]; + size_t cmdLen = OBJECT_POSITION_len; + + DBG (10, "object_position: start\n"); + + if (s->u.source == SOURCE_FLATBED) { + DBG (10, "object_position: flatbed no-op\n"); + return SANE_STATUS_GOOD; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, OBJECT_POSITION_code); + + if (i_load) { + DBG (15, "object_position: load\n"); + set_OP_autofeed (cmd, OP_Feed); + } + else { + DBG (15, "object_position: eject\n"); + set_OP_autofeed (cmd, OP_Discharge); + } + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + NULL, NULL + ); + if (ret != SANE_STATUS_GOOD) + return ret; + + DBG (10, "object_position: finish\n"); + + return ret; +} + +/* + * Issues SCAN command. + * + * (This doesn't actually read anything, it just tells the scanner + * to start scanning.) + */ +static SANE_Status +start_scan (struct scanner *s, int type) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[SCAN_len]; + size_t cmdLen = SCAN_len; + + unsigned char out[] = {WD_wid_front, WD_wid_back}; + size_t outLen = 2; + + DBG (10, "start_scan: start\n"); + + /* calibration scans use 0xff or 0xfe */ + if(type){ + out[0] = type; + out[1] = type; + } + + if (s->s.source != SOURCE_ADF_DUPLEX) { + outLen--; + if(s->s.source == SOURCE_ADF_BACK) { + out[0] = WD_wid_back; + } + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, SCAN_code); + set_SC_xfer_length (cmd, outLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + out, outLen, + NULL, NULL + ); + + DBG (10, "start_scan: finish\n"); + + return ret; +} + +/* + * Called by SANE to read data. + * + * From the SANE spec: + * This function is used to read image data from the device + * represented by handle h. Argument buf is a pointer to a memory + * area that is at least maxlen bytes long. The number of bytes + * returned is stored in *len. A backend must set this to zero when + * the call fails (i.e., when a status other than SANE_STATUS_GOOD is + * returned). + * + * When the call succeeds, the number of bytes returned can be + * anywhere in the range from 0 to maxlen bytes. + */ +SANE_Status +sane_read (SANE_Handle handle, SANE_Byte * buf, SANE_Int max_len, SANE_Int * len) +{ + struct scanner *s = (struct scanner *) handle; + SANE_Status ret=SANE_STATUS_GOOD; + + DBG (10, "sane_read: start\n"); + + *len=0; + + /* maybe cancelled? */ + if(!s->started){ + DBG (5, "sane_read: not started, call sane_start\n"); + return SANE_STATUS_CANCELLED; + } + + /* sane_start required between sides */ + if(s->u.bytes_sent[s->side] == s->i.bytes_tot[s->side]){ + s->u.eof[s->side] = 1; + DBG (15, "sane_read: returning eof\n"); + return SANE_STATUS_EOF; + } + + s->reading = 1; + + /* double width pnm interlacing */ + if(s->s.source == SOURCE_ADF_DUPLEX + && s->s.format <= SANE_FRAME_RGB + && s->duplex_interlace != DUPLEX_INTERLACE_NONE + ){ + + /* buffer both sides */ + if(!s->s.eof[SIDE_FRONT] || !s->s.eof[SIDE_BACK]){ + ret = read_from_scanner_duplex(s, 0); + if(ret){ + DBG(5,"sane_read: front returning %d\n",ret); + goto errors; + } + /*read last block, update counter*/ + if(s->s.eof[SIDE_FRONT] && s->s.eof[SIDE_BACK]){ + s->prev_page++; + DBG(15,"sane_read: duplex counter %d\n",s->prev_page); + } + } + } + + /* simplex or non-alternating duplex */ + else{ + if(!s->s.eof[s->side]){ + ret = read_from_scanner(s, s->side, 0); + if(ret){ + DBG(5,"sane_read: side %d returning %d\n",s->side,ret); + goto errors; + } + /*read last block, update counter*/ + if(s->s.eof[s->side]){ + s->prev_page++; + DBG(15,"sane_read: side %d counter %d\n",s->side,s->prev_page); + } + } + } + + /* copy a block from buffer to frontend */ + ret = read_from_buffer(s,buf,max_len,len,s->side); + if(ret) + goto errors; + + ret = check_for_cancel(s); + s->reading = 0; + + DBG (10, "sane_read: finish %d\n", ret); + return ret; + + errors: + DBG (10, "sane_read: error %d\n", ret); + s->reading = 0; + s->cancelled = 0; + s->started = 0; + return ret; +} + +static SANE_Status +read_from_scanner(struct scanner *s, int side, int exact) +{ + SANE_Status ret=SANE_STATUS_GOOD; + + unsigned char cmd[READ_len]; + size_t cmdLen = READ_len; + + unsigned char * in; + size_t inLen = 0; + + size_t bytes = s->buffer_size; + size_t remain = s->s.bytes_tot[side] - s->s.bytes_sent[side]; + + DBG (10, "read_from_scanner: start\n"); + + /* all requests must end on line boundary */ + bytes -= (bytes % s->s.Bpl); + + /* some larger scanners require even bytes per block */ + if(bytes % 2){ + bytes -= s->s.Bpl; + } + + /* usually (image) we want to read too much data, and get RS */ + /* sometimes (calib) we want to do an exact read */ + if(exact && bytes > remain){ + bytes = remain; + } + + DBG(15, "read_from_scanner: si:%d to:%d rx:%d re:%lu bu:%d pa:%lu ex:%d\n", + side, s->s.bytes_tot[side], s->s.bytes_sent[side], + (unsigned long)remain, s->buffer_size, (unsigned long)bytes, exact); + + inLen = bytes; + in = malloc(inLen); + if(!in){ + DBG(5, "read_from_scanner: not enough mem for buffer: %d\n",(int)inLen); + return SANE_STATUS_NO_MEM; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, READ_code); + set_R_datatype_code (cmd, SR_datatype_image); + + set_R_xfer_length (cmd, inLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + + if (ret == SANE_STATUS_GOOD) { + DBG(15, "read_from_scanner: got GOOD, returning GOOD %lu\n", (unsigned long)inLen); + } + else if (ret == SANE_STATUS_EOF) { + DBG(15, "read_from_scanner: got EOF, finishing %lu\n", (unsigned long)inLen); + } + else if (ret == SANE_STATUS_DEVICE_BUSY) { + DBG(5, "read_from_scanner: got BUSY, returning GOOD\n"); + inLen = 0; + ret = SANE_STATUS_GOOD; + } + else { + DBG(5, "read_from_scanner: error reading data block status = %d\n",ret); + inLen = 0; + } + +#ifdef SANE_FRAME_JPEG + /* this is jpeg data, we need to fix the missing image size */ + if(s->s.format == SANE_FRAME_JPEG){ + + /* look for the SOF header near the beginning */ + if(s->jpeg_stage == JPEG_STAGE_NONE || s->jpeg_ff_offset < 0x0d){ + + size_t i; + + for(i=0;i<inLen;i++){ + + /* about to change stage */ + if(s->jpeg_stage == JPEG_STAGE_NONE && in[i] == 0xff){ + s->jpeg_ff_offset=0; + continue; + } + + s->jpeg_ff_offset++; + + /* last byte was an ff, this byte is SOF */ + if(s->jpeg_ff_offset == 1 && in[i] == 0xc0){ + s->jpeg_stage = JPEG_STAGE_SOF; + continue; + } + + if(s->jpeg_stage == JPEG_STAGE_SOF){ + + /* lines in start of frame, overwrite it */ + if(s->jpeg_ff_offset == 5){ + in[i] = (s->s.height >> 8) & 0xff; + continue; + } + if(s->jpeg_ff_offset == 6){ + in[i] = s->s.height & 0xff; + continue; + } + + /* width in start of frame, overwrite it */ + if(s->jpeg_ff_offset == 7){ + in[i] = (s->s.width >> 8) & 0xff; + continue; + } + if(s->jpeg_ff_offset == 8){ + in[i] = s->s.width & 0xff; + continue; + } + } + } + } + } +#endif + + /*scanner may have sent more data than we asked for, chop it*/ + if(inLen > remain){ + inLen = remain; + } + + /* we've got some data, descramble and store it */ + if(inLen){ + copy_simplex(s,in,inLen,side); + } + + free(in); + + /* we've read all data, but not eof. clear and pretend */ + if(exact && inLen == remain){ + DBG (10, "read_from_scanner: exact read, clearing\n"); + ret = object_position (s,SANE_FALSE); + if(ret){ + return ret; + } + ret = SANE_STATUS_EOF; + } + + if(ret == SANE_STATUS_EOF){ + + switch (s->s.format){ + +#ifdef SANE_FRAME_JPEG + /* this is jpeg data, we need to change the total size */ + case SANE_FRAME_JPEG: + s->s.bytes_tot[side] = s->s.bytes_sent[side]; + s->i.bytes_tot[side] = s->i.bytes_sent[side]; + s->u.bytes_tot[side] = s->i.bytes_sent[side]; + break; +#endif + + /* this is non-jpeg data, fill remainder, change rx'd size */ + default: + + DBG (15, "read_from_scanner: eof: %d %d\n", s->i.bytes_tot[side], s->i.bytes_sent[side]); + + /* clone the last line repeatedly until the end */ + while(s->i.bytes_tot[side] > s->i.bytes_sent[side]){ + memcpy( + s->buffers[side]+s->i.bytes_sent[side]-s->i.Bpl, + s->buffers[side]+s->i.bytes_sent[side], + s->i.Bpl + ); + s->i.bytes_sent[side] += s->i.Bpl; + } + + DBG (15, "read_from_scanner: eof2: %d %d\n", s->i.bytes_tot[side], s->i.bytes_sent[side]); + + /* pretend we got all the data from scanner */ + s->s.bytes_sent[side] = s->s.bytes_tot[side]; + break; + } + + s->i.eof[side] = 1; + s->s.eof[side] = 1; + ret = SANE_STATUS_GOOD; + } + + DBG(15, "read_from_scanner: sto:%d srx:%d sef:%d uto:%d urx:%d uef:%d\n", + s->s.bytes_tot[side], s->s.bytes_sent[side], s->s.eof[side], + s->u.bytes_tot[side], s->u.bytes_sent[side], s->u.eof[side]); + + DBG (10, "read_from_scanner: finish\n"); + + return ret; +} + +/* cheaper scanners interlace duplex scans on a byte basis + * this code requests double width lines from scanner */ +static SANE_Status +read_from_scanner_duplex(struct scanner *s,int exact) +{ + SANE_Status ret=SANE_STATUS_GOOD; + + unsigned char cmd[READ_len]; + size_t cmdLen = READ_len; + + unsigned char * in; + size_t inLen = 0; + + size_t bytes = s->buffer_size; + size_t remain = s->s.bytes_tot[SIDE_FRONT] + s->s.bytes_tot[SIDE_BACK] + - s->s.bytes_sent[SIDE_FRONT] - s->s.bytes_sent[SIDE_BACK]; + + DBG (10, "read_from_scanner_duplex: start\n"); + + /* all requests must end on WIDE line boundary */ + bytes -= (bytes % (s->s.Bpl*2)); + + /* usually (image) we want to read too much data, and get RS */ + /* sometimes (calib) we want to do an exact read */ + if(exact && bytes > remain){ + bytes = remain; + } + + DBG(15, "read_from_scanner_duplex: re:%lu bu:%d pa:%lu ex:%d\n", + (unsigned long)remain, s->buffer_size, (unsigned long)bytes, exact); + + inLen = bytes; + in = malloc(inLen); + if(!in){ + DBG(5, "read_from_scanner_duplex: not enough mem for buffer: %d\n", + (int)inLen); + return SANE_STATUS_NO_MEM; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, READ_code); + set_R_datatype_code (cmd, SR_datatype_image); + + set_R_xfer_length (cmd, inLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + + if (ret == SANE_STATUS_GOOD) { + DBG(15, "read_from_scanner_duplex: got GOOD, returning GOOD %lu\n", (unsigned long)inLen); + } + else if (ret == SANE_STATUS_EOF) { + DBG(15, "read_from_scanner_duplex: got EOF, finishing %lu\n", (unsigned long)inLen); + } + else if (ret == SANE_STATUS_DEVICE_BUSY) { + DBG(5, "read_from_scanner_duplex: got BUSY, returning GOOD\n"); + inLen = 0; + ret = SANE_STATUS_GOOD; + } + else { + DBG(5, "read_from_scanner_duplex: error reading data block status = %d\n", + ret); + inLen = 0; + } + + /*scanner may have sent more data than we asked for, chop it*/ + if(inLen > remain){ + inLen = remain; + } + + /* we've got some data, descramble and store it */ + if(inLen){ + copy_duplex(s,in,inLen); + } + + free(in); + + /* we've read all data, but not eof. clear and pretend */ + if(exact && inLen == remain){ + DBG (10, "read_from_scanner_duplex: exact read, clearing\n"); + ret = object_position (s,SANE_FALSE); + if(ret){ + return ret; + } + ret = SANE_STATUS_EOF; + } + + if(ret == SANE_STATUS_EOF){ + + switch (s->s.format){ + +#ifdef SANE_FRAME_JPEG + /* this is jpeg data, we need to change the total size */ + case SANE_FRAME_JPEG: + s->s.bytes_tot[SIDE_FRONT] = s->s.bytes_sent[SIDE_FRONT]; + s->s.bytes_tot[SIDE_BACK] = s->s.bytes_sent[SIDE_BACK]; + s->i.bytes_tot[SIDE_FRONT] = s->i.bytes_sent[SIDE_FRONT]; + s->i.bytes_tot[SIDE_BACK] = s->i.bytes_sent[SIDE_BACK]; + s->u.bytes_tot[SIDE_FRONT] = s->i.bytes_sent[SIDE_FRONT]; + s->u.bytes_tot[SIDE_BACK] = s->i.bytes_sent[SIDE_BACK]; + break; +#endif + + /* this is non-jpeg data, fill remainder, change rx'd size */ + default: + + DBG (15, "read_from_scanner_duplex: eof: %d %d %d %d\n", + s->i.bytes_tot[SIDE_FRONT], s->i.bytes_sent[SIDE_FRONT], + s->i.bytes_tot[SIDE_BACK], s->i.bytes_sent[SIDE_BACK] + ); + + /* clone the last line repeatedly until the end */ + while(s->i.bytes_tot[SIDE_FRONT] > s->i.bytes_sent[SIDE_FRONT]){ + memcpy( + s->buffers[SIDE_FRONT]+s->i.bytes_sent[SIDE_FRONT]-s->i.Bpl, + s->buffers[SIDE_FRONT]+s->i.bytes_sent[SIDE_FRONT], + s->i.Bpl + ); + s->i.bytes_sent[SIDE_FRONT] += s->i.Bpl; + } + + /* clone the last line repeatedly until the end */ + while(s->i.bytes_tot[SIDE_BACK] > s->i.bytes_sent[SIDE_BACK]){ + memcpy( + s->buffers[SIDE_BACK]+s->i.bytes_sent[SIDE_BACK]-s->i.Bpl, + s->buffers[SIDE_BACK]+s->i.bytes_sent[SIDE_BACK], + s->i.Bpl + ); + s->i.bytes_sent[SIDE_BACK] += s->i.Bpl; + } + + DBG (15, "read_from_scanner_duplex: eof2: %d %d %d %d\n", + s->i.bytes_tot[SIDE_FRONT], s->i.bytes_sent[SIDE_FRONT], + s->i.bytes_tot[SIDE_BACK], s->i.bytes_sent[SIDE_BACK] + ); + + /* pretend we got all the data from scanner */ + s->s.bytes_sent[SIDE_FRONT] = s->s.bytes_tot[SIDE_FRONT]; + s->s.bytes_sent[SIDE_BACK] = s->s.bytes_tot[SIDE_BACK]; + break; + } + + s->i.eof[SIDE_FRONT] = 1; + s->i.eof[SIDE_BACK] = 1; + s->s.eof[SIDE_FRONT] = 1; + s->s.eof[SIDE_BACK] = 1; + ret = SANE_STATUS_GOOD; + } + + DBG (10, "read_from_scanner_duplex: finish\n"); + + return ret; +} + +/* these functions copy image data from input buffer to scanner struct + * descrambling it, and putting it in the right side buffer */ +/* NOTE: they assume buffer is scanline aligned */ +static SANE_Status +copy_simplex(struct scanner *s, unsigned char * buf, int len, int side) +{ + SANE_Status ret=SANE_STATUS_GOOD; + int i, j; + int bwidth = s->s.Bpl; + int pwidth = s->s.width; + int t = bwidth/3; + int f = bwidth/4; + int tw = bwidth/12; + + unsigned char * line = NULL; + int line_next = 0; + + /* jpeg data should not pass thru this function, so copy and bail out */ + if(s->s.format > SANE_FRAME_RGB){ + DBG (15, "copy_simplex: jpeg bulk copy\n"); + memcpy(s->buffers[side]+s->i.bytes_sent[side], buf, len); + s->i.bytes_sent[side] += len; + s->s.bytes_sent[side] += len; + return ret; + } + + DBG (15, "copy_simplex: per-line copy\n"); + + line = malloc(bwidth); + if(!line) return SANE_STATUS_NO_MEM; + + /* ingest each line */ + for(i=0; i<len; i+=bwidth){ + + int lineNum = s->s.bytes_sent[side] / bwidth; + + /*increment number of bytes rx'd from scanner*/ + s->s.bytes_sent[side] += bwidth; + + /*have some padding from scanner to drop*/ + if ( lineNum < s->i.skip_lines[side] + || lineNum - s->i.skip_lines[side] >= s->i.height + ){ + continue; + } + + line_next = 0; + + if(s->s.format == SANE_FRAME_GRAY){ + + switch (s->gray_interlace[side]) { + + /* one line has the following format: ggg...GGG + * where the 'capital' letters are the beginning of the line */ + case GRAY_INTERLACE_gG: + DBG (17, "copy_simplex: gray, gG\n"); + for (j=bwidth-1; j>=0; j--){ + line[line_next++] = buf[i+j]; + } + break; + + case GRAY_INTERLACE_2510: + DBG (17, "copy_simplex: gray, 2510\n"); + + /* first read head (third byte of every three) */ + for(j=bwidth-1;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + } + /* second read head (first byte of every three) */ + for(j=bwidth*3/4-3;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + } + /* third read head (second byte of every three) */ + for(j=bwidth-2;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + } + /* padding */ + for(j=0;j<tw;j++){ + line[line_next++] = 0; + } + break; + } + } + + else if (s->s.format == SANE_FRAME_RGB){ + + switch (s->color_interlace[side]) { + + /* scanner returns color data as bgrbgr... */ + case COLOR_INTERLACE_BGR: + DBG (17, "copy_simplex: color, BGR\n"); + for (j=0; j<pwidth; j++){ + line[line_next++] = buf[i+j*3+2]; + line[line_next++] = buf[i+j*3+1]; + line[line_next++] = buf[i+j*3]; + } + break; + + /* one line has the following format: RRR...rrrGGG...gggBBB...bbb */ + case COLOR_INTERLACE_RRGGBB: + DBG (17, "copy_simplex: color, RRGGBB\n"); + for (j=0; j<pwidth; j++){ + line[line_next++] = buf[i+j]; + line[line_next++] = buf[i+pwidth+j]; + line[line_next++] = buf[i+2*pwidth+j]; + } + break; + + /* one line has the following format: rrr...RRRggg...GGGbbb...BBB + * where the 'capital' letters are the beginning of the line */ + case COLOR_INTERLACE_rRgGbB: + DBG (17, "copy_simplex: color, rRgGbB\n"); + for (j=pwidth-1; j>=0; j--){ + line[line_next++] = buf[i+j]; + line[line_next++] = buf[i+pwidth+j]; + line[line_next++] = buf[i+2*pwidth+j]; + } + break; + + case COLOR_INTERLACE_2510: + DBG (17, "copy_simplex: color, 2510\n"); + + /* first read head (third byte of every three) */ + for(j=t-1;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + line[line_next++] = buf[i+t+j]; + line[line_next++] = buf[i+2*t+j]; + } + /* second read head (first byte of every three) */ + for(j=f-3;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + line[line_next++] = buf[i+t+j]; + line[line_next++] = buf[i+2*t+j]; + } + /* third read head (second byte of every three) */ + for(j=t-2;j>=0;j-=3){ + line[line_next++] = buf[i+j]; + line[line_next++] = buf[i+t+j]; + line[line_next++] = buf[i+2*t+j]; + } + /* padding */ + for(j=0;j<tw;j++){ + line[line_next++] = 0; + } + break; + } + } + + /* nothing sent above? just copy one line of the block */ + /* used by uninterlaced gray/color */ + if(!line_next){ + DBG (17, "copy_simplex: default\n"); + memcpy(line+line_next,buf+i,bwidth); + line_next = bwidth; + } + + /* invert image if scanner needs it for this mode */ + if(s->reverse_by_mode[s->s.mode]){ + for(j=0; j<line_next; j++){ + line[j] ^= 0xff; + } + } + + /* apply calibration if we have it */ + if(s->f_offset[side]){ + DBG (17, "copy_simplex: apply offset\n"); + for(j=0; j<s->s.valid_Bpl; j++){ + int curr = line[j] - s->f_offset[side][j]; + if(curr < 0) curr = 0; + line[j] = curr; + } + } + + if(s->f_gain[side]){ + DBG (17, "copy_simplex: apply gain\n"); + for(j=0; j<s->s.valid_Bpl; j++){ + int curr = line[j] * 240/s->f_gain[side][j]; + if(curr > 255) curr = 255; + line[j] = curr; + } + } + + /* apply brightness and contrast if hardware cannot do it */ + if(s->sw_lut && (s->s.mode == MODE_COLOR || s->s.mode == MODE_GRAYSCALE)){ + DBG (17, "copy_simplex: apply brightness/contrast\n"); + for(j=0; j<s->s.valid_Bpl; j++){ + line[j] = s->lut[line[j]]; + } + } + + /*copy the line into the buffer*/ + ret = copy_line(s,line,side); + if(ret){ + break; + } + } + + free(line); + + DBG (10, "copy_simplex: finished\n"); + + return ret; +} + +/* split the data between two buffers, hand them to copy_simplex() + * assumes that the buffer aligns to a double-wide line boundary */ +static SANE_Status +copy_duplex(struct scanner *s, unsigned char * buf, int len) +{ + SANE_Status ret=SANE_STATUS_GOOD; + int i,j; + int bwidth = s->s.Bpl; + int dbwidth = 2*bwidth; + unsigned char * front; + unsigned char * back; + int flen=0, blen=0; + + DBG (10, "copy_duplex: start\n"); + + /*split the input into two simplex output buffers*/ + front = calloc(1,len/2); + if(!front){ + DBG (5, "copy_duplex: no front mem\n"); + return SANE_STATUS_NO_MEM; + } + back = calloc(1,len/2); + if(!back){ + DBG (5, "copy_duplex: no back mem\n"); + free(front); + return SANE_STATUS_NO_MEM; + } + + if(s->duplex_interlace == DUPLEX_INTERLACE_2510){ + + DBG (10, "copy_duplex: 2510\n"); + + for(i=0; i<len; i+=dbwidth){ + + for(j=0;j<dbwidth;j+=6){ + + /* we are actually only partially descrambling, + * copy_simplex() does the rest */ + + /* front */ + /* 2nd head: 2nd byte -> 1st byte */ + /* 3rd head: 4th byte -> 2nd byte */ + /* 1st head: 5th byte -> 3rd byte */ + front[flen++] = buf[i+j+2]; + front[flen++] = buf[i+j+4]; + front[flen++] = buf[i+j+5]; + + /* back */ + /* 2nd head: 3rd byte -> 1st byte */ + /* 3rd head: 0th byte -> 2nd byte */ + /* 1st head: 1st byte -> 3rd byte */ + back[blen++] = buf[i+j+3]; + back[blen++] = buf[i+j]; + back[blen++] = buf[i+j+1]; + } + } + } + + /* no scanners use this? */ + else if(s->duplex_interlace == DUPLEX_INTERLACE_FFBB){ + for(i=0; i<len; i+=dbwidth){ + memcpy(front+flen,buf+i,bwidth); + flen+=bwidth; + memcpy(back+blen,buf+i+bwidth,bwidth); + blen+=bwidth; + } + } + + /*just alternating bytes, FBFBFB*/ + else { + for(i=0; i<len; i+=2){ + front[flen++] = buf[i]; + back[blen++] = buf[i+1]; + } + } + + copy_simplex(s,front,flen,SIDE_FRONT); + copy_simplex(s,back,blen,SIDE_BACK); + + free(front); + free(back); + + DBG (10, "copy_duplex: finished\n"); + + return ret; +} + +/* downsample a single line from scanner's size to user's size */ +/* and copy into final buffer */ +static SANE_Status +copy_line(struct scanner *s, unsigned char * buff, int side) +{ + SANE_Status ret=SANE_STATUS_GOOD; + int spwidth = s->s.width; + int sbwidth = s->s.Bpl; + int ibwidth = s->i.Bpl; + unsigned char * line; + int offset = 0; + int i, j; + + DBG (20, "copy_line: start\n"); + + /* the 'standard' case: non-stupid scan */ + if(s->s.width == s->i.width + && s->s.dpi_x == s->i.dpi_x + && s->s.mode == s->i.mode + ){ + + memcpy(s->buffers[side]+s->i.bytes_sent[side], buff, sbwidth); + s->i.bytes_sent[side] += sbwidth; + + DBG (20, "copy_line: finished smart\n"); + return ret; + } + + /* the 'corner' case: stupid scan */ + + /*setup 24 bit color single line buffer*/ + line = malloc(spwidth*3); + if(!line) return SANE_STATUS_NO_MEM; + + /*load single line color buffer*/ + switch (s->s.mode) { + + case MODE_COLOR: + memcpy(line, buff, sbwidth); + break; + + case MODE_GRAYSCALE: + for(i=0;i<spwidth;i++){ + line[i*3] = line[i*3+1] = line[i*3+2] = buff[i]; + } + break; + + default: + for(i=0;i<sbwidth;i++){ + unsigned char curr = buff[i]; + + line[i*24+0] = line[i*24+1] = line[i*24+2] = ((curr >> 7) & 1) ?0:255; + line[i*24+3] = line[i*24+4] = line[i*24+5] = ((curr >> 6) & 1) ?0:255; + line[i*24+6] = line[i*24+7] = line[i*24+8] = ((curr >> 5) & 1) ?0:255; + line[i*24+9] = line[i*24+10] = line[i*24+11] = ((curr >> 4) & 1) ?0:255; + line[i*24+12] = line[i*24+13] = line[i*24+14] =((curr >> 3) & 1) ?0:255; + line[i*24+15] = line[i*24+16] = line[i*24+17] =((curr >> 2) & 1) ?0:255; + line[i*24+18] = line[i*24+19] = line[i*24+20] =((curr >> 1) & 1) ?0:255; + line[i*24+21] = line[i*24+22] = line[i*24+23] =((curr >> 0) & 1) ?0:255; + } + break; + } + + /* scan is higher res than user wanted, scale it */ + /*FIXME: interpolate instead */ + if(s->i.dpi_x != s->s.dpi_x){ + for(i=0;i<spwidth;i++){ + int source = i * s->s.dpi_x/s->i.dpi_x * 3; + + if(source+2 >= spwidth*3) + break; + + line[i*3] = line[source]; + line[i*3+1] = line[source+1]; + line[i*3+2] = line[source+2]; + } + } + + /* scan is wider than user wanted, skip some pixels on left side */ + if(s->i.width != s->s.width){ + offset = ((s->valid_x-s->i.page_x) / 2 + s->i.tl_x) * s->i.dpi_x/1200*3; + } + + /* change mode, store line in buffer */ + switch (s->i.mode) { + + case MODE_COLOR: + memcpy(s->buffers[side]+s->i.bytes_sent[side], line+offset, ibwidth); + s->i.bytes_sent[side] += ibwidth; + break; + + case MODE_GRAYSCALE: + for(i=0;i<ibwidth;i++){ + int source = (offset+i)*3; + s->buffers[side][s->i.bytes_sent[side]++] + = ((int)line[source] + line[source+1] + line[source+2])/3; + } + break; + + default: + /*loop over output bytes*/ + for(i=0;i<ibwidth;i++){ + + unsigned char curr = 0; + int thresh = s->threshold*3; + + /*loop over output bits*/ + for(j=0;j<8;j++){ + int source = (offset+i)*24 + j*3; + if( (line[source] + line[source+1] + line[source+2]) < thresh ){ + curr |= 1 << (7-j); + } + } + + s->buffers[side][s->i.bytes_sent[side]++] = curr; + } + break; + } + + free(line); + + DBG (20, "copy_line: finish stupid\n"); + + return ret; +} + +static SANE_Status +read_from_buffer(struct scanner *s, SANE_Byte * buf, SANE_Int max_len, + SANE_Int * len, int side) +{ + SANE_Status ret=SANE_STATUS_GOOD; + int bytes = max_len; + int remain = s->i.bytes_sent[side] - s->u.bytes_sent[side]; + + DBG (10, "read_from_buffer: start\n"); + + /* figure out the max amount to transfer */ + if(bytes > remain) + bytes = remain; + + *len = bytes; + + /*FIXME this needs to timeout eventually */ + if(!bytes){ + DBG(5,"read_from_buffer: nothing to do\n"); + return SANE_STATUS_GOOD; + } + + DBG(15, "read_from_buffer: si:%d to:%d tx:%d bu:%d pa:%d\n", side, + s->i.bytes_tot[side], s->u.bytes_sent[side], max_len, bytes); + + /* copy to caller */ + memcpy(buf,s->buffers[side]+s->u.bytes_sent[side],bytes); + s->u.bytes_sent[side] += bytes; + + DBG (10, "read_from_buffer: finished\n"); + + return ret; +} + +/* + * @@ Section 5 - calibration functions + */ + +#if 0 +static SANE_Status +foo_AFE(struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[] = { + 0x3b, 0x00, 0x10, 0x08, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00 + }; + size_t cmdLen = 12; + + unsigned char in[4]; + size_t inLen = 4; + + DBG (10, "foo_AFE: start\n"); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + if (ret != SANE_STATUS_GOOD) + return ret; + + DBG (10, "foo_AFE: finish\n"); + + return ret; +} +#endif + +/* + * makes several scans, adjusts coarse calibration + */ +static SANE_Status +calibrate_AFE (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int i, j, k; + int min, max; + int lines = 8; + + /*buffer these for later*/ + int old_tl_y = s->u.tl_y; + int old_br_y = s->u.br_y; + int old_mode = s->u.mode; + int old_source = s->u.source; + + DBG (10, "calibrate_AFE: start\n"); + + if(!s->need_ccal){ + DBG (10, "calibrate_AFE: not required\n"); + return ret; + } + + /* always cal with a short scan in duplex color */ + s->u.tl_y = 0; + s->u.br_y = lines * 1200 / s->u.dpi_y; + s->u.mode = MODE_COLOR; + s->u.source = SOURCE_ADF_DUPLEX; + + /* load our own private copy of scan params */ + ret = update_params(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot update_params\n"); + goto cleanup; + } + + if(s->c_res == s->s.dpi_x && s->c_mode == s->s.mode){ + DBG (10, "calibrate_AFE: already done\n"); + goto cleanup; + } + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot clean_params\n"); + goto cleanup; + } + + /* make buffers to hold the images */ + ret = image_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot load buffers\n"); + goto cleanup; + } + + /*blast the existing fine cal data so reading code wont apply it*/ + ret = offset_buffers(s,0); + ret = gain_buffers(s,0); + + /* need to tell it we want duplex */ + ret = ssm_buffer(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot ssm buffer\n"); + goto cleanup; + } + + /* set window command */ + ret = set_window(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot set window\n"); + goto cleanup; + } + + /* first pass (black offset), lamp off, no offset/gain/exposure */ + DBG (15, "calibrate_AFE: offset\n"); + + /* blast all the existing coarse cal data */ + for(i=0;i<2;i++){ + s->c_gain[i] = 1; + s->c_offset[i] = 1; + for(j=0;j<3;j++){ + s->c_exposure[i][j] = 0; + } + } + + ret = write_AFE(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot write afe\n"); + goto cleanup; + } + + ret = calibration_scan(s,0xff); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot make offset cal scan\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ + min = 255; + for(j=0; j<s->s.valid_Bpl; j++){ + if(s->buffers[i][j] < min) + min = s->buffers[i][j]; + } + s->c_offset[i] = min*3-2; + DBG (15, "calibrate_AFE: offset %d %d %02x\n", i, min, s->c_offset[i]); + } + + /*handle second pass (per channel exposure), lamp on, overexposed*/ + DBG (15, "calibrate_AFE: exposure\n"); + for(i=0;i<2;i++){ + for(j=0; j<3; j++){ + s->c_exposure[i][j] = 0x320; + } + } + + ret = write_AFE(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot write afe\n"); + goto cleanup; + } + + ret = calibration_scan(s,0xfe); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot make exposure cal scan\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ /*sides*/ + for(j=0;j<3;j++){ /*channels*/ + max = 0; + for(k=j; k<s->s.valid_Bpl; k+=3){ /*bytes*/ + if(s->buffers[i][k] > max) + max = s->buffers[i][k]; + } + + /*generally we reduce the exposure (smaller number) */ + if(old_mode == MODE_COLOR) + s->c_exposure[i][j] = s->c_exposure[i][j] * 102/max; + else + s->c_exposure[i][j] = s->c_exposure[i][j] * 64/max; + + DBG (15, "calibrate_AFE: exp %d %d %d %02x\n", i, j, max, + s->c_exposure[i][j]); + } + } + + /*handle third pass (gain), lamp on with current offset/exposure */ + DBG (15, "calibrate_AFE: gain\n"); + + ret = write_AFE(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot write afe\n"); + goto cleanup; + } + + ret = calibration_scan(s,0xfe); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot make gain cal scan\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ + max = 0; + for(j=0; j<s->s.valid_Bpl; j++){ + if(s->buffers[i][j] > max) + max = s->buffers[i][j]; + } + + if(old_mode == MODE_COLOR) + s->c_gain[i] = (250-max)*4/5; + else + s->c_gain[i] = (125-max)*4/5; + + if(s->c_gain[i] < 1) + s->c_gain[i] = 1; + + DBG (15, "calibrate_AFE: gain %d %d %02x\n", i, max, s->c_gain[i]); + } + + /*handle fourth pass (offset again), lamp off*/ +#if 0 + DBG (15, "calibrate_AFE: offset2\n"); + + ret = write_AFE(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot write afe\n"); + goto cleanup; + } + + ret = calibration_scan(s,0xff); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot make offset2 cal scan\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ + min = 255; + for(j=0; j<s->s.valid_Bpl; j++){ + if(s->buffers[i][j] < min) + min = s->buffers[i][j]; + } + /*s->c_offset[i] += min*3-2;*/ + DBG (15, "calibrate_AFE: offset2 %d %d %02x\n", i, min, s->c_offset[i]); + } +#endif + + /*send final afe params to scanner*/ + ret = write_AFE(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_AFE: ERROR: cannot write afe\n"); + goto cleanup; + } + + /* log current cal type */ + s->c_res = s->s.dpi_x; + s->c_mode = s->s.mode; + + cleanup: + + /* recover user settings */ + s->u.tl_y = old_tl_y; + s->u.br_y = old_br_y; + s->u.mode = old_mode; + s->u.source = old_source; + + DBG (10, "calibrate_AFE: finish %d\n",ret); + + return ret; +} + + +/* alternative version- extracts data from scanner memory */ +static SANE_Status +calibrate_fine_buffer (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int i, j, k; + + unsigned char cmd[READ_len]; + size_t cmdLen = READ_len; + + unsigned char * in = NULL; + size_t inLen = 0, reqLen = 0; + + /*buffer these for later*/ + int old_tl_y = s->u.tl_y; + int old_br_y = s->u.br_y; + int old_source = s->u.source; + + DBG (10, "calibrate_fine_buffer: start\n"); + + if(!s->need_fcal_buffer){ + DBG (10, "calibrate_fine_buffer: not required\n"); + return ret; + } + + /* pretend we are doing a 1 line scan in duplex */ + s->u.tl_y = 0; + s->u.br_y = 1200 / s->u.dpi_y; + s->u.source = SOURCE_ADF_DUPLEX; + + /* load our own private copy of scan params */ + ret = update_params(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine_buffer: ERROR: cannot update_params\n"); + goto cleanup; + } + + if(s->f_res == s->s.dpi_x && s->f_mode == s->s.mode){ + DBG (10, "calibrate_fine_buffer: already done\n"); + goto cleanup; + } + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine_buffer: ERROR: cannot clean_params\n"); + goto cleanup; + } + + /*calibration buffers in scanner are single color channel, but duplex*/ + reqLen = s->s.width*2; + + in = malloc(reqLen); + if (!in) { + DBG (5, "calibrate_fine_buffer: ERROR: cannot malloc in\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + /*fine offset*/ + ret = offset_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine_buffer: ERROR: cannot load offset buffers\n"); + goto cleanup; + } + + DBG (5, "calibrate_fine_buffer: %d %x\n", s->s.dpi_x/10, s->s.dpi_x/10); + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, READ_code); + set_R_datatype_code (cmd, SR_datatype_fineoffset); + set_R_xfer_lid (cmd, s->s.dpi_x/10); + set_R_xfer_length (cmd, reqLen); + + inLen = reqLen; + + hexdump(15, "cmd:", cmd, cmdLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + if (ret != SANE_STATUS_GOOD) + goto cleanup; + + for(i=0;i<2;i++){ + + /*color mode, expand offset across all three channels? */ + if(s->s.format == SANE_FRAME_RGB){ + for(j=0; j<s->s.valid_width; j++){ + + /*red*/ + s->f_offset[i][j*3] = in[j*2+i]; + if(s->f_offset[i][j*3] < 1) + s->f_offset[i][j*3] = 1; + + /*green and blue, same as red*/ + s->f_offset[i][j*3+1] = s->f_offset[i][j*3+2] = s->f_offset[i][j*3]; + } + } + + /*gray mode, copy*/ + else{ + for(j=0; j<s->s.valid_width; j++){ + + s->f_offset[i][j] = in[j*2+i]; + if(s->f_offset[i][j] < 1) + s->f_offset[i][j] = 1; + } + } + + hexdump(15, "off:", s->f_offset[i], s->s.valid_Bpl); + } + + /*fine gain*/ + ret = gain_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine_buffer: ERROR: cannot load gain buffers\n"); + goto cleanup; + } + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, READ_code); + set_R_datatype_code (cmd, SR_datatype_finegain); + set_R_xfer_lid (cmd, s->s.dpi_x/10); + set_R_xfer_length (cmd, reqLen); + + /*color gain split into three buffers, grab them and merge*/ + if(s->s.format == SANE_FRAME_RGB){ + + int codes[] = {R_FINE_uid_red,R_FINE_uid_green,R_FINE_uid_blue}; + + for(k=0;k<3;k++){ + + set_R_xfer_uid (cmd, codes[k]); + inLen = reqLen; + + hexdump(15, "cmd:", cmd, cmdLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + if (ret != SANE_STATUS_GOOD) + goto cleanup; + + for(i=0;i<2;i++){ + for(j=0; j<s->s.valid_width; j++){ + + s->f_gain[i][j*3+k] = in[j*2+i]*3/4; + + if(s->f_gain[i][j*3+k] < 1) + s->f_gain[i][j*3+k] = 1; + } + } + } + } + + /*gray gain, copy*/ + else{ + + set_R_xfer_uid (cmd, R_FINE_uid_gray); + inLen = reqLen; + + hexdump(15, "cmd:", cmd, cmdLen); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + in, &inLen + ); + if (ret != SANE_STATUS_GOOD) + goto cleanup; + + for(i=0;i<2;i++){ + for(j=0; j<s->s.valid_width; j++){ + + s->f_gain[i][j] = in[j*2+i]*3/4; + + if(s->f_gain[i][j] < 1) + s->f_gain[i][j] = 1; + } + } + } + + for(i=0;i<2;i++){ + hexdump(15, "gain:", s->f_gain[i], s->s.valid_Bpl); + } + + /* log current cal type */ + s->f_res = s->s.dpi_x; + s->f_mode = s->s.mode; + + cleanup: + + if(in){ + free(in); + } + + /* recover user settings */ + s->u.tl_y = old_tl_y; + s->u.br_y = old_br_y; + s->u.source = old_source; + + DBG (10, "calibrate_fine_buffer: finish %d\n",ret); + + return ret; +} + +/* + * makes several scans, adjusts fine calibration + */ +static SANE_Status +calibrate_fine (struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int i, j, k; + int min, max; + int lines = 8; + + /*buffer these for later*/ + int old_tl_y = s->u.tl_y; + int old_br_y = s->u.br_y; + int old_source = s->u.source; + + DBG (10, "calibrate_fine: start\n"); + + if(!s->need_fcal){ + DBG (10, "calibrate_fine: not required\n"); + return ret; + } + + /* always cal with a short scan in duplex */ + s->u.tl_y = 0; + s->u.br_y = lines * 1200 / s->u.dpi_y; + s->u.source = SOURCE_ADF_DUPLEX; + + /* load our own private copy of scan params */ + ret = update_params(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot update_params\n"); + goto cleanup; + } + + if(s->f_res == s->s.dpi_x && s->f_mode == s->s.mode){ + DBG (10, "calibrate_fine: already done\n"); + goto cleanup; + } + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibration_fine: ERROR: cannot clean_params\n"); + goto cleanup; + } + + /* make buffers to hold the images */ + ret = image_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot load buffers\n"); + goto cleanup; + } + + /*blast the existing fine cal data so reading code wont apply it*/ + ret = offset_buffers(s,0); + ret = gain_buffers(s,0); + + /* need to tell it we want duplex */ + ret = ssm_buffer(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot ssm buffer\n"); + goto cleanup; + } + + /* set window command */ + ret = set_window(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot set window\n"); + goto cleanup; + } + + /*handle fifth pass (fine offset), lamp off*/ + DBG (15, "calibrate_fine: offset\n"); + ret = calibration_scan(s,0xff); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot make offset cal scan\n"); + goto cleanup; + } + + ret = offset_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot load offset buffers\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ + for(j=0; j<s->s.valid_Bpl; j++){ + min = 0; + for(k=j;k<lines*s->s.Bpl;k+=s->s.Bpl){ + min += s->buffers[i][k]; + } + s->f_offset[i][j] = min/lines; + } + hexdump(15, "off:", s->f_offset[i], s->s.valid_Bpl); + } + + /*handle sixth pass (fine gain), lamp on*/ + DBG (15, "calibrate_fine: gain\n"); + ret = calibration_scan(s,0xfe); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot make gain cal scan\n"); + goto cleanup; + } + + ret = gain_buffers(s,1); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibrate_fine: ERROR: cannot load gain buffers\n"); + goto cleanup; + } + + for(i=0;i<2;i++){ + for(j=0; j<s->s.valid_Bpl; j++){ + max = 0; + for(k=j;k<lines*s->s.Bpl;k+=s->s.Bpl){ + max += s->buffers[i][k]; + } + s->f_gain[i][j] = max/lines; + + if(s->f_gain[i][j] < 1) + s->f_gain[i][j] = 1; + } + hexdump(15, "gain:", s->f_gain[i], s->s.valid_Bpl); + } + + /* log current cal type */ + s->f_res = s->s.dpi_x; + s->f_mode = s->s.mode; + + cleanup: + + /* recover user settings */ + s->u.tl_y = old_tl_y; + s->u.br_y = old_br_y; + s->u.source = old_source; + + DBG (10, "calibrate_fine: finish %d\n",ret); + + return ret; +} + +/* + * sends AFE params, and ingests entire duplex image into buffers + */ +static SANE_Status +calibration_scan (struct scanner *s, int scan) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + DBG (10, "calibration_scan: start\n"); + + /* clean scan params for new scan */ + ret = clean_params(s); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibration_scan: ERROR: cannot clean_params\n"); + return ret; + } + + /* start scanning */ + ret = start_scan (s,scan); + if (ret != SANE_STATUS_GOOD) { + DBG (5, "calibration_scan: ERROR: cannot start_scan\n"); + return ret; + } + + while(!s->s.eof[SIDE_FRONT] && !s->s.eof[SIDE_BACK]){ + ret = read_from_scanner_duplex(s,1); + } + + DBG (10, "calibration_scan: finished\n"); + + return ret; +} + +/* + * sends AFE and exposure params + */ +static SANE_Status +write_AFE(struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[COR_CAL_len]; + size_t cmdLen = COR_CAL_len; + + /*use the longest payload for buffer*/ + unsigned char pay[CC3_pay_len]; + size_t payLen = CC3_pay_len; + + DBG (10, "write_AFE: start\n"); + + /* newer scanners use a longer cc payload */ + if(s->ccal_version == 3){ + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, COR_CAL_code); + set_CC_version(cmd,CC3_pay_ver); + set_CC_xferlen(cmd,payLen); + + memset(pay,0,payLen); + + set_CC3_gain_f_r(pay,s->c_gain[SIDE_FRONT]); + set_CC3_gain_f_g(pay,s->c_gain[SIDE_FRONT]); + set_CC3_gain_f_b(pay,s->c_gain[SIDE_FRONT]); + + set_CC3_off_f_r(pay,s->c_offset[SIDE_FRONT]); + set_CC3_off_f_g(pay,s->c_offset[SIDE_FRONT]); + set_CC3_off_f_b(pay,s->c_offset[SIDE_FRONT]); + + set_CC3_exp_f_r(pay,s->c_exposure[SIDE_FRONT][CHAN_RED]); + set_CC3_exp_f_g(pay,s->c_exposure[SIDE_FRONT][CHAN_GREEN]); + set_CC3_exp_f_b(pay,s->c_exposure[SIDE_FRONT][CHAN_BLUE]); + + set_CC3_gain_b_r(pay,s->c_gain[SIDE_BACK]); + set_CC3_gain_b_g(pay,s->c_gain[SIDE_BACK]); + set_CC3_gain_b_b(pay,s->c_gain[SIDE_BACK]); + + set_CC3_off_b_r(pay,s->c_offset[SIDE_BACK]); + set_CC3_off_b_g(pay,s->c_offset[SIDE_BACK]); + set_CC3_off_b_b(pay,s->c_offset[SIDE_BACK]); + + set_CC3_exp_b_r(pay,s->c_exposure[SIDE_BACK][CHAN_RED]); + set_CC3_exp_b_g(pay,s->c_exposure[SIDE_BACK][CHAN_GREEN]); + set_CC3_exp_b_b(pay,s->c_exposure[SIDE_BACK][CHAN_BLUE]); + } + + else{ + payLen = CC_pay_len; + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, COR_CAL_code); + set_CC_version(cmd,CC_pay_ver); + set_CC_xferlen(cmd,payLen); + + memset(pay,0,payLen); + set_CC_f_gain(pay,s->c_gain[SIDE_FRONT]); + set_CC_unk1(pay,1); + set_CC_f_offset(pay,s->c_offset[SIDE_FRONT]); + set_CC_unk2(pay,1); + set_CC_exp_f_r1(pay,s->c_exposure[SIDE_FRONT][CHAN_RED]); + set_CC_exp_f_g1(pay,s->c_exposure[SIDE_FRONT][CHAN_GREEN]); + set_CC_exp_f_b1(pay,s->c_exposure[SIDE_FRONT][CHAN_BLUE]); + set_CC_exp_f_r2(pay,s->c_exposure[SIDE_FRONT][CHAN_RED]); + set_CC_exp_f_g2(pay,s->c_exposure[SIDE_FRONT][CHAN_GREEN]); + set_CC_exp_f_b2(pay,s->c_exposure[SIDE_FRONT][CHAN_BLUE]); + + set_CC_b_gain(pay,s->c_gain[SIDE_BACK]); + set_CC_b_offset(pay,s->c_offset[SIDE_BACK]); + set_CC_exp_b_r1(pay,s->c_exposure[SIDE_BACK][CHAN_RED]); + set_CC_exp_b_g1(pay,s->c_exposure[SIDE_BACK][CHAN_GREEN]); + set_CC_exp_b_b1(pay,s->c_exposure[SIDE_BACK][CHAN_BLUE]); + set_CC_exp_b_r2(pay,s->c_exposure[SIDE_BACK][CHAN_RED]); + set_CC_exp_b_g2(pay,s->c_exposure[SIDE_BACK][CHAN_GREEN]); + set_CC_exp_b_b2(pay,s->c_exposure[SIDE_BACK][CHAN_BLUE]); + } + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + pay, payLen, + NULL, NULL + ); + if (ret != SANE_STATUS_GOOD) + return ret; + + DBG (10, "write_AFE: finish\n"); + + return ret; +} + +/* + * frees/callocs buffers to hold the fine cal offset data + */ +static SANE_Status +offset_buffers (struct scanner *s, int setup) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int side; + + DBG (10, "offset_buffers: start\n"); + + for(side=0;side<2;side++){ + + if (s->f_offset[side]) { + DBG (15, "offset_buffers: free f_offset %d.\n",side); + free(s->f_offset[side]); + s->f_offset[side] = NULL; + } + + if(setup){ + s->f_offset[side] = calloc (1,s->s.Bpl); + if (!s->f_offset[side]) { + DBG (5, "offset_buffers: error, no f_offset %d.\n",side); + return SANE_STATUS_NO_MEM; + } + } + } + + DBG (10, "offset_buffers: finish\n"); + + return ret; +} + +/* + * frees/callocs buffers to hold the fine cal gain data + */ +static SANE_Status +gain_buffers (struct scanner *s, int setup) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int side; + + DBG (10, "gain_buffers: start\n"); + + for(side=0;side<2;side++){ + + if (s->f_gain[side]) { + DBG (15, "gain_buffers: free f_gain %d.\n",side); + free(s->f_gain[side]); + s->f_gain[side] = NULL; + } + + if(setup){ + s->f_gain[side] = calloc (1,s->s.Bpl); + if (!s->f_gain[side]) { + DBG (5, "gain_buffers: error, no f_gain %d.\n",side); + return SANE_STATUS_NO_MEM; + } + } + } + + DBG (10, "gain_buffers: finish\n"); + + return ret; +} + +/* + * @@ Section 6 - SANE cleanup functions + */ +/* + * Cancels a scan. + * + * It has been said on the mailing list that sane_cancel is a bit of a + * misnomer because it is routinely called to signal the end of a + * batch - quoting David Mosberger-Tang: + * + * > In other words, the idea is to have sane_start() be called, and + * > collect as many images as the frontend wants (which could in turn + * > consist of multiple frames each as indicated by frame-type) and + * > when the frontend is done, it should call sane_cancel(). + * > Sometimes it's better to think of sane_cancel() as "sane_stop()" + * > but that name would have had some misleading connotations as + * > well, that's why we stuck with "cancel". + * + * The current consensus regarding duplex and ADF scans seems to be + * the following call sequence: sane_start; sane_read (repeat until + * EOF); sane_start; sane_read... and then call sane_cancel if the + * batch is at an end. I.e. do not call sane_cancel during the run but + * as soon as you get a SANE_STATUS_NO_DOCS. + * + * From the SANE spec: + * This function is used to immediately or as quickly as possible + * cancel the currently pending operation of the device represented by + * handle h. This function can be called at any time (as long as + * handle h is a valid handle) but usually affects long-running + * operations only (such as image acquisition). It is safe to call + * this function asynchronously (e.g., from within a signal handler). + * It is important to note that completion of this operaton does not + * imply that the currently pending operation has been cancelled. It + * only guarantees that cancellation has been initiated. Cancellation + * completes only when the cancelled call returns (typically with a + * status value of SANE_STATUS_CANCELLED). Since the SANE API does + * not require any other operations to be re-entrant, this implies + * that a frontend must not call any other operation until the + * cancelled operation has returned. + */ +void +sane_cancel (SANE_Handle handle) +{ + struct scanner * s = (struct scanner *) handle; + + DBG (10, "sane_cancel: start\n"); + s->cancelled = 1; + + /* if there is no other running function to check, we do it */ + if(!s->reading) + check_for_cancel(s); + + DBG (10, "sane_cancel: finish\n"); +} + +/* checks started and cancelled flags in scanner struct, + * sends cancel command to scanner if required. don't call + * this function asyncronously, wait for pending operation */ +static SANE_Status +check_for_cancel(struct scanner *s) +{ + SANE_Status ret=SANE_STATUS_GOOD; + + DBG (10, "check_for_cancel: start\n"); + + if(s->started && s->cancelled){ + unsigned char cmd[CANCEL_len]; + size_t cmdLen = CANCEL_len; + + DBG (15, "check_for_cancel: cancelling\n"); + + /* cancel scan */ + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd, CANCEL_code); + + ret = do_cmd ( + s, 1, 0, + cmd, cmdLen, + NULL, 0, + NULL, NULL + ); + if(ret){ + DBG (5, "check_for_cancel: ignoring bad cancel: %d\n",ret); + } + + ret = object_position(s,SANE_FALSE); + if(ret){ + DBG (5, "check_for_cancel: ignoring bad eject: %d\n",ret); + } + + s->started = 0; + s->cancelled = 0; + ret = SANE_STATUS_CANCELLED; + } + else if(s->cancelled){ + DBG (15, "check_for_cancel: already cancelled\n"); + s->cancelled = 0; + ret = SANE_STATUS_CANCELLED; + } + + DBG (10, "check_for_cancel: finish %d\n",ret); + return ret; +} + +/* + * Ends use of the scanner. + * + * From the SANE spec: + * This function terminates the association between the device handle + * passed in argument h and the device it represents. If the device is + * presently active, a call to sane_cancel() is performed first. After + * this function returns, handle h must not be used anymore. + */ +void +sane_close (SANE_Handle handle) +{ + struct scanner * s = (struct scanner *) handle; + + DBG (10, "sane_close: start\n"); + disconnect_fd(s); + image_buffers(s,0); + offset_buffers(s,0); + gain_buffers(s,0); + DBG (10, "sane_close: finish\n"); +} + +static SANE_Status +disconnect_fd (struct scanner *s) +{ + DBG (10, "disconnect_fd: start\n"); + + if(s->fd > -1){ + if (s->connection == CONNECTION_USB) { + DBG (15, "disconnecting usb device\n"); + sanei_usb_close (s->fd); + } + else if (s->connection == CONNECTION_SCSI) { + DBG (15, "disconnecting scsi device\n"); + sanei_scsi_close (s->fd); + } + s->fd = -1; + } + + DBG (10, "disconnect_fd: finish\n"); + + return SANE_STATUS_GOOD; +} + +/* + * Terminates the backend. + * + * From the SANE spec: + * This function must be called to terminate use of a backend. The + * function will first close all device handles that still might be + * open (it is recommended to close device handles explicitly through + * a call to sane_close(), but backends are required to release all + * resources upon a call to this function). After this function + * returns, no function other than sane_init() may be called + * (regardless of the status value returned by sane_exit(). Neglecting + * to call this function may result in some resources not being + * released properly. + */ +void +sane_exit (void) +{ + struct scanner *dev, *next; + + DBG (10, "sane_exit: start\n"); + + for (dev = scanner_devList; dev; dev = next) { + disconnect_fd(dev); + next = dev->next; + free (dev); + } + + if (sane_devArray) + free (sane_devArray); + + scanner_devList = NULL; + sane_devArray = NULL; + + DBG (10, "sane_exit: finish\n"); +} + + +/* + * @@ Section 7 - misc helper functions + */ +static void +default_globals(void) +{ + global_buffer_size = global_buffer_size_default; + global_padded_read = global_padded_read_default; + global_duplex_offset = global_duplex_offset_default; + global_vendor_name[0] = 0; + global_model_name[0] = 0; + global_version_name[0] = 0; +} + +/* + * Called by the SANE SCSI core and our usb code on device errors + * parses the request sense return data buffer, + * decides the best SANE_Status for the problem, produces debug msgs, + * and copies the sense buffer into the scanner struct + */ +static SANE_Status +sense_handler (int fd, unsigned char * sensed_data, void *arg) +{ + struct scanner *s = arg; + unsigned int sense = get_RS_sense_key (sensed_data); + unsigned int asc = get_RS_ASC (sensed_data); + unsigned int ascq = get_RS_ASCQ (sensed_data); + unsigned int eom = get_RS_EOM (sensed_data); + unsigned int ili = get_RS_ILI (sensed_data); + unsigned int info = get_RS_information (sensed_data); + + DBG (5, "sense_handler: start\n"); + + /* kill compiler warning */ + fd = fd; + + /* copy the rs return data into the scanner struct + so that the caller can use it if he wants + memcpy(&s->rs_buffer,sensed_data,RS_return_size); + */ + + DBG (5, "Sense=%#02x, ASC=%#02x, ASCQ=%#02x, EOM=%d, ILI=%d, info=%#08x\n", sense, asc, ascq, eom, ili, info); + + switch (sense) { + case 0: + if (ili == 1) { + s->rs_info = info; + DBG (5, "No sense: EOM remainder:%d\n",info); + return SANE_STATUS_EOF; + } + DBG (5, "No sense: unknown asc/ascq\n"); + return SANE_STATUS_GOOD; + + case 1: + if (asc == 0x37 && ascq == 0x00) { + DBG (5, "Recovered error: parameter rounded\n"); + return SANE_STATUS_GOOD; + } + DBG (5, "Recovered error: unknown asc/ascq\n"); + return SANE_STATUS_GOOD; + + case 2: + if (asc == 0x04 && ascq == 0x01) { + DBG (5, "Not ready: previous command unfinished\n"); + return SANE_STATUS_DEVICE_BUSY; + } + DBG (5, "Not ready: unknown asc/ascq\n"); + return SANE_STATUS_DEVICE_BUSY; + + case 3: + if (asc == 0x36 && ascq == 0x00) { + DBG (5, "Medium error: no cartridge\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x3a && ascq == 0x00) { + DBG (5, "Medium error: hopper empty\n"); + return SANE_STATUS_NO_DOCS; + } + if (asc == 0x80 && ascq == 0x00) { + DBG (5, "Medium error: paper jam\n"); + return SANE_STATUS_JAMMED; + } + if (asc == 0x80 && ascq == 0x01) { + DBG (5, "Medium error: cover open\n"); + return SANE_STATUS_COVER_OPEN; + } + if (asc == 0x81 && ascq == 0x01) { + DBG (5, "Medium error: double feed\n"); + return SANE_STATUS_JAMMED; + } + if (asc == 0x81 && ascq == 0x02) { + DBG (5, "Medium error: skew detected\n"); + return SANE_STATUS_JAMMED; + } + if (asc == 0x81 && ascq == 0x04) { + DBG (5, "Medium error: staple detected\n"); + return SANE_STATUS_JAMMED; + } + DBG (5, "Medium error: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 4: + if (asc == 0x60 && ascq == 0x00) { + DBG (5, "Hardware error: lamp error\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x80 && ascq == 0x01) { + DBG (5, "Hardware error: CPU check error\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x80 && ascq == 0x02) { + DBG (5, "Hardware error: RAM check error\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x80 && ascq == 0x03) { + DBG (5, "Hardware error: ROM check error\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x80 && ascq == 0x04) { + DBG (5, "Hardware error: hardware check error\n"); + return SANE_STATUS_IO_ERROR; + } + DBG (5, "Hardware error: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 5: + if (asc == 0x1a && ascq == 0x00) { + DBG (5, "Illegal request: Parameter list error\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x20 && ascq == 0x00) { + DBG (5, "Illegal request: invalid command\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x24 && ascq == 0x00) { + DBG (5, "Illegal request: invalid CDB field\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x25 && ascq == 0x00) { + DBG (5, "Illegal request: unsupported logical unit\n"); + return SANE_STATUS_UNSUPPORTED; + } + if (asc == 0x26 && ascq == 0x00) { + DBG (5, "Illegal request: invalid field in parm list\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x2c && ascq == 0x00) { + DBG (5, "Illegal request: command sequence error\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x2c && ascq == 0x01) { + DBG (5, "Illegal request: too many windows\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x3a && ascq == 0x00) { + DBG (5, "Illegal request: no paper\n"); + return SANE_STATUS_NO_DOCS; + } + if (asc == 0x3d && ascq == 0x00) { + DBG (5, "Illegal request: invalid IDENTIFY\n"); + return SANE_STATUS_INVAL; + } + if (asc == 0x55 && ascq == 0x00) { + DBG (5, "Illegal request: scanner out of memory\n"); + return SANE_STATUS_NO_MEM; + } + DBG (5, "Illegal request: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + break; + + case 6: + if (asc == 0x29 && ascq == 0x00) { + DBG (5, "Unit attention: device reset\n"); + return SANE_STATUS_GOOD; + } + if (asc == 0x2a && ascq == 0x00) { + DBG (5, "Unit attention: param changed by 2nd initiator\n"); + return SANE_STATUS_GOOD; + } + DBG (5, "Unit attention: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + break; + + case 7: + DBG (5, "Data protect: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 8: + DBG (5, "Blank check: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 9: + DBG (5, "Vendor defined: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 0xa: + DBG (5, "Copy aborted: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 0xb: + if (asc == 0x00 && ascq == 0x00) { + DBG (5, "Aborted command: no sense/cancelled\n"); + return SANE_STATUS_CANCELLED; + } + if (asc == 0x45 && ascq == 0x00) { + DBG (5, "Aborted command: reselect failure\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x47 && ascq == 0x00) { + DBG (5, "Aborted command: SCSI parity error\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x48 && ascq == 0x00) { + DBG (5, "Aborted command: initiator error message\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x49 && ascq == 0x00) { + DBG (5, "Aborted command: invalid message\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x80 && ascq == 0x00) { + DBG (5, "Aborted command: timeout\n"); + return SANE_STATUS_IO_ERROR; + } + DBG (5, "Aborted command: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + break; + + case 0xc: + DBG (5, "Equal: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 0xd: + DBG (5, "Volume overflow: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + case 0xe: + if (asc == 0x3b && ascq == 0x0d) { + DBG (5, "Miscompare: too many docs\n"); + return SANE_STATUS_IO_ERROR; + } + if (asc == 0x3b && ascq == 0x0e) { + DBG (5, "Miscompare: too few docs\n"); + return SANE_STATUS_IO_ERROR; + } + DBG (5, "Miscompare: unknown asc/ascq\n"); + return SANE_STATUS_IO_ERROR; + + default: + DBG (5, "Unknown Sense Code\n"); + return SANE_STATUS_IO_ERROR; + } + + DBG (5, "sense_handler: should never happen!\n"); + + return SANE_STATUS_IO_ERROR; +} + +/* + * take a bunch of pointers, send commands to scanner + */ +static SANE_Status +do_cmd(struct scanner *s, int runRS, int shortTime, + unsigned char * cmdBuff, size_t cmdLen, + unsigned char * outBuff, size_t outLen, + unsigned char * inBuff, size_t * inLen +) +{ + if (s->connection == CONNECTION_SCSI) { + return do_scsi_cmd(s, runRS, shortTime, + cmdBuff, cmdLen, + outBuff, outLen, + inBuff, inLen + ); + } + if (s->connection == CONNECTION_USB) { + return do_usb_cmd(s, runRS, shortTime, + cmdBuff, cmdLen, + outBuff, outLen, + inBuff, inLen + ); + } + return SANE_STATUS_INVAL; +} + +static SANE_Status +do_scsi_cmd(struct scanner *s, int runRS, int shortTime, + unsigned char * cmdBuff, size_t cmdLen, + unsigned char * outBuff, size_t outLen, + unsigned char * inBuff, size_t * inLen +) +{ + int ret; + size_t actLen = 0; + + /*shut up compiler*/ + runRS=runRS; + shortTime=shortTime; + + DBG(10, "do_scsi_cmd: start\n"); + + DBG(25, "cmd: writing %d bytes\n", (int)cmdLen); + hexdump(30, "cmd: >>", cmdBuff, cmdLen); + + if(outBuff && outLen){ + DBG(25, "out: writing %d bytes\n", (int)outLen); + hexdump(30, "out: >>", outBuff, outLen); + } + if (inBuff && inLen){ + DBG(25, "in: reading %d bytes\n", (int)*inLen); + memset(inBuff,0,*inLen); + actLen = *inLen; + } + + ret = sanei_scsi_cmd2(s->fd, cmdBuff, cmdLen, outBuff, outLen, inBuff, inLen); + + if(ret != SANE_STATUS_GOOD && ret != SANE_STATUS_EOF){ + DBG(5,"do_scsi_cmd: return '%s'\n",sane_strstatus(ret)); + return ret; + } + + if (inBuff && inLen){ + if(ret == SANE_STATUS_EOF){ + DBG(25, "in: short read, remainder %lu bytes\n", (u_long)s->rs_info); + *inLen -= s->rs_info; + } + hexdump(30, "in: <<", inBuff, *inLen); + DBG(25, "in: read %d bytes\n", (int)*inLen); + } + + DBG(10, "do_scsi_cmd: finish\n"); + + return ret; +} + +static SANE_Status +do_usb_cmd(struct scanner *s, int runRS, int shortTime, + unsigned char * cmdBuff, size_t cmdLen, + unsigned char * outBuff, size_t outLen, + unsigned char * inBuff, size_t * inLen +) +{ + size_t cmdOffset = 0; + size_t cmdLength = 0; + size_t cmdActual = 0; + unsigned char * cmdBuffer = NULL; + int cmdTimeout = 0; + + size_t outOffset = 0; + size_t outLength = 0; + size_t outActual = 0; + unsigned char * outBuffer = NULL; + int outTimeout = 0; + + size_t inOffset = 0; + size_t inLength = 0; + size_t inActual = 0; + unsigned char * inBuffer = NULL; + int inTimeout = 0; + + size_t statOffset = 0; + size_t statLength = 0; + size_t statActual = 0; + unsigned char * statBuffer = NULL; + int statTimeout = 0; + + int ret = 0; + int ret2 = 0; + + DBG (10, "do_usb_cmd: start\n"); + + /****************************************************************/ + /* the command stage */ + { + cmdOffset = USB_HEADER_LEN; + cmdLength = cmdOffset+USB_COMMAND_LEN; + cmdActual = cmdLength; + cmdTimeout = USB_COMMAND_TIME; + + /* change timeout */ + if(shortTime) + cmdTimeout/=60; + sanei_usb_set_timeout(cmdTimeout); + + /* build buffer */ + cmdBuffer = calloc(cmdLength,1); + if(!cmdBuffer){ + DBG(5,"cmd: no mem\n"); + return SANE_STATUS_NO_MEM; + } + + /* build a USB packet around the SCSI command */ + cmdBuffer[3] = cmdLength-4; + cmdBuffer[5] = 1; + cmdBuffer[6] = 0x90; + memcpy(cmdBuffer+cmdOffset,cmdBuff,cmdLen); + + /* write the command out */ + DBG(25, "cmd: writing %d bytes, timeout %d\n", (int)cmdLength, cmdTimeout); + hexdump(30, "cmd: >>", cmdBuffer, cmdLength); + ret = sanei_usb_write_bulk(s->fd, cmdBuffer, &cmdActual); + DBG(25, "cmd: wrote %d bytes, retVal %d\n", (int)cmdActual, ret); + + if(cmdLength != cmdActual){ + DBG(5,"cmd: wrong size %d/%d\n", (int)cmdLength, (int)cmdActual); + free(cmdBuffer); + return SANE_STATUS_IO_ERROR; + } + if(ret != SANE_STATUS_GOOD){ + DBG(5,"cmd: write error '%s'\n",sane_strstatus(ret)); + free(cmdBuffer); + return ret; + } + free(cmdBuffer); + } + + /****************************************************************/ + /* the output stage */ + if(outBuff && outLen){ + + outOffset = USB_HEADER_LEN; + outLength = outOffset+outLen; + outActual = outLength; + outTimeout = USB_DATA_TIME; + + /* change timeout */ + if(shortTime) + outTimeout/=60; + sanei_usb_set_timeout(outTimeout); + + /* build outBuffer */ + outBuffer = calloc(outLength,1); + if(!outBuffer){ + DBG(5,"out: no mem\n"); + return SANE_STATUS_NO_MEM; + } + + /* build a USB packet around the SCSI command */ + outBuffer[3] = outLength-4; + outBuffer[5] = 2; + outBuffer[6] = 0xb0; + memcpy(outBuffer+outOffset,outBuff,outLen); + + /* write the command out */ + DBG(25, "out: writing %d bytes, timeout %d\n", (int)outLength, outTimeout); + hexdump(30, "out: >>", outBuffer, outLength); + ret = sanei_usb_write_bulk(s->fd, outBuffer, &outActual); + DBG(25, "out: wrote %d bytes, retVal %d\n", (int)outActual, ret); + + if(outLength != outActual){ + DBG(5,"out: wrong size %d/%d\n", (int)outLength, (int)outActual); + free(outBuffer); + return SANE_STATUS_IO_ERROR; + } + if(ret != SANE_STATUS_GOOD){ + DBG(5,"out: write error '%s'\n",sane_strstatus(ret)); + free(outBuffer); + return ret; + } + free(outBuffer); + } + + /****************************************************************/ + /* the input stage */ + if(inBuff && inLen){ + + inOffset = 0; + if(s->padded_read) + inOffset = USB_HEADER_LEN; + + inLength = inOffset+*inLen; + inActual = inLength; + + /*blast caller's copy in case we error out*/ + *inLen = 0; + + inTimeout = USB_DATA_TIME; + + /* change timeout */ + if(shortTime) + inTimeout/=60; + sanei_usb_set_timeout(inTimeout); + + /* build inBuffer */ + inBuffer = calloc(inLength,1); + if(!inBuffer){ + DBG(5,"in: no mem\n"); + return SANE_STATUS_NO_MEM; + } + + DBG(25, "in: reading %d bytes, timeout %d\n", (int)inLength, inTimeout); + ret = sanei_usb_read_bulk(s->fd, inBuffer, &inActual); + DBG(25, "in: read %d bytes, retval %d\n", (int)inActual, ret); + hexdump(30, "in: <<", inBuffer, inActual); + + if(!inActual){ + DBG(5,"in: got no data, clearing\n"); + free(inBuffer); + return do_usb_clear(s,1,runRS); + } + if(inActual < inOffset){ + DBG(5,"in: read shorter than inOffset\n"); + free(inBuffer); + return SANE_STATUS_IO_ERROR; + } + if(ret != SANE_STATUS_GOOD){ + DBG(5,"in: return error '%s'\n",sane_strstatus(ret)); + free(inBuffer); + return ret; + } + + /* note that inBuffer is not copied and freed here...*/ + } + + /****************************************************************/ + /* the status stage */ + statOffset = 0; + if(s->padded_read) + statOffset = USB_HEADER_LEN; + + statLength = statOffset+USB_STATUS_LEN; + statActual = statLength; + statTimeout = USB_STATUS_TIME; + + /* change timeout */ + if(shortTime) + statTimeout/=60; + sanei_usb_set_timeout(statTimeout); + + /* build statBuffer */ + statBuffer = calloc(statLength,1); + if(!statBuffer){ + DBG(5,"stat: no mem\n"); + if(inBuffer) free(inBuffer); + return SANE_STATUS_NO_MEM; + } + + DBG(25, "stat: reading %d bytes, timeout %d\n", (int)statLength, statTimeout); + ret2 = sanei_usb_read_bulk(s->fd, statBuffer, &statActual); + DBG(25, "stat: read %d bytes, retval %d\n", (int)statActual, ret2); + hexdump(30, "stat: <<", statBuffer, statActual); + + /*weird status*/ + if(ret2 != SANE_STATUS_GOOD){ + DBG(5,"stat: clearing error '%s'\n",sane_strstatus(ret2)); + ret2 = do_usb_clear(s,1,runRS); + } + /*short read*/ + else if(statLength != statActual){ + DBG(5,"stat: clearing short %d/%d\n",(int)statLength,(int)statActual); + ret2 = do_usb_clear(s,1,runRS); + } + /*inspect the last byte of the status response*/ + else if(statBuffer[statLength-1]){ + DBG(5,"stat: status %d\n",statBuffer[statLength-1]); + ret2 = do_usb_clear(s,0,runRS); + } + free(statBuffer); + + /* if status said EOF, adjust input with remainder count */ + if(ret2 == SANE_STATUS_EOF && inBuffer){ + + /* EOF is ok */ + ret2 = SANE_STATUS_GOOD; + + if(inActual <= inLength - s->rs_info){ + DBG(5,"in: we read <= RS, ignoring RS: %d <= %d (%d-%d)\n", + (int)inActual,(int)(inLength-s->rs_info),(int)inLength,(int)s->rs_info); + } + else if(s->rs_info){ + DBG(5,"in: we read > RS, using RS: %d to %d (%d-%d)\n", + (int)inActual,(int)(inLength-s->rs_info),(int)inLength,(int)s->rs_info); + inActual = inLength - s->rs_info; + } + } + + /* bail out on bad RS status */ + if(ret2){ + if(inBuffer) free(inBuffer); + DBG(5,"stat: bad RS status, %d\n", ret2); + return ret2; + } + + /* now that we have read status, deal with input buffer */ + if(inBuffer){ + if(inLength != inActual){ + ret = SANE_STATUS_EOF; + DBG(5,"in: short read, %d/%d\n", (int)inLength,(int)inActual); + } + + /* ignore the USB packet around the SCSI command */ + *inLen = inActual - inOffset; + memcpy(inBuff,inBuffer+inOffset,*inLen); + + free(inBuffer); + } + + DBG (10, "do_usb_cmd: finish\n"); + + return ret; +} + +static SANE_Status +do_usb_clear(struct scanner *s, int clear, int runRS) +{ + SANE_Status ret, ret2; + + DBG (10, "do_usb_clear: start\n"); + + usleep(100000); + + if(clear){ + DBG (15, "do_usb_clear: clear halt\n"); + ret = sanei_usb_clear_halt(s->fd); + if(ret != SANE_STATUS_GOOD){ + DBG(5,"do_usb_clear: cant clear halt, returning %d\n", ret); + return ret; + } + } + + /* caller is interested in having RS run on errors */ + if(runRS){ + + unsigned char rs_cmd[REQUEST_SENSE_len]; + size_t rs_cmdLen = REQUEST_SENSE_len; + + unsigned char rs_in[RS_return_size]; + size_t rs_inLen = RS_return_size; + + memset(rs_cmd,0,rs_cmdLen); + set_SCSI_opcode(rs_cmd, REQUEST_SENSE_code); + set_RS_return_size(rs_cmd, rs_inLen); + + DBG(25,"rs sub call >>\n"); + ret2 = do_cmd( + s,0,0, + rs_cmd, rs_cmdLen, + NULL,0, + rs_in, &rs_inLen + ); + DBG(25,"rs sub call <<\n"); + + if(ret2 == SANE_STATUS_EOF){ + DBG(5,"rs: got EOF, returning IO_ERROR\n"); + return SANE_STATUS_IO_ERROR; + } + if(ret2 != SANE_STATUS_GOOD){ + DBG(5,"rs: return error '%s'\n",sane_strstatus(ret2)); + return ret2; + } + + /* parse the rs data */ + ret2 = sense_handler( 0, rs_in, (void *)s ); + + DBG (10, "do_usb_clear: finish after RS\n"); + return ret2; + } + + DBG (10, "do_usb_clear: finish with io error\n"); + + return SANE_STATUS_IO_ERROR; +} + +static SANE_Status +wait_scanner(struct scanner *s) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + unsigned char cmd[TEST_UNIT_READY_len]; + size_t cmdLen = TEST_UNIT_READY_len; + + DBG (10, "wait_scanner: start\n"); + + memset(cmd,0,cmdLen); + set_SCSI_opcode(cmd,TEST_UNIT_READY_code); + + ret = do_cmd ( + s, 0, 1, + cmd, cmdLen, + NULL, 0, + NULL, NULL + ); + + if (ret != SANE_STATUS_GOOD) { + DBG(5,"WARNING: Brain-dead scanner. Hitting with stick\n"); + ret = do_cmd ( + s, 0, 1, + cmd, cmdLen, + NULL, 0, + NULL, NULL + ); + } + if (ret != SANE_STATUS_GOOD) { + DBG(5,"WARNING: Brain-dead scanner. Hitting with stick again\n"); + ret = do_cmd ( + s, 0, 1, + cmd, cmdLen, + NULL, 0, + NULL, NULL + ); + } + + if (ret != SANE_STATUS_GOOD) { + DBG (5, "wait_scanner: error '%s'\n", sane_strstatus (ret)); + } + + DBG (10, "wait_scanner: finish\n"); + + return ret; +} + +/* s->u.page_x stores the user setting + * for the paper width in adf. sometimes, + * we need a value that differs from this + * due to using FB or overscan. + */ +static int +get_page_width(struct scanner *s) +{ + int width = s->u.page_x; + + /* scanner max for fb */ + if(s->u.source == SOURCE_FLATBED){ + return s->max_x_fb; + } + + /* cant overscan larger than scanner max */ + if(width > s->valid_x){ + return s->valid_x; + } + + /* overscan adds a margin to both sides */ + return width; +} + +/* s->u.page_y stores the user setting + * for the paper height in adf. sometimes, + * we need a value that differs from this + * due to using FB or overscan. + */ +static int +get_page_height(struct scanner *s) +{ + int height = s->u.page_y; + + /* scanner max for fb */ + if(s->u.source == SOURCE_FLATBED){ + return s->max_y_fb; + } + + /* cant overscan larger than scanner max */ + if(height > s->max_y){ + return s->max_y; + } + + /* overscan adds a margin to both sides */ + return height; +} + + +/** + * Convenience method to determine longest string size in a list. + */ +static size_t +maxStringSize (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; +} + +/* + * Prints a hex dump of the given buffer onto the debug output stream. + */ +static void +hexdump (int level, char *comment, unsigned char *p, int l) +{ + int i; + char line[70]; /* 'xxx: xx xx ... xx xx abc */ + char *hex = line+4; + char *bin = line+53; + + if(DBG_LEVEL < level) + return; + + line[0] = 0; + + DBG (level, "%s\n", comment); + + for (i = 0; i < l; i++, p++) { + + /* at start of line */ + if ((i % 16) == 0) { + + /* not at start of first line, print current, reset */ + if (i) { + DBG (level, "%s\n", line); + } + + memset(line,0x20,69); + line[69] = 0; + hex = line + 4; + bin = line + 53; + + sprintf (line, "%3.3x:", i); + } + + /* the hex section */ + sprintf (hex, " %2.2x", *p); + hex += 3; + *hex = ' '; + + /* the char section */ + if(*p >= 0x20 && *p <= 0x7e){ + *bin=*p; + } + else{ + *bin='.'; + } + bin++; + } + + /* print last (partial) line */ + DBG (level, "%s\n", line); +} + +/** + * An advanced method we don't support but have to define. + */ +SANE_Status +sane_set_io_mode (SANE_Handle h, SANE_Bool non_blocking) +{ + DBG (10, "sane_set_io_mode\n"); + DBG (15, "%d %p\n", non_blocking, h); + return SANE_STATUS_UNSUPPORTED; +} + +/** + * An advanced method we don't support but have to define. + */ +SANE_Status +sane_get_select_fd (SANE_Handle h, SANE_Int *fdp) +{ + DBG (10, "sane_get_select_fd\n"); + DBG (15, "%p %d\n", h, *fdp); + return SANE_STATUS_UNSUPPORTED; +} + +/* + * @@ Section 8 - Image processing functions + */ + +/* Look in image for likely upper and left paper edges, then rotate + * image so that upper left corner of paper is upper left of image. + * FIXME: should we do this before we binarize instead of after? */ +static SANE_Status +buffer_deskew(struct scanner *s, int side) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + int pwidth = s->i.width; + int width = s->i.Bpl; + int height = s->i.height; + + double TSlope = 0; + int TXInter = 0; + int TYInter = 0; + double TSlopeHalf = 0; + int TOffsetHalf = 0; + + double LSlope = 0; + int LXInter = 0; + int LYInter = 0; + double LSlopeHalf = 0; + int LOffsetHalf = 0; + + int rotateX = 0; + int rotateY = 0; + + int * topBuf = NULL, * botBuf = NULL; + + DBG (10, "buffer_deskew: start\n"); + + /* get buffers for edge detection */ + topBuf = getTransitionsY(s,side,1); + if(!topBuf){ + DBG (5, "buffer_deskew: cant gTY\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + if(0){ + int i; + for(i=0;i<width;i++){ + if(topBuf[i] >=0 && topBuf[i] < height) + s->buffers[side][topBuf[i]*width+i] = 0; + } + } + + botBuf = getTransitionsY(s,side,0); + if(!botBuf){ + DBG (5, "buffer_deskew: cant gTY\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + /* find best top line */ + ret = getEdgeIterate (pwidth, height, s->i.dpi_y, topBuf, + &TSlope, &TXInter, &TYInter); + if(ret){ + DBG(5,"buffer_deskew: gEI error: %d",ret); + goto cleanup; + } + DBG(15,"top: %04.04f %d %d\n",TSlope,TXInter,TYInter); + + /* slope is too shallow, don't want to divide by 0 */ + if(fabs(TSlope) < 0.0001){ + DBG(15,"buffer_deskew: slope too shallow: %0.08f\n",TSlope); + goto cleanup; + } + + /* find best left line, perpendicular to top line */ + LSlope = (double)-1/TSlope; + ret = getEdgeSlope (pwidth, height, topBuf, botBuf, LSlope, + &LXInter, &LYInter); + if(ret){ + DBG(5,"buffer_deskew: gES error: %d",ret); + goto cleanup; + } + DBG(15,"buffer_deskew: left: %04.04f %d %d\n",LSlope,LXInter,LYInter); + + /* find point about which to rotate */ + TSlopeHalf = tan(atan(TSlope)/2); + TOffsetHalf = LYInter; + DBG(15,"buffer_deskew: top half: %04.04f %d\n",TSlopeHalf,TOffsetHalf); + + LSlopeHalf = tan((atan(LSlope) + ((LSlope < 0)?-M_PI_2:M_PI_2))/2); + LOffsetHalf = - LSlopeHalf * TXInter; + DBG(15,"buffer_deskew: left half: %04.04f %d\n",LSlopeHalf,LOffsetHalf); + + rotateX = (LOffsetHalf-TOffsetHalf) / (TSlopeHalf-LSlopeHalf); + rotateY = TSlopeHalf * rotateX + TOffsetHalf; + DBG(15,"buffer_deskew: rotate: %d %d\n",rotateX,rotateY); + + ret = rotateOnCenter (s, side, rotateX, rotateY, TSlope); + if(ret){ + DBG(5,"buffer_deskew: gES error: %d",ret); + goto cleanup; + } + + cleanup: + if(topBuf) + free(topBuf); + if(botBuf) + free(botBuf); + + DBG (10, "buffer_deskew: finish\n"); + return ret; +} + +/* Look in image for likely left/right/bottom paper edges, then crop + * image to match. Does not attempt to rotate the image. + * FIXME: should we do this before we binarize instead of after? */ +static SANE_Status +buffer_crop(struct scanner *s, int side) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + int bwidth = s->i.Bpl; + int width = s->i.width; + int height = s->i.height; + + int top = 0; + int bot = 0; + int left = width; + int right = 0; + + int * topBuf = NULL, * botBuf = NULL; + int * leftBuf = NULL, * rightBuf = NULL; + int leftCount = 0, rightCount = 0, botCount = 0; + int i; + + DBG (10, "buffer_crop: start\n"); + + /* get buffers to find sides and bottom */ + topBuf = getTransitionsY(s,side,1); + if(!topBuf){ + DBG (5, "buffer_crop: no topBuf\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + botBuf = getTransitionsY(s,side,0); + if(!botBuf){ + DBG (5, "buffer_crop: no botBuf\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + leftBuf = getTransitionsX(s,side,1); + if(!leftBuf){ + DBG (5, "buffer_crop: no leftBuf\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + rightBuf = getTransitionsX(s,side,0); + if(!rightBuf){ + DBG (5, "buffer_crop: no rightBuf\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + /* loop thru top and bottom lists, look for l and r extremes */ + for(i=0; i<width; i++){ + if(botBuf[i] > topBuf[i]){ + if(left > i){ + left = i; + } + + leftCount++; + if(leftCount > 3){ + break; + } + } + else{ + leftCount = 0; + left = width; + } + } + + for(i=width-1; i>=0; i--){ + if(botBuf[i] > topBuf[i]){ + if(right < i){ + right = i; + } + + rightCount++; + if(rightCount > 3){ + break; + } + } + else{ + rightCount = 0; + right = -1; + } + } + + /* loop thru left and right lists, look for bottom extreme */ + for(i=height-1; i>=0; i--){ + if(rightBuf[i] > leftBuf[i]){ + if(bot < i){ + bot = i; + } + + botCount++; + if(botCount > 3){ + break; + } + } + else{ + botCount = 0; + bot = -1; + } + } + + DBG (15, "buffer_crop: t:%d b:%d l:%d r:%d\n",top,bot,left,right); + + /* now crop the image */ + /*FIXME: crop duplex backside at same time?*/ + if(left < right && top < bot){ + + int pixels = 0; + int bytes = 0; + unsigned char * line = NULL; + + /*convert left and right to bytes, figure new byte and pixel width */ + switch (s->i.mode) { + + case MODE_COLOR: + pixels = right-left; + bytes = pixels * 3; + left *= 3; + right *= 3; + break; + + case MODE_GRAYSCALE: + pixels = right-left; + bytes = right-left; + break; + + case MODE_LINEART: + case MODE_HALFTONE: + left /= 8; + right = (right+7)/8; + bytes = right-left; + pixels = bytes * 8; + break; + } + + DBG (15, "buffer_crop: l:%d r:%d p:%d b:%d\n",left,right,pixels,bytes); + + line = malloc(bytes); + if(!line){ + DBG (5, "buffer_crop: no line\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + s->i.bytes_sent[side] = 0; + + for(i=top; i<bot; i++){ + memcpy(line, s->buffers[side] + i*bwidth + left, bytes); + memcpy(s->buffers[side] + s->i.bytes_sent[side], line, bytes); + s->i.bytes_sent[side] += bytes; + } + + s->i.bytes_tot[side] = s->i.bytes_sent[side]; + s->i.width = pixels; + s->i.height = bot-top; + s->i.Bpl = bytes; + + free(line); + } + + cleanup: + if(topBuf) + free(topBuf); + if(botBuf) + free(botBuf); + if(leftBuf) + free(leftBuf); + if(rightBuf) + free(rightBuf); + + DBG (10, "buffer_crop: finish\n"); + return ret; +} + +/* Look in image for disconnected 'spots' of the requested size. + * Replace the spots with the average color of the surrounding pixels. + * FIXME: should we do this before we binarize instead of after? */ +static SANE_Status +buffer_despeck(struct scanner *s, int side) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int i,j,k,l,n; + int w = s->i.Bpl; + int pw = s->i.width; + int h = s->i.height; + int t = w*h; + int d = s->swdespeck; + + DBG (10, "buffer_despeck: start\n"); + + switch (s->i.mode){ + + case MODE_COLOR: + for(i=w; i<t-w-(w*d); i+=w){ + for(j=1; j<pw-1-d; j++){ + + int thresh = 255*3; + int outer[] = {0,0,0}; + int hits = 0; + + /*loop over rows and columns in window */ + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + int tmp = 0; + + for(n=0; n<3; n++){ + tmp += s->buffers[side][i + j*3 + k*w + l*3 + n]; + } + + if(tmp < thresh) + thresh = tmp; + } + } + + thresh = (thresh + 255*3 + 255*3)/3; + + /*loop over rows and columns around window */ + for(k=-1; k<d+1; k++){ + for(l=-1; l<d+1; l++){ + + int tmp[3]; + + /* dont count pixels in the window */ + if(k != -1 && k != d && l != -1 && l != d) + continue; + + for(n=0; n<3; n++){ + tmp[n] = s->buffers[side][i + j*3 + k*w + l*3 + n]; + outer[n] += tmp[n]; + } + if(tmp[0]+tmp[1]+tmp[2] < thresh){ + hits++; + break; + } + } + } + + for(n=0; n<3; n++){ + outer[n] /= (4*d + 4); + } + + /*no hits, overwrite with avg surrounding color*/ + if(!hits){ + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + for(n=0; n<3; n++){ + s->buffers[side][i + j*3 + k*w + l*3 + n] = outer[n]; + } + } + } + } + + } + } + break; + + case MODE_GRAYSCALE: + for(i=w; i<t-w-(w*d); i+=w){ + for(j=1; j<w-1-d; j++){ + + int thresh = 255; + int outer = 0; + int hits = 0; + + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + if(s->buffers[side][i + j + k*w + l] < thresh) + thresh = s->buffers[side][i + j + k*w + l]; + } + } + + thresh = (thresh + 255 + 255)/3; + + /*loop over rows and columns around window */ + for(k=-1; k<d+1; k++){ + for(l=-1; l<d+1; l++){ + + int tmp = 0; + + /* dont count pixels in the window */ + if(k != -1 && k != d && l != -1 && l != d) + continue; + + tmp = s->buffers[side][i + j + k*w + l]; + + if(tmp < thresh){ + hits++; + break; + } + + outer += tmp; + } + } + + outer /= (4*d + 4); + + /*no hits, overwrite with avg surrounding color*/ + if(!hits){ + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + s->buffers[side][i + j + k*w + l] = outer; + } + } + } + + } + } + break; + + case MODE_LINEART: + case MODE_HALFTONE: + for(i=w; i<t-w-(w*d); i+=w){ + for(j=1; j<pw-1-d; j++){ + + int curr = 0; + int hits = 0; + + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + curr += s->buffers[side][i + k*w + (j+l)/8] >> (7-(j+l)%8) & 1; + } + } + + if(!curr) + continue; + + /*loop over rows and columns around window */ + for(k=-1; k<d+1; k++){ + for(l=-1; l<d+1; l++){ + + /* dont count pixels in the window */ + if(k != -1 && k != d && l != -1 && l != d) + continue; + + hits += s->buffers[side][i + k*w + (j+l)/8] >> (7-(j+l)%8) & 1; + + if(hits) + break; + } + } + + /*no hits, overwrite with white*/ + if(!hits){ + for(k=0; k<d; k++){ + for(l=0; l<d; l++){ + s->buffers[side][i + k*w + (j+l)/8] &= ~(1 << (7-(j+l)%8)); + } + } + } + + } + } + break; + + default: + break; + } + + DBG (10, "buffer_despeck: finish\n"); + return ret; +} + +/* Loop thru the image width and look for first color change in each column. + * Return a malloc'd array. Caller is responsible for freeing. */ +int * +getTransitionsY (struct scanner *s, int side, int top) +{ + int * buff; + + int i, j, k; + int near, far; + int winLen = 9; + + int width = s->i.width; + int height = s->i.height; + int depth = 1; + + /* defaults for bottom-up */ + int firstLine = height-1; + int lastLine = -1; + int direction = -1; + + DBG (10, "getTransitionsY: start\n"); + + buff = calloc(width,sizeof(int)); + if(!buff){ + DBG (5, "getTransitionsY: no buff\n"); + return NULL; + } + + /* override for top-down */ + if(top){ + firstLine = 0; + lastLine = height; + direction = 1; + } + + /* load the buff array with y value for first color change from edge + * gray/color uses a different algo from binary/halftone */ + switch (s->i.mode) { + + case MODE_COLOR: + depth = 3; + + case MODE_GRAYSCALE: + + for(i=0; i<width; i++){ + buff[i] = lastLine; + + /* load the near and far windows with repeated copy of first pixel */ + near = 0; + for(k=0; k<depth; k++){ + near += s->buffers[side][(firstLine*width+i) * depth + k]; + } + near *= winLen; + far = near; + + /* move windows, check delta */ + for(j=firstLine+direction; j!=lastLine; j+=direction){ + + int farLine = j-winLen*2*direction; + int nearLine = j-winLen*direction; + + if(farLine < 0 || farLine >= height){ + farLine = firstLine; + } + if(nearLine < 0 || nearLine >= height){ + nearLine = firstLine; + } + + for(k=0; k<depth; k++){ + far -= s->buffers[side][(farLine*width+i)*depth+k]; + far += s->buffers[side][(nearLine*width+i)*depth+k]; + + near -= s->buffers[side][(nearLine*width+i)*depth+k]; + near += s->buffers[side][(j*width+i)*depth+k]; + } + + if(abs(near - far) > winLen*depth*9){ + buff[i] = j; + break; + } + } + } + break; + + case MODE_LINEART: + case MODE_HALFTONE: + for(i=0; i<width; i++){ + buff[i] = lastLine; + + /* load the near window with first pixel */ + near = s->buffers[side][(firstLine*width+i)/8] >> (7-(i%8)) & 1; + + /* move */ + for(j=firstLine+direction; j!=lastLine; j+=direction){ + if((s->buffers[side][(j*width+i)/8] >> (7-(i%8)) & 1) != near){ + buff[i] = j; + break; + } + } + } + break; + + } + + /* blast any stragglers with no neighbors within .5 inch */ + for(i=0;i<width-7;i++){ + int sum = 0; + for(j=1;j<=7;j++){ + if(abs(buff[i+j] - buff[i]) < s->i.dpi_y/2) + sum++; + } + if(sum < 2) + buff[i] = lastLine; + } + + DBG (10, "getTransitionsY: finish\n"); + + return buff; +} + +/* Loop thru the image height and look for first color change in each row. + * Return a malloc'd array. Caller is responsible for freeing. */ +int * +getTransitionsX (struct scanner *s, int side, int left) +{ + int * buff; + + int i, j, k; + int near, far; + int winLen = 9; + + int bwidth = s->i.Bpl; + int width = s->i.width; + int height = s->i.height; + int depth = 1; + + /* defaults for right-first */ + int firstCol = width-1; + int lastCol = -1; + int direction = -1; + + DBG (10, "getTransitionsX: start\n"); + + buff = calloc(height,sizeof(int)); + if(!buff){ + DBG (5, "getTransitionsY: no buff\n"); + return NULL; + } + + /* override for left-first*/ + if(left){ + firstCol = 0; + lastCol = width; + direction = 1; + } + + /* load the buff array with x value for first color change from edge + * gray/color uses a different algo from binary/halftone */ + switch (s->i.mode) { + + case MODE_COLOR: + depth = 3; + + case MODE_GRAYSCALE: + + for(i=0; i<height; i++){ + buff[i] = lastCol; + + /* load the near and far windows with repeated copy of first pixel */ + near = 0; + for(k=0; k<depth; k++){ + near += s->buffers[side][i*bwidth + k]; + } + near *= winLen; + far = near; + + /* move windows, check delta */ + for(j=firstCol+direction; j!=lastCol; j+=direction){ + + int farCol = j-winLen*2*direction; + int nearCol = j-winLen*direction; + + if(farCol < 0 || farCol >= width){ + farCol = firstCol; + } + if(nearCol < 0 || nearCol >= width){ + nearCol = firstCol; + } + + for(k=0; k<depth; k++){ + far -= s->buffers[side][i*bwidth + farCol*depth + k]; + far += s->buffers[side][i*bwidth + nearCol*depth + k]; + + near -= s->buffers[side][i*bwidth + nearCol*depth + k]; + near += s->buffers[side][i*bwidth + j*depth + k]; + } + + if(abs(near - far) > winLen*depth*9){ + buff[i] = j; + break; + } + } + } + break; + + case MODE_LINEART: + case MODE_HALFTONE: + for(i=0; i<height; i++){ + buff[i] = lastCol; + + /* load the near window with first pixel */ + near = s->buffers[side][i*bwidth + firstCol/8] >> (7-(firstCol%8)) & 1; + + /* move */ + for(j=firstCol+direction; j!=lastCol; j+=direction){ + if((s->buffers[side][i*bwidth + j/8] >> (7-(j%8)) & 1) != near){ + buff[i] = j; + break; + } + } + } + break; + + } + + /* blast any stragglers with no neighbors within .5 inch */ + for(i=0;i<height-7;i++){ + int sum = 0; + for(j=1;j<=7;j++){ + if(abs(buff[i+j] - buff[i]) < s->i.dpi_x/2) + sum++; + } + if(sum < 2) + buff[i] = lastCol; + } + + DBG (10, "getTransitionsX: finish\n"); + + return buff; +} + +/* Loop thru a getTransitions array, and use a simplified Hough transform + * to divide likely edges into a 2-d array of bins. Then weight each + * bin based on its angle and offset. Return the 'best' bin. */ +static SANE_Status +getLine (int height, int width, int * buff, + int slopes, double minSlope, double maxSlope, + int offsets, int minOffset, int maxOffset, + double * finSlope, int * finOffset, int * finDensity) +{ + SANE_Status ret = 0; + + int ** lines = NULL; + int i, j; + int rise, run; + double slope; + int offset; + int sIndex, oIndex; + int hWidth = width/2; + + double * slopeCenter = NULL; + int * slopeScale = NULL; + double * offsetCenter = NULL; + int * offsetScale = NULL; + + int maxDensity = 1; + double absMaxSlope = fabs(maxSlope); + double absMinSlope = fabs(minSlope); + int absMaxOffset = abs(maxOffset); + int absMinOffset = abs(minOffset); + + DBG(10,"getLine: start %+0.4f %+0.4f %d %d\n", + minSlope,maxSlope,minOffset,maxOffset); + + /*silence compiler*/ + height = height; + + if(absMaxSlope < absMinSlope) + absMaxSlope = absMinSlope; + + if(absMaxOffset < absMinOffset) + absMaxOffset = absMinOffset; + + /* build an array of pretty-print values for slope */ + slopeCenter = calloc(slopes,sizeof(double)); + if(!slopeCenter){ + DBG(5,"getLine: cant load slopeCenter\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + /* build an array of scaling factors for slope */ + slopeScale = calloc(slopes,sizeof(int)); + if(!slopeScale){ + DBG(5,"getLine: cant load slopeScale\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + for(j=0;j<slopes;j++){ + + /* find central value of this 'bucket' */ + slopeCenter[j] = ( + (double)j*(maxSlope-minSlope)/slopes+minSlope + + (double)(j+1)*(maxSlope-minSlope)/slopes+minSlope + )/2; + + /* scale value from the requested range into an inverted 100-1 range + * input close to 0 makes output close to 100 */ + slopeScale[j] = 101 - fabs(slopeCenter[j])*100/absMaxSlope; + } + + /* build an array of pretty-print values for offset */ + offsetCenter = calloc(offsets,sizeof(double)); + if(!offsetCenter){ + DBG(5,"getLine: cant load offsetCenter\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + /* build an array of scaling factors for offset */ + offsetScale = calloc(offsets,sizeof(int)); + if(!offsetScale){ + DBG(5,"getLine: cant load offsetScale\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + for(j=0;j<offsets;j++){ + + /* find central value of this 'bucket'*/ + offsetCenter[j] = ( + (double)j/offsets*(maxOffset-minOffset)+minOffset + + (double)(j+1)/offsets*(maxOffset-minOffset)+minOffset + )/2; + + /* scale value from the requested range into an inverted 100-1 range + * input close to 0 makes output close to 100 */ + offsetScale[j] = 101 - fabs(offsetCenter[j])*100/absMaxOffset; + } + + /* build 2-d array of 'density', divided into slope and offset ranges */ + lines = calloc(slopes, sizeof(int *)); + if(!lines){ + DBG(5,"getLine: cant load lines\n"); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + + for(i=0;i<slopes;i++){ + if(!(lines[i] = calloc(offsets, sizeof(int)))){ + DBG(5,"getLine: cant load lines %d\n",i); + ret = SANE_STATUS_NO_MEM; + goto cleanup; + } + } + + for(i=0;i<width;i++){ + for(j=i+1;j<width && j<i+width/3;j++){ + + /*FIXME: check for invalid (min/max) values?*/ + rise = buff[j] - buff[i]; + run = j-i; + + slope = (double)rise/run; + if(slope >= maxSlope || slope < minSlope) + continue; + + /* offset in center of width, not y intercept! */ + offset = slope * hWidth + buff[i] - slope * i; + if(offset >= maxOffset || offset < minOffset) + continue; + + sIndex = (slope - minSlope) * slopes/(maxSlope-minSlope); + if(sIndex >= slopes) + continue; + + oIndex = (offset - minOffset) * offsets/(maxOffset-minOffset); + if(oIndex >= offsets) + continue; + + lines[sIndex][oIndex]++; + } + } + + /* go thru array, and find most dense line (highest number) */ + for(i=0;i<slopes;i++){ + for(j=0;j<offsets;j++){ + if(lines[i][j] > maxDensity) + maxDensity = lines[i][j]; + } + } + + DBG(15,"getLine: maxDensity %d\n",maxDensity); + + *finSlope = 0; + *finOffset = 0; + *finDensity = 0; + + /* go thru array, and scale densities to % of maximum, plus adjust for + * prefered (smaller absolute value) slope and offset */ + for(i=0;i<slopes;i++){ + for(j=0;j<offsets;j++){ + lines[i][j] = lines[i][j] * slopeScale[i] * offsetScale[j] / maxDensity; + if(lines[i][j] > *finDensity){ + *finDensity = lines[i][j]; + *finSlope = slopeCenter[i]; + *finOffset = offsetCenter[j]; + } + } + } + + if(0){ + DBG(15,"offsetCenter: "); + for(j=0;j<offsets;j++){ + DBG(15," %+04.0f",offsetCenter[j]); + } + DBG(15,"\n"); + + DBG(15,"offsetScale: "); + for(j=0;j<offsets;j++){ + DBG(15," %04d",offsetScale[j]); + } + DBG(15,"\n"); + + for(i=0;i<slopes;i++){ + DBG(15,"slope: %02d %+02.2f %03d:",i,slopeCenter[i],slopeScale[i]); + for(j=0;j<offsets;j++){ + DBG(15,"% 5d",lines[i][j]/100); + } + DBG(15,"\n"); + } + } + + /* dont forget to cleanup */ + cleanup: + for(i=0;i<10;i++){ + if(lines[i]) + free(lines[i]); + } + if(lines) + free(lines); + if(slopeCenter) + free(slopeCenter); + if(slopeScale) + free(slopeScale); + if(offsetCenter) + free(offsetCenter); + if(offsetScale) + free(offsetScale); + + DBG(10,"getLine: finish\n"); + + return ret; +} + +/* Repeatedly find the best range of slope and offset via Hough transform. + * Shift the ranges thru 4 different positions to avoid splitting data + * across multiple bins (false positive). Home-in on the most likely upper + * line of the paper inside the image. Return the 'best' line. */ +SANE_Status +getEdgeIterate (int width, int height, int resolution, +int * buff, double * finSlope, int * finXInter, int * finYInter) +{ + SANE_Status ret = SANE_STATUS_GOOD; + + int slopes = 11; + int offsets = 11; + double maxSlope = 1; + double minSlope = -1; + int maxOffset = resolution/6; + int minOffset = -resolution/6; + + double topSlope = 0; + int topOffset = 0; + int topDensity = 0; + + int i,j; + int pass = 0; + + DBG(10,"getEdgeIterate: start\n"); + + while(pass++ < 7){ + double sStep = (maxSlope-minSlope)/slopes; + int oStep = (maxOffset-minOffset)/offsets; + + double slope = 0; + int offset = 0; + int density = 0; + int go = 0; + + topSlope = 0; + topOffset = 0; + topDensity = 0; + + /* find lines 4 times with slightly moved params, + * to bypass binning errors, highest density wins */ + for(i=0;i<2;i++){ + double sStep2 = sStep*i/2; + for(j=0;j<2;j++){ + int oStep2 = oStep*j/2; + ret = getLine(height,width,buff,slopes,minSlope+sStep2,maxSlope+sStep2,offsets,minOffset+oStep2,maxOffset+oStep2,&slope,&offset,&density); + if(ret){ + DBG(5,"getEdgeIterate: getLine error %d\n",ret); + return ret; + } + DBG(15,"getEdgeIterate: %d %d %+0.4f %d %d\n",i,j,slope,offset,density); + + if(density > topDensity){ + topSlope = slope; + topOffset = offset; + topDensity = density; + } + } + } + + DBG(15,"getEdgeIterate: ok %+0.4f %d %d\n",topSlope,topOffset,topDensity); + + /* did not find anything promising on first pass, + * give up instead of fixating on some small, pointless feature */ + if(pass == 1 && topDensity < width/5){ + DBG(5,"getEdgeIterate: density too small %d %d\n",topDensity,width); + topOffset = 0; + topSlope = 0; + break; + } + + /* if slope can zoom in some more, do so. */ + if(sStep >= 0.0001){ + minSlope = topSlope - sStep; + maxSlope = topSlope + sStep; + go = 1; + } + + /* if offset can zoom in some more, do so. */ + if(oStep){ + minOffset = topOffset - oStep; + maxOffset = topOffset + oStep; + go = 1; + } + + /* cannot zoom in more, bail out */ + if(!go){ + break; + } + + DBG(15,"getEdgeIterate: zoom: %+0.4f %+0.4f %d %d\n", + minSlope,maxSlope,minOffset,maxOffset); + } + + /* topOffset is in the center of the image, + * convert to x and y intercept */ + if(topSlope != 0){ + *finYInter = topOffset - topSlope * width/2; + *finXInter = *finYInter / -topSlope; + *finSlope = topSlope; + } + else{ + *finYInter = 0; + *finXInter = 0; + *finSlope = 0; + } + + DBG(10,"getEdgeIterate: finish\n"); + + return 0; +} + +/* find the left side of paper by moving a line + * perpendicular to top slope across the image + * the 'left-most' point on the paper is the + * one with the smallest X intercept + * return x and y intercepts */ +SANE_Status +getEdgeSlope (int width, int height, int * top, int * bot, + double slope, int * finXInter, int * finYInter) +{ + + int i; + int topXInter, topYInter; + int botXInter, botYInter; + int leftCount; + + DBG(10,"getEdgeSlope: start\n"); + + topXInter = width; + topYInter = 0; + leftCount = 0; + + for(i=0;i<width;i++){ + + if(top[i] < height){ + int tyi = top[i] - (slope * i); + int txi = tyi/-slope; + + if(topXInter > txi){ + topXInter = txi; + topYInter = tyi; + } + + leftCount++; + if(leftCount > 5){ + break; + } + } + else{ + topXInter = width; + topYInter = 0; + leftCount = 0; + } + } + + botXInter = width; + botYInter = 0; + leftCount = 0; + + for(i=0;i<width;i++){ + + if(bot[i] > -1){ + + int byi = bot[i] - (slope * i); + int bxi = byi/-slope; + + if(botXInter > bxi){ + botXInter = bxi; + botYInter = byi; + } + + leftCount++; + if(leftCount > 5){ + break; + } + } + else{ + botXInter = width; + botYInter = 0; + leftCount = 0; + } + } + + if(botXInter < topXInter){ + *finXInter = botXInter; + *finYInter = botYInter; + } + else{ + *finXInter = topXInter; + *finYInter = topYInter; + } + + DBG(10,"getEdgeSlope: finish\n"); + + return 0; +} + +/* function to do a simple rotation by a given slope, around + * a given point. The point can be outside of image to get + * proper edge alignment. Unused areas filled with bg color + * FIXME: Do in-place rotation to save memory */ +SANE_Status +rotateOnCenter (struct scanner *s, int side, + int centerX, int centerY, double slope) +{ + double slopeRad = -atan(slope); + double slopeSin = sin(slopeRad); + double slopeCos = cos(slopeRad); + + int bwidth = s->i.Bpl; + int pwidth = s->i.width; + int height = s->i.height; + int depth = 1; + int bg_color = s->lut[s->bg_color]; + + unsigned char * outbuf; + int i, j, k; + + DBG(10,"rotateOnCenter: start: %d %d\n",centerX,centerY); + + outbuf = malloc(s->i.bytes_tot[side]); + if(!outbuf){ + DBG(15,"rotateOnCenter: no outbuf\n"); + return SANE_STATUS_NO_MEM; + } + + switch (s->i.mode){ + + case MODE_COLOR: + depth = 3; + + case MODE_GRAYSCALE: + memset(outbuf,bg_color,s->i.bytes_tot[side]); + + for (i=0; i<height; i++) { + int shiftY = centerY - i; + + for (j=0; j<pwidth; j++) { + int shiftX = centerX - j; + int sourceX, sourceY; + + sourceX = centerX - (int)(shiftX * slopeCos + shiftY * slopeSin); + if (sourceX < 0 || sourceX >= pwidth) + continue; + + sourceY = centerY + (int)(-shiftY * slopeCos + shiftX * slopeSin); + if (sourceY < 0 || sourceY >= height) + continue; + + for (k=0; k<depth; k++) { + outbuf[i*bwidth+j*depth+k] + = s->buffers[side][sourceY*bwidth+sourceX*depth+k]; + } + } + } + break; + + case MODE_LINEART: + case MODE_HALFTONE: + memset(outbuf,(bg_color<s->threshold)?0xff:0x00,s->i.bytes_tot[side]); + + for (i=0; i<height; i++) { + int shiftY = centerY - i; + + for (j=0; j<pwidth; j++) { + int shiftX = centerX - j; + int sourceX, sourceY; + + sourceX = centerX - (int)(shiftX * slopeCos + shiftY * slopeSin); + if (sourceX < 0 || sourceX >= pwidth) + continue; + + sourceY = centerY + (int)(-shiftY * slopeCos + shiftX * slopeSin); + if (sourceY < 0 || sourceY >= height) + continue; + + /* wipe out old bit */ + outbuf[i*bwidth + j/8] &= ~(1 << (7-(j%8))); + + /* fill in new bit */ + outbuf[i*bwidth + j/8] |= + ((s->buffers[side][sourceY*bwidth + sourceX/8] + >> (7-(sourceX%8))) & 1) << (7-(j%8)); + } + } + break; + } + + memcpy(s->buffers[side],outbuf,s->i.bytes_tot[side]); + + free(outbuf); + + DBG(10,"rotateOnCenter: finish\n"); + + return 0; +} + +/* Function to build a lookup table (LUT), often + used by scanners to implement brightness/contrast/gamma + or by backends to speed binarization/thresholding + + offset and slope inputs are -127 to +127 + + slope rotates line around central input/output val, + 0 makes horizontal line + + pos zero neg + . x . . x + . x . . x + out . x .xxxxxxxxxxx . x + . x . . x + ....x....... ............ .......x.... + in in in + + offset moves line vertically, and clamps to output range + 0 keeps the line crossing the center of the table + + pos zero neg + . xxxxxxxx . xx . + . x . x . + out x . x . x + . . x . x + ............ xx.......... xxxxxxxx.... + in in + + out_min/max provide bounds on output values, + useful when building thresholding lut. + 0 and 255 are good defaults otherwise. + */ +static SANE_Status +load_lut (unsigned char * lut, + int in_bits, int out_bits, + int out_min, int out_max, + int slope, int offset) +{ + SANE_Status ret = SANE_STATUS_GOOD; + int i, j; + double shift, rise; + int max_in_val = (1 << in_bits) - 1; + int max_out_val = (1 << out_bits) - 1; + unsigned char * lut_p = lut; + + DBG (10, "load_lut: start %d %d\n", slope, offset); + + /* slope is converted to rise per unit run: + * first [-127,127] to [-.999,.999] + * then to [-PI/4,PI/4] then [0,PI/2] + * then take the tangent (T.O.A) + * then multiply by the normal linear slope + * because the table may not be square, i.e. 1024x256*/ + rise = tan((double)slope/128 * M_PI_4 + M_PI_4) * max_out_val / max_in_val; + + /* line must stay vertically centered, so figure + * out vertical offset at central input value */ + shift = (double)max_out_val/2 - (rise*max_in_val/2); + + /* convert the user offset setting to scale of output + * first [-127,127] to [-1,1] + * then to [-max_out_val/2,max_out_val/2]*/ + shift += (double)offset / 127 * max_out_val / 2; + + for(i=0;i<=max_in_val;i++){ + j = rise*i + shift; + + if(j<out_min){ + j=out_min; + } + else if(j>out_max){ + j=out_max; + } + + *lut_p=j; + lut_p++; + } + + hexdump(5, "load_lut: ", lut, max_in_val+1); + + DBG (10, "load_lut: finish\n"); + return ret; +} + |