summaryrefslogtreecommitdiff
path: root/backend/genesys_gl841.cc
diff options
context:
space:
mode:
Diffstat (limited to 'backend/genesys_gl841.cc')
-rw-r--r--backend/genesys_gl841.cc5624
1 files changed, 5624 insertions, 0 deletions
diff --git a/backend/genesys_gl841.cc b/backend/genesys_gl841.cc
new file mode 100644
index 0000000..9e8fc15
--- /dev/null
+++ b/backend/genesys_gl841.cc
@@ -0,0 +1,5624 @@
+/* sane - Scanner Access Now Easy.
+
+ Copyright (C) 2003 Oliver Rauch
+ Copyright (C) 2003, 2004 Henning Meier-Geinitz <henning@meier-geinitz.de>
+ Copyright (C) 2004 Gerhard Jaeger <gerhard@gjaeger.de>
+ Copyright (C) 2004-2013 Stéphane Voltz <stef.dev@free.fr>
+ Copyright (C) 2005 Philipp Schmid <philipp8288@web.de>
+ Copyright (C) 2005-2009 Pierre Willenbrock <pierre@pirsoft.dnsalias.org>
+ Copyright (C) 2006 Laurent Charpentier <laurent_pubs@yahoo.com>
+ Copyright (C) 2010 Chris Berry <s0457957@sms.ed.ac.uk> and Michael Rickmann <mrickma@gwdg.de>
+ for Plustek Opticbook 3600 support
+
+
+ This file is part of the SANE package.
+
+ This program is free software; you can redistribute it and/or
+ modify it under the terms of the GNU General Public License as
+ published by the Free Software Foundation; either version 2 of the
+ License, or (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful, but
+ WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place - Suite 330, Boston,
+ MA 02111-1307, USA.
+
+ As a special exception, the authors of SANE give permission for
+ additional uses of the libraries contained in this release of SANE.
+
+ The exception is that, if you link a SANE library with other files
+ to produce an executable, this does not by itself cause the
+ resulting executable to be covered by the GNU General Public
+ License. Your use of that executable is in no way restricted on
+ account of linking the SANE library code into it.
+
+ This exception does not, however, invalidate any other reasons why
+ the executable file might be covered by the GNU General Public
+ License.
+
+ If you submit changes to SANE to the maintainers to be included in
+ a subsequent release, you agree by submitting the changes that
+ those changes may be distributed with this exception intact.
+
+ If you write modifications of your own for SANE, it is your choice
+ whether to permit this exception to apply to your modifications.
+ If you do not wish that, delete this exception notice.
+*/
+
+#define DEBUG_DECLARE_ONLY
+
+#include "genesys_gl841.h"
+
+#include <vector>
+
+/****************************************************************************
+ Low level function
+ ****************************************************************************/
+
+/* ------------------------------------------------------------------------ */
+/* Read and write RAM, registers and AFE */
+/* ------------------------------------------------------------------------ */
+
+/* Set address for writing data */
+static SANE_Status
+gl841_set_buffer_address_gamma (Genesys_Device * dev, uint32_t addr)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBG(DBG_io, "%s: setting address to 0x%05x\n", __func__, addr & 0xfffffff0);
+
+ addr = addr >> 4;
+
+ status = sanei_genesys_write_register (dev, 0x5c, (addr & 0xff));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed while writing low byte: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ addr = addr >> 8;
+ status = sanei_genesys_write_register (dev, 0x5b, (addr & 0xff));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed while writing high byte: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_io, "%s: completed\n", __func__);
+
+ return status;
+}
+
+/****************************************************************************
+ Mid level functions
+ ****************************************************************************/
+
+static SANE_Bool
+gl841_get_fast_feed_bit (Genesys_Register_Set * regs)
+{
+ GenesysRegister *r = NULL;
+
+ r = sanei_genesys_get_address (regs, 0x02);
+ if (r && (r->value & REG02_FASTFED))
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_get_filter_bit (Genesys_Register_Set * regs)
+{
+ GenesysRegister *r = NULL;
+
+ r = sanei_genesys_get_address (regs, 0x04);
+ if (r && (r->value & REG04_FILTER))
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_get_lineart_bit (Genesys_Register_Set * regs)
+{
+ GenesysRegister *r = NULL;
+
+ r = sanei_genesys_get_address (regs, 0x04);
+ if (r && (r->value & REG04_LINEART))
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_get_bitset_bit (Genesys_Register_Set * regs)
+{
+ GenesysRegister *r = NULL;
+
+ r = sanei_genesys_get_address (regs, 0x04);
+ if (r && (r->value & REG04_BITSET))
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_get_gain4_bit (Genesys_Register_Set * regs)
+{
+ GenesysRegister *r = NULL;
+
+ r = sanei_genesys_get_address (regs, 0x06);
+ if (r && (r->value & REG06_GAIN4))
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_test_buffer_empty_bit (SANE_Byte val)
+{
+ if (val & REG41_BUFEMPTY)
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+static SANE_Bool
+gl841_test_motor_flag_bit (SANE_Byte val)
+{
+ if (val & REG41_MOTORENB)
+ return SANE_TRUE;
+ return SANE_FALSE;
+}
+
+/** copy sensor specific settings */
+/* *dev : device infos
+ *regs : registers to be set
+ extended : do extended set up
+ half_ccd: set up for half ccd resolution
+ all registers 08-0B, 10-1D, 52-59 are set up. They shouldn't
+ appear anywhere else but in register_ini
+
+Responsible for signals to CCD/CIS:
+ CCD_CK1X (CK1INV(0x16),CKDIS(0x16),CKTOGGLE(0x18),CKDELAY(0x18),MANUAL1(0x1A),CK1MTGL(0x1C),CK1LOW(0x1D),CK1MAP(0x74,0x75,0x76),CK1NEG(0x7D))
+ CCD_CK2X (CK2INV(0x16),CKDIS(0x16),CKTOGGLE(0x18),CKDELAY(0x18),MANUAL1(0x1A),CK1LOW(0x1D),CK1NEG(0x7D))
+ CCD_CK3X (MANUAL3(0x1A),CK3INV(0x1A),CK3MTGL(0x1C),CK3LOW(0x1D),CK3MAP(0x77,0x78,0x79),CK3NEG(0x7D))
+ CCD_CK4X (MANUAL3(0x1A),CK4INV(0x1A),CK4MTGL(0x1C),CK4LOW(0x1D),CK4MAP(0x7A,0x7B,0x7C),CK4NEG(0x7D))
+ CCD_CPX (CTRLHI(0x16),CTRLINV(0x16),CTRLDIS(0x16),CPH(0x72),CPL(0x73),CPNEG(0x7D))
+ CCD_RSX (CTRLHI(0x16),CTRLINV(0x16),CTRLDIS(0x16),RSH(0x70),RSL(0x71),RSNEG(0x7D))
+ CCD_TGX (TGINV(0x16),TGMODE(0x17),TGW(0x17),EXPR(0x10,0x11),TGSHLD(0x1D))
+ CCD_TGG (TGINV(0x16),TGMODE(0x17),TGW(0x17),EXPG(0x12,0x13),TGSHLD(0x1D))
+ CCD_TGB (TGINV(0x16),TGMODE(0x17),TGW(0x17),EXPB(0x14,0x15),TGSHLD(0x1D))
+ LAMP_SW (EXPR(0x10,0x11),XPA_SEL(0x03),LAMP_PWR(0x03),LAMPTIM(0x03),MTLLAMP(0x04),LAMPPWM(0x29))
+ XPA_SW (EXPG(0x12,0x13),XPA_SEL(0x03),LAMP_PWR(0x03),LAMPTIM(0x03),MTLLAMP(0x04),LAMPPWM(0x29))
+ LAMP_B (EXPB(0x14,0x15),LAMP_PWR(0x03))
+
+other registers:
+ CISSET(0x01),CNSET(0x18),DCKSEL(0x18),SCANMOD(0x18),EXPDMY(0x19),LINECLP(0x1A),CKAREA(0x1C),TGTIME(0x1C),LINESEL(0x1E),DUMMY(0x34)
+
+Responsible for signals to AFE:
+ VSMP (VSMP(0x58),VSMPW(0x58))
+ BSMP (BSMP(0x59),BSMPW(0x59))
+
+other register settings depending on this:
+ RHI(0x52),RLOW(0x53),GHI(0x54),GLOW(0x55),BHI(0x56),BLOW(0x57),
+
+*/
+static void sanei_gl841_setup_sensor(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set * regs,
+ SANE_Bool extended, SANE_Bool half_ccd)
+{
+ DBG(DBG_proc, "%s\n", __func__);
+
+ // that one is tricky at least
+ for (uint16_t addr = 0x08; addr <= 0x0b; ++addr) {
+ regs->set8(0x70 + addr - 0x08, sensor.custom_regs.get_value(addr));
+ }
+
+ // ignore registers in range [0x10..0x16)
+ for (uint16_t addr = 0x16; addr < 0x1e; ++addr) {
+ regs->set8(addr, sensor.custom_regs.get_value(addr));
+ }
+
+ // ignore registers in range [0x5b..0x5e]
+ for (uint16_t addr = 0x52; addr < 0x52 + 9; ++addr) {
+ regs->set8(addr, sensor.custom_regs.get_value(addr));
+ }
+
+ /* don't go any further if no extended setup */
+ if (!extended)
+ return;
+
+ /* todo : add more CCD types if needed */
+ /* we might want to expand the Sensor struct to have these
+ 2 kind of settings */
+ if (dev->model->ccd_type == CCD_5345)
+ {
+ if (half_ccd)
+ {
+ GenesysRegister* r;
+ /* settings for CCD used at half is max resolution */
+ r = sanei_genesys_get_address (regs, 0x70);
+ r->value = 0x00;
+ r = sanei_genesys_get_address (regs, 0x71);
+ r->value = 0x05;
+ r = sanei_genesys_get_address (regs, 0x72);
+ r->value = 0x06;
+ r = sanei_genesys_get_address (regs, 0x73);
+ r->value = 0x08;
+ r = sanei_genesys_get_address (regs, 0x18);
+ r->value = 0x28;
+ r = sanei_genesys_get_address (regs, 0x58);
+ r->value = 0x80 | (r->value & 0x03); /* VSMP=16 */
+ }
+ else
+ {
+ GenesysRegister* r;
+ /* swap latch times */
+ r = sanei_genesys_get_address (regs, 0x18);
+ r->value = 0x30;
+ regs->set8(0x52, sensor.custom_regs.get_value(0x55));
+ regs->set8(0x53, sensor.custom_regs.get_value(0x56));
+ regs->set8(0x54, sensor.custom_regs.get_value(0x57));
+ regs->set8(0x55, sensor.custom_regs.get_value(0x52));
+ regs->set8(0x56, sensor.custom_regs.get_value(0x53));
+ regs->set8(0x57, sensor.custom_regs.get_value(0x54));
+ r = sanei_genesys_get_address (regs, 0x58);
+ r->value = 0x20 | (r->value & 0x03); /* VSMP=4 */
+ }
+ return;
+ }
+
+ if (dev->model->ccd_type == CCD_HP2300)
+ {
+ /* settings for CCD used at half is max resolution */
+ GenesysRegister* r;
+ if (half_ccd)
+ {
+ r = sanei_genesys_get_address (regs, 0x70);
+ r->value = 0x16;
+ r = sanei_genesys_get_address (regs, 0x71);
+ r->value = 0x00;
+ r = sanei_genesys_get_address (regs, 0x72);
+ r->value = 0x01;
+ r = sanei_genesys_get_address (regs, 0x73);
+ r->value = 0x03;
+ /* manual clock programming */
+ r = sanei_genesys_get_address (regs, 0x1d);
+ r->value |= 0x80;
+ }
+ else
+ {
+ r = sanei_genesys_get_address (regs, 0x70);
+ r->value = 1;
+ r = sanei_genesys_get_address (regs, 0x71);
+ r->value = 3;
+ r = sanei_genesys_get_address (regs, 0x72);
+ r->value = 4;
+ r = sanei_genesys_get_address (regs, 0x73);
+ r->value = 6;
+ }
+ r = sanei_genesys_get_address (regs, 0x58);
+ r->value = 0x80 | (r->value & 0x03); /* VSMP=16 */
+ return;
+ }
+}
+
+/** Test if the ASIC works
+ */
+/*TODO: make this functional*/
+static SANE_Status
+sanei_gl841_asic_test (Genesys_Device * dev)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+ size_t size, verify_size;
+ unsigned int i;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ return SANE_STATUS_INVAL;
+
+ /* set and read exposure time, compare if it's the same */
+ status = sanei_genesys_write_register (dev, 0x38, 0xde);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to write register: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_write_register (dev, 0x39, 0xad);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to write register: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_read_register (dev, 0x38, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read register: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ if (val != 0xde) /* value of register 0x38 */
+ {
+ DBG(DBG_error, "%s: register contains invalid value\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+ }
+
+ status = sanei_genesys_read_register (dev, 0x39, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read register: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ if (val != 0xad) /* value of register 0x39 */
+ {
+ DBG(DBG_error, "%s: register contains invalid value\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+ }
+
+ /* ram test: */
+ size = 0x40000;
+ verify_size = size + 0x80;
+ /* todo: looks like the read size must be a multiple of 128?
+ otherwise the read doesn't succeed the second time after the scanner has
+ been plugged in. Very strange. */
+
+ std::vector<uint8_t> data(size);
+ std::vector<uint8_t> verify_data(verify_size);
+
+ for (i = 0; i < (size - 1); i += 2)
+ {
+ data[i] = i / 512;
+ data[i + 1] = (i / 2) % 256;
+ }
+
+ status = sanei_genesys_set_buffer_address (dev, 0x0000);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set buffer address: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+/* status = sanei_genesys_bulk_write_data(dev, 0x3c, data, size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write data: %s\n", __func__, sane_strstatus(status));
+ free (data);
+ free (verify_data);
+ return status;
+ }*/
+
+ status = sanei_genesys_set_buffer_address (dev, 0x0000);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set buffer address: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_bulk_read_data(dev, 0x45, verify_data.data(), verify_size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk read data: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* todo: why i + 2 ? */
+ for (i = 0; i < size; i++)
+ {
+ if (verify_data[i] != data[i])
+ {
+ DBG(DBG_error, "%s: data verification error\n", __func__);
+ DBG(DBG_info, "0x%.8x: got %.2x %.2x %.2x %.2x, expected %.2x %.2x %.2x %.2x\n",
+ i,
+ verify_data[i],
+ verify_data[i+1],
+ verify_data[i+2],
+ verify_data[i+3],
+ data[i],
+ data[i+1],
+ data[i+2],
+ data[i+3]);
+ return SANE_STATUS_IO_ERROR;
+ }
+ }
+
+ DBG(DBG_info, "%s: completed\n", __func__);
+
+ return SANE_STATUS_GOOD;
+}
+
+/*
+ * Set all registers LiDE 80 to default values
+ * (function called only once at the beginning)
+ * we are doing a special case to ease development
+ */
+static void
+gl841_init_lide80 (Genesys_Device * dev)
+{
+ uint8_t val;
+
+ INITREG (0x01, 0x82); /* 0x02 = SHDAREA and no CISSET ! */
+ INITREG (0x02, 0x10);
+ INITREG (0x03, 0x50);
+ INITREG (0x04, 0x02);
+ INITREG (0x05, 0x4c); /* 1200 DPI */
+ INITREG (0x06, 0x38); /* 0x38 scanmod=1, pwrbit, GAIN4 */
+ INITREG (0x07, 0x00);
+ INITREG (0x08, 0x00);
+ INITREG (0x09, 0x11);
+ INITREG (0x0a, 0x00);
+
+ INITREG (0x10, 0x40);
+ INITREG (0x11, 0x00);
+ INITREG (0x12, 0x40);
+ INITREG (0x13, 0x00);
+ INITREG (0x14, 0x40);
+ INITREG (0x15, 0x00);
+ INITREG (0x16, 0x00);
+ INITREG (0x17, 0x01);
+ INITREG (0x18, 0x00);
+ INITREG (0x19, 0x06);
+ INITREG (0x1a, 0x00);
+ INITREG (0x1b, 0x00);
+ INITREG (0x1c, 0x00);
+ INITREG (0x1d, 0x04);
+ INITREG (0x1e, 0x10);
+ INITREG (0x1f, 0x04);
+ INITREG (0x20, 0x02);
+ INITREG (0x21, 0x10);
+ INITREG (0x22, 0x20);
+ INITREG (0x23, 0x20);
+ INITREG (0x24, 0x10);
+ INITREG (0x25, 0x00);
+ INITREG (0x26, 0x00);
+ INITREG (0x27, 0x00);
+
+ INITREG (0x29, 0xff);
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+ INITREG (0x2c, sensor.optical_res>>8);
+ INITREG (0x2d, sensor.optical_res & 0xff);
+ INITREG (0x2e, 0x80);
+ INITREG (0x2f, 0x80);
+ INITREG (0x30, 0x00);
+ INITREG (0x31, 0x10);
+ INITREG (0x32, 0x15);
+ INITREG (0x33, 0x0e);
+ INITREG (0x34, 0x40);
+ INITREG (0x35, 0x00);
+ INITREG (0x36, 0x2a);
+ INITREG (0x37, 0x30);
+ INITREG (0x38, 0x2a);
+ INITREG (0x39, 0xf8);
+
+ INITREG (0x3d, 0x00);
+ INITREG (0x3e, 0x00);
+ INITREG (0x3f, 0x00);
+
+ INITREG (0x52, 0x03);
+ INITREG (0x53, 0x07);
+ INITREG (0x54, 0x00);
+ INITREG (0x55, 0x00);
+ INITREG (0x56, 0x00);
+ INITREG (0x57, 0x00);
+ INITREG (0x58, 0x29);
+ INITREG (0x59, 0x69);
+ INITREG (0x5a, 0x55);
+
+ INITREG (0x5d, 0x20);
+ INITREG (0x5e, 0x41);
+ INITREG (0x5f, 0x40);
+ INITREG (0x60, 0x00);
+ INITREG (0x61, 0x00);
+ INITREG (0x62, 0x00);
+ INITREG (0x63, 0x00);
+ INITREG (0x64, 0x00);
+ INITREG (0x65, 0x00);
+ INITREG (0x66, 0x00);
+ INITREG (0x67, 0x40);
+ INITREG (0x68, 0x40);
+ INITREG (0x69, 0x20);
+ INITREG (0x6a, 0x20);
+ INITREG (0x6c, dev->gpo.value[0]);
+ INITREG (0x6d, dev->gpo.value[1]);
+ INITREG (0x6e, dev->gpo.enable[0]);
+ INITREG (0x6f, dev->gpo.enable[1]);
+ INITREG (0x70, 0x00);
+ INITREG (0x71, 0x05);
+ INITREG (0x72, 0x07);
+ INITREG (0x73, 0x09);
+ INITREG (0x74, 0x00);
+ INITREG (0x75, 0x01);
+ INITREG (0x76, 0xff);
+ INITREG (0x77, 0x00);
+ INITREG (0x78, 0x0f);
+ INITREG (0x79, 0xf0);
+ INITREG (0x7a, 0xf0);
+ INITREG (0x7b, 0x00);
+ INITREG (0x7c, 0x1e);
+ INITREG (0x7d, 0x11);
+ INITREG (0x7e, 0x00);
+ INITREG (0x7f, 0x50);
+ INITREG (0x80, 0x00);
+ INITREG (0x81, 0x00);
+ INITREG (0x82, 0x0f);
+ INITREG (0x83, 0x00);
+ INITREG (0x84, 0x0e);
+ INITREG (0x85, 0x00);
+ INITREG (0x86, 0x0d);
+ INITREG (0x87, 0x02);
+ INITREG (0x88, 0x00);
+ INITREG (0x89, 0x00);
+
+ /* specific scanner settings, clock and gpio first */
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x0c);
+ sanei_genesys_write_register (dev, 0x06, 0x10);
+ sanei_genesys_write_register (dev, REG6E, 0x6d);
+ sanei_genesys_write_register (dev, REG6F, 0x80);
+ sanei_genesys_write_register (dev, REG6B, 0x0e);
+ sanei_genesys_read_register (dev, REG6C, &val);
+ sanei_genesys_write_register (dev, REG6C, 0x00);
+ sanei_genesys_read_register (dev, REG6D, &val);
+ sanei_genesys_write_register (dev, REG6D, 0x8f);
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x0e);
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x0e);
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x0a);
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x02);
+ sanei_genesys_read_register (dev, REG6B, &val);
+ sanei_genesys_write_register (dev, REG6B, 0x06);
+
+ sanei_genesys_write_0x8c (dev, 0x10, 0x94);
+ sanei_genesys_write_register (dev, 0x09, 0x10);
+
+ /* set up GPIO : no address, so no bulk write, doesn't written directly either ? */
+ /*
+ dev->reg.find_reg(0x6c).value = dev->gpo.value[0];
+ dev->reg.find_reg(0x6d).value = dev->gpo.value[1];
+ dev->reg.find_reg(0x6e).value = dev->gpo.enable[0];
+ dev->reg.find_reg(0x6f).value = dev->gpo.enable[1]; */
+
+ // FIXME: the following code originally changed 0x6b, but due to bug the 0x6c register was
+ // effectively changed. The current behavior matches the old code, but should probably be fixed.
+ dev->reg.find_reg(0x6c).value |= REG6B_GPO18;
+ dev->reg.find_reg(0x6c).value &= ~REG6B_GPO17;
+
+ sanei_gl841_setup_sensor(dev, sensor, &dev->reg, 0, 0);
+}
+
+/*
+ * Set all registers to default values
+ * (function called only once at the beginning)
+ */
+static void
+gl841_init_registers (Genesys_Device * dev)
+{
+ int addr;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ dev->reg.clear();
+ if (dev->model->model_id == MODEL_CANON_LIDE_80)
+ {
+ gl841_init_lide80(dev);
+ return ;
+ }
+
+ for (addr = 1; addr <= 0x0a; addr++) {
+ dev->reg.init_reg(addr, 0);
+ }
+ for (addr = 0x10; addr <= 0x27; addr++) {
+ dev->reg.init_reg(addr, 0);
+ }
+ dev->reg.init_reg(0x29, 0);
+ for (addr = 0x2c; addr <= 0x39; addr++)
+ dev->reg.init_reg(addr, 0);
+ for (addr = 0x3d; addr <= 0x3f; addr++)
+ dev->reg.init_reg(addr, 0);
+ for (addr = 0x52; addr <= 0x5a; addr++)
+ dev->reg.init_reg(addr, 0);
+ for (addr = 0x5d; addr <= 0x87; addr++)
+ dev->reg.init_reg(addr, 0);
+
+
+ dev->reg.find_reg(0x01).value = 0x20; /* (enable shading), CCD, color, 1M */
+ if (dev->model->is_cis == SANE_TRUE) {
+ dev->reg.find_reg(0x01).value |= REG01_CISSET;
+ } else {
+ dev->reg.find_reg(0x01).value &= ~REG01_CISSET;
+ }
+
+ dev->reg.find_reg(0x02).value = 0x30 /*0x38 */ ; /* auto home, one-table-move, full step */
+ dev->reg.find_reg(0x02).value |= REG02_AGOHOME;
+ sanei_genesys_set_motor_power(dev->reg, true);
+ dev->reg.find_reg(0x02).value |= REG02_FASTFED;
+
+ dev->reg.find_reg(0x03).value = 0x1f /*0x17 */ ; /* lamp on */
+ dev->reg.find_reg(0x03).value |= REG03_AVEENB;
+
+ if (dev->model->ccd_type == CCD_PLUSTEK_3600) /* AD front end */
+ {
+ dev->reg.find_reg(0x04).value = (2 << REG04S_AFEMOD) | 0x02;
+ }
+ else /* Wolfson front end */
+ {
+ dev->reg.find_reg(0x04).value |= 1 << REG04S_AFEMOD;
+ }
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+
+ dev->reg.find_reg(0x05).value = 0x00; /* disable gamma, 24 clocks/pixel */
+ if (sensor.sensor_pixels < 0x1500)
+ dev->reg.find_reg(0x05).value |= REG05_DPIHW_600;
+ else if (sensor.sensor_pixels < 0x2a80)
+ dev->reg.find_reg(0x05).value |= REG05_DPIHW_1200;
+ else if (sensor.sensor_pixels < 0x5400)
+ dev->reg.find_reg(0x05).value |= REG05_DPIHW_2400;
+ else
+ {
+ dev->reg.find_reg(0x05).value |= REG05_DPIHW_2400;
+ DBG(DBG_warn, "%s: Cannot handle sensor pixel count %d\n", __func__,
+ sensor.sensor_pixels);
+ }
+
+
+ dev->reg.find_reg(0x06).value |= REG06_PWRBIT;
+ dev->reg.find_reg(0x06).value |= REG06_GAIN4;
+
+ /* XP300 CCD needs different clock and clock/pixels values */
+ if (dev->model->ccd_type != CCD_XP300 && dev->model->ccd_type != CCD_DP685
+ && dev->model->ccd_type != CCD_PLUSTEK_3600)
+ {
+ dev->reg.find_reg(0x06).value |= 0 << REG06S_SCANMOD;
+ dev->reg.find_reg(0x09).value |= 1 << REG09S_CLKSET;
+ }
+ else
+ {
+ dev->reg.find_reg(0x06).value |= 0x05 << REG06S_SCANMOD; /* 15 clocks/pixel */
+ dev->reg.find_reg(0x09).value = 0; /* 24 MHz CLKSET */
+ }
+
+ dev->reg.find_reg(0x1e).value = 0xf0; /* watch-dog time */
+
+ dev->reg.find_reg(0x17).value |= 1 << REG17S_TGW;
+
+ dev->reg.find_reg(0x19).value = 0x50;
+
+ dev->reg.find_reg(0x1d).value |= 1 << REG1DS_TGSHLD;
+
+ dev->reg.find_reg(0x1e).value |= 1 << REG1ES_WDTIME;
+
+/*SCANFED*/
+ dev->reg.find_reg(0x1f).value = 0x01;
+
+/*BUFSEL*/
+ dev->reg.find_reg(0x20).value = 0x20;
+
+/*LAMPPWM*/
+ dev->reg.find_reg(0x29).value = 0xff;
+
+/*BWHI*/
+ dev->reg.find_reg(0x2e).value = 0x80;
+
+/*BWLOW*/
+ dev->reg.find_reg(0x2f).value = 0x80;
+
+/*LPERIOD*/
+ dev->reg.find_reg(0x38).value = 0x4f;
+ dev->reg.find_reg(0x39).value = 0xc1;
+
+/*VSMPW*/
+ dev->reg.find_reg(0x58).value |= 3 << REG58S_VSMPW;
+
+/*BSMPW*/
+ dev->reg.find_reg(0x59).value |= 3 << REG59S_BSMPW;
+
+/*RLCSEL*/
+ dev->reg.find_reg(0x5a).value |= REG5A_RLCSEL;
+
+/*STOPTIM*/
+ dev->reg.find_reg(0x5e).value |= 0x2 << REG5ES_STOPTIM;
+
+ sanei_gl841_setup_sensor(dev, sensor, &dev->reg, 0, 0);
+
+ /* set up GPIO */
+ dev->reg.find_reg(0x6c).value = dev->gpo.value[0];
+ dev->reg.find_reg(0x6d).value = dev->gpo.value[1];
+ dev->reg.find_reg(0x6e).value = dev->gpo.enable[0];
+ dev->reg.find_reg(0x6f).value = dev->gpo.enable[1];
+
+ /* TODO there is a switch calling to be written here */
+ if (dev->model->gpo_type == GPO_CANONLIDE35)
+ {
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO18;
+ dev->reg.find_reg(0x6b).value &= ~REG6B_GPO17;
+ }
+
+ if (dev->model->gpo_type == GPO_XP300)
+ {
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO17;
+ }
+
+ if (dev->model->gpo_type == GPO_DP685)
+ {
+ /* REG6B_GPO18 lights on green led */
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO17|REG6B_GPO18;
+ }
+
+ DBG(DBG_proc, "%s complete\n", __func__);
+}
+
+/* Send slope table for motor movement
+ slope_table in machine byte order
+ */
+static SANE_Status
+gl841_send_slope_table (Genesys_Device * dev, int table_nr,
+ uint16_t * slope_table, int steps)
+{
+ int dpihw;
+ int start_address;
+ SANE_Status status = SANE_STATUS_GOOD;
+ char msg[4000];
+/*#ifdef WORDS_BIGENDIAN*/
+ int i;
+/*#endif*/
+
+ DBG(DBG_proc, "%s (table_nr = %d, steps = %d)\n", __func__, table_nr, steps);
+
+ dpihw = dev->reg.find_reg(0x05).value >> 6;
+
+ if (dpihw == 0) /* 600 dpi */
+ start_address = 0x08000;
+ else if (dpihw == 1) /* 1200 dpi */
+ start_address = 0x10000;
+ else if (dpihw == 2) /* 2400 dpi */
+ start_address = 0x20000;
+ else /* reserved */
+ return SANE_STATUS_INVAL;
+
+ std::vector<uint8_t> table(steps * 2);
+ for(i = 0; i < steps; i++) {
+ table[i * 2] = slope_table[i] & 0xff;
+ table[i * 2 + 1] = slope_table[i] >> 8;
+ }
+
+ if (DBG_LEVEL >= DBG_io)
+ {
+ sprintf (msg, "write slope %d (%d)=", table_nr, steps);
+ for (i = 0; i < steps; i++)
+ {
+ sprintf (msg+strlen(msg), ",%d", slope_table[i]);
+ }
+ DBG(DBG_io, "%s: %s\n", __func__, msg);
+ }
+
+ status =
+ sanei_genesys_set_buffer_address (dev, start_address + table_nr * 0x200);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set buffer address: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_bulk_write_data(dev, 0x3c, table.data(), steps * 2);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to send slope table: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return status;
+}
+
+static SANE_Status
+gl841_set_lide80_fe (Genesys_Device * dev, uint8_t set)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBGSTART;
+
+ if (set == AFE_INIT)
+ {
+ DBG(DBG_proc, "%s(): setting DAC %u\n", __func__, dev->model->dac_type);
+
+ dev->frontend = dev->frontend_initial;
+
+ /* write them to analog frontend */
+ status = sanei_genesys_fe_write_data(dev, 0x00, dev->frontend.regs.get_value(0x00));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x00 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ status = sanei_genesys_fe_write_data(dev, 0x03, dev->frontend.regs.get_value(0x01));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x03 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ status = sanei_genesys_fe_write_data(dev, 0x06, dev->frontend.regs.get_value(0x02));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x06 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ if (set == AFE_SET)
+ {
+ status = sanei_genesys_fe_write_data(dev, 0x00, dev->frontend.regs.get_value(0x00));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x00 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ status = sanei_genesys_fe_write_data(dev, 0x06, dev->frontend.regs.get_value(0x20));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing offset failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ status = sanei_genesys_fe_write_data(dev, 0x03, dev->frontend.regs.get_value(0x28));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing gain failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ return status;
+ DBGCOMPLETED;
+}
+
+/* Set values of Analog Device type frontend */
+static SANE_Status
+gl841_set_ad_fe (Genesys_Device * dev, uint8_t set)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ int i;
+
+ /* special case for LiDE 80 analog frontend */
+ if(dev->model->dac_type==DAC_CANONLIDE80)
+ {
+ return gl841_set_lide80_fe(dev, set);
+ }
+
+ DBG(DBG_proc, "%s(): start\n", __func__);
+ if (set == AFE_INIT)
+ {
+ DBG(DBG_proc, "%s(): setting DAC %u\n", __func__, dev->model->dac_type);
+
+ dev->frontend = dev->frontend_initial;
+
+ /* write them to analog frontend */
+ status = sanei_genesys_fe_write_data(dev, 0x00, dev->frontend.regs.get_value(0x00));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x00 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data(dev, 0x01, dev->frontend.regs.get_value(0x01));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x01 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ for (i = 0; i < 6; i++)
+ {
+ status =
+ sanei_genesys_fe_write_data (dev, 0x02 + i, 0x00);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing sign[%d] failed: %s\n", __func__, 0x02 + i,
+ sane_strstatus(status));
+ return status;
+ }
+ }
+ }
+ if (set == AFE_SET)
+ {
+ /* write them to analog frontend */
+ status = sanei_genesys_fe_write_data(dev, 0x00, dev->frontend.regs.get_value(0x00));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x00 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data(dev, 0x01, dev->frontend.regs.get_value(0x01));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg 0x01 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x02 (red gain)*/
+ status = sanei_genesys_fe_write_data(dev, 0x02, dev->frontend.get_gain(0));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing fe 0x02 (gain r) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x03 (green gain)*/
+ status = sanei_genesys_fe_write_data(dev, 0x03, dev->frontend.get_gain(1));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing fe 0x03 (gain g) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x04 (blue gain)*/
+ status = sanei_genesys_fe_write_data(dev, 0x04, dev->frontend.get_gain(2));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing fe 0x04 (gain b) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x05 (red offset)*/
+ status =
+ sanei_genesys_fe_write_data(dev, 0x05, dev->frontend.get_offset(0));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: write fe 0x05 (offset r) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x06 (green offset)*/
+ status =
+ sanei_genesys_fe_write_data(dev, 0x06, dev->frontend.get_offset(1));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: write fe 0x06 (offset g) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* Write fe 0x07 (blue offset)*/
+ status =
+ sanei_genesys_fe_write_data(dev, 0x07, dev->frontend.get_offset(2));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: write fe 0x07 (offset b) fail: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+ DBG(DBG_proc, "%s(): end\n", __func__);
+
+ return status;
+}
+
+/* Set values of analog frontend */
+static SANE_Status
+gl841_set_fe(Genesys_Device * dev, const Genesys_Sensor& sensor, uint8_t set)
+{
+ (void) sensor;
+ SANE_Status status = SANE_STATUS_GOOD;
+ int i;
+
+ DBG(DBG_proc, "%s (%s)\n", __func__,
+ set == AFE_INIT ? "init" : set == AFE_SET ? "set" : set ==
+ AFE_POWER_SAVE ? "powersave" : "huh?");
+
+ /* Analog Device type frontend */
+ if ((dev->reg.find_reg(0x04).value & REG04_FESET) == 0x02)
+ {
+ return gl841_set_ad_fe (dev, set);
+ }
+
+ if ((dev->reg.find_reg(0x04).value & REG04_FESET) != 0x00)
+ {
+ DBG(DBG_proc, "%s(): unsupported frontend type %d\n", __func__,
+ dev->reg.find_reg(0x04).value & REG04_FESET);
+ return SANE_STATUS_UNSUPPORTED;
+ }
+
+ if (set == AFE_INIT)
+ {
+ DBG(DBG_proc, "%s(): setting DAC %u\n", __func__, dev->model->dac_type);
+ dev->frontend = dev->frontend_initial;
+
+ /* reset only done on init */
+ status = sanei_genesys_fe_write_data (dev, 0x04, 0x80);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: reset fe failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ DBG(DBG_proc, "%s(): frontend reset complete\n", __func__);
+ }
+
+
+ if (set == AFE_POWER_SAVE)
+ {
+ status = sanei_genesys_fe_write_data (dev, 0x01, 0x02);
+ if (status != SANE_STATUS_GOOD) {
+ DBG(DBG_error, "%s: writing data failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ return status;
+ }
+
+ /* todo : base this test on cfg reg3 or a CCD family flag to be created */
+ /*if (dev->model->ccd_type!=CCD_HP2300 && dev->model->ccd_type!=CCD_HP2400) */
+ {
+
+ status = sanei_genesys_fe_write_data(dev, 0x00, dev->frontend.regs.get_value(0x00));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg0 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ status = sanei_genesys_fe_write_data(dev, 0x02, dev->frontend.regs.get_value(0x02));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg2 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ status = sanei_genesys_fe_write_data(dev, 0x01, dev->frontend.regs.get_value(0x01));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg1 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data(dev, 0x03, dev->frontend.regs.get_value(0x03));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg3 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data (dev, 0x06, dev->frontend.reg2[0]);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg6 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data (dev, 0x08, dev->frontend.reg2[1]);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg8 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = sanei_genesys_fe_write_data (dev, 0x09, dev->frontend.reg2[2]);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing reg9 failed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ for (i = 0; i < 3; i++)
+ {
+ status =
+ sanei_genesys_fe_write_data(dev, 0x24 + i, dev->frontend.regs.get_value(0x24 + i));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing sign[%d] failed: %s\n", __func__, i,
+ sane_strstatus(status));
+ return status;
+ }
+
+ status =
+ sanei_genesys_fe_write_data(dev, 0x28 + i, dev->frontend.get_gain(i));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing gain[%d] failed: %s\n", __func__, i,
+ sane_strstatus(status));
+ return status;
+ }
+
+ status =
+ sanei_genesys_fe_write_data(dev, 0x20 + i,
+ dev->frontend.get_offset(i));
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: writing offset[%d] failed: %s\n", __func__, i,
+ sane_strstatus(status));
+ return status;
+ }
+ }
+
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+
+ return SANE_STATUS_GOOD;
+}
+
+#define MOTOR_ACTION_FEED 1
+#define MOTOR_ACTION_GO_HOME 2
+#define MOTOR_ACTION_HOME_FREE 3
+
+/** @brief turn off motor
+ *
+ */
+static SANE_Status
+gl841_init_motor_regs_off(Genesys_Register_Set * reg,
+ unsigned int scan_lines)
+{
+ unsigned int feedl;
+ GenesysRegister* r;
+
+ DBG(DBG_proc, "%s : scan_lines=%d\n", __func__, scan_lines);
+
+ feedl = 2;
+
+ r = sanei_genesys_get_address (reg, 0x3d);
+ r->value = (feedl >> 16) & 0xf;
+ r = sanei_genesys_get_address (reg, 0x3e);
+ r->value = (feedl >> 8) & 0xff;
+ r = sanei_genesys_get_address (reg, 0x3f);
+ r->value = feedl & 0xff;
+ r = sanei_genesys_get_address (reg, 0x5e);
+ r->value &= ~0xe0;
+
+ r = sanei_genesys_get_address (reg, 0x25);
+ r->value = (scan_lines >> 16) & 0xf;
+ r = sanei_genesys_get_address (reg, 0x26);
+ r->value = (scan_lines >> 8) & 0xff;
+ r = sanei_genesys_get_address (reg, 0x27);
+ r->value = scan_lines & 0xff;
+
+ r = sanei_genesys_get_address (reg, 0x02);
+ r->value &= ~0x01; /*LONGCURV OFF*/
+ r->value &= ~0x80; /*NOT_HOME OFF*/
+
+ r->value &= ~0x10;
+
+ r->value &= ~0x06;
+
+ r->value &= ~0x08;
+
+ r->value &= ~0x20;
+
+ r->value &= ~0x40;
+
+ r = sanei_genesys_get_address (reg, 0x67);
+ r->value = 0x3f;
+
+ r = sanei_genesys_get_address (reg, 0x68);
+ r->value = 0x3f;
+
+ r = sanei_genesys_get_address (reg, REG_STEPNO);
+ r->value = 0;
+
+ r = sanei_genesys_get_address (reg, REG_FASTNO);
+ r->value = 0;
+
+ r = sanei_genesys_get_address (reg, 0x69);
+ r->value = 0;
+
+ r = sanei_genesys_get_address (reg, 0x6a);
+ r->value = 0;
+
+ r = sanei_genesys_get_address (reg, 0x5f);
+ r->value = 0;
+
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+/** @brief write motor table frequency
+ * Write motor frequency data table.
+ * @param dev device to set up motor
+ * @param ydpi motor target resolution
+ * @return SANE_STATUS_GOOD on success
+ */
+static SANE_Status gl841_write_freq(Genesys_Device *dev, unsigned int ydpi)
+{
+SANE_Status status = SANE_STATUS_GOOD;
+/**< fast table */
+uint8_t tdefault[] = {0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0x36,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xb6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0xf6,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76,0x18,0x76};
+uint8_t t1200[] = {0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc7,0x31,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc0,0x11,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0xc7,0xb1,0x07,0xe0,0x07,0xe0,0x07,0xe0,0x07,0xe0,0x07,0xe0,0x07,0xe0,0x07,0xe0,0x07,0xe0,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc7,0xf1,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc0,0x51,0xc7,0x71,0xc7,0x71,0xc7,0x71,0xc7,0x71,0xc7,0x71,0xc7,0x71,0xc7,0x71,0xc7,0x71,0x07,0x20,0x07,0x20,0x07,0x20,0x07,0x20,0x07,0x20,0x07,0x20,0x07,0x20,0x07,0x20};
+uint8_t t300[] = {0x08,0x32,0x08,0x32,0x08,0x32,0x08,0x32,0x08,0x32,0x08,0x32,0x08,0x32,0x08,0x32,0x00,0x13,0x00,0x13,0x00,0x13,0x00,0x13,0x00,0x13,0x00,0x13,0x00,0x13,0x00,0x13,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x08,0xb2,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x0c,0xa0,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x08,0xf2,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x00,0xd3,0x08,0x72,0x08,0x72,0x08,0x72,0x08,0x72,0x08,0x72,0x08,0x72,0x08,0x72,0x08,0x72,0x0c,0x60,0x0c,0x60,0x0c,0x60,0x0c,0x60,0x0c,0x60,0x0c,0x60,0x0c,0x60,0x0c,0x60};
+uint8_t t150[] = {0x0c,0x33,0xcf,0x33,0xcf,0x33,0xcf,0x33,0xcf,0x33,0xcf,0x33,0xcf,0x33,0xcf,0x33,0x40,0x14,0x80,0x15,0x80,0x15,0x80,0x15,0x80,0x15,0x80,0x15,0x80,0x15,0x80,0x15,0x0c,0xb3,0xcf,0xb3,0xcf,0xb3,0xcf,0xb3,0xcf,0xb3,0xcf,0xb3,0xcf,0xb3,0xcf,0xb3,0x11,0xa0,0x16,0xa0,0x16,0xa0,0x16,0xa0,0x16,0xa0,0x16,0xa0,0x16,0xa0,0x16,0xa0,0x0c,0xf3,0xcf,0xf3,0xcf,0xf3,0xcf,0xf3,0xcf,0xf3,0xcf,0xf3,0xcf,0xf3,0xcf,0xf3,0x40,0xd4,0x80,0xd5,0x80,0xd5,0x80,0xd5,0x80,0xd5,0x80,0xd5,0x80,0xd5,0x80,0xd5,0x0c,0x73,0xcf,0x73,0xcf,0x73,0xcf,0x73,0xcf,0x73,0xcf,0x73,0xcf,0x73,0xcf,0x73,0x11,0x60,0x16,0x60,0x16,0x60,0x16,0x60,0x16,0x60,0x16,0x60,0x16,0x60,0x16,0x60};
+
+uint8_t *table;
+
+ DBGSTART;
+ if(dev->model->motor_type == MOTOR_CANONLIDE80)
+ {
+ switch(ydpi)
+ {
+ case 3600:
+ case 1200:
+ table=t1200;
+ break;
+ case 900:
+ case 300:
+ table=t300;
+ break;
+ case 450:
+ case 150:
+ table=t150;
+ break;
+ default:
+ table=tdefault;
+ }
+ RIE(sanei_genesys_write_register(dev, 0x66, 0x00));
+ RIE(sanei_genesys_write_register(dev, 0x5b, 0x0c));
+ RIE(sanei_genesys_write_register(dev, 0x5c, 0x00));
+ RIE(sanei_genesys_bulk_write_data(dev, 0x28, table, 128));
+ RIE(sanei_genesys_write_register(dev, 0x5b, 0x00));
+ RIE(sanei_genesys_write_register(dev, 0x5c, 0x00));
+ }
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+
+static SANE_Status
+gl841_init_motor_regs(Genesys_Device * dev,
+ const Genesys_Sensor& sensor,
+ Genesys_Register_Set * reg,
+ unsigned int feed_steps,/*1/base_ydpi*/
+/*maybe float for half/quarter step resolution?*/
+ unsigned int action,
+ unsigned int flags)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ unsigned int fast_exposure;
+ int scan_power_mode;
+ int use_fast_fed = 0;
+ uint16_t fast_slope_table[256];
+ unsigned int fast_slope_steps = 0;
+ unsigned int feedl;
+ GenesysRegister* r;
+/*number of scan lines to add in a scan_lines line*/
+
+ DBG(DBG_proc, "%s : feed_steps=%d, action=%d, flags=%x\n", __func__, feed_steps, action, flags);
+
+ memset(fast_slope_table,0xff,512);
+
+ gl841_send_slope_table (dev, 0, fast_slope_table, 256);
+ gl841_send_slope_table (dev, 1, fast_slope_table, 256);
+ gl841_send_slope_table (dev, 2, fast_slope_table, 256);
+ gl841_send_slope_table (dev, 3, fast_slope_table, 256);
+ gl841_send_slope_table (dev, 4, fast_slope_table, 256);
+
+ gl841_write_freq(dev, dev->motor.base_ydpi / 4);
+
+ fast_slope_steps = 256;
+ if (action == MOTOR_ACTION_FEED || action == MOTOR_ACTION_GO_HOME)
+ {
+ /* FEED and GO_HOME can use fastest slopes available */
+ fast_exposure = gl841_exposure_time(dev, sensor,
+ dev->motor.base_ydpi / 4,
+ 0,
+ 0,
+ 0,
+ &scan_power_mode);
+ DBG(DBG_info, "%s : fast_exposure=%d pixels\n", __func__, fast_exposure);
+ }
+
+ if (action == MOTOR_ACTION_HOME_FREE) {
+/* HOME_FREE must be able to stop in one step, so do not try to get faster */
+ fast_exposure = dev->motor.slopes[0][0].maximum_start_speed;
+ }
+
+ sanei_genesys_create_slope_table3 (
+ dev,
+ fast_slope_table,
+ 256,
+ fast_slope_steps,
+ 0,
+ fast_exposure,
+ dev->motor.base_ydpi / 4,
+ &fast_slope_steps,
+ &fast_exposure, 0);
+
+ feedl = feed_steps - fast_slope_steps*2;
+ use_fast_fed = 1;
+
+/* all needed slopes available. we did even decide which mode to use.
+ what next?
+ - transfer slopes
+SCAN:
+flags \ use_fast_fed ! 0 1
+------------------------\--------------------
+ 0 ! 0,1,2 0,1,2,3
+MOTOR_FLAG_AUTO_GO_HOME ! 0,1,2,4 0,1,2,3,4
+OFF: none
+FEED: 3
+GO_HOME: 3
+HOME_FREE: 3
+ - setup registers
+ * slope specific registers (already done)
+ * DECSEL for HOME_FREE/GO_HOME/SCAN
+ * FEEDL
+ * MTRREV
+ * MTRPWR
+ * FASTFED
+ * STEPSEL
+ * MTRPWM
+ * FSTPSEL
+ * FASTPWM
+ * HOMENEG
+ * BWDSTEP
+ * FWDSTEP
+ * Z1
+ * Z2
+ */
+
+ r = sanei_genesys_get_address(reg, 0x3d);
+ r->value = (feedl >> 16) & 0xf;
+ r = sanei_genesys_get_address(reg, 0x3e);
+ r->value = (feedl >> 8) & 0xff;
+ r = sanei_genesys_get_address(reg, 0x3f);
+ r->value = feedl & 0xff;
+ r = sanei_genesys_get_address(reg, 0x5e);
+ r->value &= ~0xe0;
+
+ r = sanei_genesys_get_address(reg, 0x25);
+ r->value = 0;
+ r = sanei_genesys_get_address(reg, 0x26);
+ r->value = 0;
+ r = sanei_genesys_get_address(reg, 0x27);
+ r->value = 0;
+
+ r = sanei_genesys_get_address(reg, 0x02);
+ r->value &= ~0x01; /*LONGCURV OFF*/
+ r->value &= ~0x80; /*NOT_HOME OFF*/
+
+ r->value |= 0x10;
+
+ if (action == MOTOR_ACTION_GO_HOME)
+ r->value |= 0x06;
+ else
+ r->value &= ~0x06;
+
+ if (use_fast_fed)
+ r->value |= 0x08;
+ else
+ r->value &= ~0x08;
+
+ if (flags & MOTOR_FLAG_AUTO_GO_HOME)
+ r->value |= 0x20;
+ else
+ r->value &= ~0x20;
+
+ r->value &= ~0x40;
+
+ status = gl841_send_slope_table (dev, 3, fast_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+ r = sanei_genesys_get_address(reg, 0x67);
+ r->value = 0x3f;
+
+ r = sanei_genesys_get_address(reg, 0x68);
+ r->value = 0x3f;
+
+ r = sanei_genesys_get_address(reg, REG_STEPNO);
+ r->value = 0;
+
+ r = sanei_genesys_get_address(reg, REG_FASTNO);
+ r->value = 0;
+
+ r = sanei_genesys_get_address(reg, 0x69);
+ r->value = 0;
+
+ r = sanei_genesys_get_address(reg, 0x6a);
+ r->value = (fast_slope_steps >> 1) + (fast_slope_steps & 1);
+
+ r = sanei_genesys_get_address(reg, 0x5f);
+ r->value = (fast_slope_steps >> 1) + (fast_slope_steps & 1);
+
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+static SANE_Status
+gl841_init_motor_regs_scan(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set * reg,
+ unsigned int scan_exposure_time,/*pixel*/
+ float scan_yres,/*dpi, motor resolution*/
+ int scan_step_type,/*0: full, 1: half, 2: quarter*/
+ unsigned int scan_lines,/*lines, scan resolution*/
+ unsigned int scan_dummy,
+/*number of scan lines to add in a scan_lines line*/
+ unsigned int feed_steps,/*1/base_ydpi*/
+/*maybe float for half/quarter step resolution?*/
+ int scan_power_mode,
+ unsigned int flags)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ unsigned int fast_exposure;
+ int use_fast_fed = 0;
+ int dummy_power_mode;
+ unsigned int fast_time;
+ unsigned int slow_time;
+ uint16_t slow_slope_table[256];
+ uint16_t fast_slope_table[256];
+ uint16_t back_slope_table[256];
+ unsigned int slow_slope_time;
+ unsigned int fast_slope_time;
+ unsigned int slow_slope_steps = 0;
+ unsigned int fast_slope_steps = 0;
+ unsigned int back_slope_steps = 0;
+ unsigned int feedl;
+ GenesysRegister* r;
+ unsigned int min_restep = 0x20;
+ uint32_t z1, z2;
+
+ DBG(DBG_proc, "%s : scan_exposure_time=%d, scan_yres=%g, scan_step_type=%d, scan_lines=%d,"
+ " scan_dummy=%d, feed_steps=%d, scan_power_mode=%d, flags=%x\n", __func__,
+ scan_exposure_time,
+ scan_yres,
+ scan_step_type,
+ scan_lines,
+ scan_dummy,
+ feed_steps,
+ scan_power_mode,
+ flags);
+
+ fast_exposure = gl841_exposure_time(dev, sensor,
+ dev->motor.base_ydpi / 4,
+ 0,
+ 0,
+ 0,
+ &dummy_power_mode);
+
+ DBG(DBG_info, "%s : fast_exposure=%d pixels\n", __func__, fast_exposure);
+
+ memset(slow_slope_table,0xff,512);
+
+ gl841_send_slope_table (dev, 0, slow_slope_table, 256);
+ gl841_send_slope_table (dev, 1, slow_slope_table, 256);
+ gl841_send_slope_table (dev, 2, slow_slope_table, 256);
+ gl841_send_slope_table (dev, 3, slow_slope_table, 256);
+ gl841_send_slope_table (dev, 4, slow_slope_table, 256);
+
+ /* motor frequency table */
+ gl841_write_freq(dev, scan_yres);
+
+/*
+ we calculate both tables for SCAN. the fast slope step count depends on
+ how many steps we need for slow acceleration and how much steps we are
+ allowed to use.
+ */
+ slow_slope_time = sanei_genesys_create_slope_table3 (
+ dev,
+ slow_slope_table, 256,
+ 256,
+ scan_step_type,
+ scan_exposure_time,
+ scan_yres,
+ &slow_slope_steps,
+ NULL,
+ scan_power_mode);
+
+ sanei_genesys_create_slope_table3 (
+ dev,
+ back_slope_table, 256,
+ 256,
+ scan_step_type,
+ 0,
+ scan_yres,
+ &back_slope_steps,
+ NULL,
+ scan_power_mode);
+
+ if (feed_steps < (slow_slope_steps >> scan_step_type)) {
+ /*TODO: what should we do here?? go back to exposure calculation?*/
+ feed_steps = slow_slope_steps >> scan_step_type;
+ }
+
+ if (feed_steps > fast_slope_steps*2 -
+ (slow_slope_steps >> scan_step_type))
+ fast_slope_steps = 256;
+ else
+/* we need to shorten fast_slope_steps here. */
+ fast_slope_steps = (feed_steps -
+ (slow_slope_steps >> scan_step_type))/2;
+
+ DBG(DBG_info, "%s: Maximum allowed slope steps for fast slope: %d\n", __func__,
+ fast_slope_steps);
+
+ fast_slope_time = sanei_genesys_create_slope_table3 (
+ dev,
+ fast_slope_table, 256,
+ fast_slope_steps,
+ 0,
+ fast_exposure,
+ dev->motor.base_ydpi / 4,
+ &fast_slope_steps,
+ &fast_exposure,
+ scan_power_mode);
+
+ /* fast fed special cases handling */
+ if (dev->model->gpo_type == GPO_XP300
+ || dev->model->gpo_type == GPO_DP685)
+ {
+ /* quirk: looks like at least this scanner is unable to use
+ 2-feed mode */
+ use_fast_fed = 0;
+ }
+ else if (feed_steps < fast_slope_steps*2 + (slow_slope_steps >> scan_step_type)) {
+ use_fast_fed = 0;
+ DBG(DBG_info, "%s: feed too short, slow move forced.\n", __func__);
+ } else {
+/* for deciding whether we should use fast mode we need to check how long we
+ need for (fast)accelerating, moving, decelerating, (TODO: stopping?)
+ (slow)accelerating again versus (slow)accelerating and moving. we need
+ fast and slow tables here.
+*/
+/*NOTE: scan_exposure_time is per scan_yres*/
+/*NOTE: fast_exposure is per base_ydpi/4*/
+/*we use full steps as base unit here*/
+ fast_time =
+ fast_exposure / 4 *
+ (feed_steps - fast_slope_steps*2 -
+ (slow_slope_steps >> scan_step_type))
+ + fast_slope_time*2 + slow_slope_time;
+ slow_time =
+ (scan_exposure_time * scan_yres) / dev->motor.base_ydpi *
+ (feed_steps - (slow_slope_steps >> scan_step_type))
+ + slow_slope_time;
+
+ DBG(DBG_info, "%s: Time for slow move: %d\n", __func__, slow_time);
+ DBG(DBG_info, "%s: Time for fast move: %d\n", __func__, fast_time);
+
+ use_fast_fed = fast_time < slow_time;
+ }
+
+ if (use_fast_fed)
+ feedl = feed_steps - fast_slope_steps*2 -
+ (slow_slope_steps >> scan_step_type);
+ else
+ if ((feed_steps << scan_step_type) < slow_slope_steps)
+ feedl = 0;
+ else
+ feedl = (feed_steps << scan_step_type) - slow_slope_steps;
+ DBG(DBG_info, "%s: Decided to use %s mode\n", __func__, use_fast_fed?"fast feed":"slow feed");
+
+/* all needed slopes available. we did even decide which mode to use.
+ what next?
+ - transfer slopes
+SCAN:
+flags \ use_fast_fed ! 0 1
+------------------------\--------------------
+ 0 ! 0,1,2 0,1,2,3
+MOTOR_FLAG_AUTO_GO_HOME ! 0,1,2,4 0,1,2,3,4
+OFF: none
+FEED: 3
+GO_HOME: 3
+HOME_FREE: 3
+ - setup registers
+ * slope specific registers (already done)
+ * DECSEL for HOME_FREE/GO_HOME/SCAN
+ * FEEDL
+ * MTRREV
+ * MTRPWR
+ * FASTFED
+ * STEPSEL
+ * MTRPWM
+ * FSTPSEL
+ * FASTPWM
+ * HOMENEG
+ * BWDSTEP
+ * FWDSTEP
+ * Z1
+ * Z2
+ */
+
+ r = sanei_genesys_get_address (reg, 0x3d);
+ r->value = (feedl >> 16) & 0xf;
+ r = sanei_genesys_get_address (reg, 0x3e);
+ r->value = (feedl >> 8) & 0xff;
+ r = sanei_genesys_get_address (reg, 0x3f);
+ r->value = feedl & 0xff;
+ r = sanei_genesys_get_address (reg, 0x5e);
+ r->value &= ~0xe0;
+
+ r = sanei_genesys_get_address (reg, 0x25);
+ r->value = (scan_lines >> 16) & 0xf;
+ r = sanei_genesys_get_address (reg, 0x26);
+ r->value = (scan_lines >> 8) & 0xff;
+ r = sanei_genesys_get_address (reg, 0x27);
+ r->value = scan_lines & 0xff;
+
+ r = sanei_genesys_get_address (reg, 0x02);
+ r->value &= ~0x01; /*LONGCURV OFF*/
+ r->value &= ~0x80; /*NOT_HOME OFF*/
+ r->value |= 0x10;
+
+ r->value &= ~0x06;
+
+ if (use_fast_fed)
+ r->value |= 0x08;
+ else
+ r->value &= ~0x08;
+
+ if (flags & MOTOR_FLAG_AUTO_GO_HOME)
+ r->value |= 0x20;
+ else
+ r->value &= ~0x20;
+
+ if (flags & MOTOR_FLAG_DISABLE_BUFFER_FULL_MOVE)
+ r->value |= 0x40;
+ else
+ r->value &= ~0x40;
+
+ status = gl841_send_slope_table (dev, 0, slow_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+ status = gl841_send_slope_table (dev, 1, back_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+ status = gl841_send_slope_table (dev, 2, slow_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+ if (use_fast_fed) {
+ status = gl841_send_slope_table (dev, 3, fast_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+ }
+
+ if (flags & MOTOR_FLAG_AUTO_GO_HOME){
+ status = gl841_send_slope_table (dev, 4, fast_slope_table, 256);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+ }
+
+
+/* now reg 0x21 and 0x24 are available, we can calculate reg 0x22 and 0x23,
+ reg 0x60-0x62 and reg 0x63-0x65
+ rule:
+ 2*STEPNO+FWDSTEP=2*FASTNO+BWDSTEP
+*/
+/* steps of table 0*/
+ if (min_restep < slow_slope_steps*2+2)
+ min_restep = slow_slope_steps*2+2;
+/* steps of table 1*/
+ if (min_restep < back_slope_steps*2+2)
+ min_restep = back_slope_steps*2+2;
+/* steps of table 0*/
+ r = sanei_genesys_get_address (reg, REG_FWDSTEP);
+ r->value = min_restep - slow_slope_steps*2;
+/* steps of table 1*/
+ r = sanei_genesys_get_address (reg, REG_BWDSTEP);
+ r->value = min_restep - back_slope_steps*2;
+
+/*
+ for z1/z2:
+ in dokumentation mentioned variables a-d:
+ a = time needed for acceleration, table 1
+ b = time needed for reg 0x1f... wouldn't that be reg0x1f*exposure_time?
+ c = time needed for acceleration, table 1
+ d = time needed for reg 0x22... wouldn't that be reg0x22*exposure_time?
+ z1 = (c+d-1) % exposure_time
+ z2 = (a+b-1) % exposure_time
+*/
+/* i don't see any effect of this. i can only guess that this will enhance
+ sub-pixel accuracy
+ z1 = (slope_0_time-1) % exposure_time;
+ z2 = (slope_0_time-1) % exposure_time;
+*/
+ z1 = z2 = 0;
+
+ DBG(DBG_info, "%s: z1 = %d\n", __func__, z1);
+ DBG(DBG_info, "%s: z2 = %d\n", __func__, z2);
+ r = sanei_genesys_get_address (reg, 0x60);
+ r->value = ((z1 >> 16) & 0xff);
+ r = sanei_genesys_get_address (reg, 0x61);
+ r->value = ((z1 >> 8) & 0xff);
+ r = sanei_genesys_get_address (reg, 0x62);
+ r->value = (z1 & 0xff);
+ r = sanei_genesys_get_address (reg, 0x63);
+ r->value = ((z2 >> 16) & 0xff);
+ r = sanei_genesys_get_address (reg, 0x64);
+ r->value = ((z2 >> 8) & 0xff);
+ r = sanei_genesys_get_address (reg, 0x65);
+ r->value = (z2 & 0xff);
+
+ r = sanei_genesys_get_address (reg, REG1E);
+ r->value &= REG1E_WDTIME;
+ r->value |= scan_dummy;
+
+ r = sanei_genesys_get_address (reg, 0x67);
+ r->value = 0x3f | (scan_step_type << 6);
+
+ r = sanei_genesys_get_address (reg, 0x68);
+ r->value = 0x3f;
+
+ r = sanei_genesys_get_address (reg, REG_STEPNO);
+ r->value = (slow_slope_steps >> 1) + (slow_slope_steps & 1);
+
+ r = sanei_genesys_get_address (reg, REG_FASTNO);
+ r->value = (back_slope_steps >> 1) + (back_slope_steps & 1);
+
+ r = sanei_genesys_get_address (reg, 0x69);
+ r->value = (slow_slope_steps >> 1) + (slow_slope_steps & 1);
+
+ r = sanei_genesys_get_address (reg, 0x6a);
+ r->value = (fast_slope_steps >> 1) + (fast_slope_steps & 1);
+
+ r = sanei_genesys_get_address (reg, 0x5f);
+ r->value = (fast_slope_steps >> 1) + (fast_slope_steps & 1);
+
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+static int
+gl841_get_dpihw(Genesys_Device * dev)
+{
+ GenesysRegister* r;
+ r = sanei_genesys_get_address(&dev->reg, 0x05);
+ if ((r->value & REG05_DPIHW) == REG05_DPIHW_600)
+ return 600;
+ if ((r->value & REG05_DPIHW) == REG05_DPIHW_1200)
+ return 1200;
+ if ((r->value & REG05_DPIHW) == REG05_DPIHW_2400)
+ return 2400;
+ return 0;
+}
+
+static SANE_Status
+gl841_init_optical_regs_off(Genesys_Register_Set * reg)
+{
+ GenesysRegister* r;
+
+ DBGSTART;
+
+ r = sanei_genesys_get_address(reg, 0x01);
+ r->value &= ~REG01_SCAN;
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+static SANE_Status
+gl841_init_optical_regs_scan(Genesys_Device * dev,
+ const Genesys_Sensor& sensor,
+ Genesys_Register_Set * reg,
+ unsigned int exposure_time,
+ unsigned int used_res,
+ unsigned int start,
+ unsigned int pixels,
+ int channels,
+ int depth,
+ SANE_Bool half_ccd,
+ ColorFilter color_filter,
+ int flags
+ )
+{
+ unsigned int words_per_line;
+ unsigned int end;
+ unsigned int dpiset;
+ GenesysRegister* r;
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint16_t expavg, expr, expb, expg;
+
+ DBG(DBG_proc, "%s : exposure_time=%d, used_res=%d, start=%d, pixels=%d, channels=%d, depth=%d, "
+ "half_ccd=%d, flags=%x\n", __func__,
+ exposure_time,
+ used_res,
+ start,
+ pixels,
+ channels,
+ depth,
+ half_ccd,
+ flags);
+
+ end = start + pixels;
+
+ status = gl841_set_fe(dev, sensor, AFE_SET);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set frontend: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* adjust used_res for chosen dpihw */
+ used_res = used_res * gl841_get_dpihw(dev) / sensor.optical_res;
+
+/*
+ with half_ccd the optical resolution of the ccd is halved. We don't apply this
+ to dpihw, so we need to double dpiset.
+
+ For the scanner only the ratio of dpiset and dpihw is of relevance to scale
+ down properly.
+*/
+ if (half_ccd)
+ dpiset = used_res * 2;
+ else
+ dpiset = used_res;
+
+ /* gpio part.*/
+ if (dev->model->gpo_type == GPO_CANONLIDE35)
+ {
+ r = sanei_genesys_get_address (reg, REG6C);
+ if (half_ccd)
+ r->value &= ~0x80;
+ else
+ r->value |= 0x80;
+ }
+ if (dev->model->gpo_type == GPO_CANONLIDE80)
+ {
+ r = sanei_genesys_get_address (reg, REG6C);
+ if (half_ccd)
+ {
+ r->value &= ~0x40;
+ r->value |= 0x20;
+ }
+ else
+ {
+ r->value &= ~0x20;
+ r->value |= 0x40;
+ }
+ }
+
+ /* enable shading */
+ r = sanei_genesys_get_address (reg, 0x01);
+ r->value |= REG01_SCAN;
+ if ((flags & OPTICAL_FLAG_DISABLE_SHADING) ||
+ (dev->model->flags & GENESYS_FLAG_NO_CALIBRATION))
+ r->value &= ~REG01_DVDSET;
+ else
+ r->value |= REG01_DVDSET;
+
+ /* average looks better than deletion, and we are already set up to
+ use one of the average enabled resolutions
+ */
+ r = sanei_genesys_get_address (reg, 0x03);
+ r->value |= REG03_AVEENB;
+ sanei_genesys_set_lamp_power(dev, sensor, *reg, !(flags & OPTICAL_FLAG_DISABLE_LAMP));
+
+ /* BW threshold */
+ r = sanei_genesys_get_address (reg, 0x2e);
+ r->value = dev->settings.threshold;
+ r = sanei_genesys_get_address (reg, 0x2f);
+ r->value = dev->settings.threshold;
+
+
+ /* monochrome / color scan */
+ r = sanei_genesys_get_address (reg, 0x04);
+ switch (depth) {
+ case 1:
+ r->value &= ~REG04_BITSET;
+ r->value |= REG04_LINEART;
+ break;
+ case 8:
+ r->value &= ~(REG04_LINEART | REG04_BITSET);
+ break;
+ case 16:
+ r->value &= ~REG04_LINEART;
+ r->value |= REG04_BITSET;
+ break;
+ }
+
+ /* AFEMOD should depend on FESET, and we should set these
+ * bits separately */
+ r->value &= ~(REG04_FILTER | REG04_AFEMOD);
+ if (flags & OPTICAL_FLAG_ENABLE_LEDADD)
+ {
+ r->value |= 0x10; /* no filter */
+ }
+ else if (channels == 1)
+ {
+ switch (color_filter)
+ {
+ case ColorFilter::RED:
+ r->value |= 0x14;
+ break;
+ case ColorFilter::GREEN:
+ r->value |= 0x18;
+ break;
+ case ColorFilter::BLUE:
+ r->value |= 0x1c;
+ break;
+ default:
+ r->value |= 0x10;
+ break;
+ }
+ }
+ else
+ {
+ if (dev->model->ccd_type == CCD_PLUSTEK_3600)
+ {
+ r->value |= 0x22; /* slow color pixel by pixel */
+ }
+ else
+ {
+ r->value |= 0x10; /* color pixel by pixel */
+ }
+ }
+
+ /* CIS scanners can do true gray by setting LEDADD */
+ r = sanei_genesys_get_address (reg, 0x87);
+ r->value &= ~REG87_LEDADD;
+ if (flags & OPTICAL_FLAG_ENABLE_LEDADD)
+ {
+ r->value |= REG87_LEDADD;
+ sanei_genesys_get_double (reg, REG_EXPR, &expr);
+ sanei_genesys_get_double (reg, REG_EXPG, &expg);
+ sanei_genesys_get_double (reg, REG_EXPB, &expb);
+
+ /* use minimal exposure for best image quality */
+ expavg = expg;
+ if (expr < expg)
+ expavg = expr;
+ if (expb < expavg)
+ expavg = expb;
+
+ sanei_genesys_set_double(&dev->reg, REG_EXPR, expavg);
+ sanei_genesys_set_double(&dev->reg, REG_EXPG, expavg);
+ sanei_genesys_set_double(&dev->reg, REG_EXPB, expavg);
+ }
+
+ /* enable gamma tables */
+ r = sanei_genesys_get_address (reg, 0x05);
+ if (flags & OPTICAL_FLAG_DISABLE_GAMMA)
+ r->value &= ~REG05_GMMENB;
+ else
+ r->value |= REG05_GMMENB;
+
+ /* sensor parameters */
+ sanei_gl841_setup_sensor(dev, sensor, &dev->reg, 1, half_ccd);
+
+ r = sanei_genesys_get_address (reg, 0x29);
+ r->value = 255; /*<<<"magic" number, only suitable for cis*/
+
+ sanei_genesys_set_double(reg, REG_DPISET, dpiset);
+ sanei_genesys_set_double(reg, REG_STRPIXEL, start);
+ sanei_genesys_set_double(reg, REG_ENDPIXEL, end);
+ DBG(DBG_io2, "%s: STRPIXEL=%d, ENDPIXEL=%d\n", __func__, start, end);
+
+ /* words(16bit) before gamma, conversion to 8 bit or lineart*/
+ words_per_line = (pixels * dpiset) / gl841_get_dpihw(dev);
+
+ words_per_line *= channels;
+
+ if (depth == 1)
+ words_per_line = (words_per_line >> 3) + ((words_per_line & 7)?1:0);
+ else
+ words_per_line *= depth / 8;
+
+ dev->wpl = words_per_line;
+ dev->bpl = words_per_line;
+
+ r = sanei_genesys_get_address (reg, 0x35);
+ r->value = LOBYTE (HIWORD (words_per_line));
+ r = sanei_genesys_get_address (reg, 0x36);
+ r->value = HIBYTE (LOWORD (words_per_line));
+ r = sanei_genesys_get_address (reg, 0x37);
+ r->value = LOBYTE (LOWORD (words_per_line));
+
+ sanei_genesys_set_double(reg, REG_LPERIOD, exposure_time);
+
+ r = sanei_genesys_get_address (reg, 0x34);
+ r->value = sensor.dummy_pixel;
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+static int
+gl841_get_led_exposure(Genesys_Device * dev, const Genesys_Sensor& sensor)
+{
+ int d,r,g,b,m;
+ if (!dev->model->is_cis)
+ return 0;
+ d = dev->reg.find_reg(0x19).value;
+
+ r = sensor.exposure.red;
+ g = sensor.exposure.green;
+ b = sensor.exposure.blue;
+
+ m = r;
+ if (m < g)
+ m = g;
+ if (m < b)
+ m = b;
+
+ return m + d;
+}
+
+/** @brief compute exposure time
+ * Compute exposure time for the device and the given scan resolution,
+ * also compute scan_power_mode
+ */
+static int
+gl841_exposure_time(Genesys_Device *dev, const Genesys_Sensor& sensor,
+ float slope_dpi,
+ int scan_step_type,
+ int start,
+ int used_pixels,
+ int *scan_power_mode)
+{
+int exposure_time = 0;
+int exposure_time2 = 0;
+int led_exposure;
+
+ *scan_power_mode=0;
+ led_exposure=gl841_get_led_exposure(dev, sensor);
+ exposure_time = sanei_genesys_exposure_time2(
+ dev,
+ slope_dpi,
+ scan_step_type,
+ start+used_pixels,/*+tgtime? currently done in sanei_genesys_exposure_time2 with tgtime = 32 pixel*/
+ led_exposure,
+ *scan_power_mode);
+
+ while(*scan_power_mode + 1 < dev->motor.power_mode_count) {
+ exposure_time2 = sanei_genesys_exposure_time2(
+ dev,
+ slope_dpi,
+ scan_step_type,
+ start+used_pixels,/*+tgtime? currently done in sanei_genesys_exposure_time2 with tgtime = 32 pixel*/
+ led_exposure,
+ *scan_power_mode + 1);
+ if (exposure_time < exposure_time2)
+ break;
+ exposure_time = exposure_time2;
+ (*scan_power_mode)++;
+ }
+
+ return exposure_time;
+}
+
+/**@brief compute scan_step_type
+ * Try to do at least 4 steps per line. if that is impossible we will have to
+ * live with that.
+ * @param dev device
+ * @param yres motor resolution
+ */
+static int
+gl841_scan_step_type(Genesys_Device *dev, int yres)
+{
+int scan_step_type=0;
+
+ /* TODO : check if there is a bug around the use of max_step_type */
+ /* should be <=1, need to chek all devices entry in genesys_devices */
+ if (yres*4 < dev->motor.base_ydpi || dev->motor.max_step_type <= 0)
+ {
+ scan_step_type = 0;
+ }
+ else if (yres*4 < dev->motor.base_ydpi*2 || dev->motor.max_step_type <= 1)
+ {
+ scan_step_type = 1;
+ }
+ else
+ {
+ scan_step_type = 2;
+ }
+
+ /* this motor behaves differently */
+ if (dev->model->motor_type==MOTOR_CANONLIDE80)
+ {
+ /* driven by 'frequency' tables ? */
+ scan_step_type = 0;
+ }
+
+ return scan_step_type;
+}
+
+/* set up registers for an actual scan
+ *
+ * this function sets up the scanner to scan in normal or single line mode
+ */
+static
+SANE_Status
+gl841_init_scan_regs(Genesys_Device * dev, const Genesys_Sensor& sensor, Genesys_Register_Set * reg,
+ SetupParams& params)
+{
+ params.assert_valid();
+
+ int used_res;
+ int start, used_pixels;
+ int bytes_per_line;
+ int move;
+ unsigned int lincnt;
+ int exposure_time;
+ int scan_power_mode;
+ int i;
+ int stagger;
+ int avg;
+
+ int slope_dpi = 0;
+ int dummy = 0;
+ int scan_step_type = 1;
+ int max_shift;
+ size_t requested_buffer_size, read_buffer_size;
+
+ SANE_Bool half_ccd; /* false: full CCD res is used, true, half max CCD res is used */
+ int optical_res;
+ SANE_Status status = SANE_STATUS_GOOD;
+ unsigned int oflags; /**> optical flags */
+
+ DBG(DBG_info, "%s ", __func__);
+ debug_dump(DBG_info, params);
+
+/*
+results:
+
+for scanner:
+half_ccd
+start
+end
+dpiset
+exposure_time
+dummy
+z1
+z2
+
+for ordered_read:
+ dev->words_per_line
+ dev->read_factor
+ dev->requested_buffer_size
+ dev->read_buffer_size
+ dev->read_pos
+ dev->read_bytes_in_buffer
+ dev->read_bytes_left
+ dev->max_shift
+ dev->stagger
+
+independent of our calculated values:
+ dev->total_bytes_read
+ dev->bytes_to_read
+ */
+
+/* half_ccd */
+ /* we have 2 domains for ccd: xres below or above half ccd max dpi */
+ if (sensor.get_ccd_size_divisor_for_dpi(params.xres) > 1) {
+ half_ccd = SANE_TRUE;
+ } else {
+ half_ccd = SANE_FALSE;
+ }
+
+/* optical_res */
+
+ optical_res = sensor.optical_res;
+ if (half_ccd)
+ optical_res /= 2;
+
+/* stagger */
+
+ if ((!half_ccd) && (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE))
+ stagger = (4 * params.yres) / dev->motor.base_ydpi;
+ else
+ stagger = 0;
+ DBG(DBG_info, "%s : stagger=%d lines\n", __func__, stagger);
+
+/* used_res */
+ i = optical_res / params.xres;
+
+/* gl841 supports 1/1 1/2 1/3 1/4 1/5 1/6 1/8 1/10 1/12 1/15 averaging */
+
+ if (i < 2 || (params.flags & SCAN_FLAG_USE_OPTICAL_RES)) /* optical_res >= xres > optical_res/2 */
+ used_res = optical_res;
+ else if (i < 3) /* optical_res/2 >= xres > optical_res/3 */
+ used_res = optical_res/2;
+ else if (i < 4) /* optical_res/3 >= xres > optical_res/4 */
+ used_res = optical_res/3;
+ else if (i < 5) /* optical_res/4 >= xres > optical_res/5 */
+ used_res = optical_res/4;
+ else if (i < 6) /* optical_res/5 >= xres > optical_res/6 */
+ used_res = optical_res/5;
+ else if (i < 8) /* optical_res/6 >= xres > optical_res/8 */
+ used_res = optical_res/6;
+ else if (i < 10) /* optical_res/8 >= xres > optical_res/10 */
+ used_res = optical_res/8;
+ else if (i < 12) /* optical_res/10 >= xres > optical_res/12 */
+ used_res = optical_res/10;
+ else if (i < 15) /* optical_res/12 >= xres > optical_res/15 */
+ used_res = optical_res/12;
+ else
+ used_res = optical_res/15;
+
+ /* compute scan parameters values */
+ /* pixels are allways given at half or full CCD optical resolution */
+ /* use detected left margin and fixed value */
+ /* start */
+ /* add x coordinates */
+ start = ((sensor.CCD_start_xoffset + params.startx) * used_res) / sensor.optical_res;
+
+ /* needs to be aligned for used_res */
+ start = (start * optical_res) / used_res;
+
+ start += sensor.dummy_pixel + 1;
+
+ if (stagger > 0)
+ start |= 1;
+
+ /* in case of SHDAREA, we need to align start
+ * on pixel average factor, startx is different of
+ * 0 only when calling for function to setup for
+ * scan, where shading data needs to be align */
+ if((dev->reg.find_reg(0x01).value & REG01_SHDAREA) != 0)
+ {
+ avg=optical_res/used_res;
+ start=(start/avg)*avg;
+ }
+
+ /* compute correct pixels number */
+ /* pixels */
+ used_pixels = (params.pixels * optical_res) / params.xres;
+
+ /* round up pixels number if needed */
+ if (used_pixels * params.xres < params.pixels * optical_res)
+ used_pixels++;
+
+/* dummy */
+ /* dummy lines: may not be usefull, for instance 250 dpi works with 0 or 1
+ dummy line. Maybe the dummy line adds correctness since the motor runs
+ slower (higher dpi)
+ */
+/* for cis this creates better aligned color lines:
+dummy \ scanned lines
+ 0: R G B R ...
+ 1: R G B - R ...
+ 2: R G B - - R ...
+ 3: R G B - - - R ...
+ 4: R G B - - - - R ...
+ 5: R G B - - - - - R ...
+ 6: R G B - - - - - - R ...
+ 7: R G B - - - - - - - R ...
+ 8: R G B - - - - - - - - R ...
+ 9: R G B - - - - - - - - - R ...
+ 10: R G B - - - - - - - - - - R ...
+ 11: R G B - - - - - - - - - - - R ...
+ 12: R G B - - - - - - - - - - - - R ...
+ 13: R G B - - - - - - - - - - - - - R ...
+ 14: R G B - - - - - - - - - - - - - - R ...
+ 15: R G B - - - - - - - - - - - - - - - R ...
+ -- pierre
+ */
+ dummy = 0;
+
+/* slope_dpi */
+/* cis color scan is effectively a gray scan with 3 gray lines per color
+ line and a FILTER of 0 */
+ if (dev->model->is_cis)
+ slope_dpi = params.yres* params.channels;
+ else
+ slope_dpi = params.yres;
+
+ slope_dpi = slope_dpi * (1 + dummy);
+
+ scan_step_type = gl841_scan_step_type(dev, params.yres);
+ exposure_time = gl841_exposure_time(dev, sensor,
+ slope_dpi,
+ scan_step_type,
+ start,
+ used_pixels,
+ &scan_power_mode);
+ DBG(DBG_info, "%s : exposure_time=%d pixels\n", __func__, exposure_time);
+
+ /*** optical parameters ***/
+ /* in case of dynamic lineart, we use an internal 8 bit gray scan
+ * to generate 1 lineart data */
+ if(params.flags & SCAN_FLAG_DYNAMIC_LINEART)
+ {
+ params.depth=8;
+ }
+
+ oflags=0;
+ if (params.flags & SCAN_FLAG_DISABLE_SHADING)
+ {
+ oflags |= OPTICAL_FLAG_DISABLE_SHADING;
+ }
+ if ((params.flags & SCAN_FLAG_DISABLE_GAMMA) || (params.depth==16))
+ {
+ oflags |= OPTICAL_FLAG_DISABLE_GAMMA;
+ }
+ if (params.flags & SCAN_FLAG_DISABLE_LAMP)
+ {
+ oflags |= OPTICAL_FLAG_DISABLE_LAMP;
+ }
+ if (params.flags & SCAN_FLAG_ENABLE_LEDADD)
+ {
+ oflags |= OPTICAL_FLAG_ENABLE_LEDADD;
+ }
+
+ status = gl841_init_optical_regs_scan(dev, sensor,
+ reg,
+ exposure_time,
+ used_res,
+ start,
+ used_pixels,
+ params.channels,
+ params.depth,
+ half_ccd,
+ params.color_filter,
+ oflags);
+ if (status != SANE_STATUS_GOOD)
+ {
+ return status;
+ }
+
+/*** motor parameters ***/
+
+ /* scanned area must be enlarged by max color shift needed */
+ max_shift=sanei_genesys_compute_max_shift(dev, params.channels,params.yres,params.flags);
+
+ /* lincnt */
+ lincnt = params.lines + max_shift + stagger;
+
+ /* add tl_y to base movement */
+ move = params.starty;
+ DBG(DBG_info, "%s: move=%d steps\n", __func__, move);
+
+ /* subtract current head position */
+ move -= dev->scanhead_position_in_steps;
+ DBG(DBG_info, "%s: move=%d steps\n", __func__, move);
+
+ if (move < 0)
+ move = 0;
+
+ /* round it */
+/* the move is not affected by dummy -- pierre */
+/* move = ((move + dummy) / (dummy + 1)) * (dummy + 1);
+ DBG(DBG_info, "%s: move=%d steps\n", __func__, move);*/
+
+ if (params.flags & SCAN_FLAG_SINGLE_LINE)
+ status = gl841_init_motor_regs_off(reg, dev->model->is_cis?lincnt* params.channels:lincnt);
+ else
+ status = gl841_init_motor_regs_scan(dev, sensor,
+ reg,
+ exposure_time,
+ slope_dpi,
+ scan_step_type,
+ dev->model->is_cis?lincnt* params.channels:lincnt,
+ dummy,
+ move,
+ scan_power_mode,
+ (params.flags & SCAN_FLAG_DISABLE_BUFFER_FULL_MOVE)?
+ MOTOR_FLAG_DISABLE_BUFFER_FULL_MOVE:0
+ );
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+
+ /*** prepares data reordering ***/
+
+/* words_per_line */
+ bytes_per_line = (used_pixels * used_res) / optical_res;
+ bytes_per_line = (bytes_per_line * params.channels * params.depth) / 8;
+
+ requested_buffer_size = 8 * bytes_per_line;
+ /* we must use a round number of bytes_per_line */
+ if (requested_buffer_size > sanei_genesys_get_bulk_max_size(dev))
+ requested_buffer_size =
+ (sanei_genesys_get_bulk_max_size(dev) / bytes_per_line) * bytes_per_line;
+
+ read_buffer_size =
+ 2 * requested_buffer_size +
+ ((max_shift + stagger) * used_pixels * params.channels * params.depth) / 8;
+
+ dev->read_buffer.clear();
+ dev->read_buffer.alloc(read_buffer_size);
+
+ dev->lines_buffer.clear();
+ dev->lines_buffer.alloc(read_buffer_size);
+
+ dev->shrink_buffer.clear();
+ dev->shrink_buffer.alloc(requested_buffer_size);
+
+ dev->out_buffer.clear();
+ dev->out_buffer.alloc((8 * dev->settings.pixels * params.channels * params.depth) / 8);
+
+ dev->read_bytes_left = bytes_per_line * lincnt;
+
+ DBG(DBG_info, "%s: physical bytes to read = %lu\n", __func__, (u_long) dev->read_bytes_left);
+ dev->read_active = SANE_TRUE;
+
+ dev->current_setup.params = params;
+ dev->current_setup.pixels = (used_pixels * used_res)/optical_res;
+ dev->current_setup.lines = lincnt;
+ dev->current_setup.depth = params.depth;
+ dev->current_setup.channels = params.channels;
+ dev->current_setup.exposure_time = exposure_time;
+ dev->current_setup.xres = used_res;
+ dev->current_setup.yres = params.yres;
+ dev->current_setup.ccd_size_divisor = half_ccd ? 2 : 1;
+ dev->current_setup.stagger = stagger;
+ dev->current_setup.max_shift = max_shift + stagger;
+
+/* TODO: should this be done elsewhere? */
+ /* scan bytes to send to the frontend */
+ /* theory :
+ target_size =
+ (dev->settings.pixels * dev->settings.lines * channels * depth) / 8;
+ but it suffers from integer overflow so we do the following:
+
+ 1 bit color images store color data byte-wise, eg byte 0 contains
+ 8 bits of red data, byte 1 contains 8 bits of green, byte 2 contains
+ 8 bits of blue.
+ This does not fix the overflow, though.
+ 644mp*16 = 10gp, leading to an overflow
+ -- pierre
+ */
+
+ dev->total_bytes_read = 0;
+ if (params.depth == 1)
+ dev->total_bytes_to_read =
+ ((dev->settings.pixels * dev->settings.lines) / 8 +
+ (((dev->settings.pixels * dev->settings.lines)%8)?1:0)
+ ) * params.channels;
+ else
+ dev->total_bytes_to_read =
+ dev->settings.pixels * dev->settings.lines * params.channels * (params.depth / 8);
+
+ DBG(DBG_info, "%s: total bytes to send = %lu\n", __func__, (u_long) dev->total_bytes_to_read);
+/* END TODO */
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+static void gl841_calculate_current_setup(Genesys_Device * dev, const Genesys_Sensor& sensor)
+{
+ int channels;
+ int depth;
+ int start;
+
+ int used_res;
+ int used_pixels;
+ unsigned int lincnt;
+ int exposure_time;
+ int scan_power_mode;
+ int i;
+ int stagger;
+
+ int slope_dpi = 0;
+ int dummy = 0;
+ int scan_step_type = 1;
+ int max_shift;
+
+ SANE_Bool half_ccd; /* false: full CCD res is used, true, half max CCD res is used */
+ int optical_res;
+
+ DBG(DBG_info, "%s ", __func__);
+ debug_dump(DBG_info, dev->settings);
+
+/* channels */
+ if (dev->settings.scan_mode == ScanColorMode::COLOR_SINGLE_PASS)
+ channels = 3;
+ else
+ channels = 1;
+
+/* depth */
+ depth = dev->settings.depth;
+ if (dev->settings.scan_mode == ScanColorMode::LINEART)
+ depth = 1;
+
+/* start */
+ start = SANE_UNFIX (dev->model->x_offset);
+
+ start += dev->settings.tl_x;
+
+ start = (start * sensor.optical_res) / MM_PER_INCH;
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = start;
+ params.starty = 0; // not used
+ params.pixels = dev->settings.pixels;
+ params.lines = dev->settings.lines;
+ params.depth = depth;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = dev->settings.scan_mode;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = 0;
+
+ DBG(DBG_info, "%s ", __func__);
+ debug_dump(DBG_info, params);
+
+/* half_ccd */
+ /* we have 2 domains for ccd: xres below or above half ccd max dpi */
+ if (sensor.get_ccd_size_divisor_for_dpi(params.xres) > 1) {
+ half_ccd = SANE_TRUE;
+ } else {
+ half_ccd = SANE_FALSE;
+ }
+
+/* optical_res */
+
+ optical_res = sensor.optical_res;
+ if (half_ccd)
+ optical_res /= 2;
+
+/* stagger */
+
+ if ((!half_ccd) && (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE))
+ stagger = (4 * params.yres) / dev->motor.base_ydpi;
+ else
+ stagger = 0;
+ DBG(DBG_info, "%s: stagger=%d lines\n", __func__, stagger);
+
+/* used_res */
+ i = optical_res / params.xres;
+
+/* gl841 supports 1/1 1/2 1/3 1/4 1/5 1/6 1/8 1/10 1/12 1/15 averaging */
+
+ if (i < 2) /* optical_res >= xres > optical_res/2 */
+ used_res = optical_res;
+ else if (i < 3) /* optical_res/2 >= xres > optical_res/3 */
+ used_res = optical_res/2;
+ else if (i < 4) /* optical_res/3 >= xres > optical_res/4 */
+ used_res = optical_res/3;
+ else if (i < 5) /* optical_res/4 >= xres > optical_res/5 */
+ used_res = optical_res/4;
+ else if (i < 6) /* optical_res/5 >= xres > optical_res/6 */
+ used_res = optical_res/5;
+ else if (i < 8) /* optical_res/6 >= xres > optical_res/8 */
+ used_res = optical_res/6;
+ else if (i < 10) /* optical_res/8 >= xres > optical_res/10 */
+ used_res = optical_res/8;
+ else if (i < 12) /* optical_res/10 >= xres > optical_res/12 */
+ used_res = optical_res/10;
+ else if (i < 15) /* optical_res/12 >= xres > optical_res/15 */
+ used_res = optical_res/12;
+ else
+ used_res = optical_res/15;
+
+ /* compute scan parameters values */
+ /* pixels are allways given at half or full CCD optical resolution */
+ /* use detected left margin and fixed value */
+ start = ((sensor.CCD_start_xoffset + params.startx) * used_res) / sensor.optical_res;
+
+/* needs to be aligned for used_res */
+ start = (start * optical_res) / used_res;
+
+ start += sensor.dummy_pixel + 1;
+
+ if (stagger > 0) {
+ start |= 1;
+ }
+
+ used_pixels = (params.pixels * optical_res) / params.xres;
+
+ // round up pixels number if needed
+ if (used_pixels * params.xres < params.pixels * optical_res) {
+ used_pixels++;
+ }
+
+ /* dummy lines: may not be usefull, for instance 250 dpi works with 0 or 1
+ dummy line. Maybe the dummy line adds correctness since the motor runs
+ slower (higher dpi)
+ */
+/* for cis this creates better aligned color lines:
+dummy \ scanned lines
+ 0: R G B R ...
+ 1: R G B - R ...
+ 2: R G B - - R ...
+ 3: R G B - - - R ...
+ 4: R G B - - - - R ...
+ 5: R G B - - - - - R ...
+ 6: R G B - - - - - - R ...
+ 7: R G B - - - - - - - R ...
+ 8: R G B - - - - - - - - R ...
+ 9: R G B - - - - - - - - - R ...
+ 10: R G B - - - - - - - - - - R ...
+ 11: R G B - - - - - - - - - - - R ...
+ 12: R G B - - - - - - - - - - - - R ...
+ 13: R G B - - - - - - - - - - - - - R ...
+ 14: R G B - - - - - - - - - - - - - - R ...
+ 15: R G B - - - - - - - - - - - - - - - R ...
+ -- pierre
+ */
+ dummy = 0;
+
+/* cis color scan is effectively a gray scan with 3 gray lines per color
+ line and a FILTER of 0 */
+ if (dev->model->is_cis) {
+ slope_dpi = params.yres * params.channels;
+ } else {
+ slope_dpi = params.yres;
+ }
+
+ slope_dpi = slope_dpi * (1 + dummy);
+
+ scan_step_type = gl841_scan_step_type(dev, params.yres);
+ exposure_time = gl841_exposure_time(dev, sensor,
+ slope_dpi,
+ scan_step_type,
+ start,
+ used_pixels,
+ &scan_power_mode);
+ DBG(DBG_info, "%s : exposure_time=%d pixels\n", __func__, exposure_time);
+
+ /* scanned area must be enlarged by max color shift needed */
+ max_shift = sanei_genesys_compute_max_shift(dev, params.channels, params.yres, 0);
+
+ lincnt = params.lines + max_shift + stagger;
+
+ dev->current_setup.params = params;
+ dev->current_setup.pixels = (used_pixels * used_res)/optical_res;
+ dev->current_setup.lines = lincnt;
+ dev->current_setup.depth = params.depth;
+ dev->current_setup.channels = params.channels;
+ dev->current_setup.exposure_time = exposure_time;
+ dev->current_setup.xres = used_res;
+ dev->current_setup.yres = params.yres;
+ dev->current_setup.ccd_size_divisor = half_ccd ? 2 : 1;
+ dev->current_setup.stagger = stagger;
+ dev->current_setup.max_shift = max_shift + stagger;
+
+ DBGCOMPLETED;
+}
+
+/*for fast power saving methods only, like disabling certain amplifiers*/
+static SANE_Status gl841_save_power(Genesys_Device * dev, SANE_Bool enable)
+{
+ uint8_t val;
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+
+ DBG(DBG_proc, "%s: enable = %d\n", __func__, enable);
+
+ if (enable)
+ {
+ if (dev->model->gpo_type == GPO_CANONLIDE35)
+ {
+/* expect GPIO17 to be enabled, and GPIO9 to be disabled,
+ while GPIO8 is disabled*/
+/* final state: GPIO8 disabled, GPIO9 enabled, GPIO17 disabled,
+ GPIO18 disabled*/
+
+ sanei_genesys_read_register(dev, REG6D, &val);
+ sanei_genesys_write_register(dev, REG6D, val | 0x80);
+
+ sanei_genesys_sleep_ms(1);
+
+ /*enable GPIO9*/
+ sanei_genesys_read_register(dev, REG6C, &val);
+ sanei_genesys_write_register(dev, REG6C, val | 0x01);
+
+ /*disable GPO17*/
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val & ~REG6B_GPO17);
+
+ /*disable GPO18*/
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val & ~REG6B_GPO18);
+
+ sanei_genesys_sleep_ms(1);
+
+ sanei_genesys_read_register(dev, REG6D, &val);
+ sanei_genesys_write_register(dev, REG6D, val & ~0x80);
+
+ }
+ if (dev->model->gpo_type == GPO_DP685)
+ {
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val & ~REG6B_GPO17);
+ dev->reg.find_reg(0x6b).value &= ~REG6B_GPO17;
+ dev->calib_reg.find_reg(0x6b).value &= ~REG6B_GPO17;
+ }
+
+ gl841_set_fe(dev, sensor, AFE_POWER_SAVE);
+
+ }
+ else
+ {
+ if (dev->model->gpo_type == GPO_CANONLIDE35)
+ {
+/* expect GPIO17 to be enabled, and GPIO9 to be disabled,
+ while GPIO8 is disabled*/
+/* final state: GPIO8 enabled, GPIO9 disabled, GPIO17 enabled,
+ GPIO18 enabled*/
+
+ sanei_genesys_read_register(dev, REG6D, &val);
+ sanei_genesys_write_register(dev, REG6D, val | 0x80);
+
+ sanei_genesys_sleep_ms(10);
+
+ /*disable GPIO9*/
+ sanei_genesys_read_register(dev, REG6C, &val);
+ sanei_genesys_write_register(dev, REG6C, val & ~0x01);
+
+ /*enable GPIO10*/
+ sanei_genesys_read_register(dev, REG6C, &val);
+ sanei_genesys_write_register(dev, REG6C, val | 0x02);
+
+ /*enable GPO17*/
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val | REG6B_GPO17);
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO17;
+ dev->calib_reg.find_reg(0x6b).value |= REG6B_GPO17;
+
+ /*enable GPO18*/
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val | REG6B_GPO18);
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO18;
+ dev->calib_reg.find_reg(0x6b).value |= REG6B_GPO18;
+
+ }
+ if (dev->model->gpo_type == GPO_DP665
+ || dev->model->gpo_type == GPO_DP685)
+ {
+ sanei_genesys_read_register(dev, REG6B, &val);
+ sanei_genesys_write_register(dev, REG6B, val | REG6B_GPO17);
+ dev->reg.find_reg(0x6b).value |= REG6B_GPO17;
+ dev->calib_reg.find_reg(0x6b).value |= REG6B_GPO17;
+ }
+
+ }
+
+ return SANE_STATUS_GOOD;
+}
+
+static SANE_Status
+gl841_set_powersaving (Genesys_Device * dev,
+ int delay /* in minutes */ )
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ // FIXME: SEQUENTIAL not really needed in this case
+ Genesys_Register_Set local_reg(Genesys_Register_Set::SEQUENTIAL);
+ int rate, exposure_time, tgtime, time;
+
+ DBG(DBG_proc, "%s (delay = %d)\n", __func__, delay);
+
+ local_reg.init_reg(0x01, dev->reg.get8(0x01)); /* disable fastmode */
+ local_reg.init_reg(0x03, dev->reg.get8(0x03)); /* Lamp power control */
+ local_reg.init_reg(0x05, dev->reg.get8(0x05)); /*& ~REG05_BASESEL*/; /* 24 clocks/pixel */
+ local_reg.init_reg(0x18, 0x00); // Set CCD type
+ local_reg.init_reg(0x38, 0x00);
+ local_reg.init_reg(0x39, 0x00);
+
+ // period times for LPeriod, expR,expG,expB, Z1MODE, Z2MODE
+ local_reg.init_reg(0x1c, dev->reg.get8(0x05) & ~REG1C_TGTIME);
+
+ if (!delay) {
+ local_reg.find_reg(0x03).value = local_reg.find_reg(0x03).value & 0xf0; /* disable lampdog and set lamptime = 0 */
+ } else if (delay < 20) {
+ local_reg.find_reg(0x03).value = (local_reg.find_reg(0x03).value & 0xf0) | 0x09; /* enable lampdog and set lamptime = 1 */
+ } else {
+ local_reg.find_reg(0x03).value = (local_reg.find_reg(0x03).value & 0xf0) | 0x0f; /* enable lampdog and set lamptime = 7 */
+ }
+
+ time = delay * 1000 * 60; /* -> msec */
+ exposure_time =
+ (uint32_t) (time * 32000.0 /
+ (24.0 * 64.0 * (local_reg.find_reg(0x03).value & REG03_LAMPTIM) *
+ 1024.0) + 0.5);
+ /* 32000 = system clock, 24 = clocks per pixel */
+ rate = (exposure_time + 65536) / 65536;
+ if (rate > 4)
+ {
+ rate = 8;
+ tgtime = 3;
+ }
+ else if (rate > 2)
+ {
+ rate = 4;
+ tgtime = 2;
+ }
+ else if (rate > 1)
+ {
+ rate = 2;
+ tgtime = 1;
+ }
+ else
+ {
+ rate = 1;
+ tgtime = 0;
+ }
+
+ local_reg.find_reg(0x1c).value |= tgtime;
+ exposure_time /= rate;
+
+ if (exposure_time > 65535)
+ exposure_time = 65535;
+
+ local_reg.set8(0x38, exposure_time >> 8);
+ local_reg.set8(0x39, exposure_time & 255); /* lowbyte */
+
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return status;
+}
+
+static SANE_Status
+gl841_start_action (Genesys_Device * dev)
+{
+ return sanei_genesys_write_register (dev, 0x0f, 0x01);
+}
+
+static SANE_Status
+gl841_stop_action (Genesys_Device * dev)
+{
+ Genesys_Register_Set local_reg;
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val40, val;
+ unsigned int loop;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ status = sanei_genesys_get_status (dev, &val);
+ if (DBG_LEVEL >= DBG_io)
+ {
+ sanei_genesys_print_status (val);
+ }
+
+ status = sanei_genesys_read_register(dev, 0x40, &val40);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__, sane_strstatus(status));
+ DBGCOMPLETED;
+ return status;
+ }
+
+ /* only stop action if needed */
+ if (!(val40 & REG40_DATAENB) && !(val40 & REG40_MOTMFLG))
+ {
+ DBG(DBG_info, "%s: already stopped\n", __func__);
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+ }
+
+ local_reg = dev->reg;
+
+ gl841_init_optical_regs_off(&local_reg);
+
+ gl841_init_motor_regs_off(&local_reg,0);
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* looks like writing the right registers to zero is enough to get the chip
+ out of scan mode into command mode, actually triggering(writing to
+ register 0x0f) seems to be unnecessary */
+
+ loop = 10;
+ while (loop > 0)
+ {
+ status = sanei_genesys_read_register(dev, 0x40, &val40);
+ if (DBG_LEVEL >= DBG_io)
+ {
+ sanei_genesys_print_status (val);
+ }
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__, sane_strstatus(status));
+ DBGCOMPLETED;
+ return status;
+ }
+
+ /* if scanner is in command mode, we are done */
+ if (!(val40 & REG40_DATAENB) && !(val40 & REG40_MOTMFLG))
+ {
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+ }
+
+ sanei_genesys_sleep_ms(100);
+ loop--;
+ }
+
+ DBGCOMPLETED;
+ return SANE_STATUS_IO_ERROR;
+}
+
+static SANE_Status
+gl841_get_paper_sensor(Genesys_Device * dev, SANE_Bool * paper_loaded)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+
+ status = sanei_genesys_read_register(dev, REG6D, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read gpio: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ *paper_loaded = (val & 0x1) == 0;
+ return SANE_STATUS_GOOD;
+}
+
+static SANE_Status
+gl841_eject_document (Genesys_Device * dev)
+{
+ Genesys_Register_Set local_reg;
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+ SANE_Bool paper_loaded;
+ unsigned int init_steps;
+ float feed_mm;
+ int loop;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ if (dev->model->is_sheetfed == SANE_FALSE)
+ {
+ DBG(DBG_proc, "%s: there is no \"eject sheet\"-concept for non sheet fed\n", __func__);
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+ }
+
+
+ local_reg.clear();
+ val = 0;
+
+ status = sanei_genesys_get_status (dev, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read status register: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop motor: %s\n", __func__, sane_strstatus(status));
+ return SANE_STATUS_IO_ERROR;
+ }
+
+ local_reg = dev->reg;
+
+ gl841_init_optical_regs_off(&local_reg);
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+ gl841_init_motor_regs(dev, sensor, &local_reg,
+ 65536,MOTOR_ACTION_FEED,0);
+
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ try {
+ status = gl841_start_action (dev);
+ } catch (...) {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ try {
+ gl841_stop_action(dev);
+ } catch (...) {}
+ try {
+ // restore original registers
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ } catch (...) {}
+ throw;
+ }
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ gl841_stop_action (dev);
+ /* send original registers */
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ return status;
+ }
+
+ RIE(gl841_get_paper_sensor(dev, &paper_loaded));
+ if (paper_loaded)
+ {
+ DBG(DBG_info, "%s: paper still loaded\n", __func__);
+ /* force document TRUE, because it is definitely present */
+ dev->document = SANE_TRUE;
+ dev->scanhead_position_in_steps = 0;
+
+ loop = 300;
+ while (loop > 0) /* do not wait longer then 30 seconds */
+ {
+
+ RIE(gl841_get_paper_sensor(dev, &paper_loaded));
+
+ if (!paper_loaded)
+ {
+ DBG(DBG_info, "%s: reached home position\n", __func__);
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ break;
+ }
+ sanei_genesys_sleep_ms(100);
+ --loop;
+ }
+
+ if (loop == 0)
+ {
+ /* when we come here then the scanner needed too much time for this, so we better stop the motor */
+ gl841_stop_action (dev);
+ DBG(DBG_error, "%s: timeout while waiting for scanhead to go home\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+ }
+ }
+
+ feed_mm = SANE_UNFIX(dev->model->eject_feed);
+ if (dev->document)
+ {
+ feed_mm += SANE_UNFIX(dev->model->post_scan);
+ }
+
+ status = sanei_genesys_read_feed_steps(dev, &init_steps);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read feed steps: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* now feed for extra <number> steps */
+ loop = 0;
+ while (loop < 300) /* do not wait longer then 30 seconds */
+ {
+ unsigned int steps;
+
+ status = sanei_genesys_read_feed_steps(dev, &steps);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read feed steps: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_info, "%s: init_steps: %d, steps: %d\n", __func__, init_steps, steps);
+
+ if (steps > init_steps + (feed_mm * dev->motor.base_ydpi) / MM_PER_INCH)
+ {
+ break;
+ }
+
+ sanei_genesys_sleep_ms(100);
+ ++loop;
+ }
+
+ status = gl841_stop_action(dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop motor: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ dev->document = SANE_FALSE;
+
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+
+static SANE_Status
+gl841_load_document (Genesys_Device * dev)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ SANE_Bool paper_loaded;
+ int loop = 300;
+ DBG(DBG_proc, "%s\n", __func__);
+ while (loop > 0) /* do not wait longer then 30 seconds */
+ {
+
+ RIE(gl841_get_paper_sensor(dev, &paper_loaded));
+
+ if (paper_loaded)
+ {
+ DBG(DBG_info, "%s: document inserted\n", __func__);
+
+ /* when loading OK, document is here */
+ dev->document = SANE_TRUE;
+
+ // give user some time to place document correctly
+ sanei_genesys_sleep_ms(1000);
+ break;
+ }
+ sanei_genesys_sleep_ms(100);
+ --loop;
+ }
+
+ if (loop == 0)
+ {
+ /* when we come here then the user needed to much time for this */
+ DBG(DBG_error, "%s: timeout while waiting for document\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+ }
+
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+/**
+ * detects end of document and adjust current scan
+ * to take it into account
+ * used by sheetfed scanners
+ */
+static SANE_Status
+gl841_detect_document_end (Genesys_Device * dev)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ SANE_Bool paper_loaded;
+ unsigned int scancnt = 0, lincnt, postcnt;
+ uint8_t val;
+ size_t total_bytes_to_read;
+
+ DBG(DBG_proc, "%s: begin\n", __func__);
+
+ RIE (gl841_get_paper_sensor (dev, &paper_loaded));
+
+ /* sheetfed scanner uses home sensor as paper present */
+ if ((dev->document == SANE_TRUE) && !paper_loaded)
+ {
+ DBG(DBG_info, "%s: no more document\n", __func__);
+ dev->document = SANE_FALSE;
+
+ /* we can't rely on total_bytes_to_read since the frontend
+ * might have been slow to read data, so we re-evaluate the
+ * amount of data to scan form the hardware settings
+ */
+ try {
+ status = sanei_genesys_read_scancnt(dev, &scancnt);
+ } catch (...) {
+ dev->total_bytes_to_read = dev->total_bytes_read;
+ dev->read_bytes_left = 0;
+ throw;
+ }
+
+ if(status!=SANE_STATUS_GOOD)
+ {
+ dev->total_bytes_to_read = dev->total_bytes_read;
+ dev->read_bytes_left = 0;
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+ }
+ if (dev->settings.scan_mode == ScanColorMode::COLOR_SINGLE_PASS && dev->model->is_cis)
+ {
+ scancnt/=3;
+ }
+ DBG(DBG_io, "%s: scancnt=%u lines\n", __func__, scancnt);
+
+ RIE(sanei_genesys_read_register(dev, 0x25, &val));
+ lincnt=65536*val;
+ RIE(sanei_genesys_read_register(dev, 0x26, &val));
+ lincnt+=256*val;
+ RIE(sanei_genesys_read_register(dev, 0x27, &val));
+ lincnt+=val;
+ DBG(DBG_io, "%s: lincnt=%u lines\n", __func__, lincnt);
+ postcnt=(SANE_UNFIX(dev->model->post_scan)/MM_PER_INCH)*dev->settings.yres;
+ DBG(DBG_io, "%s: postcnt=%u lines\n", __func__, postcnt);
+
+ /* the current scancnt is also the final one, so we use it to
+ * compute total bytes to read. We also add the line count to eject document */
+ total_bytes_to_read=(scancnt+postcnt)*dev->wpl;
+
+ DBG(DBG_io, "%s: old total_bytes_to_read=%u\n", __func__,
+ (unsigned int)dev->total_bytes_to_read);
+ DBG(DBG_io, "%s: new total_bytes_to_read=%u\n", __func__, (unsigned int)total_bytes_to_read);
+
+ /* assign new end value */
+ if(dev->total_bytes_to_read>total_bytes_to_read)
+ {
+ DBG(DBG_io, "%s: scan shorten\n", __func__);
+ dev->total_bytes_to_read=total_bytes_to_read;
+ }
+ }
+
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+/* Send the low-level scan command */
+/* todo : is this that useful ? */
+static SANE_Status
+gl841_begin_scan (Genesys_Device * dev, const Genesys_Sensor& sensor, Genesys_Register_Set * reg,
+ SANE_Bool start_motor)
+{
+ (void) sensor;
+ SANE_Status status = SANE_STATUS_GOOD;
+ // FIXME: SEQUENTIAL not really needed in this case
+ Genesys_Register_Set local_reg(Genesys_Register_Set::SEQUENTIAL);
+ uint8_t val;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ if (dev->model->gpo_type == GPO_CANONLIDE80)
+ {
+ RIE (sanei_genesys_read_register (dev, REG6B, &val));
+ val = REG6B_GPO18;
+ RIE (sanei_genesys_write_register (dev, REG6B, val));
+ }
+
+ if (dev->model->ccd_type != CCD_PLUSTEK_3600) {
+ local_reg.init_reg(0x03, sanei_genesys_read_reg_from_set(reg, 0x03) | REG03_LAMPPWR);
+ } else {
+ // TODO PLUSTEK_3600: why ??
+ local_reg.init_reg(0x03, sanei_genesys_read_reg_from_set(reg, 0x03));
+ }
+
+ local_reg.init_reg(0x01, sanei_genesys_read_reg_from_set(reg, 0x01) | REG01_SCAN); /* set scan bit */
+ local_reg.init_reg(0x0d, 0x01);
+
+ if (start_motor) {
+ local_reg.init_reg(0x0f, 0x01);
+ } else {
+ // do not start motor yet
+ local_reg.init_reg(0x0f, 0x00);
+ }
+
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+
+ return status;
+}
+
+
+/* Send the stop scan command */
+static SANE_Status
+gl841_end_scan (Genesys_Device * dev, Genesys_Register_Set __sane_unused__ * reg,
+ SANE_Bool check_stop)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBG(DBG_proc, "%s (check_stop = %d)\n", __func__, check_stop);
+
+ if (dev->model->is_sheetfed == SANE_TRUE)
+ {
+ status = SANE_STATUS_GOOD;
+ }
+ else /* flat bed scanners */
+ {
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+
+ return status;
+}
+
+/* Moves the slider to steps */
+static SANE_Status
+gl841_feed (Genesys_Device * dev, int steps)
+{
+ Genesys_Register_Set local_reg;
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+ int loop;
+
+ DBG(DBG_proc, "%s (steps = %d)\n", __func__, steps);
+
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop action: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ // FIXME: we should pick sensor according to the resolution scanner is currently operating on
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+
+ local_reg = dev->reg;
+
+ gl841_init_optical_regs_off(&local_reg);
+
+ gl841_init_motor_regs(dev, sensor, &local_reg, steps,MOTOR_ACTION_FEED,0);
+
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ try {
+ status = gl841_start_action (dev);
+ } catch (...) {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ try {
+ gl841_stop_action (dev);
+ } catch (...) {}
+ try {
+ // send original registers
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ } catch (...) {}
+ throw;
+ }
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ gl841_stop_action (dev);
+ /* send original registers */
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ return status;
+ }
+
+ loop = 0;
+ while (loop < 300) /* do not wait longer then 30 seconds */
+ {
+ status = sanei_genesys_get_status (dev, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ if (!(val & REG41_MOTORENB)) /* motor enabled */
+ {
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ dev->scanhead_position_in_steps += steps;
+ return SANE_STATUS_GOOD;
+ }
+ sanei_genesys_sleep_ms(100);
+ ++loop;
+ }
+
+ /* when we come here then the scanner needed too much time for this, so we better stop the motor */
+ gl841_stop_action (dev);
+
+ DBG(DBG_error, "%s: timeout while waiting for scanhead to go home\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+}
+
+/* Moves the slider to the home (top) position slowly */
+static SANE_Status
+gl841_slow_back_home (Genesys_Device * dev, SANE_Bool wait_until_home)
+{
+ Genesys_Register_Set local_reg;
+ SANE_Status status = SANE_STATUS_GOOD;
+ GenesysRegister *r;
+ uint8_t val;
+ int loop = 0;
+
+ DBG(DBG_proc, "%s (wait_until_home = %d)\n", __func__, wait_until_home);
+
+ if (dev->model->is_sheetfed == SANE_TRUE)
+ {
+ DBG(DBG_proc, "%s: there is no \"home\"-concept for sheet fed\n", __func__);
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+ }
+
+ /* reset gpio pin */
+ if (dev->model->gpo_type == GPO_CANONLIDE35)
+ {
+ RIE (sanei_genesys_read_register (dev, REG6C, &val));
+ val = dev->gpo.value[0];
+ RIE (sanei_genesys_write_register (dev, REG6C, val));
+ }
+ if (dev->model->gpo_type == GPO_CANONLIDE80)
+ {
+ RIE (sanei_genesys_read_register (dev, REG6B, &val));
+ val = REG6B_GPO18 | REG6B_GPO17;
+ RIE (sanei_genesys_write_register (dev, REG6B, val));
+ }
+ gl841_save_power(dev, SANE_FALSE);
+
+ /* first read gives HOME_SENSOR true */
+ status = sanei_genesys_get_status (dev, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ if (DBG_LEVEL >= DBG_io)
+ {
+ sanei_genesys_print_status (val);
+ }
+ sanei_genesys_sleep_ms(100);
+
+ /* second is reliable */
+ status = sanei_genesys_get_status (dev, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ if (DBG_LEVEL >= DBG_io)
+ {
+ sanei_genesys_print_status (val);
+ }
+
+ dev->scanhead_position_in_steps = 0;
+
+ if (val & REG41_HOMESNR) /* is sensor at home? */
+ {
+ DBG(DBG_info, "%s: already at home, completed\n", __func__);
+ dev->scanhead_position_in_steps = 0;
+ return SANE_STATUS_GOOD;
+ }
+
+ /* end previous scan if any */
+ r = sanei_genesys_get_address(&dev->reg, REG01);
+ r->value &= ~REG01_SCAN;
+ status = sanei_genesys_write_register (dev, REG01, r->value);
+
+ /* if motor is on, stop current action */
+ if (val & REG41_MOTORENB)
+ {
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop motor: %s\n", __func__, sane_strstatus(status));
+ return SANE_STATUS_IO_ERROR;
+ }
+ }
+
+ local_reg = dev->reg;
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+
+ gl841_init_motor_regs(dev, sensor, &local_reg, 65536,MOTOR_ACTION_GO_HOME,0);
+
+ /* set up for reverse and no scan */
+ r = sanei_genesys_get_address(&local_reg, REG02);
+ r->value |= REG02_MTRREV;
+ r = sanei_genesys_get_address(&local_reg, REG01);
+ r->value &= ~REG01_SCAN;
+
+ RIE (sanei_genesys_bulk_write_register(dev, local_reg));
+
+ try {
+ status = gl841_start_action (dev);
+ } catch (...) {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ try {
+ gl841_stop_action(dev);
+ } catch (...) {}
+ try {
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ } catch (...) {}
+ throw;
+ }
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to start motor: %s\n", __func__, sane_strstatus(status));
+ gl841_stop_action (dev);
+ /* send original registers */
+ sanei_genesys_bulk_write_register(dev, dev->reg);
+ return status;
+ }
+
+ if (wait_until_home)
+ {
+ while (loop < 300) /* do not wait longer then 30 seconds */
+ {
+ status = sanei_genesys_get_status (dev, &val);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read home sensor: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+
+ if (val & REG41_HOMESNR) /* home sensor */
+ {
+ DBG(DBG_info, "%s: reached home position\n", __func__);
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+ }
+ sanei_genesys_sleep_ms(100);
+ ++loop;
+ }
+
+ /* when we come here then the scanner needed too much time for this, so we better stop the motor */
+ gl841_stop_action (dev);
+ DBG(DBG_error, "%s: timeout while waiting for scanhead to go home\n", __func__);
+ return SANE_STATUS_IO_ERROR;
+ }
+
+ DBG(DBG_info, "%s: scanhead is still moving\n", __func__);
+ DBG(DBG_proc, "%s: finished\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+/* Automatically set top-left edge of the scan area by scanning a 200x200 pixels
+ area at 600 dpi from very top of scanner */
+static SANE_Status
+gl841_search_start_position (Genesys_Device * dev)
+{
+ int size;
+ SANE_Status status = SANE_STATUS_GOOD;
+ Genesys_Register_Set local_reg;
+ int steps;
+
+ int pixels = 600;
+ int dpi = 300;
+
+ DBGSTART;
+
+ local_reg = dev->reg;
+
+ /* sets for a 200 lines * 600 pixels */
+ /* normal scan with no shading */
+
+ // FIXME: the current approach of doing search only for one resolution does not work on scanners
+ // whith employ different sensors with potentially different settings.
+ auto& sensor = sanei_genesys_find_sensor_for_write(dev, dpi);
+
+ SetupParams params;
+ params.xres = dpi;
+ params.yres = dpi;
+ params.startx = 0;
+ params.starty = 0; /*we should give a small offset here~60 steps*/
+ params.pixels = 600;
+ params.lines = dev->model->search_lines;
+ params.depth = 8;
+ params.channels = 1;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::GRAY;
+ params.color_filter = ColorFilter::GREEN;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_DISABLE_BUFFER_FULL_MOVE;
+
+ status = gl841_init_scan_regs(dev, sensor, &local_reg, params);
+
+ if(status!=SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to init scan registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* send to scanner */
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ size = pixels * dev->model->search_lines;
+
+ std::vector<uint8_t> data(size);
+
+ status = gl841_begin_scan(dev, sensor, &local_reg, SANE_TRUE);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to begin scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* waits for valid data */
+ do
+ sanei_genesys_test_buffer_empty (dev, &steps);
+ while (steps);
+
+ /* now we're on target, we can read data */
+ status = sanei_genesys_read_data_from_scanner(dev, data.data(), size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read data: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ if (DBG_LEVEL >= DBG_data)
+ sanei_genesys_write_pnm_file("gl841_search_position.pnm", data.data(), 8, 1, pixels,
+ dev->model->search_lines);
+
+ status = gl841_end_scan(dev, &local_reg, SANE_TRUE);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to end scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* update regs to copy ASIC internal state */
+ dev->reg = local_reg;
+
+/*TODO: find out where sanei_genesys_search_reference_point
+ stores information, and use that correctly*/
+ status =
+ sanei_genesys_search_reference_point (dev, sensor, data.data(), 0, dpi, pixels,
+ dev->model->search_lines);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set search reference point: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+
+ return SANE_STATUS_GOOD;
+}
+
+/*
+ * sets up register for coarse gain calibration
+ * todo: check it for scanners using it */
+static SANE_Status
+gl841_init_regs_for_coarse_calibration(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set& regs)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t channels;
+ uint8_t cksel;
+
+ DBGSTART;
+
+ cksel = (regs.find_reg(0x18).value & REG18_CKSEL) + 1; /* clock speed = 1..4 clocks */
+
+ /* set line size */
+ if (dev->settings.scan_mode == ScanColorMode::COLOR_SINGLE_PASS)
+ channels = 3;
+ else {
+ channels = 1;
+ }
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = sensor.optical_res / cksel; /* XXX STEF XXX !!! */
+ params.lines = 20;
+ params.depth = 16;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = dev->settings.scan_mode;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_info, "%s: optical sensor res: %d dpi, actual res: %d\n", __func__,
+ sensor.optical_res / cksel, dev->settings.xres);
+
+ status = sanei_genesys_bulk_write_register(dev, regs);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+
+/* if (DBG_LEVEL >= DBG_info)
+ sanei_gl841_print_registers (regs);*/
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+
+/* init registers for shading calibration */
+static SANE_Status
+gl841_init_regs_for_shading(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set& regs)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ SANE_Int ydpi;
+ float starty=0;
+
+ DBGSTART;
+ DBG(DBG_proc, "%s: lines = %d\n", __func__, (int)(dev->calib_lines));
+
+ /* initial calibration reg values */
+ regs = dev->reg;
+
+ ydpi = dev->motor.base_ydpi;
+ if (dev->model->motor_type == MOTOR_PLUSTEK_3600) /* TODO PLUSTEK_3600: 1200dpi not yet working, produces dark bar */
+ {
+ ydpi = 600;
+ }
+ if (dev->model->motor_type == MOTOR_CANONLIDE80)
+ {
+ ydpi = gl841_get_dpihw(dev);
+ /* get over extra dark area for this model.
+ It looks like different devices have dark areas of different width
+ due to manufacturing variability. The initial value of starty was 140,
+ but it moves the sensor almost past the dark area completely in places
+ on certain devices.
+
+ On a particular device the black area starts at roughly position
+ 160 to 230 depending on location (the dark area is not completely
+ parallel to the frame).
+ */
+ starty = 70;
+ }
+
+ dev->calib_channels = 3;
+ dev->calib_lines = dev->model->shading_lines;
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = ydpi;
+ params.startx = 0;
+ params.starty = starty;
+ params.pixels = (sensor.sensor_pixels * dev->settings.xres) / sensor.optical_res;
+ params.lines = dev->calib_lines;
+ params.depth = 16;
+ params.channels = dev->calib_channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_USE_OPTICAL_RES |
+ /*SCAN_FLAG_DISABLE_BUFFER_FULL_MOVE |*/
+ SCAN_FLAG_IGNORE_LINE_DISTANCE;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ dev->calib_pixels = dev->current_setup.pixels;
+ dev->scanhead_position_in_steps += dev->calib_lines + starty;
+
+ status = sanei_genesys_bulk_write_register(dev, regs);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+/* set up registers for the actual scan
+ */
+static SANE_Status
+gl841_init_regs_for_scan (Genesys_Device * dev, const Genesys_Sensor& sensor)
+{
+ int channels;
+ int flags;
+ int depth;
+ float move;
+ int move_dpi;
+ float start;
+
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBG(DBG_info, "%s ", __func__);
+ debug_dump(DBG_info, dev->settings);
+
+/* channels */
+ if (dev->settings.scan_mode == ScanColorMode::COLOR_SINGLE_PASS)
+ channels = 3;
+ else
+ channels = 1;
+
+/* depth */
+ depth = dev->settings.depth;
+ if (dev->settings.scan_mode == ScanColorMode::LINEART)
+ depth = 1;
+
+
+ /* steps to move to reach scanning area:
+ - first we move to physical start of scanning
+ either by a fixed steps amount from the black strip
+ or by a fixed amount from parking position,
+ minus the steps done during shading calibration
+ - then we move by the needed offset whitin physical
+ scanning area
+
+ assumption: steps are expressed at maximum motor resolution
+
+ we need:
+ SANE_Fixed y_offset;
+ SANE_Fixed y_size;
+ SANE_Fixed y_offset_calib;
+ mm_to_steps()=motor dpi / 2.54 / 10=motor dpi / MM_PER_INCH */
+
+ /* if scanner uses GENESYS_FLAG_SEARCH_START y_offset is
+ relative from origin, else, it is from parking position */
+
+ move_dpi = dev->motor.base_ydpi;
+
+ move = 0;
+ if (dev->model->flags & GENESYS_FLAG_SEARCH_START)
+ {
+ move += SANE_UNFIX (dev->model->y_offset_calib);
+ }
+
+ DBG(DBG_info, "%s move=%f steps\n", __func__, move);
+
+ move += SANE_UNFIX (dev->model->y_offset);
+ DBG(DBG_info, "%s: move=%f steps\n", __func__, move);
+
+ move += dev->settings.tl_y;
+ DBG(DBG_info, "%s: move=%f steps\n", __func__, move);
+
+ move = (move * move_dpi) / MM_PER_INCH;
+
+/* start */
+ start = SANE_UNFIX (dev->model->x_offset);
+
+ start += dev->settings.tl_x;
+
+ start = (start * sensor.optical_res) / MM_PER_INCH;
+
+ flags=0;
+
+ /* we enable true gray for cis scanners only, and just when doing
+ * scan since color calibration is OK for this mode
+ */
+ flags = 0;
+
+ /* true gray (led add for cis scanners) */
+ if(dev->model->is_cis && dev->settings.true_gray
+ && dev->settings.scan_mode != ScanColorMode::COLOR_SINGLE_PASS
+ && dev->model->ccd_type != CIS_CANONLIDE80)
+ {
+ // on Lide 80 the LEDADD bit results in only red LED array being lit
+ DBG(DBG_io, "%s: activating LEDADD\n", __func__);
+ flags |= SCAN_FLAG_ENABLE_LEDADD;
+ }
+
+ /* enable emulated lineart from gray data */
+ if(dev->settings.scan_mode == ScanColorMode::LINEART
+ && dev->settings.dynamic_lineart)
+ {
+ flags |= SCAN_FLAG_DYNAMIC_LINEART;
+ }
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = start;
+ params.starty = move;
+ params.pixels = dev->settings.pixels;
+ params.lines = dev->settings.lines;
+ params.depth = depth;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = dev->settings.scan_mode;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = flags;
+
+ status = gl841_init_scan_regs(dev, sensor, &dev->reg, params);
+
+ if (status != SANE_STATUS_GOOD)
+ return status;
+
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return SANE_STATUS_GOOD;
+}
+
+/*
+ * this function sends generic gamma table (ie linear ones)
+ * or the Sensor specific one if provided
+ */
+static SANE_Status
+gl841_send_gamma_table(Genesys_Device * dev, const Genesys_Sensor& sensor)
+{
+ int size;
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBGSTART;
+
+ size = 256;
+
+ /* allocate temporary gamma tables: 16 bits words, 3 channels */
+ std::vector<uint8_t> gamma(size * 2 * 3);
+
+ RIE(sanei_genesys_generate_gamma_buffer(dev, sensor, 16, 65535, size, gamma.data()));
+
+ /* send address */
+ status = gl841_set_buffer_address_gamma (dev, 0x00000);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set buffer address: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* send data */
+ status = sanei_genesys_bulk_write_data(dev, 0x28, gamma.data(), size * 2 * 3);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to send gamma table: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+
+/* this function does the led calibration by scanning one line of the calibration
+ area below scanner's top on white strip.
+
+-needs working coarse/gain
+*/
+static SANE_Status
+gl841_led_calibration (Genesys_Device * dev, Genesys_Sensor& sensor, Genesys_Register_Set& regs)
+{
+ int num_pixels;
+ int total_size;
+ int i, j;
+ SANE_Status status = SANE_STATUS_GOOD;
+ int val;
+ int channels;
+ int avg[3], avga, avge;
+ int turn;
+ uint16_t exp[3], target;
+ int move;
+
+ SANE_Bool acceptable = SANE_FALSE;
+
+ /* these 2 boundaries should be per sensor */
+ uint16_t min_exposure=500;
+ uint16_t max_exposure;
+
+ DBGSTART;
+
+ /* feed to white strip if needed */
+ if (dev->model->y_offset_calib>0)
+ {
+ move = SANE_UNFIX (dev->model->y_offset_calib);
+ move = (move * (dev->motor.base_ydpi)) / MM_PER_INCH;
+ DBG(DBG_io, "%s: move=%d lines\n", __func__, move);
+ status = gl841_feed(dev, move);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to feed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ /* offset calibration is always done in color mode */
+ channels = 3;
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = (sensor.sensor_pixels*dev->settings.xres) / sensor.optical_res;
+ params.lines = 1;
+ params.depth = 16;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+
+ num_pixels = dev->current_setup.pixels;
+
+ total_size = num_pixels * channels * 2 * 1; /* colors * bytes_per_color * scan lines */
+
+ std::vector<uint8_t> line(total_size);
+
+/*
+ we try to get equal bright leds here:
+
+ loop:
+ average per color
+ adjust exposure times
+ */
+
+ exp[0] = sensor.exposure.red;
+ exp[1] = sensor.exposure.green;
+ exp[2] = sensor.exposure.blue;
+
+ turn = 0;
+ /* max exposure is set to ~2 time initial average
+ * exposure, or 2 time last calibration exposure */
+ max_exposure=((exp[0]+exp[1]+exp[2])/3)*2;
+ target=sensor.gain_white_ref*256;
+
+ do {
+ sensor.exposure.red = exp[0];
+ sensor.exposure.green = exp[1];
+ sensor.exposure.blue = exp[2];
+
+ sanei_genesys_set_exposure(regs, sensor.exposure);
+ RIE(sanei_genesys_write_register(dev, 0x10, (sensor.exposure.red >> 8) & 0xff));
+ RIE(sanei_genesys_write_register(dev, 0x11, sensor.exposure.red & 0xff));
+ RIE(sanei_genesys_write_register(dev, 0x12, (sensor.exposure.green >> 8) & 0xff));
+ RIE(sanei_genesys_write_register(dev, 0x13, sensor.exposure.green & 0xff));
+ RIE(sanei_genesys_write_register(dev, 0x14, (sensor.exposure.blue >> 8) & 0xff));
+ RIE(sanei_genesys_write_register(dev, 0x15, sensor.exposure.blue & 0xff));
+
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+
+ DBG(DBG_info, "%s: starting line reading\n", __func__);
+ RIE(gl841_begin_scan(dev, sensor, &regs, SANE_TRUE));
+ RIE(sanei_genesys_read_data_from_scanner(dev, line.data(), total_size));
+
+ if (DBG_LEVEL >= DBG_data) {
+ char fn[30];
+ snprintf(fn, 30, "gl841_led_%d.pnm", turn);
+ sanei_genesys_write_pnm_file(fn, line.data(), 16, channels, num_pixels, 1);
+ }
+
+ /* compute average */
+ for (j = 0; j < channels; j++)
+ {
+ avg[j] = 0;
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ line[i * 2 * channels + 2 * j + 1] * 256 +
+ line[i * 2 * channels + 2 * j];
+ avg[j] += val;
+ }
+
+ avg[j] /= num_pixels;
+ }
+
+ DBG(DBG_info,"%s: average: %d,%d,%d\n", __func__, avg[0], avg[1], avg[2]);
+
+ acceptable = SANE_TRUE;
+
+ /* exposure is acceptable if each color is in the %5 range
+ * of other color channels */
+ if (avg[0] < avg[1] * 0.95 || avg[1] < avg[0] * 0.95 ||
+ avg[0] < avg[2] * 0.95 || avg[2] < avg[0] * 0.95 ||
+ avg[1] < avg[2] * 0.95 || avg[2] < avg[1] * 0.95)
+ {
+ acceptable = SANE_FALSE;
+ }
+
+ /* led exposure is not acceptable if white level is too low
+ * ~80 hardcoded value for white level */
+ if(avg[0]<20000 || avg[1]<20000 || avg[2]<20000)
+ {
+ acceptable = SANE_FALSE;
+ }
+
+ /* for scanners using target value */
+ if(target>0)
+ {
+ acceptable = SANE_TRUE;
+ for(i=0;i<3;i++)
+ {
+ /* we accept +- 2% delta from target */
+ if(abs(avg[i]-target)>target/50)
+ {
+ exp[i]=(exp[i]*target)/avg[i];
+ acceptable = SANE_FALSE;
+ }
+ }
+ }
+ else
+ {
+ if (!acceptable)
+ {
+ avga = (avg[0]+avg[1]+avg[2])/3;
+ exp[0] = (exp[0] * avga) / avg[0];
+ exp[1] = (exp[1] * avga) / avg[1];
+ exp[2] = (exp[2] * avga) / avg[2];
+ /*
+ keep the resulting exposures below this value.
+ too long exposure drives the ccd into saturation.
+ we may fix this by relying on the fact that
+ we get a striped scan without shading, by means of
+ statistical calculation
+ */
+ avge = (exp[0] + exp[1] + exp[2]) / 3;
+
+ if (avge > max_exposure) {
+ exp[0] = (exp[0] * max_exposure) / avge;
+ exp[1] = (exp[1] * max_exposure) / avge;
+ exp[2] = (exp[2] * max_exposure) / avge;
+ }
+ if (avge < min_exposure) {
+ exp[0] = (exp[0] * min_exposure) / avge;
+ exp[1] = (exp[1] * min_exposure) / avge;
+ exp[2] = (exp[2] * min_exposure) / avge;
+ }
+
+ }
+ }
+
+ RIE (gl841_stop_action (dev));
+
+ turn++;
+
+ } while (!acceptable && turn < 100);
+
+ DBG(DBG_info,"%s: acceptable exposure: %d,%d,%d\n", __func__, exp[0], exp[1], exp[2]);
+
+ gl841_slow_back_home(dev, SANE_TRUE);
+
+ DBGCOMPLETED;
+ return status;
+}
+
+/** @brief calibration for AD frontend devices
+ * offset calibration assumes that the scanning head is on a black area
+ * For LiDE80 analog frontend
+ * 0x0003 : is gain and belongs to [0..63]
+ * 0x0006 : is offset
+ * We scan a line with no gain until average offset reaches the target
+ */
+static SANE_Status
+ad_fe_offset_calibration (Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set& regs)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ int num_pixels;
+ int total_size;
+ int i;
+ int average;
+ int turn;
+ int top;
+ int bottom;
+ int target;
+
+ DBGSTART;
+
+ /* don't impact 3600 behavior since we can't test it */
+ if (dev->model->ccd_type == CCD_PLUSTEK_3600)
+ {
+ DBGCOMPLETED;
+ return status;
+ }
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = (sensor.sensor_pixels*dev->settings.xres) / sensor.optical_res;
+ params.lines = 1;
+ params.depth = 8;
+ params.channels = 3;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ num_pixels = dev->current_setup.pixels;
+ total_size = num_pixels * 3 * 2 * 1;
+
+ std::vector<uint8_t> line(total_size);
+
+ dev->frontend.set_gain(0, 0);
+ dev->frontend.set_gain(1, 0);
+ dev->frontend.set_gain(2, 0);
+
+ /* loop on scan until target offset is reached */
+ turn=0;
+ target=24;
+ bottom=0;
+ top=255;
+ do {
+ /* set up offset mid range */
+ dev->frontend.set_offset(0, (top + bottom) / 2);
+ dev->frontend.set_offset(1, (top + bottom) / 2);
+ dev->frontend.set_offset(2, (top + bottom) / 2);
+
+ /* scan line */
+ DBG(DBG_info, "%s: starting line reading\n", __func__);
+ sanei_genesys_bulk_write_register(dev, regs);
+ gl841_set_fe(dev, sensor, AFE_SET);
+ gl841_begin_scan(dev, sensor, &regs, SANE_TRUE);
+ sanei_genesys_read_data_from_scanner(dev, line.data(), total_size);
+ gl841_stop_action (dev);
+ if (DBG_LEVEL >= DBG_data) {
+ char fn[30];
+ snprintf(fn, 30, "gl841_offset_%02d.pnm", turn);
+ sanei_genesys_write_pnm_file(fn, line.data(), 8, 3, num_pixels, 1);
+ }
+
+ /* search for minimal value */
+ average=0;
+ for(i=0;i<total_size;i++)
+ {
+ average+=line[i];
+ }
+ average/=total_size;
+ DBG(DBG_data, "%s: average=%d\n", __func__, average);
+
+ /* if min value is above target, the current value becomes the new top
+ * else it is the new bottom */
+ if(average>target)
+ {
+ top=(top+bottom)/2;
+ }
+ else
+ {
+ bottom=(top+bottom)/2;
+ }
+ turn++;
+ } while ((top-bottom)>1 && turn < 100);
+
+ // FIXME: don't overwrite the calibrated values
+ dev->frontend.set_offset(0, 0);
+ dev->frontend.set_offset(1, 0);
+ dev->frontend.set_offset(2, 0);
+ DBG(DBG_info, "%s: offset=(%d,%d,%d)\n", __func__,
+ dev->frontend.get_offset(0),
+ dev->frontend.get_offset(1),
+ dev->frontend.get_offset(2));
+ DBGCOMPLETED;
+ return status;
+}
+
+/* this function does the offset calibration by scanning one line of the calibration
+ area below scanner's top. There is a black margin and the remaining is white.
+ sanei_genesys_search_start() must have been called so that the offsets and margins
+ are allready known.
+
+this function expects the slider to be where?
+*/
+static SANE_Status
+gl841_offset_calibration(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set& regs)
+{
+ int num_pixels;
+ int total_size;
+ int i, j;
+ SANE_Status status = SANE_STATUS_GOOD;
+ int val;
+ int channels;
+ int off[3],offh[3],offl[3],off1[3],off2[3];
+ int min1[3],min2[3];
+ int cmin[3],cmax[3];
+ int turn;
+ SANE_Bool acceptable = SANE_FALSE;
+ int mintgt = 0x400;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ /* Analog Device fronted have a different calibration */
+ if ((dev->reg.find_reg(0x04).value & REG04_FESET) == 0x02)
+ {
+ return ad_fe_offset_calibration(dev, sensor, regs);
+ }
+
+ /* offset calibration is always done in color mode */
+ channels = 3;
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = (sensor.sensor_pixels*dev->settings.xres) / sensor.optical_res;
+ params.lines = 1;
+ params.depth = 16;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES |
+ SCAN_FLAG_DISABLE_LAMP;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ num_pixels = dev->current_setup.pixels;
+
+ total_size = num_pixels * channels * 2 * 1; /* colors * bytes_per_color * scan lines */
+
+ std::vector<uint8_t> first_line(total_size);
+ std::vector<uint8_t> second_line(total_size);
+
+ /* scan first line of data with no offset nor gain */
+/*WM8199: gain=0.73; offset=-260mV*/
+/*okay. the sensor black level is now at -260mV. we only get 0 from AFE...*/
+/* we should probably do real calibration here:
+ * -detect acceptable offset with binary search
+ * -calculate offset from this last version
+ *
+ * acceptable offset means
+ * - few completely black pixels(<10%?)
+ * - few completely white pixels(<10%?)
+ *
+ * final offset should map the minimum not completely black
+ * pixel to 0(16 bits)
+ *
+ * this does account for dummy pixels at the end of ccd
+ * this assumes slider is at black strip(which is not quite as black as "no
+ * signal").
+ *
+ */
+ dev->frontend.set_gain(0, 0);
+ dev->frontend.set_gain(1, 0);
+ dev->frontend.set_gain(2, 0);
+ offh[0] = 0xff;
+ offh[1] = 0xff;
+ offh[2] = 0xff;
+ offl[0] = 0x00;
+ offl[1] = 0x00;
+ offl[2] = 0x00;
+ turn = 0;
+
+ do {
+
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+
+ for (j=0; j < channels; j++) {
+ off[j] = (offh[j]+offl[j])/2;
+ dev->frontend.set_offset(j, off[j]);
+ }
+
+ status = gl841_set_fe(dev, sensor, AFE_SET);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup frontend: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_info, "%s: starting first line reading\n", __func__);
+ RIE(gl841_begin_scan(dev, sensor, &regs, SANE_TRUE));
+
+ RIE(sanei_genesys_read_data_from_scanner (dev, first_line.data(), total_size));
+
+ if (DBG_LEVEL >= DBG_data) {
+ char fn[30];
+ snprintf(fn, 30, "gl841_offset1_%02d.pnm", turn);
+ sanei_genesys_write_pnm_file(fn, first_line.data(), 16, channels, num_pixels, 1);
+ }
+
+ acceptable = SANE_TRUE;
+
+ for (j = 0; j < channels; j++)
+ {
+ cmin[j] = 0;
+ cmax[j] = 0;
+
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ first_line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ first_line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ first_line[i * 2 * channels + 2 * j + 1] * 256 +
+ first_line[i * 2 * channels + 2 * j];
+ if (val < 10)
+ cmin[j]++;
+ if (val > 65525)
+ cmax[j]++;
+ }
+
+ /* TODO the DP685 has a black strip in the middle of the sensor
+ * should be handled in a more elegant way , could be a bug */
+ if (dev->model->ccd_type == CCD_DP685)
+ cmin[j] -= 20;
+
+ if (cmin[j] > num_pixels/100) {
+ acceptable = SANE_FALSE;
+ if (dev->model->is_cis)
+ offl[0] = off[0];
+ else
+ offl[j] = off[j];
+ }
+ if (cmax[j] > num_pixels/100) {
+ acceptable = SANE_FALSE;
+ if (dev->model->is_cis)
+ offh[0] = off[0];
+ else
+ offh[j] = off[j];
+ }
+ }
+
+ DBG(DBG_info,"%s: black/white pixels: %d/%d,%d/%d,%d/%d\n", __func__, cmin[0], cmax[0],
+ cmin[1], cmax[1], cmin[2], cmax[2]);
+
+ if (dev->model->is_cis) {
+ offh[2] = offh[1] = offh[0];
+ offl[2] = offl[1] = offl[0];
+ }
+
+ RIE(gl841_stop_action(dev));
+
+ turn++;
+ } while (!acceptable && turn < 100);
+
+ DBG(DBG_info,"%s: acceptable offsets: %d,%d,%d\n", __func__, off[0], off[1], off[2]);
+
+
+ for (j = 0; j < channels; j++)
+ {
+ off1[j] = off[j];
+
+ min1[j] = 65536;
+
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ first_line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ first_line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ first_line[i * 2 * channels + 2 * j + 1] * 256 +
+ first_line[i * 2 * channels + 2 * j];
+ if (min1[j] > val && val >= 10)
+ min1[j] = val;
+ }
+ }
+
+
+ offl[0] = off[0];
+ offl[1] = off[0];
+ offl[2] = off[0];
+ turn = 0;
+
+ do {
+
+ for (j=0; j < channels; j++) {
+ off[j] = (offh[j]+offl[j])/2;
+ dev->frontend.set_offset(j, off[j]);
+ }
+
+ status = gl841_set_fe(dev, sensor, AFE_SET);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup frontend: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ DBG(DBG_info, "%s: starting second line reading\n", __func__);
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+ RIE(gl841_begin_scan(dev, sensor, &regs, SANE_TRUE));
+ RIE(sanei_genesys_read_data_from_scanner (dev, second_line.data(), total_size));
+
+ if (DBG_LEVEL >= DBG_data) {
+ char fn[30];
+ snprintf(fn, 30, "gl841_offset2_%02d.pnm", turn);
+ sanei_genesys_write_pnm_file(fn, second_line.data(), 16, channels, num_pixels, 1);
+ }
+
+ acceptable = SANE_TRUE;
+
+ for (j = 0; j < channels; j++)
+ {
+ cmin[j] = 0;
+ cmax[j] = 0;
+
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ second_line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ second_line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ second_line[i * 2 * channels + 2 * j + 1] * 256 +
+ second_line[i * 2 * channels + 2 * j];
+ if (val < 10)
+ cmin[j]++;
+ if (val > 65525)
+ cmax[j]++;
+ }
+
+ if (cmin[j] > num_pixels/100) {
+ acceptable = SANE_FALSE;
+ if (dev->model->is_cis)
+ offl[0] = off[0];
+ else
+ offl[j] = off[j];
+ }
+ if (cmax[j] > num_pixels/100) {
+ acceptable = SANE_FALSE;
+ if (dev->model->is_cis)
+ offh[0] = off[0];
+ else
+ offh[j] = off[j];
+ }
+ }
+
+ DBG(DBG_info, "%s: black/white pixels: %d/%d,%d/%d,%d/%d\n", __func__, cmin[0], cmax[0],
+ cmin[1], cmax[1], cmin[2], cmax[2]);
+
+ if (dev->model->is_cis) {
+ offh[2] = offh[1] = offh[0];
+ offl[2] = offl[1] = offl[0];
+ }
+
+ RIE(gl841_stop_action (dev));
+
+ turn++;
+
+ } while (!acceptable && turn < 100);
+
+ DBG(DBG_info, "%s: acceptable offsets: %d,%d,%d\n", __func__, off[0], off[1], off[2]);
+
+
+ for (j = 0; j < channels; j++)
+ {
+ off2[j] = off[j];
+
+ min2[j] = 65536;
+
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ second_line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ second_line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ second_line[i * 2 * channels + 2 * j + 1] * 256 +
+ second_line[i * 2 * channels + 2 * j];
+ if (min2[j] > val && val != 0)
+ min2[j] = val;
+ }
+ }
+
+ DBG(DBG_info, "%s: first set: %d/%d,%d/%d,%d/%d\n", __func__, off1[0], min1[0], off1[1], min1[1],
+ off1[2], min1[2]);
+
+ DBG(DBG_info, "%s: second set: %d/%d,%d/%d,%d/%d\n", __func__, off2[0], min2[0], off2[1], min2[1],
+ off2[2], min2[2]);
+
+/*
+ calculate offset for each channel
+ based on minimal pixel value min1 at offset off1 and minimal pixel value min2
+ at offset off2
+
+ to get min at off, values are linearly interpolated:
+ min=real+off*fact
+ min1=real+off1*fact
+ min2=real+off2*fact
+
+ fact=(min1-min2)/(off1-off2)
+ real=min1-off1*(min1-min2)/(off1-off2)
+
+ off=(min-min1+off1*(min1-min2)/(off1-off2))/((min1-min2)/(off1-off2))
+
+ off=(min*(off1-off2)+min1*off2-off1*min2)/(min1-min2)
+
+ */
+ for (j = 0; j < channels; j++)
+ {
+ if (min2[j]-min1[j] == 0) {
+/*TODO: try to avoid this*/
+ DBG(DBG_warn, "%s: difference too small\n", __func__);
+ if (mintgt * (off1[j] - off2[j]) + min1[j] * off2[j] - min2[j] * off1[j] >= 0)
+ off[j] = 0x0000;
+ else
+ off[j] = 0xffff;
+ } else
+ off[j] = (mintgt * (off1[j] - off2[j]) + min1[j] * off2[j] - min2[j] * off1[j])/(min1[j]-min2[j]);
+ if (off[j] > 255)
+ off[j] = 255;
+ if (off[j] < 0)
+ off[j] = 0;
+ dev->frontend.set_offset(j, off[j]);
+ }
+
+ DBG(DBG_info, "%s: final offsets: %d,%d,%d\n", __func__, off[0], off[1], off[2]);
+
+ if (dev->model->is_cis) {
+ if (off[0] < off[1])
+ off[0] = off[1];
+ if (off[0] < off[2])
+ off[0] = off[2];
+ dev->frontend.set_offset(0, off[0]);
+ dev->frontend.set_offset(1, off[0]);
+ dev->frontend.set_offset(2, off[0]);
+ }
+
+ if (channels == 1)
+ {
+ dev->frontend.set_offset(1, dev->frontend.get_offset(0));
+ dev->frontend.set_offset(2, dev->frontend.get_offset(0));
+ }
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return status;
+}
+
+
+/* alternative coarse gain calibration
+ this on uses the settings from offset_calibration and
+ uses only one scanline
+ */
+/*
+ with offset and coarse calibration we only want to get our input range into
+ a reasonable shape. the fine calibration of the upper and lower bounds will
+ be done with shading.
+ */
+static SANE_Status
+gl841_coarse_gain_calibration(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Register_Set& regs, int dpi)
+{
+ int num_pixels;
+ int total_size;
+ int i, j, channels;
+ SANE_Status status = SANE_STATUS_GOOD;
+ int max[3];
+ float gain[3];
+ int val;
+ int lines=1;
+ int move;
+
+ DBG(DBG_proc, "%s: dpi=%d\n", __func__, dpi);
+
+ /* feed to white strip if needed */
+ if (dev->model->y_offset_calib>0)
+ {
+ move = SANE_UNFIX (dev->model->y_offset_calib);
+ move = (move * (dev->motor.base_ydpi)) / MM_PER_INCH;
+ DBG(DBG_io, "%s: move=%d lines\n", __func__, move);
+ status = gl841_feed(dev, move);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to feed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+ }
+
+ /* coarse gain calibration is allways done in color mode */
+ channels = 3;
+
+ SetupParams params;
+ params.xres = dev->settings.xres;
+ params.yres = dev->settings.yres;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = (sensor.sensor_pixels*dev->settings.xres) / sensor.optical_res;
+ params.lines = lines;
+ params.depth = 16;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+
+ num_pixels = dev->current_setup.pixels;
+
+ total_size = num_pixels * channels * 2 * lines; /* colors * bytes_per_color * scan lines */
+
+ std::vector<uint8_t> line(total_size);
+
+ RIE(gl841_begin_scan(dev, sensor, &regs, SANE_TRUE));
+ RIE(sanei_genesys_read_data_from_scanner(dev, line.data(), total_size));
+
+ if (DBG_LEVEL >= DBG_data)
+ sanei_genesys_write_pnm_file("gl841_gain.pnm", line.data(), 16, channels, num_pixels, lines);
+
+ /* average high level for each channel and compute gain
+ to reach the target code
+ we only use the central half of the CCD data */
+ for (j = 0; j < channels; j++)
+ {
+ max[j] = 0;
+ for (i = 0; i < num_pixels; i++)
+ {
+ if (dev->model->is_cis)
+ val =
+ line[i * 2 + j * 2 * num_pixels + 1] * 256 +
+ line[i * 2 + j * 2 * num_pixels];
+ else
+ val =
+ line[i * 2 * channels + 2 * j + 1] * 256 +
+ line[i * 2 * channels + 2 * j];
+
+ if (val > max[j])
+ max[j] = val;
+ }
+
+ gain[j] = 65535.0/max[j];
+
+ uint8_t out_gain = 0;
+
+ if (dev->model->dac_type == DAC_CANONLIDE35 ||
+ dev->model->dac_type == DAC_WOLFSON_XP300 ||
+ dev->model->dac_type == DAC_WOLFSON_DSM600)
+ {
+ gain[j] *= 0.69;/*seems we don't get the real maximum. empirically derived*/
+ if (283 - 208/gain[j] > 255)
+ out_gain = 255;
+ else if (283 - 208/gain[j] < 0)
+ out_gain = 0;
+ else
+ out_gain = 283 - 208/gain[j];
+ }
+ else if (dev->model->dac_type == DAC_CANONLIDE80)
+ {
+ out_gain = gain[j]*12;
+ }
+ dev->frontend.set_gain(j, out_gain);
+
+ DBG(DBG_proc, "%s: channel %d, max=%d, gain = %f, setting:%d\n", __func__, j, max[j], gain[j],
+ out_gain);
+ }
+
+ for (j = 0; j < channels; j++)
+ {
+ if(gain[j] > 10)
+ {
+ DBG (DBG_error0, "**********************************************\n");
+ DBG (DBG_error0, "**********************************************\n");
+ DBG (DBG_error0, "**** ****\n");
+ DBG (DBG_error0, "**** Extremely low Brightness detected. ****\n");
+ DBG (DBG_error0, "**** Check the scanning head is ****\n");
+ DBG (DBG_error0, "**** unlocked and moving. ****\n");
+ DBG (DBG_error0, "**** ****\n");
+ DBG (DBG_error0, "**********************************************\n");
+ DBG (DBG_error0, "**********************************************\n");
+ return SANE_STATUS_JAMMED;
+ }
+
+ }
+
+ if (dev->model->is_cis) {
+ uint8_t gain0 = dev->frontend.get_gain(0);
+ if (gain0 > dev->frontend.get_gain(1)) {
+ gain0 = dev->frontend.get_gain(1);
+ }
+ if (gain0 > dev->frontend.get_gain(2)) {
+ gain0 = dev->frontend.get_gain(2);
+ }
+ dev->frontend.set_gain(0, gain0);
+ dev->frontend.set_gain(1, gain0);
+ dev->frontend.set_gain(2, gain0);
+ }
+
+ if (channels == 1) {
+ dev->frontend.set_gain(0, dev->frontend.get_gain(1));
+ dev->frontend.set_gain(2, dev->frontend.get_gain(1));
+ }
+
+ DBG(DBG_info, "%s: gain=(%d,%d,%d)\n", __func__,
+ dev->frontend.get_gain(0),
+ dev->frontend.get_gain(1),
+ dev->frontend.get_gain(2));
+
+ RIE (gl841_stop_action (dev));
+
+ gl841_slow_back_home(dev, SANE_TRUE);
+
+ DBGCOMPLETED;
+ return status;
+}
+
+/*
+ * wait for lamp warmup by scanning the same line until difference
+ * between 2 scans is below a threshold
+ */
+static SANE_Status
+gl841_init_regs_for_warmup (Genesys_Device * dev,
+ const Genesys_Sensor& sensor,
+ Genesys_Register_Set * local_reg,
+ int *channels, int *total_size)
+{
+ int num_pixels = (int) (4 * 300);
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ *local_reg = dev->reg;
+
+/* okay.. these should be defaults stored somewhere */
+ dev->frontend.set_gain(0, 0);
+ dev->frontend.set_gain(1, 0);
+ dev->frontend.set_gain(2, 0);
+ dev->frontend.set_offset(0, 0x80);
+ dev->frontend.set_offset(1, 0x80);
+ dev->frontend.set_offset(2, 0x80);
+
+ SetupParams params;
+ params.xres = sensor.optical_res;
+ params.yres = dev->settings.yres;
+ params.startx = sensor.dummy_pixel;
+ params.starty = 0;
+ params.pixels = num_pixels;
+ params.lines = 1;
+ params.depth = 16;
+ params.channels = *channels;
+ params.scan_method = dev->settings.scan_method;
+ if (*channels == 3) {
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ } else {
+ params.scan_mode = ScanColorMode::GRAY;
+ }
+ params.color_filter = dev->settings.color_filter;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES;
+
+ status = gl841_init_scan_regs(dev, sensor, local_reg, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ num_pixels = dev->current_setup.pixels;
+
+ *total_size = num_pixels * 3 * 2 * 1; /* colors * bytes_per_color * scan lines */
+
+ RIE(sanei_genesys_bulk_write_register(dev, *local_reg));
+
+ return status;
+}
+
+
+/*
+ * this function moves head without scanning, forward, then backward
+ * so that the head goes to park position.
+ * as a by-product, also check for lock
+ */
+static SANE_Status
+sanei_gl841_repark_head (Genesys_Device * dev)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ DBG(DBG_proc, "%s\n", __func__);
+
+ status = gl841_feed(dev,232);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to feed: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* toggle motor flag, put an huge step number and redo move backward */
+ status = gl841_slow_back_home (dev, SANE_TRUE);
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return status;
+}
+
+static bool
+gl841_is_compatible_calibration (Genesys_Device * dev, const Genesys_Sensor& sensor,
+ Genesys_Calibration_Cache *cache,
+ int for_overwrite)
+{
+#ifdef HAVE_SYS_TIME_H
+ struct timeval time;
+#endif
+
+ DBGSTART;
+
+ /* calibration cache not working yet for this model */
+ if (dev->model->ccd_type == CCD_PLUSTEK_3600)
+ {
+ return false;
+ }
+
+ gl841_calculate_current_setup (dev, sensor);
+
+ DBG(DBG_proc, "%s: checking\n", __func__);
+
+ if (dev->current_setup.ccd_size_divisor != cache->used_setup.ccd_size_divisor)
+ return false;
+
+ /* a cache entry expires after 30 minutes for non sheetfed scanners */
+ /* this is not taken into account when overwriting cache entries */
+#ifdef HAVE_SYS_TIME_H
+ if(for_overwrite == SANE_FALSE)
+ {
+ gettimeofday (&time, NULL);
+ if ((time.tv_sec - cache->last_calibration > 30 * 60)
+ && (dev->model->is_sheetfed == SANE_FALSE))
+ {
+ DBG(DBG_proc, "%s: expired entry, non compatible cache\n", __func__);
+ return false;
+ }
+ }
+#endif
+
+ DBGCOMPLETED;
+ return true;
+}
+
+/*
+ * initialize ASIC : registers, motor tables, and gamma tables
+ * then ensure scanner's head is at home
+ */
+static SANE_Status
+gl841_init (Genesys_Device * dev)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+ size_t size;
+
+ DBG_INIT ();
+ DBGSTART;
+
+ dev->scanhead_position_in_steps = 0;
+
+ /* Check if the device has already been initialized and powered up */
+ if (dev->already_initialized)
+ {
+ RIE (sanei_genesys_get_status (dev, &val));
+ if (val & REG41_PWRBIT)
+ {
+ DBG(DBG_info, "%s: already initialized\n", __func__);
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+ }
+ }
+
+ dev->dark_average_data.clear();
+ dev->white_average_data.clear();
+
+ dev->settings.color_filter = ColorFilter::RED;
+
+ /* ASIC reset */
+ RIE (sanei_genesys_write_register (dev, 0x0e, 0x01));
+ RIE (sanei_genesys_write_register (dev, 0x0e, 0x00));
+
+ /* Set default values for registers */
+ gl841_init_registers (dev);
+
+ /* Write initial registers */
+ RIE(sanei_genesys_bulk_write_register(dev, dev->reg));
+
+ /* Test ASIC and RAM */
+ if (!(dev->model->flags & GENESYS_FLAG_LAZY_INIT))
+ {
+ RIE (sanei_gl841_asic_test (dev));
+ }
+
+ const auto& sensor = sanei_genesys_find_sensor_any(dev);
+
+ /* Set analog frontend */
+ RIE (gl841_set_fe(dev, sensor, AFE_INIT));
+
+ /* Move home */
+ RIE (gl841_slow_back_home (dev, SANE_TRUE));
+
+ /* Init shading data */
+ RIE (sanei_genesys_init_shading_data(dev, sensor, sensor.sensor_pixels));
+
+ /* ensure head is correctly parked, and check lock */
+ if (dev->model->flags & GENESYS_FLAG_REPARK)
+ {
+ status = sanei_gl841_repark_head (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ if (status == SANE_STATUS_INVAL)
+ DBG(DBG_error0, "Your scanner is locked. Please move the lock switch to the unlocked "
+ "position\n");
+ else
+ DBG(DBG_error, "%s: sanei_gl841_repark_head failed: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+ }
+
+ /* send gamma tables */
+ status = gl841_send_gamma_table(dev, sensor);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to send initial gamma tables: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+
+ /* initial calibration reg values */
+ Genesys_Register_Set& regs = dev->calib_reg;
+ regs = dev->reg;
+
+ SetupParams params;
+ params.xres = 300;
+ params.yres = 300;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = (16 * 300) / sensor.optical_res;
+ params.lines = 1;
+ params.depth = 16;
+ params.channels = 3;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS;
+ params.color_filter = ColorFilter::RED;
+ params.flags = SCAN_FLAG_DISABLE_SHADING |
+ SCAN_FLAG_DISABLE_GAMMA |
+ SCAN_FLAG_SINGLE_LINE |
+ SCAN_FLAG_IGNORE_LINE_DISTANCE |
+ SCAN_FLAG_USE_OPTICAL_RES;
+
+ status = gl841_init_scan_regs(dev, sensor, &regs, params);
+
+ RIE(sanei_genesys_bulk_write_register(dev, regs));
+
+ size = dev->current_setup.pixels * 3 * 2 * 1; /* colors * bytes_per_color * scan lines */
+
+ std::vector<uint8_t> line(size);
+
+ DBG(DBG_info, "%s: starting dummy data reading\n", __func__);
+ RIE(gl841_begin_scan(dev, sensor, &regs, SANE_TRUE));
+
+ sanei_usb_set_timeout(1000);/* 1 second*/
+
+/*ignore errors. next read will succeed*/
+ sanei_genesys_read_data_from_scanner(dev, line.data(), size);
+
+ sanei_usb_set_timeout(30 * 1000);/* 30 seconds*/
+
+ RIE (gl841_end_scan(dev, &regs, SANE_TRUE));
+
+ regs = dev->reg;
+
+ /* Set powersaving (default = 15 minutes) */
+ RIE (gl841_set_powersaving (dev, 15));
+ dev->already_initialized = SANE_TRUE;
+
+ DBGCOMPLETED;
+ return SANE_STATUS_GOOD;
+}
+
+static SANE_Status
+gl841_update_hardware_sensors (Genesys_Scanner * s)
+{
+ /* do what is needed to get a new set of events, but try to not lose
+ any of them.
+ */
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint8_t val;
+
+ if (s->dev->model->gpo_type == GPO_CANONLIDE35
+ || s->dev->model->gpo_type == GPO_CANONLIDE80)
+ {
+ RIE(sanei_genesys_read_register(s->dev, REG6D, &val));
+ s->buttons[BUTTON_SCAN_SW].write((val & 0x01) == 0);
+ s->buttons[BUTTON_FILE_SW].write((val & 0x02) == 0);
+ s->buttons[BUTTON_EMAIL_SW].write((val & 0x04) == 0);
+ s->buttons[BUTTON_COPY_SW].write((val & 0x08) == 0);
+ }
+
+ if (s->dev->model->gpo_type == GPO_XP300 ||
+ s->dev->model->gpo_type == GPO_DP665 ||
+ s->dev->model->gpo_type == GPO_DP685)
+ {
+ RIE(sanei_genesys_read_register(s->dev, REG6D, &val));
+
+ s->buttons[BUTTON_PAGE_LOADED_SW].write((val & 0x01) == 0);
+ s->buttons[BUTTON_SCAN_SW].write((val & 0x02) == 0);
+ }
+
+ return status;
+}
+
+/** @brief search for a full width black or white strip.
+ * This function searches for a black or white stripe across the scanning area.
+ * When searching backward, the searched area must completely be of the desired
+ * color since this area will be used for calibration which scans forward.
+ * @param dev scanner device
+ * @param forward SANE_TRUE if searching forward, SANE_FALSE if searching backward
+ * @param black SANE_TRUE if searching for a black strip, SANE_FALSE for a white strip
+ * @return SANE_STATUS_GOOD if a matching strip is found, SANE_STATUS_UNSUPPORTED if not
+ */
+static SANE_Status
+gl841_search_strip(Genesys_Device * dev, const Genesys_Sensor& sensor,
+ SANE_Bool forward, SANE_Bool black)
+{
+ unsigned int pixels, lines, channels;
+ SANE_Status status = SANE_STATUS_GOOD;
+ Genesys_Register_Set local_reg;
+ size_t size;
+ int steps, depth, dpi;
+ unsigned int pass, count, found, x, y, length;
+ char title[80];
+ GenesysRegister *r;
+ uint8_t white_level=90; /**< default white level to detect white dots */
+ uint8_t black_level=60; /**< default black level to detect black dots */
+
+ DBG(DBG_proc, "%s %s %s\n", __func__, black ? "black" : "white", forward ? "forward" : "reverse");
+
+ /* use maximum gain when doing forward white strip detection
+ * since we don't have calibrated the sensor yet */
+ if(!black && forward)
+ {
+ dev->frontend.set_gain(0, 0xff);
+ dev->frontend.set_gain(1, 0xff);
+ dev->frontend.set_gain(2, 0xff);
+ }
+
+ gl841_set_fe(dev, sensor, AFE_SET);
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to stop: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* set up for a gray scan at lowest dpi */
+ dpi = 9600;
+ for (x = 0; x < MAX_RESOLUTIONS; x++)
+ {
+ if (dev->model->xdpi_values[x] > 0 && dev->model->xdpi_values[x] < dpi)
+ dpi = dev->model->xdpi_values[x];
+ }
+ channels = 1;
+
+ /* shading calibation is done with dev->motor.base_ydpi */
+ /* lines = (dev->model->shading_lines * dpi) / dev->motor.base_ydpi; */
+ lines = (10*dpi)/MM_PER_INCH;
+
+ depth = 8;
+ pixels = (sensor.sensor_pixels * dpi) / sensor.optical_res;
+ size = pixels * channels * lines * (depth / 8);
+ std::vector<uint8_t> data(size);
+
+ /* 20 cm max length for calibration sheet */
+ length = ((200 * dpi) / MM_PER_INCH)/lines;
+
+ dev->scanhead_position_in_steps = 0;
+
+ local_reg = dev->reg;
+
+ SetupParams params;
+ params.xres = dpi;
+ params.yres = dpi;
+ params.startx = 0;
+ params.starty = 0;
+ params.pixels = pixels;
+ params.lines = lines;
+ params.depth = depth;
+ params.channels = channels;
+ params.scan_method = dev->settings.scan_method;
+ params.scan_mode = ScanColorMode::GRAY;
+ params.color_filter = ColorFilter::RED;
+ params.flags = SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA;
+
+ status = gl841_init_scan_regs(dev, sensor, &local_reg, params);
+
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to setup for scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* set up for reverse or forward */
+ r = sanei_genesys_get_address(&local_reg, 0x02);
+ if (forward)
+ r->value &= ~4;
+ else
+ r->value |= 4;
+
+
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = gl841_begin_scan(dev, sensor, &local_reg, SANE_TRUE);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to begin scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* waits for valid data */
+ do
+ sanei_genesys_test_buffer_empty (dev, &steps);
+ while (steps);
+
+ /* now we're on target, we can read data */
+ status = sanei_genesys_read_data_from_scanner(dev, data.data(), size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to read data: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: gl841_stop_action failed\n", __func__);
+ return status;
+ }
+
+ pass = 0;
+ if (DBG_LEVEL >= DBG_data)
+ {
+ sprintf(title, "gl841_search_strip_%s_%s%02u.pnm", black ? "black" : "white",
+ forward ? "fwd" : "bwd", pass);
+ sanei_genesys_write_pnm_file(title, data.data(), depth, channels, pixels, lines);
+ }
+
+ /* loop until strip is found or maximum pass number done */
+ found = 0;
+ while (pass < length && !found)
+ {
+ status = sanei_genesys_bulk_write_register(dev, local_reg);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to bulk write registers: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+
+ /* now start scan */
+ status = gl841_begin_scan(dev, sensor, &local_reg, SANE_TRUE);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error,"%s: failed to begin scan: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ /* waits for valid data */
+ do
+ sanei_genesys_test_buffer_empty (dev, &steps);
+ while (steps);
+
+ /* now we're on target, we can read data */
+ status = sanei_genesys_read_data_from_scanner (dev, data.data(), size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "g%s: failed to read data: %s\n", __func__, sane_strstatus(status));
+ return status;
+ }
+
+ status = gl841_stop_action (dev);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: gl841_stop_action failed\n", __func__);
+ return status;
+ }
+
+ if (DBG_LEVEL >= DBG_data)
+ {
+ sprintf(title, "gl841_search_strip_%s_%s%02u.pnm",
+ black ? "black" : "white", forward ? "fwd" : "bwd", pass);
+ sanei_genesys_write_pnm_file(title, data.data(), depth, channels, pixels, lines);
+ }
+
+ /* search data to find black strip */
+ /* when searching forward, we only need one line of the searched color since we
+ * will scan forward. But when doing backward search, we need all the area of the
+ * same color */
+ if (forward)
+ {
+ for (y = 0; y < lines && !found; y++)
+ {
+ count = 0;
+ /* count of white/black pixels depending on the color searched */
+ for (x = 0; x < pixels; x++)
+ {
+ /* when searching for black, detect white pixels */
+ if (black && data[y * pixels + x] > white_level)
+ {
+ count++;
+ }
+ /* when searching for white, detect black pixels */
+ if (!black && data[y * pixels + x] < black_level)
+ {
+ count++;
+ }
+ }
+
+ /* at end of line, if count >= 3%, line is not fully of the desired color
+ * so we must go to next line of the buffer */
+ /* count*100/pixels < 3 */
+ if ((count * 100) / pixels < 3)
+ {
+ found = 1;
+ DBG(DBG_data, "%s: strip found forward during pass %d at line %d\n", __func__,
+ pass, y);
+ }
+ else
+ {
+ DBG(DBG_data, "%s: pixels=%d, count=%d (%d%%)\n", __func__, pixels, count,
+ (100 * count) / pixels);
+ }
+ }
+ }
+ else /* since calibration scans are done forward, we need the whole area
+ to be of the required color when searching backward */
+ {
+ count = 0;
+ for (y = 0; y < lines; y++)
+ {
+ /* count of white/black pixels depending on the color searched */
+ for (x = 0; x < pixels; x++)
+ {
+ /* when searching for black, detect white pixels */
+ if (black && data[y * pixels + x] > white_level)
+ {
+ count++;
+ }
+ /* when searching for white, detect black pixels */
+ if (!black && data[y * pixels + x] < black_level)
+ {
+ count++;
+ }
+ }
+ }
+
+ /* at end of area, if count >= 3%, area is not fully of the desired color
+ * so we must go to next buffer */
+ if ((count * 100) / (pixels * lines) < 3)
+ {
+ found = 1;
+ DBG(DBG_data, "%s: strip found backward during pass %d \n", __func__, pass);
+ }
+ else
+ {
+ DBG(DBG_data, "%s: pixels=%d, count=%d (%d%%)\n", __func__, pixels, count,
+ (100 * count) / pixels);
+ }
+ }
+ pass++;
+ }
+
+ if (found)
+ {
+ status = SANE_STATUS_GOOD;
+ DBG(DBG_info, "%s: %s strip found\n", __func__, black ? "black" : "white");
+ }
+ else
+ {
+ status = SANE_STATUS_UNSUPPORTED;
+ DBG(DBG_info, "%s: %s strip not found\n", __func__, black ? "black" : "white");
+ }
+
+ DBG(DBG_proc, "%s: completed\n", __func__);
+ return status;
+}
+
+/**
+ * Send shading calibration data. The buffer is considered to always hold values
+ * for all the channels.
+ */
+static
+SANE_Status
+gl841_send_shading_data (Genesys_Device * dev, const Genesys_Sensor& sensor,
+ uint8_t * data, int size)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ uint32_t length, x, factor, pixels, i;
+ uint32_t lines, channels;
+ uint16_t dpiset, dpihw, strpixel ,endpixel, beginpixel;
+ uint8_t *ptr,*src;
+
+ DBGSTART;
+ DBG(DBG_io2, "%s: writing %d bytes of shading data\n", __func__, size);
+
+ /* old method if no SHDAREA */
+ if((dev->reg.find_reg(0x01).value & REG01_SHDAREA) == 0)
+ {
+ /* start address */
+ status = sanei_genesys_set_buffer_address (dev, 0x0000);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to set buffer address: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+
+ /* shading data whole line */
+ status = dev->model->cmd_set->bulk_write_data (dev, 0x3c, data, size);
+ if (status != SANE_STATUS_GOOD)
+ {
+ DBG(DBG_error, "%s: failed to send shading table: %s\n", __func__,
+ sane_strstatus(status));
+ return status;
+ }
+ DBGCOMPLETED;
+ return status;
+ }
+
+ /* data is whole line, we extract only the part for the scanned area */
+ length = (uint32_t) (size / 3);
+ sanei_genesys_get_double(&dev->reg,REG_STRPIXEL,&strpixel);
+ sanei_genesys_get_double(&dev->reg,REG_ENDPIXEL,&endpixel);
+ DBG(DBG_io2, "%s: STRPIXEL=%d, ENDPIXEL=%d, PIXELS=%d\n", __func__, strpixel, endpixel,
+ endpixel-strpixel);
+
+ /* compute deletion/average factor */
+ sanei_genesys_get_double(&dev->reg,REG_DPISET,&dpiset);
+ dpihw = gl841_get_dpihw(dev);
+ unsigned ccd_size_divisor = dev->current_setup.ccd_size_divisor;
+ factor=dpihw/dpiset;
+ DBG(DBG_io2, "%s: dpihw=%d, dpiset=%d, ccd_size_divisor=%d, factor=%d\n", __func__, dpihw, dpiset,
+ ccd_size_divisor, factor);
+
+ /* binary data logging */
+ if(DBG_LEVEL>=DBG_data)
+ {
+ dev->binary=fopen("binary.pnm","wb");
+ sanei_genesys_get_triple(&dev->reg, REG_LINCNT, &lines);
+ channels=dev->current_setup.channels;
+ if(dev->binary!=NULL)
+ {
+ fprintf(dev->binary,"P5\n%d %d\n%d\n",(endpixel-strpixel)/factor*channels,lines/channels,255);
+ }
+ }
+
+ /* turn pixel value into bytes 2x16 bits words */
+ strpixel*=2*2; /* 2 words of 2 bytes */
+ endpixel*=2*2;
+ pixels=endpixel-strpixel;
+
+ /* shading pixel begin is start pixel minus start pixel during shading
+ * calibration. Currently only cases handled are full and half ccd resolution.
+ */
+ beginpixel = sensor.CCD_start_xoffset / ccd_size_divisor;
+ beginpixel += sensor.dummy_pixel + 1;
+ DBG(DBG_io2, "%s: ORIGIN PIXEL=%d\n", __func__, beginpixel);
+ beginpixel = (strpixel-beginpixel*2*2)/factor;
+ DBG(DBG_io2, "%s: BEGIN PIXEL=%d\n", __func__, beginpixel/4);
+
+ DBG(DBG_io2, "%s: using chunks of %d bytes (%d shading data pixels)\n", __func__, length,
+ length/4);
+ std::vector<uint8_t> buffer(pixels, 0);
+
+ /* write actual shading data contigously
+ * channel by channel, starting at addr 0x0000
+ * */
+ for(i=0;i<3;i++)
+ {
+ /* copy data to work buffer and process it */
+ /* coefficent destination */
+ ptr=buffer.data();
+
+ /* iterate on both sensor segment, data has been averaged,
+ * so is in the right order and we only have to copy it */
+ for(x=0;x<pixels;x+=4)
+ {
+ /* coefficient source */
+ src=data+x+beginpixel+i*length;
+ ptr[0]=src[0];
+ ptr[1]=src[1];
+ ptr[2]=src[2];
+ ptr[3]=src[3];
+
+ /* next shading coefficient */
+ ptr+=4;
+ }
+
+ /* 0x5400 alignment for LIDE80 internal memory */
+ RIE(sanei_genesys_set_buffer_address(dev, 0x5400*i));
+ RIE(dev->model->cmd_set->bulk_write_data(dev, 0x3c, buffer.data(), pixels));
+ }
+
+ DBGCOMPLETED;
+
+ return status;
+}
+
+
+/** the gl841 command set */
+static Genesys_Command_Set gl841_cmd_set = {
+ "gl841-generic", /* the name of this set */
+
+ [](Genesys_Device* dev) -> bool { (void) dev; return true; },
+
+ gl841_init,
+ gl841_init_regs_for_warmup,
+ gl841_init_regs_for_coarse_calibration,
+ gl841_init_regs_for_shading,
+ gl841_init_regs_for_scan,
+
+ gl841_get_filter_bit,
+ gl841_get_lineart_bit,
+ gl841_get_bitset_bit,
+ gl841_get_gain4_bit,
+ gl841_get_fast_feed_bit,
+ gl841_test_buffer_empty_bit,
+ gl841_test_motor_flag_bit,
+
+ gl841_set_fe,
+ gl841_set_powersaving,
+ gl841_save_power,
+
+ gl841_begin_scan,
+ gl841_end_scan,
+
+ gl841_send_gamma_table,
+
+ gl841_search_start_position,
+
+ gl841_offset_calibration,
+ gl841_coarse_gain_calibration,
+ gl841_led_calibration,
+
+ NULL,
+ gl841_slow_back_home,
+ NULL,
+
+ sanei_genesys_bulk_write_register,
+ sanei_genesys_bulk_write_data,
+ sanei_genesys_bulk_read_data,
+
+ gl841_update_hardware_sensors,
+
+ gl841_load_document,
+ gl841_detect_document_end,
+ gl841_eject_document,
+ gl841_search_strip,
+
+ gl841_is_compatible_calibration,
+ NULL,
+ gl841_send_shading_data,
+ gl841_calculate_current_setup,
+ NULL
+};
+
+SANE_Status
+sanei_gl841_init_cmd_set (Genesys_Device * dev)
+{
+ dev->model->cmd_set = &gl841_cmd_set;
+ return SANE_STATUS_GOOD;
+}